Compare commits
	
		
			407 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 1f7272d139 | ||
|  | f6ba1ad788 | ||
|  | 294d59c9bb | ||
|  | 759f72169a | ||
|  | 1f7135be61 | ||
|  | 6942f9c1cf | ||
|  | d9da75d1c0 | ||
|  | 3503c98857 | ||
|  | 708c3f1e2a | ||
|  | 6f645e8619 | ||
|  | bce7ca7ac4 | ||
|  | 350465c25d | ||
|  | 5b9c70ae22 | ||
|  | 9b30afeca9 | ||
|  | c1b202c119 | ||
|  | 41cfe5d2ca | ||
|  | 05339e184f | ||
|  | 447127d956 | ||
|  | 394334fbea | ||
|  | 9f8cd33d43 | ||
|  | f066e28c35 | ||
|  | b349a449bb | ||
|  | 1c5898d396 | ||
|  | 6802967863 | ||
|  | 0462f18680 | ||
|  | af6699098f | ||
|  | 6b7e7dc124 | ||
|  | 6bae4c6a66 | ||
|  | 46da918dbe | ||
|  | bb7e5f17b5 | ||
|  | b9d03114c2 | ||
|  | 436b1ce176 | ||
|  | 50fb5d83f1 | ||
|  | fda672f806 | ||
|  | 2bf783b04d | ||
|  | 2f72b23a0d | ||
|  | 85336f9777 | ||
|  | 174d964553 | ||
|  | cf8677248e | ||
|  | 1e6a3163af | ||
|  | e008919978 | ||
|  | 4814066c67 | ||
|  | f17f8b48c2 | ||
|  | ab0aec0ac5 | ||
|  | b49a641ba5 | ||
|  | 2f50051426 | ||
|  | 43cc32db40 | ||
|  | b4d6f6b947 | ||
|  | 71ff533623 | ||
|  | e33a5bbef5 | ||
|  | 6c0112c2be | ||
|  | 15bbf26b93 | ||
|  | 87c97efce0 | ||
|  | 6c4aee1479 | ||
|  | 73549a9044 | ||
|  | 30fdd3e184 | ||
|  | c97eb5d63f | ||
|  | 5729c7d5e7 | ||
|  | d77b13efcb | ||
|  | c43faca7b9 | ||
|  | 892ddd5724 | ||
|  | a9de779f33 | ||
|  | 1c2f016ba0 | ||
|  | 7b4d9140af | ||
|  | c1fc87ff4e | ||
|  | cd5ea5d4e0 | ||
|  | 30c01089f5 | ||
|  | 89825a2b21 | ||
|  | a743b75bb4 | ||
|  | f7ebf8dedd | ||
|  | f6220cab3b | ||
|  | 0c5e1c4138 | ||
|  | 03fe431f1a | ||
|  | a8e4554fec | ||
|  | e81b09b9aa | ||
|  | c6e846e0ae | ||
|  | 03dcfb5c4b | ||
|  | 3e54da03e2 | ||
|  | c4b3196917 | ||
|  | 0d81e7933e | ||
|  | b2a2735034 | ||
|  | f865c5de90 | ||
|  | 4159369e8b | ||
|  | 170693cf0b | ||
|  | 4e7b5d4af8 | ||
|  | 67bf789fcf | ||
|  | f5cf616c2f | ||
|  | 7975f19817 | ||
|  | 017602056d | ||
|  | c63f43854b | ||
|  | 5cc71ec2ad | ||
|  | 80e81f8475 | ||
|  | 3685c8e015 | ||
|  | 99e943c365 | ||
|  | 21818e71f5 | ||
|  | bcc6d25e21 | ||
|  | 7b885ee0d3 | ||
|  | c10e808a4f | ||
|  | 54e9be0ed8 | ||
|  | 938cdf316a | ||
|  | 27c33911e6 | ||
|  | e88f8759e7 | ||
|  | f2992e3165 | ||
|  | c71fd1ee3b | ||
|  | fb45b19fdc | ||
|  | c4ea8d4942 | ||
|  | 646aa131ef | ||
|  | 0adb40bf92 | ||
|  | 17d6014bf1 | ||
|  | ff57cd4eaf | ||
|  | 74bd7c3744 | ||
|  | cfbb283f85 | ||
|  | 74a3c4451b | ||
|  | be3643c962 | ||
|  | f4aa546af8 | ||
|  | 67b876a7f4 | ||
|  | 94e177c0ef | ||
|  | 1bd83cc9bc | ||
|  | ecda3f4a7d | ||
|  | 8f972a965d | ||
|  | 0f051fc57c | ||
|  | c3f8925f46 | ||
|  | 5d0cab2052 | ||
|  | 4d7492f682 | ||
|  | fc9d99080f | ||
|  | 47ebac0276 | ||
|  | cb3fca03e9 | ||
|  | abbbd83729 | ||
|  | 1743ab7812 | ||
|  | 324e3972a6 | ||
|  | 1502dda2ab | ||
|  | f31b2c4a79 | ||
|  | 89b9b60e0c | ||
|  | de9ba12779 | ||
|  | 9cc4359c04 | ||
|  | 67eaf120b9 | ||
|  | b8353c4a33 | ||
|  | 7013033ae4 | ||
|  | cb8cd03852 | ||
|  | f63fb62014 | ||
|  | 2e4fb86b86 | ||
|  | 5e776a07dd | ||
|  | 81e637e50e | ||
|  | 0971ad0a80 | ||
|  | 8267ded7ec | ||
|  | 7f36ea55f5 | ||
|  | 72a051f2d3 | ||
|  | 51b197888c | ||
|  | cd63865d31 | ||
|  | 5be5685a09 | ||
|  | 76b2f25d46 | ||
|  | 58607d4a7f | ||
|  | c0a5b16a7f | ||
|  | 3a0c69005b | ||
|  | 5c295fb9e3 | ||
|  | 4ee212e7d5 | ||
|  | 70651ce994 | ||
|  | a778a91106 | ||
|  | cfc31eead3 | ||
|  | da0a1bbe9f | ||
|  | bc66fb33e9 | ||
|  | b1b6493755 | ||
|  | 1d189f239b | ||
|  | 5b90691bcc | ||
|  | d1d5972277 | ||
|  | 2c07d77368 | ||
|  | 642cfbf59a | ||
|  | bb1367cfb9 | ||
|  | 11724aa555 | ||
|  | 4d374712de | ||
|  | eb9003187d | ||
|  | caba444962 | ||
|  | 5b6c8c191f | ||
|  | dd51589f67 | ||
|  | b02a31d4b9 | ||
|  | 0e7878b406 | ||
|  | cae91ce0c5 | ||
|  | 67a65a2aa9 | ||
|  | 364b0a7163 | ||
|  | d6419f2059 | ||
|  | 6f7ad7ef91 | ||
|  | 5ae588833b | ||
|  | a70dbac0e6 | ||
|  | 4d34a02afe | ||
|  | 4db4f45897 | ||
|  | 2d5280fc95 | ||
|  | b8d568761e | ||
|  | 29309dac9a | ||
|  | 7f7745071a | ||
|  | 1914032e35 | ||
|  | f44c8f1205 | ||
|  | fe2ef4e61c | ||
|  | fc3eda55c7 | ||
|  | 8adf1cdd02 | ||
|  | adbbc656d4 | ||
|  | 8e852bce02 | ||
|  | bb461b009f | ||
|  | 03559a3cc4 | ||
|  | 7bb2fe128a | ||
|  | 2312e17a8e | ||
|  | 9835b382da | ||
|  | 1eacc6fbff | ||
|  | 85187239b6 | ||
|  | 819ff2a902 | ||
|  | c744104a18 | ||
|  | c87801f0a9 | ||
|  | 39735594bd | ||
|  | 30964f65e4 | ||
|  | ee0c7fd8bf | ||
|  | dfdecef8e7 | ||
|  | edcdfeb057 | ||
|  | 47f0de9836 | ||
|  | 9ba657797e | ||
|  | 07442a6f84 | ||
|  | 3faf3c84be | ||
|  | abcacc82f3 | ||
|  | 9544b7d968 | ||
|  | babbc8bcd6 | ||
|  | 12809ebc74 | ||
|  | b45a601ad2 | ||
|  | f099dc6a37 | ||
|  | 803caddbd4 | ||
|  | 4d7b988018 | ||
|  | c1f88a4e14 | ||
|  | 5d9ec0b208 | ||
|  | 1877cacf9c | ||
|  | 2f4978cfea | ||
|  | d27a1103fa | ||
|  | b85bb95082 | ||
|  | db7f93cff3 | ||
|  | 85e271098f | ||
|  | 17001e2f74 | ||
|  | c82f4f0d45 | ||
|  | 88247a3af9 | ||
|  | 158578a406 | ||
|  | 19314e7e06 | ||
|  | 8bcbc6d545 | ||
|  | ef55e6d476 | ||
|  | 295ef3dc1d | ||
|  | 9d125c9e79 | ||
|  | 86363986fc | ||
|  | 0a2dbbc58b | ||
|  | 673a966541 | ||
|  | db1e69813b | ||
|  | e60d56f060 | ||
|  | 328e062ae9 | ||
|  | 0523c2ea4b | ||
|  | c5c7378c63 | ||
|  | 9b2080d036 | ||
|  | d4b3649640 | ||
|  | b085993901 | ||
|  | 0d4afad342 | ||
|  | 0da694b845 | ||
|  | 6d5e7d9e81 | ||
|  | bc08bea284 | ||
|  | 0e5a0661e1 | ||
|  | a839bd428f | ||
|  | 0277062693 | ||
|  | 7affa5ab69 | ||
|  | ed22af4e73 | ||
|  | 63ebb6998e | ||
|  | 7914cd47ca | ||
|  | 708dbac70e | ||
|  | 1b62dd5c40 | ||
|  | 4911545843 | ||
|  | c5cc4b7867 | ||
|  | eacb614750 | ||
|  | 341e1e7a6d | ||
|  | a02c820c2d | ||
|  | 2f6890c78a | ||
|  | 516591fe88 | ||
|  | d2941a9110 | ||
|  | f7302f710b | ||
|  | 6a02ac7e80 | ||
|  | d1b86fdef5 | ||
|  | 57ac38ddca | ||
|  | 7a73a92074 | ||
|  | d1b30f4792 | ||
|  | 16dcf78cab | ||
|  | d868cfdeb0 | ||
|  | c074f4d925 | ||
|  | 453024c58d | ||
|  | fe8340617a | ||
|  | b024dd913d | ||
|  | a2a698ab0e | ||
|  | bb56f92213 | ||
|  | 8dcd998945 | ||
|  | bcbbbe4046 | ||
|  | 7200a8cb84 | ||
|  | 6925344807 | ||
|  | 60ceeb0ddd | ||
|  | 06caabf333 | ||
|  | 954131bd51 | ||
|  | 855efe7fe8 | ||
|  | d902a74ab0 | ||
|  | 499e11f730 | ||
|  | 6db59a9c31 | ||
|  | 6465726008 | ||
|  | 3a3b96e0be | ||
|  | 992c91dc0c | ||
|  | 809473c15c | ||
|  | d79a5ec3d6 | ||
|  | 237469ceaf | ||
|  | c28d9135d9 | ||
|  | 48a5679087 | ||
|  | 7c938712f2 | ||
|  | 4df12bebc2 | ||
|  | dfe8987aaa | ||
|  | 02dbe401d8 | ||
|  | c18f8c92e7 | ||
|  | 857cd718df | ||
|  | 11d4f6499a | ||
|  | f2c25b4744 | ||
|  | 27b846717f | ||
|  | 9ed138f896 | ||
|  | 1978dc80eb | ||
|  | fc4b247f4f | ||
|  | ebf7056f4a | ||
|  | eb975d7e13 | ||
|  | a2dd8cb6b9 | ||
|  | 7c254c6136 | ||
|  | c8a33b83f1 | ||
|  | 1145c72b01 | ||
|  | 7fc45fb711 | ||
|  | e146262c38 | ||
|  | 6f808bd06e | ||
|  | 0b6ab49325 | ||
|  | 66d9182e50 | ||
|  | 654cca82a9 | ||
|  | 89785da1c5 | ||
|  | 2f9964e46e | ||
|  | 168ecd67b0 | ||
|  | bcbe740598 | ||
|  | 86c8929d77 | ||
|  | 6738a9433b | ||
|  | 23843ec86e | ||
|  | f4db0da585 | ||
|  | 9ee3b796cd | ||
|  | f57569f553 | ||
|  | fffd0e8990 | ||
|  | 200e52bab5 | ||
|  | a0ef649dd8 | ||
|  | 0dd01bda01 | ||
|  | a707598042 | ||
|  | 8a3171308a | ||
|  | 29c887f30b | ||
|  | 661398d891 | ||
|  | 2cd722d751 | ||
|  | 49f5b4fa5c | ||
|  | 67baf465f4 | ||
|  | ee7666ddea | ||
|  | 02fc41ff1c | ||
|  | d07a9d2ef8 | ||
|  | 3622ebfabd | ||
|  | 70b320633f | ||
|  | f30208f345 | ||
|  | 5bcc454678 | ||
|  | 473110568f | ||
|  | 88ca0f8196 | ||
|  | a171005010 | ||
|  | f56ad2fa58 | ||
|  | c9dc441915 | ||
|  | a0d255369a | ||
|  | 40b0a15b35 | ||
|  | b98b06ff79 | ||
|  | a448c9aebf | ||
|  | b3f462a39d | ||
|  | 7ce34ca019 | ||
|  | 719bb53c3a | ||
|  | 214415969f | ||
|  | 7431b1f123 | ||
|  | d8ffa843a9 | ||
|  | a69db231cc | ||
|  | c17f94422f | ||
|  | b4777f7f4f | ||
|  | a57d9a9303 | ||
|  | 5e70e1bcb2 | ||
|  | 0c43787996 | ||
|  | dc310b99f9 | ||
|  | e98c5e10bc | ||
|  | f1b1090263 | ||
|  | 6efd6faa3f | ||
|  | 1e4d48d371 | ||
|  | 93a2adb3e6 | ||
|  | a66d516777 | ||
|  | 7a97d42338 | ||
|  | b66cdc8fa0 | ||
|  | 67f43b2aad | ||
|  | d143e50238 | ||
|  | e27439be6a | ||
|  | 2ad5ffbda2 | ||
|  | dae9e662a5 | ||
|  | f22737d6a4 | ||
|  | a458d5a176 | ||
|  | d92ed04538 | ||
|  | 80b3df8953 | ||
|  | bcf83ec761 | ||
|  | e44e72bce3 | ||
|  | 35f2781518 | ||
|  | dc5512e403 | ||
|  | a7ca9950fc | ||
|  | e0dd33e6be | ||
|  | 2e718e1130 | ||
|  | 5cfd8909a8 | ||
|  | d92f992c01 | ||
|  | 20a5d9051d | ||
|  | 782d48594a | 
							
								
								
									
										62
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										62
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,28 +1,60 @@ | |||||||
| # http://travis-ci.org/#!/MongoEngine/mongoengine | # http://travis-ci.org/#!/MongoEngine/mongoengine | ||||||
| language: python | language: python | ||||||
| services: mongodb |  | ||||||
| python: | python: | ||||||
|     - "2.6" |     - "2.6" | ||||||
|     - "2.7" |     - "2.7" | ||||||
|     - "3.2" |     - "3.2" | ||||||
|     - "3.3" |     - "3.3" | ||||||
|  |     - "3.4" | ||||||
|  |     - "pypy" | ||||||
|  |     - "pypy3" | ||||||
| env: | env: | ||||||
|   - PYMONGO=dev DJANGO=1.5.1 |   - PYMONGO=dev DJANGO=dev | ||||||
|   - PYMONGO=dev DJANGO=1.4.2 |   - PYMONGO=dev DJANGO=1.6.5 | ||||||
|   - PYMONGO=2.5 DJANGO=1.5.1 |   - PYMONGO=dev DJANGO=1.5.8 | ||||||
|   - PYMONGO=2.5 DJANGO=1.4.2 |   - PYMONGO=2.7.1 DJANGO=dev | ||||||
|   - PYMONGO=3.2 DJANGO=1.5.1 |   - PYMONGO=2.7.1 DJANGO=1.6.5 | ||||||
|   - PYMONGO=3.3 DJANGO=1.5.1 |   - PYMONGO=2.7.1 DJANGO=1.5.8 | ||||||
|  |   - PYMONGO=2.7.2 DJANGO=dev | ||||||
|  |   - PYMONGO=2.7.2 DJANGO=1.6.5 | ||||||
|  |   - PYMONGO=2.7.2 DJANGO=1.5.8 | ||||||
|  |  | ||||||
|  | matrix: | ||||||
|  |     exclude: | ||||||
|  |         - python: "2.6" | ||||||
|  |           env: PYMONGO=dev DJANGO=dev | ||||||
|  |         - python: "2.6" | ||||||
|  |           env: PYMONGO=2.7.1 DJANGO=dev | ||||||
|  |         - python: "2.6" | ||||||
|  |           env: PYMONGO=2.7.2 DJANGO=dev | ||||||
|  |     allow_failures: | ||||||
|  |         - python: "pypy3" | ||||||
|  |     fast_finish: true | ||||||
|  |  | ||||||
|  | before_install: | ||||||
|  |   - "travis_retry sudo apt-key adv --keyserver hkp://keyserver.ubuntu.com:80 --recv 7F0CEB10" | ||||||
|  |   - "echo 'deb http://downloads-distro.mongodb.org/repo/ubuntu-upstart dist 10gen' | sudo tee /etc/apt/sources.list.d/mongodb.list" | ||||||
|  |   - "travis_retry sudo apt-get update" | ||||||
|  |   - "travis_retry sudo apt-get install mongodb-org-server" | ||||||
|  |  | ||||||
| install: | install: | ||||||
|     - if [[ $TRAVIS_PYTHON_VERSION == '2.'* ]]; then cp /usr/lib/*/libz.so $VIRTUAL_ENV/lib/; fi |     - sudo apt-get install python-dev python3-dev libopenjpeg-dev zlib1g-dev libjpeg-turbo8-dev libtiff4-dev libjpeg8-dev libfreetype6-dev liblcms2-dev libwebp-dev tcl8.5-dev tk8.5-dev python-tk | ||||||
|     - if [[ $TRAVIS_PYTHON_VERSION == '2.'* ]]; then pip install pil --use-mirrors ; true; fi |     - if [[ $PYMONGO == 'dev' ]]; then travis_retry pip install https://github.com/mongodb/mongo-python-driver/tarball/master; true; fi | ||||||
|     - if [[ $TRAVIS_PYTHON_VERSION == '2.'* ]]; then pip install django==$DJANGO --use-mirrors ; true; fi |     - if [[ $PYMONGO != 'dev' ]]; then travis_retry pip install pymongo==$PYMONGO; true; fi | ||||||
|     - if [[ $PYMONGO == 'dev' ]]; then pip install https://github.com/mongodb/mongo-python-driver/tarball/master; true; fi |     - if [[ $DJANGO == 'dev' ]]; then travis_retry pip install https://www.djangoproject.com/download/1.7c2/tarball/; fi | ||||||
|     - if [[ $PYMONGO != 'dev' ]]; then pip install pymongo==$PYMONGO --use-mirrors; true; fi |     - if [[ $DJANGO != 'dev' ]]; then travis_retry pip install Django==$DJANGO; fi | ||||||
|     - pip install https://pypi.python.org/packages/source/p/python-dateutil/python-dateutil-2.1.tar.gz#md5=1534bb15cf311f07afaa3aacba1c028b |     - travis_retry pip install https://pypi.python.org/packages/source/p/python-dateutil/python-dateutil-2.1.tar.gz#md5=1534bb15cf311f07afaa3aacba1c028b | ||||||
|     - python setup.py install |     - travis_retry pip install coveralls | ||||||
|  |     - travis_retry python setup.py install | ||||||
|  |  | ||||||
| script: | script: | ||||||
|     - python setup.py test |     - travis_retry python setup.py test | ||||||
|  |     - if [[ $TRAVIS_PYTHON_VERSION == '3.'* ]]; then 2to3 . -w; fi; | ||||||
|  |     - coverage run --source=mongoengine setup.py test | ||||||
|  |     - coverage report -m | ||||||
|  |     - python benchmark.py | ||||||
|  | after_script: | ||||||
|  |   coveralls --verbose | ||||||
| notifications: | notifications: | ||||||
|   irc: "irc.freenode.org#mongoengine" |   irc: "irc.freenode.org#mongoengine" | ||||||
| branches: | branches: | ||||||
|   | |||||||
							
								
								
									
										44
									
								
								AUTHORS
									
									
									
									
									
								
							
							
						
						
									
										44
									
								
								AUTHORS
									
									
									
									
									
								
							| @@ -134,7 +134,7 @@ that much better: | |||||||
|  * Paul Swartz |  * Paul Swartz | ||||||
|  * Sundar Raman |  * Sundar Raman | ||||||
|  * Benoit Louy |  * Benoit Louy | ||||||
|  * lraucy |  * Loic Raucy (https://github.com/lraucy) | ||||||
|  * hellysmile |  * hellysmile | ||||||
|  * Jaepil Jeong |  * Jaepil Jeong | ||||||
|  * Daniil Sharou |  * Daniil Sharou | ||||||
| @@ -142,7 +142,7 @@ that much better: | |||||||
|  * Pete Campton |  * Pete Campton | ||||||
|  * Martyn Smith |  * Martyn Smith | ||||||
|  * Marcelo Anton |  * Marcelo Anton | ||||||
|  * Aleksey Porfirov |  * Aleksey Porfirov (https://github.com/lexqt) | ||||||
|  * Nicolas Trippar |  * Nicolas Trippar | ||||||
|  * Manuel Hermann |  * Manuel Hermann | ||||||
|  * Gustavo Gawryszewski |  * Gustavo Gawryszewski | ||||||
| @@ -171,4 +171,42 @@ that much better: | |||||||
|  * Michael Bartnett (https://github.com/michaelbartnett) |  * Michael Bartnett (https://github.com/michaelbartnett) | ||||||
|  * Alon Horev (https://github.com/alonho) |  * Alon Horev (https://github.com/alonho) | ||||||
|  * Kelvin Hammond (https://github.com/kelvinhammond) |  * Kelvin Hammond (https://github.com/kelvinhammond) | ||||||
|  * Jatin- (https://github.com/jatin-) |  * Jatin Chopra (https://github.com/jatin) | ||||||
|  |  * Paul Uithol (https://github.com/PaulUithol) | ||||||
|  |  * Thom Knowles (https://github.com/fleat) | ||||||
|  |  * Paul (https://github.com/squamous) | ||||||
|  |  * Olivier Cortès (https://github.com/Karmak23) | ||||||
|  |  * crazyzubr (https://github.com/crazyzubr) | ||||||
|  |  * FrankSomething (https://github.com/FrankSomething) | ||||||
|  |  * Alexandr Morozov (https://github.com/LK4D4) | ||||||
|  |  * mishudark (https://github.com/mishudark) | ||||||
|  |  * Joe Friedl (https://github.com/grampajoe) | ||||||
|  |  * Daniel Ward (https://github.com/danielward) | ||||||
|  |  * Aniket Deshpande (https://github.com/anicake) | ||||||
|  |  * rfkrocktk (https://github.com/rfkrocktk) | ||||||
|  |  * Gustavo Andrés Angulo (https://github.com/woakas) | ||||||
|  |  * Dmytro Popovych (https://github.com/drudim) | ||||||
|  |  * Tom (https://github.com/tomprimozic) | ||||||
|  |  * j0hnsmith (https://github.com/j0hnsmith) | ||||||
|  |  * Damien Churchill (https://github.com/damoxc) | ||||||
|  |  * Jonathan Simon Prates (https://github.com/jonathansp) | ||||||
|  |  * Thiago Papageorgiou (https://github.com/tmpapageorgiou) | ||||||
|  |  * Omer Katz (https://github.com/thedrow) | ||||||
|  |  * Falcon Dai (https://github.com/falcondai) | ||||||
|  |  * Polyrabbit (https://github.com/polyrabbit) | ||||||
|  |  * Sagiv Malihi (https://github.com/sagivmalihi) | ||||||
|  |  * Dmitry Konishchev (https://github.com/KonishchevDmitry) | ||||||
|  |  * Martyn Smith (https://github.com/martynsmith) | ||||||
|  |  * Andrei Zbikowski (https://github.com/b1naryth1ef) | ||||||
|  |  * Ronald van Rij (https://github.com/ronaldvanrij) | ||||||
|  |  * François Schmidts (https://github.com/jaesivsm) | ||||||
|  |  * Eric Plumb (https://github.com/professorplumb) | ||||||
|  |  * Damien Churchill (https://github.com/damoxc) | ||||||
|  |  * Aleksandr Sorokoumov (https://github.com/Gerrrr) | ||||||
|  |  * Clay McClure (https://github.com/claymation) | ||||||
|  |  * Bruno Rocha (https://github.com/rochacbruno) | ||||||
|  |  * Norberto Leite (https://github.com/nleite) | ||||||
|  |  * Bob Cribbs (https://github.com/bocribbz) | ||||||
|  |  * Jay Shirley (https://github.com/jshirley) | ||||||
|  |  * DavidBord (https://github.com/DavidBord) | ||||||
|  |  * Axel Haustant (https://github.com/noirbizarre) | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								README.rst
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								README.rst
									
									
									
									
									
								
							| @@ -9,6 +9,13 @@ MongoEngine | |||||||
| .. image:: https://secure.travis-ci.org/MongoEngine/mongoengine.png?branch=master | .. image:: https://secure.travis-ci.org/MongoEngine/mongoengine.png?branch=master | ||||||
|   :target: http://travis-ci.org/MongoEngine/mongoengine |   :target: http://travis-ci.org/MongoEngine/mongoengine | ||||||
|    |    | ||||||
|  | .. image:: https://coveralls.io/repos/MongoEngine/mongoengine/badge.png?branch=master  | ||||||
|  |   :target: https://coveralls.io/r/MongoEngine/mongoengine?branch=master | ||||||
|  |    | ||||||
|  | .. image:: https://landscape.io/github/MongoEngine/mongoengine/master/landscape.png | ||||||
|  |    :target: https://landscape.io/github/MongoEngine/mongoengine/master | ||||||
|  |    :alt: Code Health | ||||||
|  |  | ||||||
| About | About | ||||||
| ===== | ===== | ||||||
| MongoEngine is a Python Object-Document Mapper for working with MongoDB. | MongoEngine is a Python Object-Document Mapper for working with MongoDB. | ||||||
| @@ -26,9 +33,18 @@ setup.py install``. | |||||||
|  |  | ||||||
| Dependencies | Dependencies | ||||||
| ============ | ============ | ||||||
| - pymongo 2.5+ | - pymongo>=2.7.1 | ||||||
| - sphinx (optional - for documentation generation) | - sphinx (optional - for documentation generation) | ||||||
|  |  | ||||||
|  | Optional Dependencies | ||||||
|  | --------------------- | ||||||
|  | - **Django Integration:** Django>=1.4.0 for Python 2.x or PyPy and Django>=1.5.0 for Python 3.x | ||||||
|  | - **Image Fields**: Pillow>=2.0.0 or PIL (not recommended since MongoEngine is tested with Pillow) | ||||||
|  | - dateutil>=2.1.0 | ||||||
|  |  | ||||||
|  | .. note | ||||||
|  |    MongoEngine always runs it's test suite against the latest patch version of each dependecy. e.g.: Django 1.6.5 | ||||||
|  |  | ||||||
| Examples | Examples | ||||||
| ======== | ======== | ||||||
| Some simple examples of what MongoEngine code looks like:: | Some simple examples of what MongoEngine code looks like:: | ||||||
|   | |||||||
							
								
								
									
										67
									
								
								benchmark.py
									
									
									
									
									
								
							
							
						
						
									
										67
									
								
								benchmark.py
									
									
									
									
									
								
							| @@ -15,7 +15,7 @@ def cprofile_main(): | |||||||
|     class Noddy(Document): |     class Noddy(Document): | ||||||
|         fields = DictField() |         fields = DictField() | ||||||
|  |  | ||||||
|     for i in xrange(1): |     for i in range(1): | ||||||
|         noddy = Noddy() |         noddy = Noddy() | ||||||
|         for j in range(20): |         for j in range(20): | ||||||
|             noddy.fields["key" + str(j)] = "value " + str(j) |             noddy.fields["key" + str(j)] = "value " + str(j) | ||||||
| @@ -113,6 +113,7 @@ def main(): | |||||||
|     4.68946313858 |     4.68946313858 | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     """ |     """ | ||||||
|  |     print("Benchmarking...") | ||||||
|  |  | ||||||
|     setup = """ |     setup = """ | ||||||
| from pymongo import MongoClient | from pymongo import MongoClient | ||||||
| @@ -127,7 +128,7 @@ connection = MongoClient() | |||||||
| db = connection.timeit_test | db = connection.timeit_test | ||||||
| noddy = db.noddy | noddy = db.noddy | ||||||
|  |  | ||||||
| for i in xrange(10000): | for i in range(10000): | ||||||
|     example = {'fields': {}} |     example = {'fields': {}} | ||||||
|     for j in range(20): |     for j in range(20): | ||||||
|         example['fields']["key"+str(j)] = "value "+str(j) |         example['fields']["key"+str(j)] = "value "+str(j) | ||||||
| @@ -138,10 +139,10 @@ myNoddys = noddy.find() | |||||||
| [n for n in myNoddys] # iterate | [n for n in myNoddys] # iterate | ||||||
| """ | """ | ||||||
|  |  | ||||||
|     print "-" * 100 |     print("-" * 100) | ||||||
|     print """Creating 10000 dictionaries - Pymongo""" |     print("""Creating 10000 dictionaries - Pymongo""") | ||||||
|     t = timeit.Timer(stmt=stmt, setup=setup) |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|     print t.timeit(1) |     print(t.timeit(1)) | ||||||
|  |  | ||||||
|     stmt = """ |     stmt = """ | ||||||
| from pymongo import MongoClient | from pymongo import MongoClient | ||||||
| @@ -150,7 +151,7 @@ connection = MongoClient() | |||||||
| db = connection.timeit_test | db = connection.timeit_test | ||||||
| noddy = db.noddy | noddy = db.noddy | ||||||
|  |  | ||||||
| for i in xrange(10000): | for i in range(10000): | ||||||
|     example = {'fields': {}} |     example = {'fields': {}} | ||||||
|     for j in range(20): |     for j in range(20): | ||||||
|         example['fields']["key"+str(j)] = "value "+str(j) |         example['fields']["key"+str(j)] = "value "+str(j) | ||||||
| @@ -161,10 +162,10 @@ myNoddys = noddy.find() | |||||||
| [n for n in myNoddys] # iterate | [n for n in myNoddys] # iterate | ||||||
| """ | """ | ||||||
|  |  | ||||||
|     print "-" * 100 |     print("-" * 100) | ||||||
|     print """Creating 10000 dictionaries - Pymongo write_concern={"w": 0}""" |     print("""Creating 10000 dictionaries - Pymongo write_concern={"w": 0}""") | ||||||
|     t = timeit.Timer(stmt=stmt, setup=setup) |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|     print t.timeit(1) |     print(t.timeit(1)) | ||||||
|  |  | ||||||
|     setup = """ |     setup = """ | ||||||
| from pymongo import MongoClient | from pymongo import MongoClient | ||||||
| @@ -180,7 +181,7 @@ class Noddy(Document): | |||||||
| """ | """ | ||||||
|  |  | ||||||
|     stmt = """ |     stmt = """ | ||||||
| for i in xrange(10000): | for i in range(10000): | ||||||
|     noddy = Noddy() |     noddy = Noddy() | ||||||
|     for j in range(20): |     for j in range(20): | ||||||
|         noddy.fields["key"+str(j)] = "value "+str(j) |         noddy.fields["key"+str(j)] = "value "+str(j) | ||||||
| @@ -190,13 +191,13 @@ myNoddys = Noddy.objects() | |||||||
| [n for n in myNoddys] # iterate | [n for n in myNoddys] # iterate | ||||||
| """ | """ | ||||||
|  |  | ||||||
|     print "-" * 100 |     print("-" * 100) | ||||||
|     print """Creating 10000 dictionaries - MongoEngine""" |     print("""Creating 10000 dictionaries - MongoEngine""") | ||||||
|     t = timeit.Timer(stmt=stmt, setup=setup) |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|     print t.timeit(1) |     print(t.timeit(1)) | ||||||
|  |  | ||||||
|     stmt = """ |     stmt = """ | ||||||
| for i in xrange(10000): | for i in range(10000): | ||||||
|     noddy = Noddy() |     noddy = Noddy() | ||||||
|     fields = {} |     fields = {} | ||||||
|     for j in range(20): |     for j in range(20): | ||||||
| @@ -208,13 +209,13 @@ myNoddys = Noddy.objects() | |||||||
| [n for n in myNoddys] # iterate | [n for n in myNoddys] # iterate | ||||||
| """ | """ | ||||||
|  |  | ||||||
|     print "-" * 100 |     print("-" * 100) | ||||||
|     print """Creating 10000 dictionaries without continual assign - MongoEngine""" |     print("""Creating 10000 dictionaries without continual assign - MongoEngine""") | ||||||
|     t = timeit.Timer(stmt=stmt, setup=setup) |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|     print t.timeit(1) |     print(t.timeit(1)) | ||||||
|  |  | ||||||
|     stmt = """ |     stmt = """ | ||||||
| for i in xrange(10000): | for i in range(10000): | ||||||
|     noddy = Noddy() |     noddy = Noddy() | ||||||
|     for j in range(20): |     for j in range(20): | ||||||
|         noddy.fields["key"+str(j)] = "value "+str(j) |         noddy.fields["key"+str(j)] = "value "+str(j) | ||||||
| @@ -224,13 +225,13 @@ myNoddys = Noddy.objects() | |||||||
| [n for n in myNoddys] # iterate | [n for n in myNoddys] # iterate | ||||||
| """ | """ | ||||||
|  |  | ||||||
|     print "-" * 100 |     print("-" * 100) | ||||||
|     print """Creating 10000 dictionaries - MongoEngine - write_concern={"w": 0}, cascade = True""" |     print("""Creating 10000 dictionaries - MongoEngine - write_concern={"w": 0}, cascade = True""") | ||||||
|     t = timeit.Timer(stmt=stmt, setup=setup) |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|     print t.timeit(1) |     print(t.timeit(1)) | ||||||
|  |  | ||||||
|     stmt = """ |     stmt = """ | ||||||
| for i in xrange(10000): | for i in range(10000): | ||||||
|     noddy = Noddy() |     noddy = Noddy() | ||||||
|     for j in range(20): |     for j in range(20): | ||||||
|         noddy.fields["key"+str(j)] = "value "+str(j) |         noddy.fields["key"+str(j)] = "value "+str(j) | ||||||
| @@ -240,13 +241,13 @@ myNoddys = Noddy.objects() | |||||||
| [n for n in myNoddys] # iterate | [n for n in myNoddys] # iterate | ||||||
| """ | """ | ||||||
|  |  | ||||||
|     print "-" * 100 |     print("-" * 100) | ||||||
|     print """Creating 10000 dictionaries - MongoEngine, write_concern={"w": 0}, validate=False, cascade=True""" |     print("""Creating 10000 dictionaries - MongoEngine, write_concern={"w": 0}, validate=False, cascade=True""") | ||||||
|     t = timeit.Timer(stmt=stmt, setup=setup) |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|     print t.timeit(1) |     print(t.timeit(1)) | ||||||
|  |  | ||||||
|     stmt = """ |     stmt = """ | ||||||
| for i in xrange(10000): | for i in range(10000): | ||||||
|     noddy = Noddy() |     noddy = Noddy() | ||||||
|     for j in range(20): |     for j in range(20): | ||||||
|         noddy.fields["key"+str(j)] = "value "+str(j) |         noddy.fields["key"+str(j)] = "value "+str(j) | ||||||
| @@ -256,13 +257,13 @@ myNoddys = Noddy.objects() | |||||||
| [n for n in myNoddys] # iterate | [n for n in myNoddys] # iterate | ||||||
| """ | """ | ||||||
|  |  | ||||||
|     print "-" * 100 |     print("-" * 100) | ||||||
|     print """Creating 10000 dictionaries - MongoEngine, write_concern={"w": 0}, validate=False""" |     print("""Creating 10000 dictionaries - MongoEngine, write_concern={"w": 0}, validate=False""") | ||||||
|     t = timeit.Timer(stmt=stmt, setup=setup) |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|     print t.timeit(1) |     print(t.timeit(1)) | ||||||
|  |  | ||||||
|     stmt = """ |     stmt = """ | ||||||
| for i in xrange(10000): | for i in range(10000): | ||||||
|     noddy = Noddy() |     noddy = Noddy() | ||||||
|     for j in range(20): |     for j in range(20): | ||||||
|         noddy.fields["key"+str(j)] = "value "+str(j) |         noddy.fields["key"+str(j)] = "value "+str(j) | ||||||
| @@ -272,10 +273,10 @@ myNoddys = Noddy.objects() | |||||||
| [n for n in myNoddys] # iterate | [n for n in myNoddys] # iterate | ||||||
| """ | """ | ||||||
|  |  | ||||||
|     print "-" * 100 |     print("-" * 100) | ||||||
|     print """Creating 10000 dictionaries - MongoEngine, force_insert=True, write_concern={"w": 0}, validate=False""" |     print("""Creating 10000 dictionaries - MongoEngine, force_insert=True, write_concern={"w": 0}, validate=False""") | ||||||
|     t = timeit.Timer(stmt=stmt, setup=setup) |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|     print t.timeit(1) |     print(t.timeit(1)) | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == "__main__": | if __name__ == "__main__": | ||||||
|   | |||||||
							
								
								
									
										229
									
								
								docs/_themes/nature/static/nature.css_t
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										229
									
								
								docs/_themes/nature/static/nature.css_t
									
									
									
									
										vendored
									
									
								
							| @@ -1,229 +0,0 @@ | |||||||
| /** |  | ||||||
|  * Sphinx stylesheet -- default theme |  | ||||||
|  * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ |  | ||||||
|  */ |  | ||||||
|   |  | ||||||
| @import url("basic.css"); |  | ||||||
|   |  | ||||||
| /* -- page layout ----------------------------------------------------------- */ |  | ||||||
|   |  | ||||||
| body { |  | ||||||
|     font-family: Arial, sans-serif; |  | ||||||
|     font-size: 100%; |  | ||||||
|     background-color: #111; |  | ||||||
|     color: #555; |  | ||||||
|     margin: 0; |  | ||||||
|     padding: 0; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.documentwrapper { |  | ||||||
|     float: left; |  | ||||||
|     width: 100%; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.bodywrapper { |  | ||||||
|     margin: 0 0 0 230px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| hr{ |  | ||||||
|     border: 1px solid #B1B4B6; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.document { |  | ||||||
|     background-color: #eee; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.body { |  | ||||||
|     background-color: #ffffff; |  | ||||||
|     color: #3E4349; |  | ||||||
|     padding: 0 30px 30px 30px; |  | ||||||
|     font-size: 0.8em; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.footer { |  | ||||||
|     color: #555; |  | ||||||
|     width: 100%; |  | ||||||
|     padding: 13px 0; |  | ||||||
|     text-align: center; |  | ||||||
|     font-size: 75%; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.footer a { |  | ||||||
|     color: #444; |  | ||||||
|     text-decoration: underline; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.related { |  | ||||||
|     background-color: #6BA81E; |  | ||||||
|     line-height: 32px; |  | ||||||
|     color: #fff; |  | ||||||
|     text-shadow: 0px 1px 0 #444; |  | ||||||
|     font-size: 0.80em; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.related a { |  | ||||||
|     color: #E2F3CC; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.sphinxsidebar { |  | ||||||
|     font-size: 0.75em; |  | ||||||
|     line-height: 1.5em; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.sphinxsidebarwrapper{ |  | ||||||
|     padding: 20px 0; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.sphinxsidebar h3, |  | ||||||
| div.sphinxsidebar h4 { |  | ||||||
|     font-family: Arial, sans-serif; |  | ||||||
|     color: #222; |  | ||||||
|     font-size: 1.2em; |  | ||||||
|     font-weight: normal; |  | ||||||
|     margin: 0; |  | ||||||
|     padding: 5px 10px; |  | ||||||
|     background-color: #ddd; |  | ||||||
|     text-shadow: 1px 1px 0 white |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.sphinxsidebar h4{ |  | ||||||
|     font-size: 1.1em; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.sphinxsidebar h3 a { |  | ||||||
|     color: #444; |  | ||||||
| } |  | ||||||
|   |  | ||||||
|   |  | ||||||
| div.sphinxsidebar p { |  | ||||||
|     color: #888; |  | ||||||
|     padding: 5px 20px; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.sphinxsidebar p.topless { |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.sphinxsidebar ul { |  | ||||||
|     margin: 10px 20px; |  | ||||||
|     padding: 0; |  | ||||||
|     color: #000; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.sphinxsidebar a { |  | ||||||
|     color: #444; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.sphinxsidebar input { |  | ||||||
|     border: 1px solid #ccc; |  | ||||||
|     font-family: sans-serif; |  | ||||||
|     font-size: 1em; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.sphinxsidebar input[type=text]{ |  | ||||||
|     margin-left: 20px; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| /* -- body styles ----------------------------------------------------------- */ |  | ||||||
|   |  | ||||||
| a { |  | ||||||
|     color: #005B81; |  | ||||||
|     text-decoration: none; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| a:hover { |  | ||||||
|     color: #E32E00; |  | ||||||
|     text-decoration: underline; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.body h1, |  | ||||||
| div.body h2, |  | ||||||
| div.body h3, |  | ||||||
| div.body h4, |  | ||||||
| div.body h5, |  | ||||||
| div.body h6 { |  | ||||||
|     font-family: Arial, sans-serif; |  | ||||||
|     background-color: #BED4EB; |  | ||||||
|     font-weight: normal; |  | ||||||
|     color: #212224; |  | ||||||
|     margin: 30px 0px 10px 0px; |  | ||||||
|     padding: 5px 0 5px 10px; |  | ||||||
|     text-shadow: 0px 1px 0 white |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.body h1 { border-top: 20px solid white; margin-top: 0; font-size: 200%; } |  | ||||||
| div.body h2 { font-size: 150%; background-color: #C8D5E3; } |  | ||||||
| div.body h3 { font-size: 120%; background-color: #D8DEE3; } |  | ||||||
| div.body h4 { font-size: 110%; background-color: #D8DEE3; } |  | ||||||
| div.body h5 { font-size: 100%; background-color: #D8DEE3; } |  | ||||||
| div.body h6 { font-size: 100%; background-color: #D8DEE3; } |  | ||||||
|   |  | ||||||
| a.headerlink { |  | ||||||
|     color: #c60f0f; |  | ||||||
|     font-size: 0.8em; |  | ||||||
|     padding: 0 4px 0 4px; |  | ||||||
|     text-decoration: none; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| a.headerlink:hover { |  | ||||||
|     background-color: #c60f0f; |  | ||||||
|     color: white; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.body p, div.body dd, div.body li { |  | ||||||
|     line-height: 1.5em; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.admonition p.admonition-title + p { |  | ||||||
|     display: inline; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.highlight{ |  | ||||||
|     background-color: white; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.note { |  | ||||||
|     background-color: #eee; |  | ||||||
|     border: 1px solid #ccc; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.seealso { |  | ||||||
|     background-color: #ffc; |  | ||||||
|     border: 1px solid #ff6; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.topic { |  | ||||||
|     background-color: #eee; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| div.warning { |  | ||||||
|     background-color: #ffe4e4; |  | ||||||
|     border: 1px solid #f66; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| p.admonition-title { |  | ||||||
|     display: inline; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| p.admonition-title:after { |  | ||||||
|     content: ":"; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| pre { |  | ||||||
|     padding: 10px; |  | ||||||
|     background-color: White; |  | ||||||
|     color: #222; |  | ||||||
|     line-height: 1.2em; |  | ||||||
|     border: 1px solid #C6C9CB; |  | ||||||
|     font-size: 1.2em; |  | ||||||
|     margin: 1.5em 0 1.5em 0; |  | ||||||
|     -webkit-box-shadow: 1px 1px 1px #d8d8d8; |  | ||||||
|     -moz-box-shadow: 1px 1px 1px #d8d8d8; |  | ||||||
| } |  | ||||||
|   |  | ||||||
| tt { |  | ||||||
|     background-color: #ecf0f3; |  | ||||||
|     color: #222; |  | ||||||
|     padding: 1px 2px; |  | ||||||
|     font-size: 1.2em; |  | ||||||
|     font-family: monospace; |  | ||||||
| } |  | ||||||
							
								
								
									
										54
									
								
								docs/_themes/nature/static/pygments.css
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										54
									
								
								docs/_themes/nature/static/pygments.css
									
									
									
									
										vendored
									
									
								
							| @@ -1,54 +0,0 @@ | |||||||
| .c { color: #999988; font-style: italic } /* Comment */ |  | ||||||
| .k { font-weight: bold } /* Keyword */ |  | ||||||
| .o { font-weight: bold } /* Operator */ |  | ||||||
| .cm { color: #999988; font-style: italic } /* Comment.Multiline */ |  | ||||||
| .cp { color: #999999; font-weight: bold } /* Comment.preproc */ |  | ||||||
| .c1 { color: #999988; font-style: italic } /* Comment.Single */ |  | ||||||
| .gd { color: #000000; background-color: #ffdddd } /* Generic.Deleted */ |  | ||||||
| .ge { font-style: italic } /* Generic.Emph */ |  | ||||||
| .gr { color: #aa0000 } /* Generic.Error */ |  | ||||||
| .gh { color: #999999 } /* Generic.Heading */ |  | ||||||
| .gi { color: #000000; background-color: #ddffdd } /* Generic.Inserted */ |  | ||||||
| .go { color: #111 } /* Generic.Output */ |  | ||||||
| .gp { color: #555555 } /* Generic.Prompt */ |  | ||||||
| .gs { font-weight: bold } /* Generic.Strong */ |  | ||||||
| .gu { color: #aaaaaa } /* Generic.Subheading */ |  | ||||||
| .gt { color: #aa0000 } /* Generic.Traceback */ |  | ||||||
| .kc { font-weight: bold } /* Keyword.Constant */ |  | ||||||
| .kd { font-weight: bold } /* Keyword.Declaration */ |  | ||||||
| .kp { font-weight: bold } /* Keyword.Pseudo */ |  | ||||||
| .kr { font-weight: bold } /* Keyword.Reserved */ |  | ||||||
| .kt { color: #445588; font-weight: bold } /* Keyword.Type */ |  | ||||||
| .m { color: #009999 } /* Literal.Number */ |  | ||||||
| .s { color: #bb8844 } /* Literal.String */ |  | ||||||
| .na { color: #008080 } /* Name.Attribute */ |  | ||||||
| .nb { color: #999999 } /* Name.Builtin */ |  | ||||||
| .nc { color: #445588; font-weight: bold } /* Name.Class */ |  | ||||||
| .no { color: #ff99ff } /* Name.Constant */ |  | ||||||
| .ni { color: #800080 } /* Name.Entity */ |  | ||||||
| .ne { color: #990000; font-weight: bold } /* Name.Exception */ |  | ||||||
| .nf { color: #990000; font-weight: bold } /* Name.Function */ |  | ||||||
| .nn { color: #555555 } /* Name.Namespace */ |  | ||||||
| .nt { color: #000080 } /* Name.Tag */ |  | ||||||
| .nv { color: purple } /* Name.Variable */ |  | ||||||
| .ow { font-weight: bold } /* Operator.Word */ |  | ||||||
| .mf { color: #009999 } /* Literal.Number.Float */ |  | ||||||
| .mh { color: #009999 } /* Literal.Number.Hex */ |  | ||||||
| .mi { color: #009999 } /* Literal.Number.Integer */ |  | ||||||
| .mo { color: #009999 } /* Literal.Number.Oct */ |  | ||||||
| .sb { color: #bb8844 } /* Literal.String.Backtick */ |  | ||||||
| .sc { color: #bb8844 } /* Literal.String.Char */ |  | ||||||
| .sd { color: #bb8844 } /* Literal.String.Doc */ |  | ||||||
| .s2 { color: #bb8844 } /* Literal.String.Double */ |  | ||||||
| .se { color: #bb8844 } /* Literal.String.Escape */ |  | ||||||
| .sh { color: #bb8844 } /* Literal.String.Heredoc */ |  | ||||||
| .si { color: #bb8844 } /* Literal.String.Interpol */ |  | ||||||
| .sx { color: #bb8844 } /* Literal.String.Other */ |  | ||||||
| .sr { color: #808000 } /* Literal.String.Regex */ |  | ||||||
| .s1 { color: #bb8844 } /* Literal.String.Single */ |  | ||||||
| .ss { color: #bb8844 } /* Literal.String.Symbol */ |  | ||||||
| .bp { color: #999999 } /* Name.Builtin.Pseudo */ |  | ||||||
| .vc { color: #ff99ff } /* Name.Variable.Class */ |  | ||||||
| .vg { color: #ff99ff } /* Name.Variable.Global */ |  | ||||||
| .vi { color: #ff99ff } /* Name.Variable.Instance */ |  | ||||||
| .il { color: #009999 } /* Literal.Number.Integer.Long */ |  | ||||||
							
								
								
									
										4
									
								
								docs/_themes/nature/theme.conf
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								docs/_themes/nature/theme.conf
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +0,0 @@ | |||||||
| [theme] |  | ||||||
| inherit = basic |  | ||||||
| stylesheet = nature.css |  | ||||||
| pygments_style = tango |  | ||||||
							
								
								
									
										17
									
								
								docs/_themes/sphinx_rtd_theme/__init__.py
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										17
									
								
								docs/_themes/sphinx_rtd_theme/__init__.py
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | """Sphinx ReadTheDocs theme. | ||||||
|  |  | ||||||
|  | From https://github.com/ryan-roemer/sphinx-bootstrap-theme. | ||||||
|  |  | ||||||
|  | """ | ||||||
|  | import os | ||||||
|  |  | ||||||
|  | VERSION = (0, 1, 5) | ||||||
|  |  | ||||||
|  | __version__ = ".".join(str(v) for v in VERSION) | ||||||
|  | __version_full__ = __version__ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_html_theme_path(): | ||||||
|  |     """Return list of HTML theme paths.""" | ||||||
|  |     cur_dir = os.path.abspath(os.path.dirname(os.path.dirname(__file__))) | ||||||
|  |     return cur_dir | ||||||
							
								
								
									
										15
									
								
								docs/_themes/sphinx_rtd_theme/breadcrumbs.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										15
									
								
								docs/_themes/sphinx_rtd_theme/breadcrumbs.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | <ul class="wy-breadcrumbs"> | ||||||
|  |   <li><a href="{{ pathto(master_doc) }}">Docs</a> »</li> | ||||||
|  |   <li><a href="">{{ title }}</a></li> | ||||||
|  |     <li class="wy-breadcrumbs-aside"> | ||||||
|  |       {% if display_github %} | ||||||
|  |         <a href="https://github.com/{{ github_user }}/{{ github_repo }}/blob/{{ github_version }}{{ conf_py_path }}{{ pagename }}.rst" class="icon icon-github"> Edit on GitHub</a> | ||||||
|  |       {% elif display_bitbucket %} | ||||||
|  |         <a href="https://bitbucket.org/{{ bitbucket_user }}/{{ bitbucket_repo }}/src/{{ bitbucket_version}}{{ conf_py_path }}{{ pagename }}.rst'" class="icon icon-bitbucket"> Edit on Bitbucket</a> | ||||||
|  |       {% elif show_source and has_source and sourcename %} | ||||||
|  |         <a href="{{ pathto('_sources/' + sourcename, true)|e }}" rel="nofollow"> View page source</a> | ||||||
|  |       {% endif %} | ||||||
|  |     </li> | ||||||
|  | </ul> | ||||||
|  | <hr/> | ||||||
|  |  | ||||||
							
								
								
									
										30
									
								
								docs/_themes/sphinx_rtd_theme/footer.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										30
									
								
								docs/_themes/sphinx_rtd_theme/footer.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,30 @@ | |||||||
|  | <footer> | ||||||
|  |   {% if next or prev %} | ||||||
|  |     <div class="rst-footer-buttons"> | ||||||
|  |       {% if next %} | ||||||
|  |         <a href="{{ next.link|e }}" class="btn btn-neutral float-right" title="{{ next.title|striptags|e }}"/>Next <span class="icon icon-circle-arrow-right"></span></a> | ||||||
|  |       {% endif %} | ||||||
|  |       {% if prev %} | ||||||
|  |         <a href="{{ prev.link|e }}" class="btn btn-neutral" title="{{ prev.title|striptags|e }}"><span class="icon icon-circle-arrow-left"></span> Previous</a> | ||||||
|  |       {% endif %} | ||||||
|  |     </div> | ||||||
|  |   {% endif %} | ||||||
|  |  | ||||||
|  |   <hr/> | ||||||
|  |  | ||||||
|  |   <p> | ||||||
|  |   {%- if show_copyright %} | ||||||
|  |     {%- if hasdoc('copyright') %} | ||||||
|  |       {% trans path=pathto('copyright'), copyright=copyright|e %}© <a href="{{ path }}">Copyright</a> {{ copyright }}.{% endtrans %} | ||||||
|  |     {%- else %} | ||||||
|  |       {% trans copyright=copyright|e %}© Copyright {{ copyright }}.{% endtrans %} | ||||||
|  |     {%- endif %} | ||||||
|  |   {%- endif %} | ||||||
|  |  | ||||||
|  |   {%- if last_updated %} | ||||||
|  |     {% trans last_updated=last_updated|e %}Last updated on {{ last_updated }}.{% endtrans %} | ||||||
|  |   {%- endif %} | ||||||
|  |   </p> | ||||||
|  |  | ||||||
|  |   {% trans %}<a href="https://www.github.com/snide/sphinx_rtd_theme">Sphinx theme</a> provided by <a href="http://readthedocs.org">Read the Docs</a>{% endtrans %} | ||||||
|  | </footer> | ||||||
							
								
								
									
										142
									
								
								docs/_themes/sphinx_rtd_theme/layout.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										142
									
								
								docs/_themes/sphinx_rtd_theme/layout.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,142 @@ | |||||||
|  | {# TEMPLATE VAR SETTINGS #} | ||||||
|  | {%- set url_root = pathto('', 1) %} | ||||||
|  | {%- if url_root == '#' %}{% set url_root = '' %}{% endif %} | ||||||
|  | {%- if not embedded and docstitle %} | ||||||
|  |   {%- set titlesuffix = " — "|safe + docstitle|e %} | ||||||
|  | {%- else %} | ||||||
|  |   {%- set titlesuffix = "" %} | ||||||
|  | {%- endif %} | ||||||
|  |  | ||||||
|  | <!DOCTYPE html> | ||||||
|  | <!--[if IE 8]><html class="no-js lt-ie9" lang="en" > <![endif]--> | ||||||
|  | <!--[if gt IE 8]><!--> <html class="no-js" lang="en" > <!--<![endif]--> | ||||||
|  | <head> | ||||||
|  |   <meta charset="utf-8"> | ||||||
|  |   <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||||
|  |   {% block htmltitle %} | ||||||
|  |   <title>{{ title|striptags|e }}{{ titlesuffix }}</title> | ||||||
|  |   {% endblock %} | ||||||
|  |  | ||||||
|  |   {# FAVICON #} | ||||||
|  |   {% if favicon %} | ||||||
|  |     <link rel="shortcut icon" href="{{ pathto('_static/' + favicon, 1) }}"/> | ||||||
|  |   {% endif %} | ||||||
|  |   {# CANONICAL #} | ||||||
|  |   {%- if theme_canonical_url %} | ||||||
|  |     <link rel="canonical" href="{{ theme_canonical_url }}{{ pagename }}.html"/> | ||||||
|  |   {%- endif %} | ||||||
|  |  | ||||||
|  |   {# CSS #} | ||||||
|  |   <link href='https://fonts.googleapis.com/css?family=Lato:400,700|Roboto+Slab:400,700|Inconsolata:400,700' rel='stylesheet' type='text/css'> | ||||||
|  |  | ||||||
|  |   {# JS #} | ||||||
|  |   {% if not embedded %} | ||||||
|  |  | ||||||
|  |     <script type="text/javascript"> | ||||||
|  |       var DOCUMENTATION_OPTIONS = { | ||||||
|  |         URL_ROOT:'{{ url_root }}', | ||||||
|  |         VERSION:'{{ release|e }}', | ||||||
|  |         COLLAPSE_INDEX:false, | ||||||
|  |         FILE_SUFFIX:'{{ '' if no_search_suffix else file_suffix }}', | ||||||
|  |         HAS_SOURCE:  {{ has_source|lower }} | ||||||
|  |       }; | ||||||
|  |     </script> | ||||||
|  |     {%- for scriptfile in script_files %} | ||||||
|  |       <script type="text/javascript" src="{{ pathto(scriptfile, 1) }}"></script> | ||||||
|  |     {%- endfor %} | ||||||
|  |  | ||||||
|  |     {% if use_opensearch %} | ||||||
|  |       <link rel="search" type="application/opensearchdescription+xml" title="{% trans docstitle=docstitle|e %}Search within {{ docstitle }}{% endtrans %}" href="{{ pathto('_static/opensearch.xml', 1) }}"/> | ||||||
|  |     {% endif %} | ||||||
|  |  | ||||||
|  |   {% endif %} | ||||||
|  |  | ||||||
|  |   {# RTD hosts these file themselves, so just load on non RTD builds #} | ||||||
|  |   {% if not READTHEDOCS %} | ||||||
|  |     <link rel="stylesheet" href="{{ pathto('_static/' + style, 1) }}" type="text/css" /> | ||||||
|  |     <script type="text/javascript" src="_static/js/theme.js"></script> | ||||||
|  |   {% endif %} | ||||||
|  |  | ||||||
|  |   {% for cssfile in css_files %} | ||||||
|  |     <link rel="stylesheet" href="{{ pathto(cssfile, 1) }}" type="text/css" /> | ||||||
|  |   {% endfor %} | ||||||
|  |  | ||||||
|  |   {%- block linktags %} | ||||||
|  |     {%- if hasdoc('about') %} | ||||||
|  |         <link rel="author" title="{{ _('About these documents') }}" | ||||||
|  |               href="{{ pathto('about') }}"/> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if hasdoc('genindex') %} | ||||||
|  |         <link rel="index" title="{{ _('Index') }}" | ||||||
|  |               href="{{ pathto('genindex') }}"/> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if hasdoc('search') %} | ||||||
|  |         <link rel="search" title="{{ _('Search') }}" href="{{ pathto('search') }}"/> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if hasdoc('copyright') %} | ||||||
|  |         <link rel="copyright" title="{{ _('Copyright') }}" href="{{ pathto('copyright') }}"/> | ||||||
|  |     {%- endif %} | ||||||
|  |     <link rel="top" title="{{ docstitle|e }}" href="{{ pathto('index') }}"/> | ||||||
|  |     {%- if parents %} | ||||||
|  |         <link rel="up" title="{{ parents[-1].title|striptags|e }}" href="{{ parents[-1].link|e }}"/> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if next %} | ||||||
|  |         <link rel="next" title="{{ next.title|striptags|e }}" href="{{ next.link|e }}"/> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if prev %} | ||||||
|  |         <link rel="prev" title="{{ prev.title|striptags|e }}" href="{{ prev.link|e }}"/> | ||||||
|  |     {%- endif %} | ||||||
|  |   {%- endblock %} | ||||||
|  |   {%- block extrahead %} {% endblock %} | ||||||
|  |  | ||||||
|  |   <script src="//cdnjs.cloudflare.com/ajax/libs/modernizr/2.6.2/modernizr.min.js"></script> | ||||||
|  |  | ||||||
|  | </head> | ||||||
|  |  | ||||||
|  | <body class="wy-body-for-nav"> | ||||||
|  |  | ||||||
|  |   <div class="wy-grid-for-nav"> | ||||||
|  |  | ||||||
|  |     {# SIDE NAV, TOGGLES ON MOBILE #} | ||||||
|  |     <nav data-toggle="wy-nav-shift" class="wy-nav-side"> | ||||||
|  |       <div class="wy-side-nav-search"> | ||||||
|  |         <a href="{{ pathto(master_doc) }}" class="icon icon-home"> {{ project }}</a> | ||||||
|  |         {% include "searchbox.html" %} | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       <div class="wy-menu wy-menu-vertical" data-spy="affix"> | ||||||
|  |         {% set toctree = toctree(maxdepth=2, collapse=False, includehidden=True) %} | ||||||
|  |         {% if toctree %} | ||||||
|  |             {{ toctree }} | ||||||
|  |         {% else %} | ||||||
|  |             <!-- Local TOC --> | ||||||
|  |             <div class="local-toc">{{ toc }}</div> | ||||||
|  |         {% endif %} | ||||||
|  |       </div> | ||||||
|  |         | ||||||
|  |     </nav> | ||||||
|  |  | ||||||
|  |     <section data-toggle="wy-nav-shift" class="wy-nav-content-wrap"> | ||||||
|  |  | ||||||
|  |       {# MOBILE NAV, TRIGGLES SIDE NAV ON TOGGLE #} | ||||||
|  |       <nav class="wy-nav-top"> | ||||||
|  |         <i data-toggle="wy-nav-top" class="icon icon-reorder"></i> | ||||||
|  |         <a href="{{ pathto(master_doc) }}">{{ project }}</a> | ||||||
|  |       </nav> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |       {# PAGE CONTENT #} | ||||||
|  |       <div class="wy-nav-content"> | ||||||
|  |         <div class="rst-content"> | ||||||
|  |           {% include "breadcrumbs.html" %} | ||||||
|  |           {% block body %}{% endblock %} | ||||||
|  |           {% include "footer.html" %} | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |     </section> | ||||||
|  |  | ||||||
|  |   </div> | ||||||
|  |   {% include "versions.html" %} | ||||||
|  | </body> | ||||||
|  | </html> | ||||||
							
								
								
									
										205
									
								
								docs/_themes/sphinx_rtd_theme/layout_old.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										205
									
								
								docs/_themes/sphinx_rtd_theme/layout_old.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,205 @@ | |||||||
|  | {# | ||||||
|  |     basic/layout.html | ||||||
|  |     ~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|  |     Master layout template for Sphinx themes. | ||||||
|  |  | ||||||
|  |     :copyright: Copyright 2007-2013 by the Sphinx team, see AUTHORS. | ||||||
|  |     :license: BSD, see LICENSE for details. | ||||||
|  | #} | ||||||
|  | {%- block doctype -%} | ||||||
|  | <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" | ||||||
|  |   "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> | ||||||
|  | {%- endblock %} | ||||||
|  | {%- set reldelim1 = reldelim1 is not defined and ' »' or reldelim1 %} | ||||||
|  | {%- set reldelim2 = reldelim2 is not defined and ' |' or reldelim2 %} | ||||||
|  | {%- set render_sidebar = (not embedded) and (not theme_nosidebar|tobool) and | ||||||
|  |                          (sidebars != []) %} | ||||||
|  | {%- set url_root = pathto('', 1) %} | ||||||
|  | {# XXX necessary? #} | ||||||
|  | {%- if url_root == '#' %}{% set url_root = '' %}{% endif %} | ||||||
|  | {%- if not embedded and docstitle %} | ||||||
|  |   {%- set titlesuffix = " — "|safe + docstitle|e %} | ||||||
|  | {%- else %} | ||||||
|  |   {%- set titlesuffix = "" %} | ||||||
|  | {%- endif %} | ||||||
|  |  | ||||||
|  | {%- macro relbar() %} | ||||||
|  |     <div class="related"> | ||||||
|  |       <h3>{{ _('Navigation') }}</h3> | ||||||
|  |       <ul> | ||||||
|  |         {%- for rellink in rellinks %} | ||||||
|  |         <li class="right" {% if loop.first %}style="margin-right: 10px"{% endif %}> | ||||||
|  |           <a href="{{ pathto(rellink[0]) }}" title="{{ rellink[1]|striptags|e }}" | ||||||
|  |              {{ accesskey(rellink[2]) }}>{{ rellink[3] }}</a> | ||||||
|  |           {%- if not loop.first %}{{ reldelim2 }}{% endif %}</li> | ||||||
|  |         {%- endfor %} | ||||||
|  |         {%- block rootrellink %} | ||||||
|  |         <li><a href="{{ pathto(master_doc) }}">{{ shorttitle|e }}</a>{{ reldelim1 }}</li> | ||||||
|  |         {%- endblock %} | ||||||
|  |         {%- for parent in parents %} | ||||||
|  |           <li><a href="{{ parent.link|e }}" {% if loop.last %}{{ accesskey("U") }}{% endif %}>{{ parent.title }}</a>{{ reldelim1 }}</li> | ||||||
|  |         {%- endfor %} | ||||||
|  |         {%- block relbaritems %} {% endblock %} | ||||||
|  |       </ul> | ||||||
|  |     </div> | ||||||
|  | {%- endmacro %} | ||||||
|  |  | ||||||
|  | {%- macro sidebar() %} | ||||||
|  |       {%- if render_sidebar %} | ||||||
|  |       <div class="sphinxsidebar"> | ||||||
|  |         <div class="sphinxsidebarwrapper"> | ||||||
|  |           {%- block sidebarlogo %} | ||||||
|  |           {%- if logo %} | ||||||
|  |             <p class="logo"><a href="{{ pathto(master_doc) }}"> | ||||||
|  |               <img class="logo" src="{{ pathto('_static/' + logo, 1) }}" alt="Logo"/> | ||||||
|  |             </a></p> | ||||||
|  |           {%- endif %} | ||||||
|  |           {%- endblock %} | ||||||
|  |           {%- if sidebars != None %} | ||||||
|  |             {#- new style sidebar: explicitly include/exclude templates #} | ||||||
|  |             {%- for sidebartemplate in sidebars %} | ||||||
|  |             {%- include sidebartemplate %} | ||||||
|  |             {%- endfor %} | ||||||
|  |           {%- else %} | ||||||
|  |             {#- old style sidebars: using blocks -- should be deprecated #} | ||||||
|  |             {%- block sidebartoc %} | ||||||
|  |             {%- include "localtoc.html" %} | ||||||
|  |             {%- endblock %} | ||||||
|  |             {%- block sidebarrel %} | ||||||
|  |             {%- include "relations.html" %} | ||||||
|  |             {%- endblock %} | ||||||
|  |             {%- block sidebarsourcelink %} | ||||||
|  |             {%- include "sourcelink.html" %} | ||||||
|  |             {%- endblock %} | ||||||
|  |             {%- if customsidebar %} | ||||||
|  |             {%- include customsidebar %} | ||||||
|  |             {%- endif %} | ||||||
|  |             {%- block sidebarsearch %} | ||||||
|  |             {%- include "searchbox.html" %} | ||||||
|  |             {%- endblock %} | ||||||
|  |           {%- endif %} | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |       {%- endif %} | ||||||
|  | {%- endmacro %} | ||||||
|  |  | ||||||
|  | {%- macro script() %} | ||||||
|  |     <script type="text/javascript"> | ||||||
|  |       var DOCUMENTATION_OPTIONS = { | ||||||
|  |         URL_ROOT:    '{{ url_root }}', | ||||||
|  |         VERSION:     '{{ release|e }}', | ||||||
|  |         COLLAPSE_INDEX: false, | ||||||
|  |         FILE_SUFFIX: '{{ '' if no_search_suffix else file_suffix }}', | ||||||
|  |         HAS_SOURCE:  {{ has_source|lower }} | ||||||
|  |       }; | ||||||
|  |     </script> | ||||||
|  |     {%- for scriptfile in script_files %} | ||||||
|  |     <script type="text/javascript" src="{{ pathto(scriptfile, 1) }}"></script> | ||||||
|  |     {%- endfor %} | ||||||
|  | {%- endmacro %} | ||||||
|  |  | ||||||
|  | {%- macro css() %} | ||||||
|  |     <link rel="stylesheet" href="{{ pathto('_static/' + style, 1) }}" type="text/css" /> | ||||||
|  |     <link rel="stylesheet" href="{{ pathto('_static/pygments.css', 1) }}" type="text/css" /> | ||||||
|  |     {%- for cssfile in css_files %} | ||||||
|  |     <link rel="stylesheet" href="{{ pathto(cssfile, 1) }}" type="text/css" /> | ||||||
|  |     {%- endfor %} | ||||||
|  | {%- endmacro %} | ||||||
|  |  | ||||||
|  | <html xmlns="http://www.w3.org/1999/xhtml"> | ||||||
|  |   <head> | ||||||
|  |     <meta http-equiv="Content-Type" content="text/html; charset={{ encoding }}" /> | ||||||
|  |     {{ metatags }} | ||||||
|  |     {%- block htmltitle %} | ||||||
|  |     <title>{{ title|striptags|e }}{{ titlesuffix }}</title> | ||||||
|  |     {%- endblock %} | ||||||
|  |     {{ css() }} | ||||||
|  |     {%- if not embedded %} | ||||||
|  |     {{ script() }} | ||||||
|  |     {%- if use_opensearch %} | ||||||
|  |     <link rel="search" type="application/opensearchdescription+xml" | ||||||
|  |           title="{% trans docstitle=docstitle|e %}Search within {{ docstitle }}{% endtrans %}" | ||||||
|  |           href="{{ pathto('_static/opensearch.xml', 1) }}"/> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if favicon %} | ||||||
|  |     <link rel="shortcut icon" href="{{ pathto('_static/' + favicon, 1) }}"/> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- endif %} | ||||||
|  | {%- block linktags %} | ||||||
|  |     {%- if hasdoc('about') %} | ||||||
|  |     <link rel="author" title="{{ _('About these documents') }}" href="{{ pathto('about') }}" /> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if hasdoc('genindex') %} | ||||||
|  |     <link rel="index" title="{{ _('Index') }}" href="{{ pathto('genindex') }}" /> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if hasdoc('search') %} | ||||||
|  |     <link rel="search" title="{{ _('Search') }}" href="{{ pathto('search') }}" /> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if hasdoc('copyright') %} | ||||||
|  |     <link rel="copyright" title="{{ _('Copyright') }}" href="{{ pathto('copyright') }}" /> | ||||||
|  |     {%- endif %} | ||||||
|  |     <link rel="top" title="{{ docstitle|e }}" href="{{ pathto('index') }}" /> | ||||||
|  |     {%- if parents %} | ||||||
|  |     <link rel="up" title="{{ parents[-1].title|striptags|e }}" href="{{ parents[-1].link|e }}" /> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if next %} | ||||||
|  |     <link rel="next" title="{{ next.title|striptags|e }}" href="{{ next.link|e }}" /> | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if prev %} | ||||||
|  |     <link rel="prev" title="{{ prev.title|striptags|e }}" href="{{ prev.link|e }}" /> | ||||||
|  |     {%- endif %} | ||||||
|  | {%- endblock %} | ||||||
|  | {%- block extrahead %} {% endblock %} | ||||||
|  |   </head> | ||||||
|  |   <body> | ||||||
|  | {%- block header %}{% endblock %} | ||||||
|  |  | ||||||
|  | {%- block relbar1 %}{{ relbar() }}{% endblock %} | ||||||
|  |  | ||||||
|  | {%- block content %} | ||||||
|  |   {%- block sidebar1 %} {# possible location for sidebar #} {% endblock %} | ||||||
|  |  | ||||||
|  |     <div class="document"> | ||||||
|  |   {%- block document %} | ||||||
|  |       <div class="documentwrapper"> | ||||||
|  |       {%- if render_sidebar %} | ||||||
|  |         <div class="bodywrapper"> | ||||||
|  |       {%- endif %} | ||||||
|  |           <div class="body"> | ||||||
|  |             {% block body %} {% endblock %} | ||||||
|  |           </div> | ||||||
|  |       {%- if render_sidebar %} | ||||||
|  |         </div> | ||||||
|  |       {%- endif %} | ||||||
|  |       </div> | ||||||
|  |   {%- endblock %} | ||||||
|  |  | ||||||
|  |   {%- block sidebar2 %}{{ sidebar() }}{% endblock %} | ||||||
|  |       <div class="clearer"></div> | ||||||
|  |     </div> | ||||||
|  | {%- endblock %} | ||||||
|  |  | ||||||
|  | {%- block relbar2 %}{{ relbar() }}{% endblock %} | ||||||
|  |  | ||||||
|  | {%- block footer %} | ||||||
|  |     <div class="footer"> | ||||||
|  |     {%- if show_copyright %} | ||||||
|  |       {%- if hasdoc('copyright') %} | ||||||
|  |         {% trans path=pathto('copyright'), copyright=copyright|e %}© <a href="{{ path }}">Copyright</a> {{ copyright }}.{% endtrans %} | ||||||
|  |       {%- else %} | ||||||
|  |         {% trans copyright=copyright|e %}© Copyright {{ copyright }}.{% endtrans %} | ||||||
|  |       {%- endif %} | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if last_updated %} | ||||||
|  |       {% trans last_updated=last_updated|e %}Last updated on {{ last_updated }}.{% endtrans %} | ||||||
|  |     {%- endif %} | ||||||
|  |     {%- if show_sphinx %} | ||||||
|  |       {% trans sphinx_version=sphinx_version|e %}Created using <a href="http://sphinx-doc.org/">Sphinx</a> {{ sphinx_version }}.{% endtrans %} | ||||||
|  |     {%- endif %} | ||||||
|  |     </div> | ||||||
|  |     <p>asdf asdf asdf asdf 22</p> | ||||||
|  | {%- endblock %} | ||||||
|  |   </body> | ||||||
|  | </html> | ||||||
|  |  | ||||||
							
								
								
									
										50
									
								
								docs/_themes/sphinx_rtd_theme/search.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										50
									
								
								docs/_themes/sphinx_rtd_theme/search.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,50 @@ | |||||||
|  | {# | ||||||
|  |     basic/search.html | ||||||
|  |     ~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|  |     Template for the search page. | ||||||
|  |  | ||||||
|  |     :copyright: Copyright 2007-2013 by the Sphinx team, see AUTHORS. | ||||||
|  |     :license: BSD, see LICENSE for details. | ||||||
|  | #} | ||||||
|  | {%- extends "layout.html" %} | ||||||
|  | {% set title = _('Search') %} | ||||||
|  | {% set script_files = script_files + ['_static/searchtools.js'] %} | ||||||
|  | {% block extrahead %} | ||||||
|  |   <script type="text/javascript"> | ||||||
|  |     jQuery(function() { Search.loadIndex("{{ pathto('searchindex.js', 1) }}"); }); | ||||||
|  |   </script> | ||||||
|  |   {# this is used when loading the search index using $.ajax fails, | ||||||
|  |      such as on Chrome for documents on localhost #} | ||||||
|  |   <script type="text/javascript" id="searchindexloader"></script> | ||||||
|  |   {{ super() }} | ||||||
|  | {% endblock %} | ||||||
|  | {% block body %} | ||||||
|  |   <noscript> | ||||||
|  |   <div id="fallback" class="admonition warning"> | ||||||
|  |     <p class="last"> | ||||||
|  |       {% trans %}Please activate JavaScript to enable the search | ||||||
|  |       functionality.{% endtrans %} | ||||||
|  |     </p> | ||||||
|  |   </div> | ||||||
|  |   </noscript> | ||||||
|  |  | ||||||
|  |   {% if search_performed %} | ||||||
|  |     <h2>{{ _('Search Results') }}</h2> | ||||||
|  |     {% if not search_results %} | ||||||
|  |       <p>{{ _('Your search did not match any documents. Please make sure that all words are spelled correctly and that you\'ve selected enough categories.') }}</p> | ||||||
|  |     {% endif %} | ||||||
|  |   {% endif %} | ||||||
|  |   <div id="search-results"> | ||||||
|  |   {% if search_results %} | ||||||
|  |     <ul> | ||||||
|  |     {% for href, caption, context in search_results %} | ||||||
|  |       <li> | ||||||
|  |         <a href="{{ pathto(item.href) }}">{{ caption }}</a> | ||||||
|  |         <p class="context">{{ context|e }}</p> | ||||||
|  |       </li> | ||||||
|  |     {% endfor %} | ||||||
|  |     </ul> | ||||||
|  |   {% endif %} | ||||||
|  |   </div> | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										5
									
								
								docs/_themes/sphinx_rtd_theme/searchbox.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										5
									
								
								docs/_themes/sphinx_rtd_theme/searchbox.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | <form id ="rtd-search-form" class="wy-form" action="{{ pathto('search') }}" method="get"> | ||||||
|  |   <input type="text" name="q" placeholder="Search docs" /> | ||||||
|  |   <input type="hidden" name="check_keywords" value="yes" /> | ||||||
|  |   <input type="hidden" name="area" value="default" /> | ||||||
|  | </form> | ||||||
							
								
								
									
										1
									
								
								docs/_themes/sphinx_rtd_theme/static/css/badge_only.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										1
									
								
								docs/_themes/sphinx_rtd_theme/static/css/badge_only.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | .font-smooth,.icon:before{-webkit-font-smoothing:antialiased}.clearfix{*zoom:1}.clearfix:before,.clearfix:after{display:table;content:""}.clearfix:after{clear:both}@font-face{font-family:fontawesome-webfont;font-weight:normal;font-style:normal;src:url("../font/fontawesome_webfont.eot");src:url("../font/fontawesome_webfont.eot?#iefix") format("embedded-opentype"),url("../font/fontawesome_webfont.woff") format("woff"),url("../font/fontawesome_webfont.ttf") format("truetype"),url("../font/fontawesome_webfont.svg#fontawesome-webfont") format("svg")}.icon:before{display:inline-block;font-family:fontawesome-webfont;font-style:normal;font-weight:normal;line-height:1;text-decoration:inherit}a .icon{display:inline-block;text-decoration:inherit}li .icon{display:inline-block}li .icon-large:before,li .icon-large:before{width:1.875em}ul.icons{list-style-type:none;margin-left:2em;text-indent:-0.8em}ul.icons li .icon{width:0.8em}ul.icons li .icon-large:before,ul.icons li .icon-large:before{vertical-align:baseline}.icon-book:before{content:"\f02d"}.icon-caret-down:before{content:"\f0d7"}.icon-caret-up:before{content:"\f0d8"}.icon-caret-left:before{content:"\f0d9"}.icon-caret-right:before{content:"\f0da"}.rst-versions{position:fixed;bottom:0;left:0;width:300px;color:#fcfcfc;background:#1f1d1d;border-top:solid 10px #343131;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;z-index:400}.rst-versions a{color:#2980b9;text-decoration:none}.rst-versions .rst-badge-small{display:none}.rst-versions .rst-current-version{padding:12px;background-color:#272525;display:block;text-align:right;font-size:90%;cursor:pointer;color:#27ae60;*zoom:1}.rst-versions .rst-current-version:before,.rst-versions .rst-current-version:after{display:table;content:""}.rst-versions .rst-current-version:after{clear:both}.rst-versions .rst-current-version .icon{color:#fcfcfc}.rst-versions .rst-current-version .icon-book{float:left}.rst-versions .rst-current-version.rst-out-of-date{background-color:#e74c3c;color:#fff}.rst-versions.shift-up .rst-other-versions{display:block}.rst-versions .rst-other-versions{font-size:90%;padding:12px;color:gray;display:none}.rst-versions .rst-other-versions hr{display:block;height:1px;border:0;margin:20px 0;padding:0;border-top:solid 1px #413d3d}.rst-versions .rst-other-versions dd{display:inline-block;margin:0}.rst-versions .rst-other-versions dd a{display:inline-block;padding:6px;color:#fcfcfc}.rst-versions.rst-badge{width:auto;bottom:20px;right:20px;left:auto;border:none;max-width:300px}.rst-versions.rst-badge .icon-book{float:none}.rst-versions.rst-badge.shift-up .rst-current-version{text-align:right}.rst-versions.rst-badge.shift-up .rst-current-version .icon-book{float:left}.rst-versions.rst-badge .rst-current-version{width:auto;height:30px;line-height:30px;padding:0 6px;display:block;text-align:center}@media screen and (max-width: 768px){.rst-versions{width:85%;display:none}.rst-versions.shift{display:block}img{width:100%;height:auto}} | ||||||
							
								
								
									
										1
									
								
								docs/_themes/sphinx_rtd_theme/static/css/theme.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										1
									
								
								docs/_themes/sphinx_rtd_theme/static/css/theme.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								docs/_themes/sphinx_rtd_theme/static/favicon.ico
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/_themes/sphinx_rtd_theme/static/favicon.ico
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 6.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/_themes/sphinx_rtd_theme/static/font/fontawesome_webfont.eot
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/_themes/sphinx_rtd_theme/static/font/fontawesome_webfont.eot
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										399
									
								
								docs/_themes/sphinx_rtd_theme/static/font/fontawesome_webfont.svg
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										399
									
								
								docs/_themes/sphinx_rtd_theme/static/font/fontawesome_webfont.svg
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,399 @@ | |||||||
|  | <?xml version="1.0" standalone="no"?> | ||||||
|  | <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" > | ||||||
|  | <svg xmlns="http://www.w3.org/2000/svg"> | ||||||
|  | <metadata></metadata> | ||||||
|  | <defs> | ||||||
|  | <font id="fontawesomeregular" horiz-adv-x="1536" > | ||||||
|  | <font-face units-per-em="1792" ascent="1536" descent="-256" /> | ||||||
|  | <missing-glyph horiz-adv-x="448" /> | ||||||
|  | <glyph unicode=" "  horiz-adv-x="448" /> | ||||||
|  | <glyph unicode="	" horiz-adv-x="448" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="448" /> | ||||||
|  | <glyph unicode="¨" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode="©" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode="®" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode="´" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode="Æ" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="768" /> | ||||||
|  | <glyph unicode=" " /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="768" /> | ||||||
|  | <glyph unicode=" " /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="512" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="384" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="256" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="256" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="192" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="307" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="85" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="307" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="384" /> | ||||||
|  | <glyph unicode="™" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode="∞" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode="≠" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="500" d="M0 0z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1699 1350q0 -35 -43 -78l-632 -632v-768h320q26 0 45 -19t19 -45t-19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45t45 19h320v768l-632 632q-43 43 -43 78q0 23 18 36.5t38 17.5t43 4h1408q23 0 43 -4t38 -17.5t18 -36.5z" /> | ||||||
|  | <glyph unicode="" d="M1536 1312v-1120q0 -50 -34 -89t-86 -60.5t-103.5 -32t-96.5 -10.5t-96.5 10.5t-103.5 32t-86 60.5t-34 89t34 89t86 60.5t103.5 32t96.5 10.5q105 0 192 -39v537l-768 -237v-709q0 -50 -34 -89t-86 -60.5t-103.5 -32t-96.5 -10.5t-96.5 10.5t-103.5 32t-86 60.5t-34 89 t34 89t86 60.5t103.5 32t96.5 10.5q105 0 192 -39v967q0 31 19 56.5t49 35.5l832 256q12 4 28 4q40 0 68 -28t28 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1152 704q0 185 -131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5t316.5 131.5t131.5 316.5zM1664 -128q0 -52 -38 -90t-90 -38q-54 0 -90 38l-343 342q-179 -124 -399 -124q-143 0 -273.5 55.5t-225 150t-150 225t-55.5 273.5 t55.5 273.5t150 225t225 150t273.5 55.5t273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -220 -124 -399l343 -343q37 -37 37 -90z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1664 32v768q-32 -36 -69 -66q-268 -206 -426 -338q-51 -43 -83 -67t-86.5 -48.5t-102.5 -24.5h-1h-1q-48 0 -102.5 24.5t-86.5 48.5t-83 67q-158 132 -426 338q-37 30 -69 66v-768q0 -13 9.5 -22.5t22.5 -9.5h1472q13 0 22.5 9.5t9.5 22.5zM1664 1083v11v13.5t-0.5 13 t-3 12.5t-5.5 9t-9 7.5t-14 2.5h-1472q-13 0 -22.5 -9.5t-9.5 -22.5q0 -168 147 -284q193 -152 401 -317q6 -5 35 -29.5t46 -37.5t44.5 -31.5t50.5 -27.5t43 -9h1h1q20 0 43 9t50.5 27.5t44.5 31.5t46 37.5t35 29.5q208 165 401 317q54 43 100.5 115.5t46.5 131.5z M1792 1120v-1088q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1472q66 0 113 -47t47 -113z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M896 -128q-26 0 -44 18l-624 602q-10 8 -27.5 26t-55.5 65.5t-68 97.5t-53.5 121t-23.5 138q0 220 127 344t351 124q62 0 126.5 -21.5t120 -58t95.5 -68.5t76 -68q36 36 76 68t95.5 68.5t120 58t126.5 21.5q224 0 351 -124t127 -344q0 -221 -229 -450l-623 -600 q-18 -18 -44 -18z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1664 889q0 -22 -26 -48l-363 -354l86 -500q1 -7 1 -20q0 -21 -10.5 -35.5t-30.5 -14.5q-19 0 -40 12l-449 236l-449 -236q-22 -12 -40 -12q-21 0 -31.5 14.5t-10.5 35.5q0 6 2 20l86 500l-364 354q-25 27 -25 48q0 37 56 46l502 73l225 455q19 41 49 41t49 -41l225 -455 l502 -73q56 -9 56 -46z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1137 532l306 297l-422 62l-189 382l-189 -382l-422 -62l306 -297l-73 -421l378 199l377 -199zM1664 889q0 -22 -26 -48l-363 -354l86 -500q1 -7 1 -20q0 -50 -41 -50q-19 0 -40 12l-449 236l-449 -236q-22 -12 -40 -12q-21 0 -31.5 14.5t-10.5 35.5q0 6 2 20l86 500 l-364 354q-25 27 -25 48q0 37 56 46l502 73l225 455q19 41 49 41t49 -41l225 -455l502 -73q56 -9 56 -46z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1408 131q0 -120 -73 -189.5t-194 -69.5h-874q-121 0 -194 69.5t-73 189.5q0 53 3.5 103.5t14 109t26.5 108.5t43 97.5t62 81t85.5 53.5t111.5 20q9 0 42 -21.5t74.5 -48t108 -48t133.5 -21.5t133.5 21.5t108 48t74.5 48t42 21.5q61 0 111.5 -20t85.5 -53.5t62 -81 t43 -97.5t26.5 -108.5t14 -109t3.5 -103.5zM1088 1024q0 -159 -112.5 -271.5t-271.5 -112.5t-271.5 112.5t-112.5 271.5t112.5 271.5t271.5 112.5t271.5 -112.5t112.5 -271.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M384 -64v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM384 320v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM384 704v128q0 26 -19 45t-45 19h-128 q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1408 -64v512q0 26 -19 45t-45 19h-768q-26 0 -45 -19t-19 -45v-512q0 -26 19 -45t45 -19h768q26 0 45 19t19 45zM384 1088v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45 t45 -19h128q26 0 45 19t19 45zM1792 -64v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1408 704v512q0 26 -19 45t-45 19h-768q-26 0 -45 -19t-19 -45v-512q0 -26 19 -45t45 -19h768q26 0 45 19t19 45zM1792 320v128 q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1792 704v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1792 1088v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19 t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1920 1248v-1344q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1344q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M768 512v-384q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90zM768 1280v-384q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90zM1664 512v-384q0 -52 -38 -90t-90 -38 h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90zM1664 1280v-384q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M512 288v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM512 800v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1152 288v-192q0 -40 -28 -68t-68 -28h-320 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM512 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1152 800v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28 h320q40 0 68 -28t28 -68zM1792 288v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1152 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 800v-192 q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M512 288v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM512 800v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 288v-192q0 -40 -28 -68t-68 -28h-960 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h960q40 0 68 -28t28 -68zM512 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 800v-192q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v192q0 40 28 68t68 28 h960q40 0 68 -28t28 -68zM1792 1312v-192q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h960q40 0 68 -28t28 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1671 970q0 -40 -28 -68l-724 -724l-136 -136q-28 -28 -68 -28t-68 28l-136 136l-362 362q-28 28 -28 68t28 68l136 136q28 28 68 28t68 -28l294 -295l656 657q28 28 68 28t68 -28l136 -136q28 -28 28 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1298 214q0 -40 -28 -68l-136 -136q-28 -28 -68 -28t-68 28l-294 294l-294 -294q-28 -28 -68 -28t-68 28l-136 136q-28 28 -28 68t28 68l294 294l-294 294q-28 28 -28 68t28 68l136 136q28 28 68 28t68 -28l294 -294l294 294q28 28 68 28t68 -28l136 -136q28 -28 28 -68 t-28 -68l-294 -294l294 -294q28 -28 28 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1024 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-224v-224q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v224h-224q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h224v224q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5v-224h224 q13 0 22.5 -9.5t9.5 -22.5zM1152 704q0 185 -131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5t316.5 131.5t131.5 316.5zM1664 -128q0 -53 -37.5 -90.5t-90.5 -37.5q-54 0 -90 38l-343 342q-179 -124 -399 -124q-143 0 -273.5 55.5 t-225 150t-150 225t-55.5 273.5t55.5 273.5t150 225t225 150t273.5 55.5t273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -220 -124 -399l343 -343q37 -37 37 -90z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1024 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-576q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h576q13 0 22.5 -9.5t9.5 -22.5zM1152 704q0 185 -131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5t316.5 131.5t131.5 316.5z M1664 -128q0 -53 -37.5 -90.5t-90.5 -37.5q-54 0 -90 38l-343 342q-179 -124 -399 -124q-143 0 -273.5 55.5t-225 150t-150 225t-55.5 273.5t55.5 273.5t150 225t225 150t273.5 55.5t273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -220 -124 -399l343 -343q37 -37 37 -90z " /> | ||||||
|  | <glyph unicode="" d="M1536 640q0 -156 -61 -298t-164 -245t-245 -164t-298 -61t-298 61t-245 164t-164 245t-61 298q0 182 80.5 343t226.5 270q43 32 95.5 25t83.5 -50q32 -42 24.5 -94.5t-49.5 -84.5q-98 -74 -151.5 -181t-53.5 -228q0 -104 40.5 -198.5t109.5 -163.5t163.5 -109.5 t198.5 -40.5t198.5 40.5t163.5 109.5t109.5 163.5t40.5 198.5q0 121 -53.5 228t-151.5 181q-42 32 -49.5 84.5t24.5 94.5q31 43 84 50t95 -25q146 -109 226.5 -270t80.5 -343zM896 1408v-640q0 -52 -38 -90t-90 -38t-90 38t-38 90v640q0 52 38 90t90 38t90 -38t38 -90z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M256 96v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM640 224v-320q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v320q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1024 480v-576q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23 v576q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1408 864v-960q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v960q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1792 1376v-1472q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v1472q0 14 9 23t23 9h192q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" d="M1024 640q0 106 -75 181t-181 75t-181 -75t-75 -181t75 -181t181 -75t181 75t75 181zM1536 749v-222q0 -12 -8 -23t-20 -13l-185 -28q-19 -54 -39 -91q35 -50 107 -138q10 -12 10 -25t-9 -23q-27 -37 -99 -108t-94 -71q-12 0 -26 9l-138 108q-44 -23 -91 -38 q-16 -136 -29 -186q-7 -28 -36 -28h-222q-14 0 -24.5 8.5t-11.5 21.5l-28 184q-49 16 -90 37l-141 -107q-10 -9 -25 -9q-14 0 -25 11q-126 114 -165 168q-7 10 -7 23q0 12 8 23q15 21 51 66.5t54 70.5q-27 50 -41 99l-183 27q-13 2 -21 12.5t-8 23.5v222q0 12 8 23t19 13 l186 28q14 46 39 92q-40 57 -107 138q-10 12 -10 24q0 10 9 23q26 36 98.5 107.5t94.5 71.5q13 0 26 -10l138 -107q44 23 91 38q16 136 29 186q7 28 36 28h222q14 0 24.5 -8.5t11.5 -21.5l28 -184q49 -16 90 -37l142 107q9 9 24 9q13 0 25 -10q129 -119 165 -170q7 -8 7 -22 q0 -12 -8 -23q-15 -21 -51 -66.5t-54 -70.5q26 -50 41 -98l183 -28q13 -2 21 -12.5t8 -23.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M512 800v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM768 800v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1024 800v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576 q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1152 76v948h-896v-948q0 -22 7 -40.5t14.5 -27t10.5 -8.5h832q3 0 10.5 8.5t14.5 27t7 40.5zM480 1152h448l-48 117q-7 9 -17 11h-317q-10 -2 -17 -11zM1408 1120v-64q0 -14 -9 -23t-23 -9h-96v-948q0 -83 -47 -143.5t-113 -60.5h-832 q-66 0 -113 58.5t-47 141.5v952h-96q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h309l70 167q15 37 54 63t79 26h320q40 0 79 -26t54 -63l70 -167h309q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1408 544v-480q0 -26 -19 -45t-45 -19h-384v384h-256v-384h-384q-26 0 -45 19t-19 45v480q0 1 0.5 3t0.5 3l575 474l575 -474q1 -2 1 -6zM1631 613l-62 -74q-8 -9 -21 -11h-3q-13 0 -21 7l-692 577l-692 -577q-12 -8 -24 -7q-13 2 -21 11l-62 74q-8 10 -7 23.5t11 21.5 l719 599q32 26 76 26t76 -26l244 -204v195q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-408l219 -182q10 -8 11 -21.5t-7 -23.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1280" d="M128 0h1024v768h-416q-40 0 -68 28t-28 68v416h-512v-1280zM768 896h376q-10 29 -22 41l-313 313q-12 12 -41 22v-376zM1280 864v-896q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h640q40 0 88 -20t76 -48l312 -312q28 -28 48 -76t20 -88z " /> | ||||||
|  | <glyph unicode="" d="M896 992v-448q0 -14 -9 -23t-23 -9h-320q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h224v352q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1111 540v4l-24 320q-1 13 -11 22.5t-23 9.5h-186q-13 0 -23 -9.5t-11 -22.5l-24 -320v-4q-1 -12 8 -20t21 -8h244q12 0 21 8t8 20zM1870 73q0 -73 -46 -73h-704q13 0 22 9.5t8 22.5l-20 256q-1 13 -11 22.5t-23 9.5h-272q-13 0 -23 -9.5t-11 -22.5l-20 -256 q-1 -13 8 -22.5t22 -9.5h-704q-46 0 -46 73q0 54 26 116l417 1044q8 19 26 33t38 14h339q-13 0 -23 -9.5t-11 -22.5l-15 -192q-1 -14 8 -23t22 -9h166q13 0 22 9t8 23l-15 192q-1 13 -11 22.5t-23 9.5h339q20 0 38 -14t26 -33l417 -1044q26 -62 26 -116z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1280 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1536 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 416v-320q0 -40 -28 -68t-68 -28h-1472q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h465l135 -136 q58 -56 136 -56t136 56l136 136h464q40 0 68 -28t28 -68zM1339 985q17 -41 -14 -70l-448 -448q-18 -19 -45 -19t-45 19l-448 448q-31 29 -14 70q17 39 59 39h256v448q0 26 19 45t45 19h256q26 0 45 -19t19 -45v-448h256q42 0 59 -39z" /> | ||||||
|  | <glyph unicode="" d="M1120 608q0 -12 -10 -24l-319 -319q-11 -9 -23 -9t-23 9l-320 320q-15 16 -7 35q8 20 30 20h192v352q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-352h192q14 0 23 -9t9 -23zM768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273 t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1118 660q-8 -20 -30 -20h-192v-352q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v352h-192q-14 0 -23 9t-9 23q0 12 10 24l319 319q11 9 23 9t23 -9l320 -320q15 -16 7 -35zM768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198 t73 273t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1023 576h316q-1 3 -2.5 8t-2.5 8l-212 496h-708l-212 -496q-1 -2 -2.5 -8t-2.5 -8h316l95 -192h320zM1536 546v-482q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v482q0 62 25 123l238 552q10 25 36.5 42t52.5 17h832q26 0 52.5 -17t36.5 -42l238 -552 q25 -61 25 -123z" /> | ||||||
|  | <glyph unicode="" d="M1184 640q0 -37 -32 -55l-544 -320q-15 -9 -32 -9q-16 0 -32 8q-32 19 -32 56v640q0 37 32 56q33 18 64 -1l544 -320q32 -18 32 -55zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1536 1280v-448q0 -26 -19 -45t-45 -19h-448q-42 0 -59 40q-17 39 14 69l138 138q-148 137 -349 137q-104 0 -198.5 -40.5t-163.5 -109.5t-109.5 -163.5t-40.5 -198.5t40.5 -198.5t109.5 -163.5t163.5 -109.5t198.5 -40.5q119 0 225 52t179 147q7 10 23 12q14 0 25 -9 l137 -138q9 -8 9.5 -20.5t-7.5 -22.5q-109 -132 -264 -204.5t-327 -72.5q-156 0 -298 61t-245 164t-164 245t-61 298t61 298t164 245t245 164t298 61q147 0 284.5 -55.5t244.5 -156.5l130 129q29 31 70 14q39 -17 39 -59z" /> | ||||||
|  | <glyph unicode="" d="M1511 480q0 -5 -1 -7q-64 -268 -268 -434.5t-478 -166.5q-146 0 -282.5 55t-243.5 157l-129 -129q-19 -19 -45 -19t-45 19t-19 45v448q0 26 19 45t45 19h448q26 0 45 -19t19 -45t-19 -45l-137 -137q71 -66 161 -102t187 -36q134 0 250 65t186 179q11 17 53 117 q8 23 30 23h192q13 0 22.5 -9.5t9.5 -22.5zM1536 1280v-448q0 -26 -19 -45t-45 -19h-448q-26 0 -45 19t-19 45t19 45l138 138q-148 137 -349 137q-134 0 -250 -65t-186 -179q-11 -17 -53 -117q-8 -23 -30 -23h-199q-13 0 -22.5 9.5t-9.5 22.5v7q65 268 270 434.5t480 166.5 q146 0 284 -55.5t245 -156.5l130 129q19 19 45 19t45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M384 352v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 608v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M384 864v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1536 352v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h960q13 0 22.5 -9.5t9.5 -22.5z M1536 608v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h960q13 0 22.5 -9.5t9.5 -22.5zM1536 864v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h960q13 0 22.5 -9.5 t9.5 -22.5zM1664 160v832q0 13 -9.5 22.5t-22.5 9.5h-1472q-13 0 -22.5 -9.5t-9.5 -22.5v-832q0 -13 9.5 -22.5t22.5 -9.5h1472q13 0 22.5 9.5t9.5 22.5zM1792 1248v-1088q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1472q66 0 113 -47 t47 -113z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M320 768h512v192q0 106 -75 181t-181 75t-181 -75t-75 -181v-192zM1152 672v-576q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v576q0 40 28 68t68 28h32v192q0 184 132 316t316 132t316 -132t132 -316v-192h32q40 0 68 -28t28 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M320 1280q0 -72 -64 -110v-1266q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v1266q-64 38 -64 110q0 53 37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1792 1216v-763q0 -25 -12.5 -38.5t-39.5 -27.5q-215 -116 -369 -116q-61 0 -123.5 22t-108.5 48 t-115.5 48t-142.5 22q-192 0 -464 -146q-17 -9 -33 -9q-26 0 -45 19t-19 45v742q0 32 31 55q21 14 79 43q236 120 421 120q107 0 200 -29t219 -88q38 -19 88 -19q54 0 117.5 21t110 47t88 47t54.5 21q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1664 650q0 -166 -60 -314l-20 -49l-185 -33q-22 -83 -90.5 -136.5t-156.5 -53.5v-32q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576q0 14 9 23t23 9h64q14 0 23 -9t9 -23v-32q71 0 130 -35.5t93 -95.5l68 12q29 95 29 193q0 148 -88 279t-236.5 209t-315.5 78 t-315.5 -78t-236.5 -209t-88 -279q0 -98 29 -193l68 -12q34 60 93 95.5t130 35.5v32q0 14 9 23t23 9h64q14 0 23 -9t9 -23v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v32q-88 0 -156.5 53.5t-90.5 136.5l-185 33l-20 49q-60 148 -60 314q0 151 67 291t179 242.5 t266 163.5t320 61t320 -61t266 -163.5t179 -242.5t67 -291z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="768" d="M768 1184v-1088q0 -26 -19 -45t-45 -19t-45 19l-333 333h-262q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h262l333 333q19 19 45 19t45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M768 1184v-1088q0 -26 -19 -45t-45 -19t-45 19l-333 333h-262q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h262l333 333q19 19 45 19t45 -19t19 -45zM1152 640q0 -76 -42.5 -141.5t-112.5 -93.5q-10 -5 -25 -5q-26 0 -45 18.5t-19 45.5q0 21 12 35.5t29 25t34 23t29 35.5 t12 57t-12 57t-29 35.5t-34 23t-29 25t-12 35.5q0 27 19 45.5t45 18.5q15 0 25 -5q70 -27 112.5 -93t42.5 -142z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M768 1184v-1088q0 -26 -19 -45t-45 -19t-45 19l-333 333h-262q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h262l333 333q19 19 45 19t45 -19t19 -45zM1152 640q0 -76 -42.5 -141.5t-112.5 -93.5q-10 -5 -25 -5q-26 0 -45 18.5t-19 45.5q0 21 12 35.5t29 25t34 23t29 35.5 t12 57t-12 57t-29 35.5t-34 23t-29 25t-12 35.5q0 27 19 45.5t45 18.5q15 0 25 -5q70 -27 112.5 -93t42.5 -142zM1408 640q0 -153 -85 -282.5t-225 -188.5q-13 -5 -25 -5q-27 0 -46 19t-19 45q0 39 39 59q56 29 76 44q74 54 115.5 135.5t41.5 173.5t-41.5 173.5 t-115.5 135.5q-20 15 -76 44q-39 20 -39 59q0 26 19 45t45 19q13 0 26 -5q140 -59 225 -188.5t85 -282.5zM1664 640q0 -230 -127 -422.5t-338 -283.5q-13 -5 -26 -5q-26 0 -45 19t-19 45q0 36 39 59q7 4 22.5 10.5t22.5 10.5q46 25 82 51q123 91 192 227t69 289t-69 289 t-192 227q-36 26 -82 51q-7 4 -22.5 10.5t-22.5 10.5q-39 23 -39 59q0 26 19 45t45 19q13 0 26 -5q211 -91 338 -283.5t127 -422.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M384 384v-128h-128v128h128zM384 1152v-128h-128v128h128zM1152 1152v-128h-128v128h128zM128 129h384v383h-384v-383zM128 896h384v384h-384v-384zM896 896h384v384h-384v-384zM640 640v-640h-640v640h640zM1152 128v-128h-128v128h128zM1408 128v-128h-128v128h128z M1408 640v-384h-384v128h-128v-384h-128v640h384v-128h128v128h128zM640 1408v-640h-640v640h640zM1408 1408v-640h-640v640h640z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M63 0h-63v1408h63v-1408zM126 1h-32v1407h32v-1407zM220 1h-31v1407h31v-1407zM377 1h-31v1407h31v-1407zM534 1h-62v1407h62v-1407zM660 1h-31v1407h31v-1407zM723 1h-31v1407h31v-1407zM786 1h-31v1407h31v-1407zM943 1h-63v1407h63v-1407zM1100 1h-63v1407h63v-1407z M1226 1h-63v1407h63v-1407zM1352 1h-63v1407h63v-1407zM1446 1h-63v1407h63v-1407zM1635 1h-94v1407h94v-1407zM1698 1h-32v1407h32v-1407zM1792 0h-63v1408h63v-1408z" /> | ||||||
|  | <glyph unicode="" d="M448 1088q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1515 512q0 -53 -37 -90l-491 -492q-39 -37 -91 -37q-53 0 -90 37l-715 716q-38 37 -64.5 101t-26.5 117v416q0 52 38 90t90 38h416q53 0 117 -26.5t102 -64.5 l715 -714q37 -39 37 -91z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M448 1088q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1515 512q0 -53 -37 -90l-491 -492q-39 -37 -91 -37q-53 0 -90 37l-715 716q-38 37 -64.5 101t-26.5 117v416q0 52 38 90t90 38h416q53 0 117 -26.5t102 -64.5 l715 -714q37 -39 37 -91zM1899 512q0 -53 -37 -90l-491 -492q-39 -37 -91 -37q-36 0 -59 14t-53 45l470 470q37 37 37 90q0 52 -37 91l-715 714q-38 38 -102 64.5t-117 26.5h224q53 0 117 -26.5t102 -64.5l715 -714q37 -39 37 -91z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1639 1058q40 -57 18 -129l-275 -906q-19 -64 -76.5 -107.5t-122.5 -43.5h-923q-77 0 -148.5 53.5t-99.5 131.5q-24 67 -2 127q0 4 3 27t4 37q1 8 -3 21.5t-3 19.5q2 11 8 21t16.5 23.5t16.5 23.5q23 38 45 91.5t30 91.5q3 10 0.5 30t-0.5 28q3 11 17 28t17 23 q21 36 42 92t25 90q1 9 -2.5 32t0.5 28q4 13 22 30.5t22 22.5q19 26 42.5 84.5t27.5 96.5q1 8 -3 25.5t-2 26.5q2 8 9 18t18 23t17 21q8 12 16.5 30.5t15 35t16 36t19.5 32t26.5 23.5t36 11.5t47.5 -5.5l-1 -3q38 9 51 9h761q74 0 114 -56t18 -130l-274 -906 q-36 -119 -71.5 -153.5t-128.5 -34.5h-869q-27 0 -38 -15q-11 -16 -1 -43q24 -70 144 -70h923q29 0 56 15.5t35 41.5l300 987q7 22 5 57q38 -15 59 -43zM575 1056q-4 -13 2 -22.5t20 -9.5h608q13 0 25.5 9.5t16.5 22.5l21 64q4 13 -2 22.5t-20 9.5h-608q-13 0 -25.5 -9.5 t-16.5 -22.5zM492 800q-4 -13 2 -22.5t20 -9.5h608q13 0 25.5 9.5t16.5 22.5l21 64q4 13 -2 22.5t-20 9.5h-608q-13 0 -25.5 -9.5t-16.5 -22.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1280" d="M1164 1408q23 0 44 -9q33 -13 52.5 -41t19.5 -62v-1289q0 -34 -19.5 -62t-52.5 -41q-19 -8 -44 -8q-48 0 -83 32l-441 424l-441 -424q-36 -33 -83 -33q-23 0 -44 9q-33 13 -52.5 41t-19.5 62v1289q0 34 19.5 62t52.5 41q21 9 44 9h1048z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M384 0h896v256h-896v-256zM384 640h896v384h-160q-40 0 -68 28t-28 68v160h-640v-640zM1536 576q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 576v-416q0 -13 -9.5 -22.5t-22.5 -9.5h-224v-160q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68 v160h-224q-13 0 -22.5 9.5t-9.5 22.5v416q0 79 56.5 135.5t135.5 56.5h64v544q0 40 28 68t68 28h672q40 0 88 -20t76 -48l152 -152q28 -28 48 -76t20 -88v-256h64q79 0 135.5 -56.5t56.5 -135.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M960 864q119 0 203.5 -84.5t84.5 -203.5t-84.5 -203.5t-203.5 -84.5t-203.5 84.5t-84.5 203.5t84.5 203.5t203.5 84.5zM1664 1280q106 0 181 -75t75 -181v-896q0 -106 -75 -181t-181 -75h-1408q-106 0 -181 75t-75 181v896q0 106 75 181t181 75h224l51 136 q19 49 69.5 84.5t103.5 35.5h512q53 0 103.5 -35.5t69.5 -84.5l51 -136h224zM960 128q185 0 316.5 131.5t131.5 316.5t-131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M725 977l-170 -450q73 -1 153.5 -2t119 -1.5t52.5 -0.5l29 2q-32 95 -92 241q-53 132 -92 211zM21 -128h-21l2 79q22 7 80 18q89 16 110 31q20 16 48 68l237 616l280 724h75h53l11 -21l205 -480q103 -242 124 -297q39 -102 96 -235q26 -58 65 -164q24 -67 65 -149 q22 -49 35 -57q22 -19 69 -23q47 -6 103 -27q6 -39 6 -57q0 -14 -1 -26q-80 0 -192 8q-93 8 -189 8q-79 0 -135 -2l-200 -11l-58 -2q0 45 4 78l131 28q56 13 68 23q12 12 12 27t-6 32l-47 114l-92 228l-450 2q-29 -65 -104 -274q-23 -64 -23 -84q0 -31 17 -43 q26 -21 103 -32q3 0 13.5 -2t30 -5t40.5 -6q1 -28 1 -58q0 -17 -2 -27q-66 0 -349 20l-48 -8q-81 -14 -167 -14z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M555 15q76 -32 140 -32q131 0 216 41t122 113q38 70 38 181q0 114 -41 180q-58 94 -141 126q-80 32 -247 32q-74 0 -101 -10v-144l-1 -173l3 -270q0 -15 12 -44zM541 761q43 -7 109 -7q175 0 264 65t89 224q0 112 -85 187q-84 75 -255 75q-52 0 -130 -13q0 -44 2 -77 q7 -122 6 -279l-1 -98q0 -43 1 -77zM0 -128l2 94q45 9 68 12q77 12 123 31q17 27 21 51q9 66 9 194l-2 497q-5 256 -9 404q-1 87 -11 109q-1 4 -12 12q-18 12 -69 15q-30 2 -114 13l-4 83l260 6l380 13l45 1q5 0 14 0.5t14 0.5q1 0 21.5 -0.5t40.5 -0.5h74q88 0 191 -27 q43 -13 96 -39q57 -29 102 -76q44 -47 65 -104t21 -122q0 -70 -32 -128t-95 -105q-26 -20 -150 -77q177 -41 267 -146q92 -106 92 -236q0 -76 -29 -161q-21 -62 -71 -117q-66 -72 -140 -108q-73 -36 -203 -60q-82 -15 -198 -11l-197 4q-84 2 -298 -11q-33 -3 -272 -11z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M0 -126l17 85q4 1 77 20q76 19 116 39q29 37 41 101l27 139l56 268l12 64q8 44 17 84.5t16 67t12.5 46.5t9 30.5t3.5 11.5l29 157l16 63l22 135l8 50v38q-41 22 -144 28q-28 2 -38 4l19 103l317 -14q39 -2 73 -2q66 0 214 9q33 2 68 4.5t36 2.5q-2 -19 -6 -38 q-7 -29 -13 -51q-55 -19 -109 -31q-64 -16 -101 -31q-12 -31 -24 -88q-9 -44 -13 -82q-44 -199 -66 -306l-61 -311l-38 -158l-43 -235l-12 -45q-2 -7 1 -27q64 -15 119 -21q36 -5 66 -10q-1 -29 -7 -58q-7 -31 -9 -41q-18 0 -23 -1q-24 -2 -42 -2q-9 0 -28 3q-19 4 -145 17 l-198 2q-41 1 -174 -11q-74 -7 -98 -9z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M81 1407l54 -27q20 -5 211 -5h130l19 3l115 1l215 -1h293l34 -2q14 -1 28 7t21 16l7 8l42 1q15 0 28 -1v-104.5t1 -131.5l1 -100l-1 -58q0 -32 -4 -51q-39 -15 -68 -18q-25 43 -54 128q-8 24 -15.5 62.5t-11.5 65.5t-6 29q-13 15 -27 19q-7 2 -42.5 2t-103.5 -1t-111 -1 q-34 0 -67 -5q-10 -97 -8 -136l1 -152v-332l3 -359l-1 -147q-1 -46 11 -85q49 -25 89 -32q2 0 18 -5t44 -13t43 -12q30 -8 50 -18q5 -45 5 -50q0 -10 -3 -29q-14 -1 -34 -1q-110 0 -187 10q-72 8 -238 8q-88 0 -233 -14q-48 -4 -70 -4q-2 22 -2 26l-1 26v9q21 33 79 49 q139 38 159 50q9 21 12 56q8 192 6 433l-5 428q-1 62 -0.5 118.5t0.5 102.5t-2 57t-6 15q-6 5 -14 6q-38 6 -148 6q-43 0 -100 -13.5t-73 -24.5q-13 -9 -22 -33t-22 -75t-24 -84q-6 -19 -19.5 -32t-20.5 -13q-44 27 -56 44v297v86zM1744 128q33 0 42 -18.5t-11 -44.5 l-126 -162q-20 -26 -49 -26t-49 26l-126 162q-20 26 -11 44.5t42 18.5h80v1024h-80q-33 0 -42 18.5t11 44.5l126 162q20 26 49 26t49 -26l126 -162q20 -26 11 -44.5t-42 -18.5h-80v-1024h80z" /> | ||||||
|  | <glyph unicode="" d="M81 1407l54 -27q20 -5 211 -5h130l19 3l115 1l446 -1h318l34 -2q14 -1 28 7t21 16l7 8l42 1q15 0 28 -1v-104.5t1 -131.5l1 -100l-1 -58q0 -32 -4 -51q-39 -15 -68 -18q-25 43 -54 128q-8 24 -15.5 62.5t-11.5 65.5t-6 29q-13 15 -27 19q-7 2 -58.5 2t-138.5 -1t-128 -1 q-94 0 -127 -5q-10 -97 -8 -136l1 -152v52l3 -359l-1 -147q-1 -46 11 -85q49 -25 89 -32q2 0 18 -5t44 -13t43 -12q30 -8 50 -18q5 -45 5 -50q0 -10 -3 -29q-14 -1 -34 -1q-110 0 -187 10q-72 8 -238 8q-82 0 -233 -13q-45 -5 -70 -5q-2 22 -2 26l-1 26v9q21 33 79 49 q139 38 159 50q9 21 12 56q6 137 6 433l-5 44q0 265 -2 278q-2 11 -6 15q-6 5 -14 6q-38 6 -148 6q-50 0 -168.5 -14t-132.5 -24q-13 -9 -22 -33t-22 -75t-24 -84q-6 -19 -19.5 -32t-20.5 -13q-44 27 -56 44v297v86zM1505 113q26 -20 26 -49t-26 -49l-162 -126 q-26 -20 -44.5 -11t-18.5 42v80h-1024v-80q0 -33 -18.5 -42t-44.5 11l-162 126q-26 20 -26 49t26 49l162 126q26 20 44.5 11t18.5 -42v-80h1024v80q0 33 18.5 42t44.5 -11z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1408 576v-128q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1280q26 0 45 -19t19 -45zM1664 960v-128q0 -26 -19 -45 t-45 -19h-1536q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1536q26 0 45 -19t19 -45zM1280 1344v-128q0 -26 -19 -45t-45 -19h-1152q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1408 576v-128q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h896q26 0 45 -19t19 -45zM1664 960v-128q0 -26 -19 -45t-45 -19 h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1280 1344v-128q0 -26 -19 -45t-45 -19h-640q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h640q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 576v-128q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1280q26 0 45 -19t19 -45zM1792 960v-128q0 -26 -19 -45 t-45 -19h-1536q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1536q26 0 45 -19t19 -45zM1792 1344v-128q0 -26 -19 -45t-45 -19h-1152q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 576v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 960v-128q0 -26 -19 -45 t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 1344v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M256 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5t9.5 -22.5zM256 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5 t9.5 -22.5zM256 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5t9.5 -22.5zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1344 q13 0 22.5 -9.5t9.5 -22.5zM256 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5t9.5 -22.5zM1792 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5 t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5zM1792 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5zM1792 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192 q0 13 9.5 22.5t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M384 992v-576q0 -13 -9.5 -22.5t-22.5 -9.5q-14 0 -23 9l-288 288q-9 9 -9 23t9 23l288 288q9 9 23 9q13 0 22.5 -9.5t9.5 -22.5zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5 t9.5 -22.5zM1792 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088q13 0 22.5 -9.5t9.5 -22.5zM1792 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088 q13 0 22.5 -9.5t9.5 -22.5zM1792 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5t9.5 -22.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M352 704q0 -14 -9 -23l-288 -288q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5v576q0 13 9.5 22.5t22.5 9.5q14 0 23 -9l288 -288q9 -9 9 -23zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5 t9.5 -22.5zM1792 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088q13 0 22.5 -9.5t9.5 -22.5zM1792 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088 q13 0 22.5 -9.5t9.5 -22.5zM1792 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5t9.5 -22.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 1184v-1088q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-403 403v-166q0 -119 -84.5 -203.5t-203.5 -84.5h-704q-119 0 -203.5 84.5t-84.5 203.5v704q0 119 84.5 203.5t203.5 84.5h704q119 0 203.5 -84.5t84.5 -203.5v-165l403 402q18 19 45 19q12 0 25 -5 q39 -17 39 -59z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M640 960q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1664 576v-448h-1408v192l320 320l160 -160l512 512zM1760 1280h-1600q-13 0 -22.5 -9.5t-9.5 -22.5v-1216q0 -13 9.5 -22.5t22.5 -9.5h1600q13 0 22.5 9.5t9.5 22.5v1216 q0 13 -9.5 22.5t-22.5 9.5zM1920 1248v-1216q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" /> | ||||||
|  | <glyph unicode="" d="M363 0l91 91l-235 235l-91 -91v-107h128v-128h107zM886 928q0 22 -22 22q-10 0 -17 -7l-542 -542q-7 -7 -7 -17q0 -22 22 -22q10 0 17 7l542 542q7 7 7 17zM832 1120l416 -416l-832 -832h-416v416zM1515 1024q0 -53 -37 -90l-166 -166l-416 416l166 165q36 38 90 38 q53 0 91 -38l235 -234q37 -39 37 -91z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M768 896q0 106 -75 181t-181 75t-181 -75t-75 -181t75 -181t181 -75t181 75t75 181zM1024 896q0 -109 -33 -179l-364 -774q-16 -33 -47.5 -52t-67.5 -19t-67.5 19t-46.5 52l-365 774q-33 70 -33 179q0 212 150 362t362 150t362 -150t150 -362z" /> | ||||||
|  | <glyph unicode="" d="M768 96v1088q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M512 384q0 36 -20 69q-1 1 -15.5 22.5t-25.5 38t-25 44t-21 50.5q-4 16 -21 16t-21 -16q-7 -23 -21 -50.5t-25 -44t-25.5 -38t-15.5 -22.5q-20 -33 -20 -69q0 -53 37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1024 512q0 -212 -150 -362t-362 -150t-362 150t-150 362 q0 145 81 275q6 9 62.5 90.5t101 151t99.5 178t83 201.5q9 30 34 47t51 17t51.5 -17t33.5 -47q28 -93 83 -201.5t99.5 -178t101 -151t62.5 -90.5q81 -127 81 -275z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M888 352l116 116l-152 152l-116 -116v-56h96v-96h56zM1328 1072q-16 16 -33 -1l-350 -350q-17 -17 -1 -33t33 1l350 350q17 17 1 33zM1408 478v-190q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832 q63 0 117 -25q15 -7 18 -23q3 -17 -9 -29l-49 -49q-14 -14 -32 -8q-23 6 -45 6h-832q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v126q0 13 9 22l64 64q15 15 35 7t20 -29zM1312 1216l288 -288l-672 -672h-288v288zM1756 1084l-92 -92 l-288 288l92 92q28 28 68 28t68 -28l152 -152q28 -28 28 -68t-28 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1408 547v-259q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h255v0q13 0 22.5 -9.5t9.5 -22.5q0 -27 -26 -32q-77 -26 -133 -60q-10 -4 -16 -4h-112q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832 q66 0 113 47t47 113v214q0 19 18 29q28 13 54 37q16 16 35 8q21 -9 21 -29zM1645 1043l-384 -384q-18 -19 -45 -19q-12 0 -25 5q-39 17 -39 59v192h-160q-323 0 -438 -131q-119 -137 -74 -473q3 -23 -20 -34q-8 -2 -12 -2q-16 0 -26 13q-10 14 -21 31t-39.5 68.5t-49.5 99.5 t-38.5 114t-17.5 122q0 49 3.5 91t14 90t28 88t47 81.5t68.5 74t94.5 61.5t124.5 48.5t159.5 30.5t196.5 11h160v192q0 42 39 59q13 5 25 5q26 0 45 -19l384 -384q19 -19 19 -45t-19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1408 606v-318q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832q63 0 117 -25q15 -7 18 -23q3 -17 -9 -29l-49 -49q-10 -10 -23 -10q-3 0 -9 2q-23 6 -45 6h-832q-66 0 -113 -47t-47 -113v-832 q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v254q0 13 9 22l64 64q10 10 23 10q6 0 12 -3q20 -8 20 -29zM1639 1095l-814 -814q-24 -24 -57 -24t-57 24l-430 430q-24 24 -24 57t24 57l110 110q24 24 57 24t57 -24l263 -263l647 647q24 24 57 24t57 -24l110 -110 q24 -24 24 -57t-24 -57z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 640q0 -26 -19 -45l-256 -256q-19 -19 -45 -19t-45 19t-19 45v128h-384v-384h128q26 0 45 -19t19 -45t-19 -45l-256 -256q-19 -19 -45 -19t-45 19l-256 256q-19 19 -19 45t19 45t45 19h128v384h-384v-128q0 -26 -19 -45t-45 -19t-45 19l-256 256q-19 19 -19 45 t19 45l256 256q19 19 45 19t45 -19t19 -45v-128h384v384h-128q-26 0 -45 19t-19 45t19 45l256 256q19 19 45 19t45 -19l256 -256q19 -19 19 -45t-19 -45t-45 -19h-128v-384h384v128q0 26 19 45t45 19t45 -19l256 -256q19 -19 19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M979 1395q19 19 32 13t13 -32v-1472q0 -26 -13 -32t-32 13l-710 710q-9 9 -13 19v-678q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-678q4 11 13 19z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1747 1395q19 19 32 13t13 -32v-1472q0 -26 -13 -32t-32 13l-710 710q-9 9 -13 19v-710q0 -26 -13 -32t-32 13l-710 710q-9 9 -13 19v-678q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-678q4 11 13 19l710 710 q19 19 32 13t13 -32v-710q4 11 13 19z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1619 1395q19 19 32 13t13 -32v-1472q0 -26 -13 -32t-32 13l-710 710q-8 9 -13 19v-710q0 -26 -13 -32t-32 13l-710 710q-19 19 -19 45t19 45l710 710q19 19 32 13t13 -32v-710q5 11 13 19z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1384 609l-1328 -738q-23 -13 -39.5 -3t-16.5 36v1472q0 26 16.5 36t39.5 -3l1328 -738q23 -13 23 -31t-23 -31z" /> | ||||||
|  | <glyph unicode="" d="M1536 1344v-1408q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h512q26 0 45 -19t19 -45zM640 1344v-1408q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h512q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" d="M1536 1344v-1408q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h1408q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M45 -115q-19 -19 -32 -13t-13 32v1472q0 26 13 32t32 -13l710 -710q8 -8 13 -19v710q0 26 13 32t32 -13l710 -710q19 -19 19 -45t-19 -45l-710 -710q-19 -19 -32 -13t-13 32v710q-5 -10 -13 -19z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M45 -115q-19 -19 -32 -13t-13 32v1472q0 26 13 32t32 -13l710 -710q8 -8 13 -19v710q0 26 13 32t32 -13l710 -710q8 -8 13 -19v678q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-1408q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v678q-5 -10 -13 -19l-710 -710 q-19 -19 -32 -13t-13 32v710q-5 -10 -13 -19z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M45 -115q-19 -19 -32 -13t-13 32v1472q0 26 13 32t32 -13l710 -710q8 -8 13 -19v678q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-1408q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v678q-5 -10 -13 -19z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1538" d="M14 557l710 710q19 19 45 19t45 -19l710 -710q19 -19 13 -32t-32 -13h-1472q-26 0 -32 13t13 32zM1473 0h-1408q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1408q26 0 45 -19t19 -45v-256q0 -26 -19 -45t-45 -19z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M742 -37l-652 651q-37 37 -37 90.5t37 90.5l652 651q37 37 90.5 37t90.5 -37l75 -75q37 -37 37 -90.5t-37 -90.5l-486 -486l486 -485q37 -38 37 -91t-37 -90l-75 -75q-37 -37 -90.5 -37t-90.5 37z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M1099 704q0 -52 -37 -91l-652 -651q-37 -37 -90 -37t-90 37l-76 75q-37 39 -37 91q0 53 37 90l486 486l-486 485q-37 39 -37 91q0 53 37 90l76 75q36 38 90 38t90 -38l652 -651q37 -37 37 -90z" /> | ||||||
|  | <glyph unicode="" d="M1216 576v128q0 26 -19 45t-45 19h-256v256q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-256h-256q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h256v-256q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v256h256q26 0 45 19t19 45zM1536 640q0 -209 -103 -385.5 t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1216 576v128q0 26 -19 45t-45 19h-768q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h768q26 0 45 19t19 45zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5 t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1149 414q0 26 -19 45l-181 181l181 181q19 19 19 45q0 27 -19 46l-90 90q-19 19 -46 19q-26 0 -45 -19l-181 -181l-181 181q-19 19 -45 19q-27 0 -46 -19l-90 -90q-19 -19 -19 -46q0 -26 19 -45l181 -181l-181 -181q-19 -19 -19 -45q0 -27 19 -46l90 -90q19 -19 46 -19 q26 0 45 19l181 181l181 -181q19 -19 45 -19q27 0 46 19l90 90q19 19 19 46zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1284 802q0 28 -18 46l-91 90q-19 19 -45 19t-45 -19l-408 -407l-226 226q-19 19 -45 19t-45 -19l-91 -90q-18 -18 -18 -46q0 -27 18 -45l362 -362q19 -19 45 -19q27 0 46 19l543 543q18 18 18 45zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103 t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M896 160v192q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h192q14 0 23 9t9 23zM1152 832q0 88 -55.5 163t-138.5 116t-170 41q-243 0 -371 -213q-15 -24 8 -42l132 -100q7 -6 19 -6q16 0 25 12q53 68 86 92q34 24 86 24q48 0 85.5 -26t37.5 -59 q0 -38 -20 -61t-68 -45q-63 -28 -115.5 -86.5t-52.5 -125.5v-36q0 -14 9 -23t23 -9h192q14 0 23 9t9 23q0 19 21.5 49.5t54.5 49.5q32 18 49 28.5t46 35t44.5 48t28 60.5t12.5 81zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1024 160v160q0 14 -9 23t-23 9h-96v512q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-160q0 -14 9 -23t23 -9h96v-320h-96q-14 0 -23 -9t-9 -23v-160q0 -14 9 -23t23 -9h448q14 0 23 9t9 23zM896 1056v160q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-160q0 -14 9 -23 t23 -9h192q14 0 23 9t9 23zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1197 512h-109q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h109q-32 108 -112.5 188.5t-188.5 112.5v-109q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v109q-108 -32 -188.5 -112.5t-112.5 -188.5h109q26 0 45 -19t19 -45v-128q0 -26 -19 -45t-45 -19h-109 q32 -108 112.5 -188.5t188.5 -112.5v109q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-109q108 32 188.5 112.5t112.5 188.5zM1536 704v-128q0 -26 -19 -45t-45 -19h-143q-37 -161 -154.5 -278.5t-278.5 -154.5v-143q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v143 q-161 37 -278.5 154.5t-154.5 278.5h-143q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h143q37 161 154.5 278.5t278.5 154.5v143q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-143q161 -37 278.5 -154.5t154.5 -278.5h143q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" d="M1097 457l-146 -146q-10 -10 -23 -10t-23 10l-137 137l-137 -137q-10 -10 -23 -10t-23 10l-146 146q-10 10 -10 23t10 23l137 137l-137 137q-10 10 -10 23t10 23l146 146q10 10 23 10t23 -10l137 -137l137 137q10 10 23 10t23 -10l146 -146q10 -10 10 -23t-10 -23 l-137 -137l137 -137q10 -10 10 -23t-10 -23zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5 t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1171 723l-422 -422q-19 -19 -45 -19t-45 19l-294 294q-19 19 -19 45t19 45l102 102q19 19 45 19t45 -19l147 -147l275 275q19 19 45 19t45 -19l102 -102q19 -19 19 -45t-19 -45zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198 t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1312 643q0 161 -87 295l-754 -753q137 -89 297 -89q111 0 211.5 43.5t173.5 116.5t116 174.5t43 212.5zM313 344l755 754q-135 91 -300 91q-148 0 -273 -73t-198 -199t-73 -274q0 -162 89 -299zM1536 643q0 -157 -61 -300t-163.5 -246t-245 -164t-298.5 -61t-298.5 61 t-245 164t-163.5 246t-61 300t61 299.5t163.5 245.5t245 164t298.5 61t298.5 -61t245 -164t163.5 -245.5t61 -299.5z" /> | ||||||
|  | <glyph unicode="" d="M1536 640v-128q0 -53 -32.5 -90.5t-84.5 -37.5h-704l293 -294q38 -36 38 -90t-38 -90l-75 -76q-37 -37 -90 -37q-52 0 -91 37l-651 652q-37 37 -37 90q0 52 37 91l651 650q38 38 91 38q52 0 90 -38l75 -74q38 -38 38 -91t-38 -91l-293 -293h704q52 0 84.5 -37.5 t32.5 -90.5z" /> | ||||||
|  | <glyph unicode="" d="M1472 576q0 -54 -37 -91l-651 -651q-39 -37 -91 -37q-51 0 -90 37l-75 75q-38 38 -38 91t38 91l293 293h-704q-52 0 -84.5 37.5t-32.5 90.5v128q0 53 32.5 90.5t84.5 37.5h704l-293 294q-38 36 -38 90t38 90l75 75q38 38 90 38q53 0 91 -38l651 -651q37 -35 37 -90z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1611 565q0 -51 -37 -90l-75 -75q-38 -38 -91 -38q-54 0 -90 38l-294 293v-704q0 -52 -37.5 -84.5t-90.5 -32.5h-128q-53 0 -90.5 32.5t-37.5 84.5v704l-294 -293q-36 -38 -90 -38t-90 38l-75 75q-38 38 -38 90q0 53 38 91l651 651q35 37 90 37q54 0 91 -37l651 -651 q37 -39 37 -91z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1611 704q0 -53 -37 -90l-651 -652q-39 -37 -91 -37q-53 0 -90 37l-651 652q-38 36 -38 90q0 53 38 91l74 75q39 37 91 37q53 0 90 -37l294 -294v704q0 52 38 90t90 38h128q52 0 90 -38t38 -90v-704l294 294q37 37 90 37q52 0 91 -37l75 -75q37 -39 37 -91z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 896q0 -26 -19 -45l-512 -512q-19 -19 -45 -19t-45 19t-19 45v256h-224q-98 0 -175.5 -6t-154 -21.5t-133 -42.5t-105.5 -69.5t-80 -101t-48.5 -138.5t-17.5 -181q0 -55 5 -123q0 -6 2.5 -23.5t2.5 -26.5q0 -15 -8.5 -25t-23.5 -10q-16 0 -28 17q-7 9 -13 22 t-13.5 30t-10.5 24q-127 285 -127 451q0 199 53 333q162 403 875 403h224v256q0 26 19 45t45 19t45 -19l512 -512q19 -19 19 -45z" /> | ||||||
|  | <glyph unicode="" d="M755 480q0 -13 -10 -23l-332 -332l144 -144q19 -19 19 -45t-19 -45t-45 -19h-448q-26 0 -45 19t-19 45v448q0 26 19 45t45 19t45 -19l144 -144l332 332q10 10 23 10t23 -10l114 -114q10 -10 10 -23zM1536 1344v-448q0 -26 -19 -45t-45 -19t-45 19l-144 144l-332 -332 q-10 -10 -23 -10t-23 10l-114 114q-10 10 -10 23t10 23l332 332l-144 144q-19 19 -19 45t19 45t45 19h448q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" d="M768 576v-448q0 -26 -19 -45t-45 -19t-45 19l-144 144l-332 -332q-10 -10 -23 -10t-23 10l-114 114q-10 10 -10 23t10 23l332 332l-144 144q-19 19 -19 45t19 45t45 19h448q26 0 45 -19t19 -45zM1523 1248q0 -13 -10 -23l-332 -332l144 -144q19 -19 19 -45t-19 -45 t-45 -19h-448q-26 0 -45 19t-19 45v448q0 26 19 45t45 19t45 -19l144 -144l332 332q10 10 23 10t23 -10l114 -114q10 -10 10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1408 800v-192q0 -40 -28 -68t-68 -28h-416v-416q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v416h-416q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h416v416q0 40 28 68t68 28h192q40 0 68 -28t28 -68v-416h416q40 0 68 -28t28 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1408 800v-192q0 -40 -28 -68t-68 -28h-1216q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h1216q40 0 68 -28t28 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1482 486q46 -26 59.5 -77.5t-12.5 -97.5l-64 -110q-26 -46 -77.5 -59.5t-97.5 12.5l-266 153v-307q0 -52 -38 -90t-90 -38h-128q-52 0 -90 38t-38 90v307l-266 -153q-46 -26 -97.5 -12.5t-77.5 59.5l-64 110q-26 46 -12.5 97.5t59.5 77.5l266 154l-266 154 q-46 26 -59.5 77.5t12.5 97.5l64 110q26 46 77.5 59.5t97.5 -12.5l266 -153v307q0 52 38 90t90 38h128q52 0 90 -38t38 -90v-307l266 153q46 26 97.5 12.5t77.5 -59.5l64 -110q26 -46 12.5 -97.5t-59.5 -77.5l-266 -154z" /> | ||||||
|  | <glyph unicode="" d="M768 1408q209 0 385.5 -103t279.5 -279.5t103 -385.5t-103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103zM896 161v190q0 14 -9 23.5t-22 9.5h-192q-13 0 -23 -10t-10 -23v-190q0 -13 10 -23t23 -10h192 q13 0 22 9.5t9 23.5zM894 505l18 621q0 12 -10 18q-10 8 -24 8h-220q-14 0 -24 -8q-10 -6 -10 -18l17 -621q0 -10 10 -17.5t24 -7.5h185q14 0 23.5 7.5t10.5 17.5z" /> | ||||||
|  | <glyph unicode="" d="M928 180v56v468v192h-320v-192v-468v-56q0 -25 18 -38.5t46 -13.5h192q28 0 46 13.5t18 38.5zM472 1024h195l-126 161q-26 31 -69 31q-40 0 -68 -28t-28 -68t28 -68t68 -28zM1160 1120q0 40 -28 68t-68 28q-43 0 -69 -31l-125 -161h194q40 0 68 28t28 68zM1536 864v-320 q0 -14 -9 -23t-23 -9h-96v-416q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28t-28 68v416h-96q-14 0 -23 9t-9 23v320q0 14 9 23t23 9h440q-93 0 -158.5 65.5t-65.5 158.5t65.5 158.5t158.5 65.5q107 0 168 -77l128 -165l128 165q61 77 168 77q93 0 158.5 -65.5t65.5 -158.5 t-65.5 -158.5t-158.5 -65.5h440q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1280 832q0 26 -19 45t-45 19q-172 0 -318 -49.5t-259.5 -134t-235.5 -219.5q-19 -21 -19 -45q0 -26 19 -45t45 -19q24 0 45 19q27 24 74 71t67 66q137 124 268.5 176t313.5 52q26 0 45 19t19 45zM1792 1030q0 -95 -20 -193q-46 -224 -184.5 -383t-357.5 -268 q-214 -108 -438 -108q-148 0 -286 47q-15 5 -88 42t-96 37q-16 0 -39.5 -32t-45 -70t-52.5 -70t-60 -32q-30 0 -51 11t-31 24t-27 42q-2 4 -6 11t-5.5 10t-3 9.5t-1.5 13.5q0 35 31 73.5t68 65.5t68 56t31 48q0 4 -14 38t-16 44q-9 51 -9 104q0 115 43.5 220t119 184.5 t170.5 139t204 95.5q55 18 145 25.5t179.5 9t178.5 6t163.5 24t113.5 56.5l29.5 29.5t29.5 28t27 20t36.5 16t43.5 4.5q39 0 70.5 -46t47.5 -112t24 -124t8 -96z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1408 -160v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5zM1152 896q0 -78 -24.5 -144t-64 -112.5t-87.5 -88t-96 -77.5t-87.5 -72t-64 -81.5t-24.5 -96.5q0 -96 67 -224l-4 1l1 -1 q-90 41 -160 83t-138.5 100t-113.5 122.5t-72.5 150.5t-27.5 184q0 78 24.5 144t64 112.5t87.5 88t96 77.5t87.5 72t64 81.5t24.5 96.5q0 94 -66 224l3 -1l-1 1q90 -41 160 -83t138.5 -100t113.5 -122.5t72.5 -150.5t27.5 -184z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1664 576q-152 236 -381 353q61 -104 61 -225q0 -185 -131.5 -316.5t-316.5 -131.5t-316.5 131.5t-131.5 316.5q0 121 61 225q-229 -117 -381 -353q133 -205 333.5 -326.5t434.5 -121.5t434.5 121.5t333.5 326.5zM944 960q0 20 -14 34t-34 14q-125 0 -214.5 -89.5 t-89.5 -214.5q0 -20 14 -34t34 -14t34 14t14 34q0 86 61 147t147 61q20 0 34 14t14 34zM1792 576q0 -34 -20 -69q-140 -230 -376.5 -368.5t-499.5 -138.5t-499.5 139t-376.5 368q-20 35 -20 69t20 69q140 229 376.5 368t499.5 139t499.5 -139t376.5 -368q20 -35 20 -69z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M555 201l78 141q-87 63 -136 159t-49 203q0 121 61 225q-229 -117 -381 -353q167 -258 427 -375zM944 960q0 20 -14 34t-34 14q-125 0 -214.5 -89.5t-89.5 -214.5q0 -20 14 -34t34 -14t34 14t14 34q0 86 61 147t147 61q20 0 34 14t14 34zM1307 1151q0 -7 -1 -9 q-105 -188 -315 -566t-316 -567l-49 -89q-10 -16 -28 -16q-12 0 -134 70q-16 10 -16 28q0 12 44 87q-143 65 -263.5 173t-208.5 245q-20 31 -20 69t20 69q153 235 380 371t496 136q89 0 180 -17l54 97q10 16 28 16q5 0 18 -6t31 -15.5t33 -18.5t31.5 -18.5t19.5 -11.5 q16 -10 16 -27zM1344 704q0 -139 -79 -253.5t-209 -164.5l280 502q8 -45 8 -84zM1792 576q0 -35 -20 -69q-39 -64 -109 -145q-150 -172 -347.5 -267t-419.5 -95l74 132q212 18 392.5 137t301.5 307q-115 179 -282 294l63 112q95 -64 182.5 -153t144.5 -184q20 -34 20 -69z " /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1024 161v190q0 14 -9.5 23.5t-22.5 9.5h-192q-13 0 -22.5 -9.5t-9.5 -23.5v-190q0 -14 9.5 -23.5t22.5 -9.5h192q13 0 22.5 9.5t9.5 23.5zM1022 535l18 459q0 12 -10 19q-13 11 -24 11h-220q-11 0 -24 -11q-10 -7 -10 -21l17 -457q0 -10 10 -16.5t24 -6.5h185 q14 0 23.5 6.5t10.5 16.5zM1008 1469l768 -1408q35 -63 -2 -126q-17 -29 -46.5 -46t-63.5 -17h-1536q-34 0 -63.5 17t-46.5 46q-37 63 -2 126l768 1408q17 31 47 49t65 18t65 -18t47 -49z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1376 1376q44 -52 12 -148t-108 -172l-161 -161l160 -696q5 -19 -12 -33l-128 -96q-7 -6 -19 -6q-4 0 -7 1q-15 3 -21 16l-279 508l-259 -259l53 -194q5 -17 -8 -31l-96 -96q-9 -9 -23 -9h-2q-15 2 -24 13l-189 252l-252 189q-11 7 -13 23q-1 13 9 25l96 97q9 9 23 9 q6 0 8 -1l194 -53l259 259l-508 279q-14 8 -17 24q-2 16 9 27l128 128q14 13 30 8l665 -159l160 160q76 76 172 108t148 -12z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M128 -128h288v288h-288v-288zM480 -128h320v288h-320v-288zM128 224h288v320h-288v-320zM480 224h320v320h-320v-320zM128 608h288v288h-288v-288zM864 -128h320v288h-320v-288zM480 608h320v288h-320v-288zM1248 -128h288v288h-288v-288zM864 224h320v320h-320v-320z M512 1088v288q0 13 -9.5 22.5t-22.5 9.5h-64q-13 0 -22.5 -9.5t-9.5 -22.5v-288q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5zM1248 224h288v320h-288v-320zM864 608h320v288h-320v-288zM1248 608h288v288h-288v-288zM1280 1088v288q0 13 -9.5 22.5t-22.5 9.5h-64 q-13 0 -22.5 -9.5t-9.5 -22.5v-288q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5zM1664 1152v-1280q0 -52 -38 -90t-90 -38h-1408q-52 0 -90 38t-38 90v1280q0 52 38 90t90 38h128v96q0 66 47 113t113 47h64q66 0 113 -47t47 -113v-96h384v96q0 66 47 113t113 47 h64q66 0 113 -47t47 -113v-96h128q52 0 90 -38t38 -90z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M666 1055q-60 -92 -137 -273q-22 45 -37 72.5t-40.5 63.5t-51 56.5t-63 35t-81.5 14.5h-224q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h224q250 0 410 -225zM1792 256q0 -14 -9 -23l-320 -320q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5v192q-32 0 -85 -0.5t-81 -1t-73 1 t-71 5t-64 10.5t-63 18.5t-58 28.5t-59 40t-55 53.5t-56 69.5q59 93 136 273q22 -45 37 -72.5t40.5 -63.5t51 -56.5t63 -35t81.5 -14.5h256v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23zM1792 1152q0 -14 -9 -23l-320 -320q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5 v192h-256q-48 0 -87 -15t-69 -45t-51 -61.5t-45 -77.5q-32 -62 -78 -171q-29 -66 -49.5 -111t-54 -105t-64 -100t-74 -83t-90 -68.5t-106.5 -42t-128 -16.5h-224q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h224q48 0 87 15t69 45t51 61.5t45 77.5q32 62 78 171q29 66 49.5 111 t54 105t64 100t74 83t90 68.5t106.5 42t128 16.5h256v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 640q0 -174 -120 -321.5t-326 -233t-450 -85.5q-70 0 -145 8q-198 -175 -460 -242q-49 -14 -114 -22q-17 -2 -30.5 9t-17.5 29v1q-3 4 -0.5 12t2 10t4.5 9.5l6 9t7 8.5t8 9q7 8 31 34.5t34.5 38t31 39.5t32.5 51t27 59t26 76q-157 89 -247.5 220t-90.5 281 q0 130 71 248.5t191 204.5t286 136.5t348 50.5q244 0 450 -85.5t326 -233t120 -321.5z" /> | ||||||
|  | <glyph unicode="" d="M1536 704v-128q0 -201 -98.5 -362t-274 -251.5t-395.5 -90.5t-395.5 90.5t-274 251.5t-98.5 362v128q0 26 19 45t45 19h384q26 0 45 -19t19 -45v-128q0 -52 23.5 -90t53.5 -57t71 -30t64 -13t44 -2t44 2t64 13t71 30t53.5 57t23.5 90v128q0 26 19 45t45 19h384 q26 0 45 -19t19 -45zM512 1344v-384q0 -26 -19 -45t-45 -19h-384q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h384q26 0 45 -19t19 -45zM1536 1344v-384q0 -26 -19 -45t-45 -19h-384q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h384q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1611 320q0 -53 -37 -90l-75 -75q-38 -38 -91 -38q-54 0 -90 38l-486 485l-486 -485q-36 -38 -90 -38t-90 38l-75 75q-38 36 -38 90q0 53 38 91l651 651q37 37 90 37q52 0 91 -37l650 -651q38 -38 38 -91z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1611 832q0 -53 -37 -90l-651 -651q-38 -38 -91 -38q-54 0 -90 38l-651 651q-38 36 -38 90q0 53 38 91l74 75q39 37 91 37q53 0 90 -37l486 -486l486 486q37 37 90 37q52 0 91 -37l75 -75q37 -39 37 -91z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1280 32q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-8 0 -13.5 2t-9 7t-5.5 8t-3 11.5t-1 11.5v13v11v160v416h-192q-26 0 -45 19t-19 45q0 24 15 41l320 384q19 22 49 22t49 -22l320 -384q15 -17 15 -41q0 -26 -19 -45t-45 -19h-192v-384h576q16 0 25 -11l160 -192q7 -11 7 -21 zM1920 448q0 -24 -15 -41l-320 -384q-20 -23 -49 -23t-49 23l-320 384q-15 17 -15 41q0 26 19 45t45 19h192v384h-576q-16 0 -25 12l-160 192q-7 9 -7 20q0 13 9.5 22.5t22.5 9.5h960q8 0 13.5 -2t9 -7t5.5 -8t3 -11.5t1 -11.5v-13v-11v-160v-416h192q26 0 45 -19t19 -45z " /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M640 0q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1536 0q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1664 1088v-512q0 -24 -16 -42.5t-41 -21.5 l-1044 -122q1 -7 4.5 -21.5t6 -26.5t2.5 -22q0 -16 -24 -64h920q26 0 45 -19t19 -45t-19 -45t-45 -19h-1024q-26 0 -45 19t-19 45q0 14 11 39.5t29.5 59.5t20.5 38l-177 823h-204q-26 0 -45 19t-19 45t19 45t45 19h256q16 0 28.5 -6.5t20 -15.5t13 -24.5t7.5 -26.5 t5.5 -29.5t4.5 -25.5h1201q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1664 928v-704q0 -92 -66 -158t-158 -66h-1216q-92 0 -158 66t-66 158v960q0 92 66 158t158 66h320q92 0 158 -66t66 -158v-32h672q92 0 158 -66t66 -158z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1879 584q0 -31 -31 -66l-336 -396q-43 -51 -120.5 -86.5t-143.5 -35.5h-1088q-34 0 -60.5 13t-26.5 43q0 31 31 66l336 396q43 51 120.5 86.5t143.5 35.5h1088q34 0 60.5 -13t26.5 -43zM1536 928v-160h-832q-94 0 -197 -47.5t-164 -119.5l-337 -396l-5 -6q0 4 -0.5 12.5 t-0.5 12.5v960q0 92 66 158t158 66h320q92 0 158 -66t66 -158v-32h544q92 0 158 -66t66 -158z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="768" d="M704 1216q0 -26 -19 -45t-45 -19h-128v-1024h128q26 0 45 -19t19 -45t-19 -45l-256 -256q-19 -19 -45 -19t-45 19l-256 256q-19 19 -19 45t19 45t45 19h128v1024h-128q-26 0 -45 19t-19 45t19 45l256 256q19 19 45 19t45 -19l256 -256q19 -19 19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 640q0 -26 -19 -45l-256 -256q-19 -19 -45 -19t-45 19t-19 45v128h-1024v-128q0 -26 -19 -45t-45 -19t-45 19l-256 256q-19 19 -19 45t19 45l256 256q19 19 45 19t45 -19t19 -45v-128h1024v128q0 26 19 45t45 19t45 -19l256 -256q19 -19 19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M512 512v-384h-256v384h256zM896 1024v-896h-256v896h256zM1280 768v-640h-256v640h256zM1664 1152v-1024h-256v1024h256zM1792 32v1216q0 13 -9.5 22.5t-22.5 9.5h-1600q-13 0 -22.5 -9.5t-9.5 -22.5v-1216q0 -13 9.5 -22.5t22.5 -9.5h1600q13 0 22.5 9.5t9.5 22.5z M1920 1248v-1216q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" /> | ||||||
|  | <glyph unicode="" d="M1280 926q-56 -25 -121 -34q68 40 93 117q-65 -38 -134 -51q-61 66 -153 66q-87 0 -148.5 -61.5t-61.5 -148.5q0 -29 5 -48q-129 7 -242 65t-192 155q-29 -50 -29 -106q0 -114 91 -175q-47 1 -100 26v-2q0 -75 50 -133.5t123 -72.5q-29 -8 -51 -8q-13 0 -39 4 q21 -63 74.5 -104t121.5 -42q-116 -90 -261 -90q-26 0 -50 3q148 -94 322 -94q112 0 210 35.5t168 95t120.5 137t75 162t24.5 168.5q0 18 -1 27q63 45 105 109zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5 t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M1307 618l23 219h-198v109q0 49 15.5 68.5t71.5 19.5h110v219h-175q-152 0 -218 -72t-66 -213v-131h-131v-219h131v-635h262v635h175zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960 q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M928 704q0 14 -9 23t-23 9q-66 0 -113 -47t-47 -113q0 -14 9 -23t23 -9t23 9t9 23q0 40 28 68t68 28q14 0 23 9t9 23zM1152 574q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181zM128 0h1536v128h-1536v-128zM1280 574q0 159 -112.5 271.5 t-271.5 112.5t-271.5 -112.5t-112.5 -271.5t112.5 -271.5t271.5 -112.5t271.5 112.5t112.5 271.5zM256 1216h384v128h-384v-128zM128 1024h1536v118v138h-828l-64 -128h-644v-128zM1792 1280v-1280q0 -53 -37.5 -90.5t-90.5 -37.5h-1536q-53 0 -90.5 37.5t-37.5 90.5v1280 q0 53 37.5 90.5t90.5 37.5h1536q53 0 90.5 -37.5t37.5 -90.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M832 1024q0 80 -56 136t-136 56t-136 -56t-56 -136q0 -42 19 -83q-41 19 -83 19q-80 0 -136 -56t-56 -136t56 -136t136 -56t136 56t56 136q0 42 -19 83q41 -19 83 -19q80 0 136 56t56 136zM1683 320q0 -17 -49 -66t-66 -49q-9 0 -28.5 16t-36.5 33t-38.5 40t-24.5 26 l-96 -96l220 -220q28 -28 28 -68q0 -42 -39 -81t-81 -39q-40 0 -68 28l-671 671q-176 -131 -365 -131q-163 0 -265.5 102.5t-102.5 265.5q0 160 95 313t248 248t313 95q163 0 265.5 -102.5t102.5 -265.5q0 -189 -131 -365l355 -355l96 96q-3 3 -26 24.5t-40 38.5t-33 36.5 t-16 28.5q0 17 49 66t66 49q13 0 23 -10q6 -6 46 -44.5t82 -79.5t86.5 -86t73 -78t28.5 -41z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M896 640q0 106 -75 181t-181 75t-181 -75t-75 -181t75 -181t181 -75t181 75t75 181zM1664 128q0 52 -38 90t-90 38t-90 -38t-38 -90q0 -53 37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1664 1152q0 52 -38 90t-90 38t-90 -38t-38 -90q0 -53 37.5 -90.5t90.5 -37.5 t90.5 37.5t37.5 90.5zM1280 731v-185q0 -10 -7 -19.5t-16 -10.5l-155 -24q-11 -35 -32 -76q34 -48 90 -115q7 -10 7 -20q0 -12 -7 -19q-23 -30 -82.5 -89.5t-78.5 -59.5q-11 0 -21 7l-115 90q-37 -19 -77 -31q-11 -108 -23 -155q-7 -24 -30 -24h-186q-11 0 -20 7.5t-10 17.5 l-23 153q-34 10 -75 31l-118 -89q-7 -7 -20 -7q-11 0 -21 8q-144 133 -144 160q0 9 7 19q10 14 41 53t47 61q-23 44 -35 82l-152 24q-10 1 -17 9.5t-7 19.5v185q0 10 7 19.5t16 10.5l155 24q11 35 32 76q-34 48 -90 115q-7 11 -7 20q0 12 7 20q22 30 82 89t79 59q11 0 21 -7 l115 -90q34 18 77 32q11 108 23 154q7 24 30 24h186q11 0 20 -7.5t10 -17.5l23 -153q34 -10 75 -31l118 89q8 7 20 7q11 0 21 -8q144 -133 144 -160q0 -9 -7 -19q-12 -16 -42 -54t-45 -60q23 -48 34 -82l152 -23q10 -2 17 -10.5t7 -19.5zM1920 198v-140q0 -16 -149 -31 q-12 -27 -30 -52q51 -113 51 -138q0 -4 -4 -7q-122 -71 -124 -71q-8 0 -46 47t-52 68q-20 -2 -30 -2t-30 2q-14 -21 -52 -68t-46 -47q-2 0 -124 71q-4 3 -4 7q0 25 51 138q-18 25 -30 52q-149 15 -149 31v140q0 16 149 31q13 29 30 52q-51 113 -51 138q0 4 4 7q4 2 35 20 t59 34t30 16q8 0 46 -46.5t52 -67.5q20 2 30 2t30 -2q51 71 92 112l6 2q4 0 124 -70q4 -3 4 -7q0 -25 -51 -138q17 -23 30 -52q149 -15 149 -31zM1920 1222v-140q0 -16 -149 -31q-12 -27 -30 -52q51 -113 51 -138q0 -4 -4 -7q-122 -71 -124 -71q-8 0 -46 47t-52 68 q-20 -2 -30 -2t-30 2q-14 -21 -52 -68t-46 -47q-2 0 -124 71q-4 3 -4 7q0 25 51 138q-18 25 -30 52q-149 15 -149 31v140q0 16 149 31q13 29 30 52q-51 113 -51 138q0 4 4 7q4 2 35 20t59 34t30 16q8 0 46 -46.5t52 -67.5q20 2 30 2t30 -2q51 71 92 112l6 2q4 0 124 -70 q4 -3 4 -7q0 -25 -51 -138q17 -23 30 -52q149 -15 149 -31z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1408 768q0 -139 -94 -257t-256.5 -186.5t-353.5 -68.5q-86 0 -176 16q-124 -88 -278 -128q-36 -9 -86 -16h-3q-11 0 -20.5 8t-11.5 21q-1 3 -1 6.5t0.5 6.5t2 6l2.5 5t3.5 5.5t4 5t4.5 5t4 4.5q5 6 23 25t26 29.5t22.5 29t25 38.5t20.5 44q-124 72 -195 177t-71 224 q0 139 94 257t256.5 186.5t353.5 68.5t353.5 -68.5t256.5 -186.5t94 -257zM1792 512q0 -120 -71 -224.5t-195 -176.5q10 -24 20.5 -44t25 -38.5t22.5 -29t26 -29.5t23 -25q1 -1 4 -4.5t4.5 -5t4 -5t3.5 -5.5l2.5 -5t2 -6t0.5 -6.5t-1 -6.5q-3 -14 -13 -22t-22 -7 q-50 7 -86 16q-154 40 -278 128q-90 -16 -176 -16q-271 0 -472 132q58 -4 88 -4q161 0 309 45t264 129q125 92 192 212t67 254q0 77 -23 152q129 -71 204 -178t75 -230z" /> | ||||||
|  | <glyph unicode="" d="M256 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 768q0 51 -39 89.5t-89 38.5h-352q0 58 48 159.5t48 160.5q0 98 -32 145t-128 47q-26 -26 -38 -85t-30.5 -125.5t-59.5 -109.5q-22 -23 -77 -91q-4 -5 -23 -30t-31.5 -41t-34.5 -42.5 t-40 -44t-38.5 -35.5t-40 -27t-35.5 -9h-32v-640h32q13 0 31.5 -3t33 -6.5t38 -11t35 -11.5t35.5 -12.5t29 -10.5q211 -73 342 -73h121q192 0 192 167q0 26 -5 56q30 16 47.5 52.5t17.5 73.5t-18 69q53 50 53 119q0 25 -10 55.5t-25 47.5q32 1 53.5 47t21.5 81zM1536 769 q0 -89 -49 -163q9 -33 9 -69q0 -77 -38 -144q3 -21 3 -43q0 -101 -60 -178q1 -139 -85 -219.5t-227 -80.5h-36h-93q-96 0 -189.5 22.5t-216.5 65.5q-116 40 -138 40h-288q-53 0 -90.5 37.5t-37.5 90.5v640q0 53 37.5 90.5t90.5 37.5h274q36 24 137 155q58 75 107 128 q24 25 35.5 85.5t30.5 126.5t62 108q39 37 90 37q84 0 151 -32.5t102 -101.5t35 -186q0 -93 -48 -192h176q104 0 180 -76t76 -179z" /> | ||||||
|  | <glyph unicode="" d="M256 1088q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 512q0 35 -21.5 81t-53.5 47q15 17 25 47.5t10 55.5q0 69 -53 119q18 32 18 69t-17.5 73.5t-47.5 52.5q5 30 5 56q0 85 -49 126t-136 41h-128q-131 0 -342 -73q-5 -2 -29 -10.5 t-35.5 -12.5t-35 -11.5t-38 -11t-33 -6.5t-31.5 -3h-32v-640h32q16 0 35.5 -9t40 -27t38.5 -35.5t40 -44t34.5 -42.5t31.5 -41t23 -30q55 -68 77 -91q41 -43 59.5 -109.5t30.5 -125.5t38 -85q96 0 128 47t32 145q0 59 -48 160.5t-48 159.5h352q50 0 89 38.5t39 89.5z M1536 511q0 -103 -76 -179t-180 -76h-176q48 -99 48 -192q0 -118 -35 -186q-35 -69 -102 -101.5t-151 -32.5q-51 0 -90 37q-34 33 -54 82t-25.5 90.5t-17.5 84.5t-31 64q-48 50 -107 127q-101 131 -137 155h-274q-53 0 -90.5 37.5t-37.5 90.5v640q0 53 37.5 90.5t90.5 37.5 h288q22 0 138 40q128 44 223 66t200 22h112q140 0 226.5 -79t85.5 -216v-5q60 -77 60 -178q0 -22 -3 -43q38 -67 38 -144q0 -36 -9 -69q49 -74 49 -163z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="896" d="M832 1504v-1339l-449 -236q-22 -12 -40 -12q-21 0 -31.5 14.5t-10.5 35.5q0 6 2 20l86 500l-364 354q-25 27 -25 48q0 37 56 46l502 73l225 455q19 41 49 41z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1664 940q0 81 -21.5 143t-55 98.5t-81.5 59.5t-94 31t-98 8t-112 -25.5t-110.5 -64t-86.5 -72t-60 -61.5q-18 -22 -49 -22t-49 22q-24 28 -60 61.5t-86.5 72t-110.5 64t-112 25.5t-98 -8t-94 -31t-81.5 -59.5t-55 -98.5t-21.5 -143q0 -168 187 -355l581 -560l580 559 q188 188 188 356zM1792 940q0 -221 -229 -450l-623 -600q-18 -18 -44 -18t-44 18l-624 602q-10 8 -27.5 26t-55.5 65.5t-68 97.5t-53.5 121t-23.5 138q0 220 127 344t351 124q62 0 126.5 -21.5t120 -58t95.5 -68.5t76 -68q36 36 76 68t95.5 68.5t120 58t126.5 21.5 q224 0 351 -124t127 -344z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M640 96q0 -4 1 -20t0.5 -26.5t-3 -23.5t-10 -19.5t-20.5 -6.5h-320q-119 0 -203.5 84.5t-84.5 203.5v704q0 119 84.5 203.5t203.5 84.5h320q13 0 22.5 -9.5t9.5 -22.5q0 -4 1 -20t0.5 -26.5t-3 -23.5t-10 -19.5t-20.5 -6.5h-320q-66 0 -113 -47t-47 -113v-704 q0 -66 47 -113t113 -47h288h11h13t11.5 -1t11.5 -3t8 -5.5t7 -9t2 -13.5zM1568 640q0 -26 -19 -45l-544 -544q-19 -19 -45 -19t-45 19t-19 45v288h-448q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h448v288q0 26 19 45t45 19t45 -19l544 -544q19 -19 19 -45z" /> | ||||||
|  | <glyph unicode="" d="M237 122h231v694h-231v-694zM483 1030q-1 52 -36 86t-93 34t-94.5 -34t-36.5 -86q0 -51 35.5 -85.5t92.5 -34.5h1q59 0 95 34.5t36 85.5zM1068 122h231v398q0 154 -73 233t-193 79q-136 0 -209 -117h2v101h-231q3 -66 0 -694h231v388q0 38 7 56q15 35 45 59.5t74 24.5 q116 0 116 -157v-371zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M480 672v448q0 14 -9 23t-23 9t-23 -9t-9 -23v-448q0 -14 9 -23t23 -9t23 9t9 23zM1152 320q0 -26 -19 -45t-45 -19h-429l-51 -483q-2 -12 -10.5 -20.5t-20.5 -8.5h-1q-27 0 -32 27l-76 485h-404q-26 0 -45 19t-19 45q0 123 78.5 221.5t177.5 98.5v512q-52 0 -90 38 t-38 90t38 90t90 38h640q52 0 90 -38t38 -90t-38 -90t-90 -38v-512q99 0 177.5 -98.5t78.5 -221.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1408 608v-320q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h704q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-704q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v320 q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1792 1472v-512q0 -26 -19 -45t-45 -19t-45 19l-176 176l-652 -652q-10 -10 -23 -10t-23 10l-114 114q-10 10 -10 23t10 23l652 652l-176 176q-19 19 -19 45t19 45t45 19h512q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" d="M1184 640q0 -26 -19 -45l-544 -544q-19 -19 -45 -19t-45 19t-19 45v288h-448q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h448v288q0 26 19 45t45 19t45 -19l544 -544q19 -19 19 -45zM1536 992v-704q0 -119 -84.5 -203.5t-203.5 -84.5h-320q-13 0 -22.5 9.5t-9.5 22.5 q0 4 -1 20t-0.5 26.5t3 23.5t10 19.5t20.5 6.5h320q66 0 113 47t47 113v704q0 66 -47 113t-113 47h-288h-11h-13t-11.5 1t-11.5 3t-8 5.5t-7 9t-2 13.5q0 4 -1 20t-0.5 26.5t3 23.5t10 19.5t20.5 6.5h320q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M458 653q-74 162 -74 371h-256v-96q0 -78 94.5 -162t235.5 -113zM1536 928v96h-256q0 -209 -74 -371q141 29 235.5 113t94.5 162zM1664 1056v-128q0 -71 -41.5 -143t-112 -130t-173 -97.5t-215.5 -44.5q-42 -54 -95 -95q-38 -34 -52.5 -72.5t-14.5 -89.5q0 -54 30.5 -91 t97.5 -37q75 0 133.5 -45.5t58.5 -114.5v-64q0 -14 -9 -23t-23 -9h-832q-14 0 -23 9t-9 23v64q0 69 58.5 114.5t133.5 45.5q67 0 97.5 37t30.5 91q0 51 -14.5 89.5t-52.5 72.5q-53 41 -95 95q-113 5 -215.5 44.5t-173 97.5t-112 130t-41.5 143v128q0 40 28 68t68 28h288v96 q0 66 47 113t113 47h576q66 0 113 -47t47 -113v-96h288q40 0 68 -28t28 -68z" /> | ||||||
|  | <glyph unicode="" d="M394 184q-8 -9 -20 3q-13 11 -4 19q8 9 20 -3q12 -11 4 -19zM352 245q9 -12 0 -19q-8 -6 -17 7t0 18q9 7 17 -6zM291 305q-5 -7 -13 -2q-10 5 -7 12q3 5 13 2q10 -5 7 -12zM322 271q-6 -7 -16 3q-9 11 -2 16q6 6 16 -3q9 -11 2 -16zM451 159q-4 -12 -19 -6q-17 4 -13 15 t19 7q16 -5 13 -16zM514 154q0 -11 -16 -11q-17 -2 -17 11q0 11 16 11q17 2 17 -11zM572 164q2 -10 -14 -14t-18 8t14 15q16 2 18 -9zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-224q-16 0 -24.5 1t-19.5 5t-16 14.5t-5 27.5v239q0 97 -52 142q57 6 102.5 18t94 39 t81 66.5t53 105t20.5 150.5q0 121 -79 206q37 91 -8 204q-28 9 -81 -11t-92 -44l-38 -24q-93 26 -192 26t-192 -26q-16 11 -42.5 27t-83.5 38.5t-86 13.5q-44 -113 -7 -204q-79 -85 -79 -206q0 -85 20.5 -150t52.5 -105t80.5 -67t94 -39t102.5 -18q-40 -36 -49 -103 q-21 -10 -45 -15t-57 -5t-65.5 21.5t-55.5 62.5q-19 32 -48.5 52t-49.5 24l-20 3q-21 0 -29 -4.5t-5 -11.5t9 -14t13 -12l7 -5q22 -10 43.5 -38t31.5 -51l10 -23q13 -38 44 -61.5t67 -30t69.5 -7t55.5 3.5l23 4q0 -38 0.5 -103t0.5 -68q0 -22 -11 -33.5t-22 -13t-33 -1.5 h-224q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1280 64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1536 64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 288v-320q0 -40 -28 -68t-68 -28h-1472q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h427q21 -56 70.5 -92 t110.5 -36h256q61 0 110.5 36t70.5 92h427q40 0 68 -28t28 -68zM1339 936q-17 -40 -59 -40h-256v-448q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v448h-256q-42 0 -59 40q-17 39 14 69l448 448q18 19 45 19t45 -19l448 -448q31 -30 14 -69z" /> | ||||||
|  | <glyph unicode="" d="M1407 710q0 44 -7 113.5t-18 96.5q-12 30 -17 44t-9 36.5t-4 48.5q0 23 5 68.5t5 67.5q0 37 -10 55q-4 1 -13 1q-19 0 -58 -4.5t-59 -4.5q-60 0 -176 24t-175 24q-43 0 -94.5 -11.5t-85 -23.5t-89.5 -34q-137 -54 -202 -103q-96 -73 -159.5 -189.5t-88 -236t-24.5 -248.5 q0 -40 12.5 -120t12.5 -121q0 -23 -11 -66.5t-11 -65.5t12 -36.5t34 -14.5q24 0 72.5 11t73.5 11q57 0 169.5 -15.5t169.5 -15.5q181 0 284 36q129 45 235.5 152.5t166 245.5t59.5 275zM1535 712q0 -165 -70 -327.5t-196 -288t-281 -180.5q-124 -44 -326 -44 q-57 0 -170 14.5t-169 14.5q-24 0 -72.5 -14.5t-73.5 -14.5q-73 0 -123.5 55.5t-50.5 128.5q0 24 11 68t11 67q0 40 -12.5 120.5t-12.5 121.5q0 111 18 217.5t54.5 209.5t100.5 194t150 156q78 59 232 120q194 78 316 78q60 0 175.5 -24t173.5 -24q19 0 57 5t58 5 q81 0 118 -50.5t37 -134.5q0 -23 -5 -68t-5 -68q0 -10 1 -18.5t3 -17t4 -13.5t6.5 -16t6.5 -17q16 -40 25 -118.5t9 -136.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1408 296q0 -27 -10 -70.5t-21 -68.5q-21 -50 -122 -106q-94 -51 -186 -51q-27 0 -52.5 3.5t-57.5 12.5t-47.5 14.5t-55.5 20.5t-49 18q-98 35 -175 83q-128 79 -264.5 215.5t-215.5 264.5q-48 77 -83 175q-3 9 -18 49t-20.5 55.5t-14.5 47.5t-12.5 57.5t-3.5 52.5 q0 92 51 186q56 101 106 122q25 11 68.5 21t70.5 10q14 0 21 -3q18 -6 53 -76q11 -19 30 -54t35 -63.5t31 -53.5q3 -4 17.5 -25t21.5 -35.5t7 -28.5q0 -20 -28.5 -50t-62 -55t-62 -53t-28.5 -46q0 -9 5 -22.5t8.5 -20.5t14 -24t11.5 -19q76 -137 174 -235t235 -174 q2 -1 19 -11.5t24 -14t20.5 -8.5t22.5 -5q18 0 46 28.5t53 62t55 62t50 28.5q14 0 28.5 -7t35.5 -21.5t25 -17.5q25 -15 53.5 -31t63.5 -35t54 -30q70 -35 76 -53q3 -7 3 -21z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1120 1280h-832q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v832q0 66 -47 113t-113 47zM1408 1120v-832q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832 q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1280" d="M1152 1280h-1024v-1242l423 406l89 85l89 -85l423 -406v1242zM1164 1408q23 0 44 -9q33 -13 52.5 -41t19.5 -62v-1289q0 -34 -19.5 -62t-52.5 -41q-19 -8 -44 -8q-48 0 -83 32l-441 424l-441 -424q-36 -33 -83 -33q-23 0 -44 9q-33 13 -52.5 41t-19.5 62v1289 q0 34 19.5 62t52.5 41q21 9 44 9h1048z" /> | ||||||
|  | <glyph unicode="" d="M1280 343q0 11 -2 16q-3 8 -38.5 29.5t-88.5 49.5l-53 29q-5 3 -19 13t-25 15t-21 5q-18 0 -47 -32.5t-57 -65.5t-44 -33q-7 0 -16.5 3.5t-15.5 6.5t-17 9.5t-14 8.5q-99 55 -170.5 126.5t-126.5 170.5q-2 3 -8.5 14t-9.5 17t-6.5 15.5t-3.5 16.5q0 13 20.5 33.5t45 38.5 t45 39.5t20.5 36.5q0 10 -5 21t-15 25t-13 19q-3 6 -15 28.5t-25 45.5t-26.5 47.5t-25 40.5t-16.5 18t-16 2q-48 0 -101 -22q-46 -21 -80 -94.5t-34 -130.5q0 -16 2.5 -34t5 -30.5t9 -33t10 -29.5t12.5 -33t11 -30q60 -164 216.5 -320.5t320.5 -216.5q6 -2 30 -11t33 -12.5 t29.5 -10t33 -9t30.5 -5t34 -2.5q57 0 130.5 34t94.5 80q22 53 22 101zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1620 1128q-67 -98 -162 -167q1 -14 1 -42q0 -130 -38 -259.5t-115.5 -248.5t-184.5 -210.5t-258 -146t-323 -54.5q-271 0 -496 145q35 -4 78 -4q225 0 401 138q-105 2 -188 64.5t-114 159.5q33 -5 61 -5q43 0 85 11q-112 23 -185.5 111.5t-73.5 205.5v4q68 -38 146 -41 q-66 44 -105 115t-39 154q0 88 44 163q121 -149 294.5 -238.5t371.5 -99.5q-8 38 -8 74q0 134 94.5 228.5t228.5 94.5q140 0 236 -102q109 21 205 78q-37 -115 -142 -178q93 10 186 50z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="768" d="M511 980h257l-30 -284h-227v-824h-341v824h-170v284h170v171q0 182 86 275.5t283 93.5h227v-284h-142q-39 0 -62.5 -6.5t-34 -23.5t-13.5 -34.5t-3 -49.5v-142z" /> | ||||||
|  | <glyph unicode="" d="M1536 640q0 -251 -146.5 -451.5t-378.5 -277.5q-27 -5 -39.5 7t-12.5 30v211q0 97 -52 142q57 6 102.5 18t94 39t81 66.5t53 105t20.5 150.5q0 121 -79 206q37 91 -8 204q-28 9 -81 -11t-92 -44l-38 -24q-93 26 -192 26t-192 -26q-16 11 -42.5 27t-83.5 38.5t-86 13.5 q-44 -113 -7 -204q-79 -85 -79 -206q0 -85 20.5 -150t52.5 -105t80.5 -67t94 -39t102.5 -18q-40 -36 -49 -103q-21 -10 -45 -15t-57 -5t-65.5 21.5t-55.5 62.5q-19 32 -48.5 52t-49.5 24l-20 3q-21 0 -29 -4.5t-5 -11.5t9 -14t13 -12l7 -5q22 -10 43.5 -38t31.5 -51l10 -23 q13 -38 44 -61.5t67 -30t69.5 -7t55.5 3.5l23 4q0 -38 0.5 -89t0.5 -54q0 -18 -13 -30t-40 -7q-232 77 -378.5 277.5t-146.5 451.5q0 209 103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1664 960v-256q0 -26 -19 -45t-45 -19h-64q-26 0 -45 19t-19 45v256q0 106 -75 181t-181 75t-181 -75t-75 -181v-192h96q40 0 68 -28t28 -68v-576q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v576q0 40 28 68t68 28h672v192q0 185 131.5 316.5t316.5 131.5 t316.5 -131.5t131.5 -316.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1760 1408q66 0 113 -47t47 -113v-1216q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1600zM160 1280q-13 0 -22.5 -9.5t-9.5 -22.5v-224h1664v224q0 13 -9.5 22.5t-22.5 9.5h-1600zM1760 0q13 0 22.5 9.5t9.5 22.5v608h-1664v-608 q0 -13 9.5 -22.5t22.5 -9.5h1600zM256 128v128h256v-128h-256zM640 128v128h384v-128h-384z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M384 192q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM896 69q2 -28 -17 -48q-18 -21 -47 -21h-135q-25 0 -43 16.5t-20 41.5q-22 229 -184.5 391.5t-391.5 184.5q-25 2 -41.5 20t-16.5 43v135q0 29 21 47q17 17 43 17h5q160 -13 306 -80.5 t259 -181.5q114 -113 181.5 -259t80.5 -306zM1408 67q2 -27 -18 -47q-18 -20 -46 -20h-143q-26 0 -44.5 17.5t-19.5 42.5q-12 215 -101 408.5t-231.5 336t-336 231.5t-408.5 102q-25 1 -42.5 19.5t-17.5 43.5v143q0 28 20 46q18 18 44 18h3q262 -13 501.5 -120t425.5 -294 q187 -186 294 -425.5t120 -501.5z" /> | ||||||
|  | <glyph unicode="" d="M1040 320q0 -33 -23.5 -56.5t-56.5 -23.5t-56.5 23.5t-23.5 56.5t23.5 56.5t56.5 23.5t56.5 -23.5t23.5 -56.5zM1296 320q0 -33 -23.5 -56.5t-56.5 -23.5t-56.5 23.5t-23.5 56.5t23.5 56.5t56.5 23.5t56.5 -23.5t23.5 -56.5zM1408 160v320q0 13 -9.5 22.5t-22.5 9.5 h-1216q-13 0 -22.5 -9.5t-9.5 -22.5v-320q0 -13 9.5 -22.5t22.5 -9.5h1216q13 0 22.5 9.5t9.5 22.5zM178 640h1180l-157 482q-4 13 -16 21.5t-26 8.5h-782q-14 0 -26 -8.5t-16 -21.5zM1536 480v-320q0 -66 -47 -113t-113 -47h-1216q-66 0 -113 47t-47 113v320q0 25 16 75 l197 606q17 53 63 86t101 33h782q55 0 101 -33t63 -86l197 -606q16 -50 16 -75z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1664 896q53 0 90.5 -37.5t37.5 -90.5t-37.5 -90.5t-90.5 -37.5v-384q0 -52 -38 -90t-90 -38q-417 347 -812 380q-58 -19 -91 -66t-31 -100.5t40 -92.5q-20 -33 -23 -65.5t6 -58t33.5 -55t48 -50t61.5 -50.5q-29 -58 -111.5 -83t-168.5 -11.5t-132 55.5q-7 23 -29.5 87.5 t-32 94.5t-23 89t-15 101t3.5 98.5t22 110.5h-122q-66 0 -113 47t-47 113v192q0 66 47 113t113 47h480q435 0 896 384q52 0 90 -38t38 -90v-384zM1536 292v954q-394 -302 -768 -343v-270q377 -42 768 -341z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M848 -160q0 16 -16 16q-59 0 -101.5 42.5t-42.5 101.5q0 16 -16 16t-16 -16q0 -73 51.5 -124.5t124.5 -51.5q16 0 16 16zM183 128h1298q-164 181 -246.5 411.5t-82.5 484.5q0 256 -320 256t-320 -256q0 -254 -82.5 -484.5t-246.5 -411.5zM1664 128q0 -52 -38 -90t-90 -38 h-448q0 -106 -75 -181t-181 -75t-181 75t-75 181h-448q-52 0 -90 38t-38 90q190 161 287 397.5t97 498.5q0 165 96 262t264 117q-8 18 -8 37q0 40 28 68t68 28t68 -28t28 -68q0 -19 -8 -37q168 -20 264 -117t96 -262q0 -262 97 -498.5t287 -397.5z" /> | ||||||
|  | <glyph unicode="" d="M1376 640l138 -135q30 -28 20 -70q-12 -41 -52 -51l-188 -48l53 -186q12 -41 -19 -70q-29 -31 -70 -19l-186 53l-48 -188q-10 -40 -51 -52q-12 -2 -19 -2q-31 0 -51 22l-135 138l-135 -138q-28 -30 -70 -20q-41 11 -51 52l-48 188l-186 -53q-41 -12 -70 19q-31 29 -19 70 l53 186l-188 48q-40 10 -52 51q-10 42 20 70l138 135l-138 135q-30 28 -20 70q12 41 52 51l188 48l-53 186q-12 41 19 70q29 31 70 19l186 -53l48 188q10 41 51 51q41 12 70 -19l135 -139l135 139q29 30 70 19q41 -10 51 -51l48 -188l186 53q41 12 70 -19q31 -29 19 -70 l-53 -186l188 -48q40 -10 52 -51q10 -42 -20 -70z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M256 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 768q0 51 -39 89.5t-89 38.5h-576q0 20 15 48.5t33 55t33 68t15 84.5q0 67 -44.5 97.5t-115.5 30.5q-24 0 -90 -139q-24 -44 -37 -65q-40 -64 -112 -145q-71 -81 -101 -106 q-69 -57 -140 -57h-32v-640h32q72 0 167 -32t193.5 -64t179.5 -32q189 0 189 167q0 26 -5 56q30 16 47.5 52.5t17.5 73.5t-18 69q53 50 53 119q0 25 -10 55.5t-25 47.5h331q52 0 90 38t38 90zM1792 769q0 -105 -75.5 -181t-180.5 -76h-169q-4 -62 -37 -119q3 -21 3 -43 q0 -101 -60 -178q1 -139 -85 -219.5t-227 -80.5q-133 0 -322 69q-164 59 -223 59h-288q-53 0 -90.5 37.5t-37.5 90.5v640q0 53 37.5 90.5t90.5 37.5h288q10 0 21.5 4.5t23.5 14t22.5 18t24 22.5t20.5 21.5t19 21.5t14 17q65 74 100 129q13 21 33 62t37 72t40.5 63t55 49.5 t69.5 17.5q125 0 206.5 -67t81.5 -189q0 -68 -22 -128h374q104 0 180 -76t76 -179z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1376 128h32v640h-32q-35 0 -67.5 12t-62.5 37t-50 46t-49 54q-2 3 -3.5 4.5t-4 4.5t-4.5 5q-72 81 -112 145q-14 22 -38 68q-1 3 -10.5 22.5t-18.5 36t-20 35.5t-21.5 30.5t-18.5 11.5q-71 0 -115.5 -30.5t-44.5 -97.5q0 -43 15 -84.5t33 -68t33 -55t15 -48.5h-576 q-50 0 -89 -38.5t-39 -89.5q0 -52 38 -90t90 -38h331q-15 -17 -25 -47.5t-10 -55.5q0 -69 53 -119q-18 -32 -18 -69t17.5 -73.5t47.5 -52.5q-4 -24 -4 -56q0 -85 48.5 -126t135.5 -41q84 0 183 32t194 64t167 32zM1664 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45 t45 -19t45 19t19 45zM1792 768v-640q0 -53 -37.5 -90.5t-90.5 -37.5h-288q-59 0 -223 -59q-190 -69 -317 -69q-142 0 -230 77.5t-87 217.5l1 5q-61 76 -61 178q0 22 3 43q-33 57 -37 119h-169q-105 0 -180.5 76t-75.5 181q0 103 76 179t180 76h374q-22 60 -22 128 q0 122 81.5 189t206.5 67q38 0 69.5 -17.5t55 -49.5t40.5 -63t37 -72t33 -62q35 -55 100 -129q2 -3 14 -17t19 -21.5t20.5 -21.5t24 -22.5t22.5 -18t23.5 -14t21.5 -4.5h288q53 0 90.5 -37.5t37.5 -90.5z" /> | ||||||
|  | <glyph unicode="" d="M1280 -64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 700q0 189 -167 189q-26 0 -56 -5q-16 30 -52.5 47.5t-73.5 17.5t-69 -18q-50 53 -119 53q-25 0 -55.5 -10t-47.5 -25v331q0 52 -38 90t-90 38q-51 0 -89.5 -39t-38.5 -89v-576 q-20 0 -48.5 15t-55 33t-68 33t-84.5 15q-67 0 -97.5 -44.5t-30.5 -115.5q0 -24 139 -90q44 -24 65 -37q64 -40 145 -112q81 -71 106 -101q57 -69 57 -140v-32h640v32q0 72 32 167t64 193.5t32 179.5zM1536 705q0 -133 -69 -322q-59 -164 -59 -223v-288q0 -53 -37.5 -90.5 t-90.5 -37.5h-640q-53 0 -90.5 37.5t-37.5 90.5v288q0 10 -4.5 21.5t-14 23.5t-18 22.5t-22.5 24t-21.5 20.5t-21.5 19t-17 14q-74 65 -129 100q-21 13 -62 33t-72 37t-63 40.5t-49.5 55t-17.5 69.5q0 125 67 206.5t189 81.5q68 0 128 -22v374q0 104 76 180t179 76 q105 0 181 -75.5t76 -180.5v-169q62 -4 119 -37q21 3 43 3q101 0 178 -60q139 1 219.5 -85t80.5 -227z" /> | ||||||
|  | <glyph unicode="" d="M1408 576q0 84 -32 183t-64 194t-32 167v32h-640v-32q0 -35 -12 -67.5t-37 -62.5t-46 -50t-54 -49q-9 -8 -14 -12q-81 -72 -145 -112q-22 -14 -68 -38q-3 -1 -22.5 -10.5t-36 -18.5t-35.5 -20t-30.5 -21.5t-11.5 -18.5q0 -71 30.5 -115.5t97.5 -44.5q43 0 84.5 15t68 33 t55 33t48.5 15v-576q0 -50 38.5 -89t89.5 -39q52 0 90 38t38 90v331q46 -35 103 -35q69 0 119 53q32 -18 69 -18t73.5 17.5t52.5 47.5q24 -4 56 -4q85 0 126 48.5t41 135.5zM1280 1344q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1536 580 q0 -142 -77.5 -230t-217.5 -87l-5 1q-76 -61 -178 -61q-22 0 -43 3q-54 -30 -119 -37v-169q0 -105 -76 -180.5t-181 -75.5q-103 0 -179 76t-76 180v374q-54 -22 -128 -22q-121 0 -188.5 81.5t-67.5 206.5q0 38 17.5 69.5t49.5 55t63 40.5t72 37t62 33q55 35 129 100 q3 2 17 14t21.5 19t21.5 20.5t22.5 24t18 22.5t14 23.5t4.5 21.5v288q0 53 37.5 90.5t90.5 37.5h640q53 0 90.5 -37.5t37.5 -90.5v-288q0 -59 59 -223q69 -190 69 -317z" /> | ||||||
|  | <glyph unicode="" d="M1280 576v128q0 26 -19 45t-45 19h-502l189 189q19 19 19 45t-19 45l-91 91q-18 18 -45 18t-45 -18l-362 -362l-91 -91q-18 -18 -18 -45t18 -45l91 -91l362 -362q18 -18 45 -18t45 18l91 91q18 18 18 45t-18 45l-189 189h502q26 0 45 19t19 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1285 640q0 27 -18 45l-91 91l-362 362q-18 18 -45 18t-45 -18l-91 -91q-18 -18 -18 -45t18 -45l189 -189h-502q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h502l-189 -189q-19 -19 -19 -45t19 -45l91 -91q18 -18 45 -18t45 18l362 362l91 91q18 18 18 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1284 641q0 27 -18 45l-362 362l-91 91q-18 18 -45 18t-45 -18l-91 -91l-362 -362q-18 -18 -18 -45t18 -45l91 -91q18 -18 45 -18t45 18l189 189v-502q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v502l189 -189q19 -19 45 -19t45 19l91 91q18 18 18 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1284 639q0 27 -18 45l-91 91q-18 18 -45 18t-45 -18l-189 -189v502q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-502l-189 189q-19 19 -45 19t-45 -19l-91 -91q-18 -18 -18 -45t18 -45l362 -362l91 -91q18 -18 45 -18t45 18l91 91l362 362q18 18 18 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M768 1408q209 0 385.5 -103t279.5 -279.5t103 -385.5t-103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103zM1042 887q-2 -1 -9.5 -9.5t-13.5 -9.5q2 0 4.5 5t5 11t3.5 7q6 7 22 15q14 6 52 12q34 8 51 -11 q-2 2 9.5 13t14.5 12q3 2 15 4.5t15 7.5l2 22q-12 -1 -17.5 7t-6.5 21q0 -2 -6 -8q0 7 -4.5 8t-11.5 -1t-9 -1q-10 3 -15 7.5t-8 16.5t-4 15q-2 5 -9.5 10.5t-9.5 10.5q-1 2 -2.5 5.5t-3 6.5t-4 5.5t-5.5 2.5t-7 -5t-7.5 -10t-4.5 -5q-3 2 -6 1.5t-4.5 -1t-4.5 -3t-5 -3.5 q-3 -2 -8.5 -3t-8.5 -2q15 5 -1 11q-10 4 -16 3q9 4 7.5 12t-8.5 14h5q-1 4 -8.5 8.5t-17.5 8.5t-13 6q-8 5 -34 9.5t-33 0.5q-5 -6 -4.5 -10.5t4 -14t3.5 -12.5q1 -6 -5.5 -13t-6.5 -12q0 -7 14 -15.5t10 -21.5q-3 -8 -16 -16t-16 -12q-5 -8 -1.5 -18.5t10.5 -16.5 q2 -2 1.5 -4t-3.5 -4.5t-5.5 -4t-6.5 -3.5l-3 -2q-11 -5 -20.5 6t-13.5 26q-7 25 -16 30q-23 8 -29 -1q-5 13 -41 26q-25 9 -58 4q6 1 0 15q-7 15 -19 12q3 6 4 17.5t1 13.5q3 13 12 23q1 1 7 8.5t9.5 13.5t0.5 6q35 -4 50 11q5 5 11.5 17t10.5 17q9 6 14 5.5t14.5 -5.5 t14.5 -5q14 -1 15.5 11t-7.5 20q12 -1 3 17q-5 7 -8 9q-12 4 -27 -5q-8 -4 2 -8q-1 1 -9.5 -10.5t-16.5 -17.5t-16 5q-1 1 -5.5 13.5t-9.5 13.5q-8 0 -16 -15q3 8 -11 15t-24 8q19 12 -8 27q-7 4 -20.5 5t-19.5 -4q-5 -7 -5.5 -11.5t5 -8t10.5 -5.5t11.5 -4t8.5 -3 q14 -10 8 -14q-2 -1 -8.5 -3.5t-11.5 -4.5t-6 -4q-3 -4 0 -14t-2 -14q-5 5 -9 17.5t-7 16.5q7 -9 -25 -6l-10 1q-4 0 -16 -2t-20.5 -1t-13.5 8q-4 8 0 20q1 4 4 2q-4 3 -11 9.5t-10 8.5q-46 -15 -94 -41q6 -1 12 1q5 2 13 6.5t10 5.5q34 14 42 7l5 5q14 -16 20 -25 q-7 4 -30 1q-20 -6 -22 -12q7 -12 5 -18q-4 3 -11.5 10t-14.5 11t-15 5q-16 0 -22 -1q-146 -80 -235 -222q7 -7 12 -8q4 -1 5 -9t2.5 -11t11.5 3q9 -8 3 -19q1 1 44 -27q19 -17 21 -21q3 -11 -10 -18q-1 2 -9 9t-9 4q-3 -5 0.5 -18.5t10.5 -12.5q-7 0 -9.5 -16t-2.5 -35.5 t-1 -23.5l2 -1q-3 -12 5.5 -34.5t21.5 -19.5q-13 -3 20 -43q6 -8 8 -9q3 -2 12 -7.5t15 -10t10 -10.5q4 -5 10 -22.5t14 -23.5q-2 -6 9.5 -20t10.5 -23q-1 0 -2.5 -1t-2.5 -1q3 -7 15.5 -14t15.5 -13q1 -3 2 -10t3 -11t8 -2q2 20 -24 62q-15 25 -17 29q-3 5 -5.5 15.5 t-4.5 14.5q2 0 6 -1.5t8.5 -3.5t7.5 -4t2 -3q-3 -7 2 -17.5t12 -18.5t17 -19t12 -13q6 -6 14 -19.5t0 -13.5q9 0 20 -10t17 -20q5 -8 8 -26t5 -24q2 -7 8.5 -13.5t12.5 -9.5l16 -8t13 -7q5 -2 18.5 -10.5t21.5 -11.5q10 -4 16 -4t14.5 2.5t13.5 3.5q15 2 29 -15t21 -21 q36 -19 55 -11q-2 -1 0.5 -7.5t8 -15.5t9 -14.5t5.5 -8.5q5 -6 18 -15t18 -15q6 4 7 9q-3 -8 7 -20t18 -10q14 3 14 32q-31 -15 -49 18q0 1 -2.5 5.5t-4 8.5t-2.5 8.5t0 7.5t5 3q9 0 10 3.5t-2 12.5t-4 13q-1 8 -11 20t-12 15q-5 -9 -16 -8t-16 9q0 -1 -1.5 -5.5t-1.5 -6.5 q-13 0 -15 1q1 3 2.5 17.5t3.5 22.5q1 4 5.5 12t7.5 14.5t4 12.5t-4.5 9.5t-17.5 2.5q-19 -1 -26 -20q-1 -3 -3 -10.5t-5 -11.5t-9 -7q-7 -3 -24 -2t-24 5q-13 8 -22.5 29t-9.5 37q0 10 2.5 26.5t3 25t-5.5 24.5q3 2 9 9.5t10 10.5q2 1 4.5 1.5t4.5 0t4 1.5t3 6q-1 1 -4 3 q-3 3 -4 3q7 -3 28.5 1.5t27.5 -1.5q15 -11 22 2q0 1 -2.5 9.5t-0.5 13.5q5 -27 29 -9q3 -3 15.5 -5t17.5 -5q3 -2 7 -5.5t5.5 -4.5t5 0.5t8.5 6.5q10 -14 12 -24q11 -40 19 -44q7 -3 11 -2t4.5 9.5t0 14t-1.5 12.5l-1 8v18l-1 8q-15 3 -18.5 12t1.5 18.5t15 18.5q1 1 8 3.5 t15.5 6.5t12.5 8q21 19 15 35q7 0 11 9q-1 0 -5 3t-7.5 5t-4.5 2q9 5 2 16q5 3 7.5 11t7.5 10q9 -12 21 -2q7 8 1 16q5 7 20.5 10.5t18.5 9.5q7 -2 8 2t1 12t3 12q4 5 15 9t13 5l17 11q3 4 0 4q18 -2 31 11q10 11 -6 20q3 6 -3 9.5t-15 5.5q3 1 11.5 0.5t10.5 1.5 q15 10 -7 16q-17 5 -43 -12zM879 10q206 36 351 189q-3 3 -12.5 4.5t-12.5 3.5q-18 7 -24 8q1 7 -2.5 13t-8 9t-12.5 8t-11 7q-2 2 -7 6t-7 5.5t-7.5 4.5t-8.5 2t-10 -1l-3 -1q-3 -1 -5.5 -2.5t-5.5 -3t-4 -3t0 -2.5q-21 17 -36 22q-5 1 -11 5.5t-10.5 7t-10 1.5t-11.5 -7 q-5 -5 -6 -15t-2 -13q-7 5 0 17.5t2 18.5q-3 6 -10.5 4.5t-12 -4.5t-11.5 -8.5t-9 -6.5t-8.5 -5.5t-8.5 -7.5q-3 -4 -6 -12t-5 -11q-2 4 -11.5 6.5t-9.5 5.5q2 -10 4 -35t5 -38q7 -31 -12 -48q-27 -25 -29 -40q-4 -22 12 -26q0 -7 -8 -20.5t-7 -21.5q0 -6 2 -16z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M384 64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1028 484l-682 -682q-37 -37 -90 -37q-52 0 -91 37l-106 108q-38 36 -38 90q0 53 38 91l681 681q39 -98 114.5 -173.5t173.5 -114.5zM1662 919q0 -39 -23 -106q-47 -134 -164.5 -217.5 t-258.5 -83.5q-185 0 -316.5 131.5t-131.5 316.5t131.5 316.5t316.5 131.5q58 0 121.5 -16.5t107.5 -46.5q16 -11 16 -28t-16 -28l-293 -169v-224l193 -107q5 3 79 48.5t135.5 81t70.5 35.5q15 0 23.5 -10t8.5 -25z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1024 128h640v128h-640v-128zM640 640h1024v128h-1024v-128zM1280 1152h384v128h-384v-128zM1792 320v-256q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 832v-256q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19 t-19 45v256q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 1344v-256q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1664q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1403 1241q17 -41 -14 -70l-493 -493v-742q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-256 256q-19 19 -19 45v486l-493 493q-31 29 -14 70q17 39 59 39h1280q42 0 59 -39z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M640 1280h512v128h-512v-128zM1792 640v-480q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v480h672v-160q0 -26 19 -45t45 -19h320q26 0 45 19t19 45v160h672zM1024 640v-128h-256v128h256zM1792 1120v-384h-1792v384q0 66 47 113t113 47h352v160q0 40 28 68 t68 28h576q40 0 68 -28t28 -68v-160h352q66 0 113 -47t47 -113z" /> | ||||||
|  | <glyph unicode="" d="M1283 995l-355 -355l355 -355l144 144q29 31 70 14q39 -17 39 -59v-448q0 -26 -19 -45t-45 -19h-448q-42 0 -59 40q-17 39 14 69l144 144l-355 355l-355 -355l144 -144q31 -30 14 -69q-17 -40 -59 -40h-448q-26 0 -45 19t-19 45v448q0 42 40 59q39 17 69 -14l144 -144 l355 355l-355 355l-144 -144q-19 -19 -45 -19q-12 0 -24 5q-40 17 -40 59v448q0 26 19 45t45 19h448q42 0 59 -40q17 -39 -14 -69l-144 -144l355 -355l355 355l-144 144q-31 30 -14 69q17 40 59 40h448q26 0 45 -19t19 -45v-448q0 -42 -39 -59q-13 -5 -25 -5q-26 0 -45 19z " /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M593 640q-162 -5 -265 -128h-134q-82 0 -138 40.5t-56 118.5q0 353 124 353q6 0 43.5 -21t97.5 -42.5t119 -21.5q67 0 133 23q-5 -37 -5 -66q0 -139 81 -256zM1664 3q0 -120 -73 -189.5t-194 -69.5h-874q-121 0 -194 69.5t-73 189.5q0 53 3.5 103.5t14 109t26.5 108.5 t43 97.5t62 81t85.5 53.5t111.5 20q10 0 43 -21.5t73 -48t107 -48t135 -21.5t135 21.5t107 48t73 48t43 21.5q61 0 111.5 -20t85.5 -53.5t62 -81t43 -97.5t26.5 -108.5t14 -109t3.5 -103.5zM640 1280q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75 t75 -181zM1344 896q0 -159 -112.5 -271.5t-271.5 -112.5t-271.5 112.5t-112.5 271.5t112.5 271.5t271.5 112.5t271.5 -112.5t112.5 -271.5zM1920 671q0 -78 -56 -118.5t-138 -40.5h-134q-103 123 -265 128q81 117 81 256q0 29 -5 66q66 -23 133 -23q59 0 119 21.5t97.5 42.5 t43.5 21q124 0 124 -353zM1792 1280q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1456 320q0 40 -28 68l-208 208q-28 28 -68 28q-42 0 -72 -32q3 -3 19 -18.5t21.5 -21.5t15 -19t13 -25.5t3.5 -27.5q0 -40 -28 -68t-68 -28q-15 0 -27.5 3.5t-25.5 13t-19 15t-21.5 21.5t-18.5 19q-33 -31 -33 -73q0 -40 28 -68l206 -207q27 -27 68 -27q40 0 68 26 l147 146q28 28 28 67zM753 1025q0 40 -28 68l-206 207q-28 28 -68 28q-39 0 -68 -27l-147 -146q-28 -28 -28 -67q0 -40 28 -68l208 -208q27 -27 68 -27q42 0 72 31q-3 3 -19 18.5t-21.5 21.5t-15 19t-13 25.5t-3.5 27.5q0 40 28 68t68 28q15 0 27.5 -3.5t25.5 -13t19 -15 t21.5 -21.5t18.5 -19q33 31 33 73zM1648 320q0 -120 -85 -203l-147 -146q-83 -83 -203 -83q-121 0 -204 85l-206 207q-83 83 -83 203q0 123 88 209l-88 88q-86 -88 -208 -88q-120 0 -204 84l-208 208q-84 84 -84 204t85 203l147 146q83 83 203 83q121 0 204 -85l206 -207 q83 -83 83 -203q0 -123 -88 -209l88 -88q86 88 208 88q120 0 204 -84l208 -208q84 -84 84 -204z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1920 384q0 -159 -112.5 -271.5t-271.5 -112.5h-1088q-185 0 -316.5 131.5t-131.5 316.5q0 132 71 241.5t187 163.5q-2 28 -2 43q0 212 150 362t362 150q158 0 286.5 -88t187.5 -230q70 62 166 62q106 0 181 -75t75 -181q0 -75 -41 -138q129 -30 213 -134.5t84 -239.5z " /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1527 88q56 -89 21.5 -152.5t-140.5 -63.5h-1152q-106 0 -140.5 63.5t21.5 152.5l503 793v399h-64q-26 0 -45 19t-19 45t19 45t45 19h512q26 0 45 -19t19 -45t-19 -45t-45 -19h-64v-399zM748 813l-272 -429h712l-272 429l-20 31v37v399h-128v-399v-37z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M960 640q26 0 45 -19t19 -45t-19 -45t-45 -19t-45 19t-19 45t19 45t45 19zM1260 576l507 -398q28 -20 25 -56q-5 -35 -35 -51l-128 -64q-13 -7 -29 -7q-17 0 -31 8l-690 387l-110 -66q-8 -4 -12 -5q14 -49 10 -97q-7 -77 -56 -147.5t-132 -123.5q-132 -84 -277 -84 q-136 0 -222 78q-90 84 -79 207q7 76 56 147t131 124q132 84 278 84q83 0 151 -31q9 13 22 22l122 73l-122 73q-13 9 -22 22q-68 -31 -151 -31q-146 0 -278 84q-82 53 -131 124t-56 147q-5 59 15.5 113t63.5 93q85 79 222 79q145 0 277 -84q83 -52 132 -123t56 -148 q4 -48 -10 -97q4 -1 12 -5l110 -66l690 387q14 8 31 8q16 0 29 -7l128 -64q30 -16 35 -51q3 -36 -25 -56zM579 836q46 42 21 108t-106 117q-92 59 -192 59q-74 0 -113 -36q-46 -42 -21 -108t106 -117q92 -59 192 -59q74 0 113 36zM494 91q81 51 106 117t-21 108 q-39 36 -113 36q-100 0 -192 -59q-81 -51 -106 -117t21 -108q39 -36 113 -36q100 0 192 59zM672 704l96 -58v11q0 36 33 56l14 8l-79 47l-26 -26q-3 -3 -10 -11t-12 -12q-2 -2 -4 -3.5t-3 -2.5zM896 480l96 -32l736 576l-128 64l-768 -431v-113l-160 -96l9 -8q2 -2 7 -6 q4 -4 11 -12t11 -12l26 -26zM1600 64l128 64l-520 408l-177 -138q-2 -3 -13 -7z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1696 1152q40 0 68 -28t28 -68v-1216q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v288h-544q-40 0 -68 28t-28 68v672q0 40 20 88t48 76l408 408q28 28 76 48t88 20h416q40 0 68 -28t28 -68v-328q68 40 128 40h416zM1152 939l-299 -299h299v299zM512 1323l-299 -299 h299v299zM708 676l316 316v416h-384v-416q0 -40 -28 -68t-68 -28h-416v-640h512v256q0 40 20 88t48 76zM1664 -128v1152h-384v-416q0 -40 -28 -68t-68 -28h-416v-640h896z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1404 151q0 -117 -79 -196t-196 -79q-135 0 -235 100l-777 776q-113 115 -113 271q0 159 110 270t269 111q158 0 273 -113l605 -606q10 -10 10 -22q0 -16 -30.5 -46.5t-46.5 -30.5q-13 0 -23 10l-606 607q-79 77 -181 77q-106 0 -179 -75t-73 -181q0 -105 76 -181 l776 -777q63 -63 145 -63q64 0 106 42t42 106q0 82 -63 145l-581 581q-26 24 -60 24q-29 0 -48 -19t-19 -48q0 -32 25 -59l410 -410q10 -10 10 -22q0 -16 -31 -47t-47 -31q-12 0 -22 10l-410 410q-63 61 -63 149q0 82 57 139t139 57q88 0 149 -63l581 -581q100 -98 100 -235 z" /> | ||||||
|  | <glyph unicode="" d="M384 0h768v384h-768v-384zM1280 0h128v896q0 14 -10 38.5t-20 34.5l-281 281q-10 10 -34 20t-39 10v-416q0 -40 -28 -68t-68 -28h-576q-40 0 -68 28t-28 68v416h-128v-1280h128v416q0 40 28 68t68 28h832q40 0 68 -28t28 -68v-416zM896 928v320q0 13 -9.5 22.5t-22.5 9.5 h-192q-13 0 -22.5 -9.5t-9.5 -22.5v-320q0 -13 9.5 -22.5t22.5 -9.5h192q13 0 22.5 9.5t9.5 22.5zM1536 896v-928q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h928q40 0 88 -20t76 -48l280 -280q28 -28 48 -76t20 -88z" /> | ||||||
|  | <glyph unicode="" d="M1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M1536 192v-128q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1536 704v-128q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1536 1216v-128q0 -26 -19 -45 t-45 -19h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M384 128q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM384 640q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5 t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5zM384 1152q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1792 736v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5z M1792 1248v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M381 -84q0 -80 -54.5 -126t-135.5 -46q-106 0 -172 66l57 88q49 -45 106 -45q29 0 50.5 14.5t21.5 42.5q0 64 -105 56l-26 56q8 10 32.5 43.5t42.5 54t37 38.5v1q-16 0 -48.5 -1t-48.5 -1v-53h-106v152h333v-88l-95 -115q51 -12 81 -49t30 -88zM383 543v-159h-362 q-6 36 -6 54q0 51 23.5 93t56.5 68t66 47.5t56.5 43.5t23.5 45q0 25 -14.5 38.5t-39.5 13.5q-46 0 -81 -58l-85 59q24 51 71.5 79.5t105.5 28.5q73 0 123 -41.5t50 -112.5q0 -50 -34 -91.5t-75 -64.5t-75.5 -50.5t-35.5 -52.5h127v60h105zM1792 224v-192q0 -13 -9.5 -22.5 t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 14 9 23t23 9h1216q13 0 22.5 -9.5t9.5 -22.5zM384 1123v-99h-335v99h107q0 41 0.5 122t0.5 121v12h-2q-8 -17 -50 -54l-71 76l136 127h106v-404h108zM1792 736v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5 t-9.5 22.5v192q0 14 9 23t23 9h1216q13 0 22.5 -9.5t9.5 -22.5zM1792 1248v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1760 640q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-1728q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h1728zM483 704q-28 35 -51 80q-48 97 -48 188q0 181 134 309q133 127 393 127q50 0 167 -19q66 -12 177 -48q10 -38 21 -118q14 -123 14 -183q0 -18 -5 -45l-12 -3l-84 6 l-14 2q-50 149 -103 205q-88 91 -210 91q-114 0 -182 -59q-67 -58 -67 -146q0 -73 66 -140t279 -129q69 -20 173 -66q58 -28 95 -52h-743zM990 448h411q7 -39 7 -92q0 -111 -41 -212q-23 -55 -71 -104q-37 -35 -109 -81q-80 -48 -153 -66q-80 -21 -203 -21q-114 0 -195 23 l-140 40q-57 16 -72 28q-8 8 -8 22v13q0 108 -2 156q-1 30 0 68l2 37v44l102 2q15 -34 30 -71t22.5 -56t12.5 -27q35 -57 80 -94q43 -36 105 -57q59 -22 132 -22q64 0 139 27q77 26 122 86q47 61 47 129q0 84 -81 157q-34 29 -137 71z" /> | ||||||
|  | <glyph unicode="" d="M48 1313q-37 2 -45 4l-3 88q13 1 40 1q60 0 112 -4q132 -7 166 -7q86 0 168 3q116 4 146 5q56 0 86 2l-1 -14l2 -64v-9q-60 -9 -124 -9q-60 0 -79 -25q-13 -14 -13 -132q0 -13 0.5 -32.5t0.5 -25.5l1 -229l14 -280q6 -124 51 -202q35 -59 96 -92q88 -47 177 -47 q104 0 191 28q56 18 99 51q48 36 65 64q36 56 53 114q21 73 21 229q0 79 -3.5 128t-11 122.5t-13.5 159.5l-4 59q-5 67 -24 88q-34 35 -77 34l-100 -2l-14 3l2 86h84l205 -10q76 -3 196 10l18 -2q6 -38 6 -51q0 -7 -4 -31q-45 -12 -84 -13q-73 -11 -79 -17q-15 -15 -15 -41 q0 -7 1.5 -27t1.5 -31q8 -19 22 -396q6 -195 -15 -304q-15 -76 -41 -122q-38 -65 -112 -123q-75 -57 -182 -89q-109 -33 -255 -33q-167 0 -284 46q-119 47 -179 122q-61 76 -83 195q-16 80 -16 237v333q0 188 -17 213q-25 36 -147 39zM1536 -96v64q0 14 -9 23t-23 9h-1472 q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h1472q14 0 23 9t9 23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M512 160v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM512 544v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1024 160v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23 v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM512 928v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1024 544v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1536 160v192 q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1024 928v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1536 544v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192 q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1536 928v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1664 1248v-1088q0 -66 -47 -113t-113 -47h-1344q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1344q66 0 113 -47t47 -113 z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1190 955l293 293l-107 107l-293 -293zM1637 1248q0 -27 -18 -45l-1286 -1286q-18 -18 -45 -18t-45 18l-198 198q-18 18 -18 45t18 45l1286 1286q18 18 45 18t45 -18l198 -198q18 -18 18 -45zM286 1438l98 -30l-98 -30l-30 -98l-30 98l-98 30l98 30l30 98zM636 1276 l196 -60l-196 -60l-60 -196l-60 196l-196 60l196 60l60 196zM1566 798l98 -30l-98 -30l-30 -98l-30 98l-98 30l98 30l30 98zM926 1438l98 -30l-98 -30l-30 -98l-30 98l-98 30l98 30l30 98z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M640 128q0 52 -38 90t-90 38t-90 -38t-38 -90t38 -90t90 -38t90 38t38 90zM256 640h384v256h-158q-13 0 -22 -9l-195 -195q-9 -9 -9 -22v-30zM1536 128q0 52 -38 90t-90 38t-90 -38t-38 -90t38 -90t90 -38t90 38t38 90zM1792 1216v-1024q0 -15 -4 -26.5t-13.5 -18.5 t-16.5 -11.5t-23.5 -6t-22.5 -2t-25.5 0t-22.5 0.5q0 -106 -75 -181t-181 -75t-181 75t-75 181h-384q0 -106 -75 -181t-181 -75t-181 75t-75 181h-64q-3 0 -22.5 -0.5t-25.5 0t-22.5 2t-23.5 6t-16.5 11.5t-13.5 18.5t-4 26.5q0 26 19 45t45 19v320q0 8 -0.5 35t0 38 t2.5 34.5t6.5 37t14 30.5t22.5 30l198 198q19 19 50.5 32t58.5 13h160v192q0 26 19 45t45 19h1024q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" d="M1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103q-111 0 -218 32q59 93 78 164q9 34 54 211q20 -39 73 -67.5t114 -28.5q121 0 216 68.5t147 188.5t52 270q0 114 -59.5 214t-172.5 163t-255 63q-105 0 -196 -29t-154.5 -77t-109 -110.5t-67 -129.5t-21.5 -134 q0 -104 40 -183t117 -111q30 -12 38 20q2 7 8 31t8 30q6 23 -11 43q-51 61 -51 151q0 151 104.5 259.5t273.5 108.5q151 0 235.5 -82t84.5 -213q0 -170 -68.5 -289t-175.5 -119q-61 0 -98 43.5t-23 104.5q8 35 26.5 93.5t30 103t11.5 75.5q0 50 -27 83t-77 33 q-62 0 -105 -57t-43 -142q0 -73 25 -122l-99 -418q-17 -70 -13 -177q-206 91 -333 281t-127 423q0 209 103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1248 1408q119 0 203.5 -84.5t84.5 -203.5v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-725q85 122 108 210q9 34 53 209q21 -39 73.5 -67t112.5 -28q181 0 295.5 147.5t114.5 373.5q0 84 -35 162.5t-96.5 139t-152.5 97t-197 36.5q-104 0 -194.5 -28.5t-153 -76.5 t-107.5 -109.5t-66.5 -128t-21.5 -132.5q0 -102 39.5 -180t116.5 -110q13 -5 23.5 0t14.5 19q10 44 15 61q6 23 -11 42q-50 62 -50 150q0 150 103.5 256.5t270.5 106.5q149 0 232.5 -81t83.5 -210q0 -168 -67.5 -286t-173.5 -118q-60 0 -97 43.5t-23 103.5q8 34 26.5 92.5 t29.5 102t11 74.5q0 49 -26.5 81.5t-75.5 32.5q-61 0 -103.5 -56.5t-42.5 -139.5q0 -72 24 -121l-98 -414q-24 -100 -7 -254h-183q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960z" /> | ||||||
|  | <glyph unicode="" d="M678 -57q0 -38 -10 -71h-380q-95 0 -171.5 56.5t-103.5 147.5q24 45 69 77.5t100 49.5t107 24t107 7q32 0 49 -2q6 -4 30.5 -21t33 -23t31 -23t32 -25.5t27.5 -25.5t26.5 -29.5t21 -30.5t17.5 -34.5t9.5 -36t4.5 -40.5zM385 294q-234 -7 -385 -85v433q103 -118 273 -118 q32 0 70 5q-21 -61 -21 -86q0 -67 63 -149zM558 805q0 -100 -43.5 -160.5t-140.5 -60.5q-51 0 -97 26t-78 67.5t-56 93.5t-35.5 104t-11.5 99q0 96 51.5 165t144.5 69q66 0 119 -41t84 -104t47 -130t16 -128zM1536 896v-736q0 -119 -84.5 -203.5t-203.5 -84.5h-468 q39 73 39 157q0 66 -22 122.5t-55.5 93t-72 71t-72 59.5t-55.5 54.5t-22 59.5q0 36 23 68t56 61.5t65.5 64.5t55.5 93t23 131t-26.5 145.5t-75.5 118.5q-6 6 -14 11t-12.5 7.5t-10 9.5t-10.5 17h135l135 64h-437q-138 0 -244.5 -38.5t-182.5 -133.5q0 126 81 213t207 87h960 q119 0 203.5 -84.5t84.5 -203.5v-96h-256v256h-128v-256h-256v-128h256v-256h128v256h256z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M876 71q0 21 -4.5 40.5t-9.5 36t-17.5 34.5t-21 30.5t-26.5 29.5t-27.5 25.5t-32 25.5t-31 23t-33 23t-30.5 21q-17 2 -50 2q-54 0 -106 -7t-108 -25t-98 -46t-69 -75t-27 -107q0 -68 35.5 -121.5t93 -84t120.5 -45.5t127 -15q59 0 112.5 12.5t100.5 39t74.5 73.5 t27.5 110zM756 933q0 60 -16.5 127.5t-47 130.5t-84 104t-119.5 41q-93 0 -144 -69t-51 -165q0 -47 11.5 -99t35.5 -104t56 -93.5t78 -67.5t97 -26q97 0 140.5 60.5t43.5 160.5zM625 1408h437l-135 -79h-135q71 -45 110 -126t39 -169q0 -74 -23 -131.5t-56 -92.5t-66 -64.5 t-56 -61t-23 -67.5q0 -26 16.5 -51t43 -48t58.5 -48t64 -55.5t58.5 -66t43 -85t16.5 -106.5q0 -160 -140 -282q-152 -131 -420 -131q-59 0 -119.5 10t-122 33.5t-108.5 58t-77 89t-30 121.5q0 61 37 135q32 64 96 110.5t145 71t155 36t150 13.5q-64 83 -64 149q0 12 2 23.5 t5 19.5t8 21.5t7 21.5q-40 -5 -70 -5q-149 0 -255.5 98t-106.5 246q0 140 95 250.5t234 141.5q94 20 187 20zM1664 1152v-128h-256v-256h-128v256h-256v128h256v256h128v-256h256z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M768 384h384v96h-128v448h-114l-148 -137l77 -80q42 37 55 57h2v-288h-128v-96zM1280 640q0 -70 -21 -142t-59.5 -134t-101.5 -101t-138 -39t-138 39t-101.5 101t-59.5 134t-21 142t21 142t59.5 134t101.5 101t138 39t138 -39t101.5 -101t59.5 -134t21 -142zM1792 384 v512q-106 0 -181 75t-75 181h-1152q0 -106 -75 -181t-181 -75v-512q106 0 181 -75t75 -181h1152q0 106 75 181t181 75zM1920 1216v-1152q0 -26 -19 -45t-45 -19h-1792q-26 0 -45 19t-19 45v1152q0 26 19 45t45 19h1792q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M1024 832q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19l-448 448q-19 19 -19 45t19 45t45 19h896q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M1024 320q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45l448 448q19 19 45 19t45 -19l448 -448q19 -19 19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="640" d="M640 1088v-896q0 -26 -19 -45t-45 -19t-45 19l-448 448q-19 19 -19 45t19 45l448 448q19 19 45 19t45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="640" d="M576 640q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19t-19 45v896q0 26 19 45t45 19t45 -19l448 -448q19 -19 19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M160 0h608v1152h-640v-1120q0 -13 9.5 -22.5t22.5 -9.5zM1536 32v1120h-640v-1152h608q13 0 22.5 9.5t9.5 22.5zM1664 1248v-1216q0 -66 -47 -113t-113 -47h-1344q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1344q66 0 113 -47t47 -113z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M1024 448q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19l-448 448q-19 19 -19 45t19 45t45 19h896q26 0 45 -19t19 -45zM1024 832q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45l448 448q19 19 45 19t45 -19l448 -448q19 -19 19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M1024 448q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19l-448 448q-19 19 -19 45t19 45t45 19h896q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M1024 832q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45l448 448q19 19 45 19t45 -19l448 -448q19 -19 19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 826v-794q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v794q44 -49 101 -87q362 -246 497 -345q57 -42 92.5 -65.5t94.5 -48t110 -24.5h1h1q51 0 110 24.5t94.5 48t92.5 65.5q170 123 498 345q57 39 100 87zM1792 1120q0 -79 -49 -151t-122 -123 q-376 -261 -468 -325q-10 -7 -42.5 -30.5t-54 -38t-52 -32.5t-57.5 -27t-50 -9h-1h-1q-23 0 -50 9t-57.5 27t-52 32.5t-54 38t-42.5 30.5q-91 64 -262 182.5t-205 142.5q-62 42 -117 115.5t-55 136.5q0 78 41.5 130t118.5 52h1472q65 0 112.5 -47t47.5 -113z" /> | ||||||
|  | <glyph unicode="" d="M349 911v-991h-330v991h330zM370 1217q1 -73 -50.5 -122t-135.5 -49h-2q-82 0 -132 49t-50 122q0 74 51.5 122.5t134.5 48.5t133 -48.5t51 -122.5zM1536 488v-568h-329v530q0 105 -40.5 164.5t-126.5 59.5q-63 0 -105.5 -34.5t-63.5 -85.5q-11 -30 -11 -81v-553h-329 q2 399 2 647t-1 296l-1 48h329v-144h-2q20 32 41 56t56.5 52t87 43.5t114.5 15.5q171 0 275 -113.5t104 -332.5z" /> | ||||||
|  | <glyph unicode="" d="M1536 640q0 -156 -61 -298t-164 -245t-245 -164t-298 -61q-172 0 -327 72.5t-264 204.5q-7 10 -6.5 22.5t8.5 20.5l137 138q10 9 25 9q16 -2 23 -12q73 -95 179 -147t225 -52q104 0 198.5 40.5t163.5 109.5t109.5 163.5t40.5 198.5t-40.5 198.5t-109.5 163.5 t-163.5 109.5t-198.5 40.5q-98 0 -188 -35.5t-160 -101.5l137 -138q31 -30 14 -69q-17 -40 -59 -40h-448q-26 0 -45 19t-19 45v448q0 42 40 59q39 17 69 -14l130 -129q107 101 244.5 156.5t284.5 55.5q156 0 298 -61t245 -164t164 -245t61 -298z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1771 0q0 -53 -37 -90l-107 -108q-39 -37 -91 -37q-53 0 -90 37l-363 364q-38 36 -38 90q0 53 43 96l-256 256l-126 -126q-14 -14 -34 -14t-34 14q2 -2 12.5 -12t12.5 -13t10 -11.5t10 -13.5t6 -13.5t5.5 -16.5t1.5 -18q0 -38 -28 -68q-3 -3 -16.5 -18t-19 -20.5 t-18.5 -16.5t-22 -15.5t-22 -9t-26 -4.5q-40 0 -68 28l-408 408q-28 28 -28 68q0 13 4.5 26t9 22t15.5 22t16.5 18.5t20.5 19t18 16.5q30 28 68 28q10 0 18 -1.5t16.5 -5.5t13.5 -6t13.5 -10t11.5 -10t13 -12.5t12 -12.5q-14 14 -14 34t14 34l348 348q14 14 34 14t34 -14 q-2 2 -12.5 12t-12.5 13t-10 11.5t-10 13.5t-6 13.5t-5.5 16.5t-1.5 18q0 38 28 68q3 3 16.5 18t19 20.5t18.5 16.5t22 15.5t22 9t26 4.5q40 0 68 -28l408 -408q28 -28 28 -68q0 -13 -4.5 -26t-9 -22t-15.5 -22t-16.5 -18.5t-20.5 -19t-18 -16.5q-30 -28 -68 -28 q-10 0 -18 1.5t-16.5 5.5t-13.5 6t-13.5 10t-11.5 10t-13 12.5t-12 12.5q14 -14 14 -34t-14 -34l-126 -126l256 -256q43 43 96 43q52 0 91 -37l363 -363q37 -39 37 -91z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M384 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM576 832q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1004 351l101 382q6 26 -7.5 48.5t-38.5 29.5 t-48 -6.5t-30 -39.5l-101 -382q-60 -5 -107 -43.5t-63 -98.5q-20 -77 20 -146t117 -89t146 20t89 117q16 60 -6 117t-72 91zM1664 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1024 1024q0 53 -37.5 90.5 t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1472 832q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1792 384q0 -261 -141 -483q-19 -29 -54 -29h-1402q-35 0 -54 29 q-141 221 -141 483q0 182 71 348t191 286t286 191t348 71t348 -71t286 -191t191 -286t71 -348z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M896 1152q-204 0 -381.5 -69.5t-282 -187.5t-104.5 -255q0 -112 71.5 -213.5t201.5 -175.5l87 -50l-27 -96q-24 -91 -70 -172q152 63 275 171l43 38l57 -6q69 -8 130 -8q204 0 381.5 69.5t282 187.5t104.5 255t-104.5 255t-282 187.5t-381.5 69.5zM1792 640 q0 -174 -120 -321.5t-326 -233t-450 -85.5q-70 0 -145 8q-198 -175 -460 -242q-49 -14 -114 -22h-5q-15 0 -27 10.5t-16 27.5v1q-3 4 -0.5 12t2 10t4.5 9.5l6 9t7 8.5t8 9q7 8 31 34.5t34.5 38t31 39.5t32.5 51t27 59t26 76q-157 89 -247.5 220t-90.5 281q0 174 120 321.5 t326 233t450 85.5t450 -85.5t326 -233t120 -321.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M704 1152q-153 0 -286 -52t-211.5 -141t-78.5 -191q0 -82 53 -158t149 -132l97 -56l-35 -84q34 20 62 39l44 31l53 -10q78 -14 153 -14q153 0 286 52t211.5 141t78.5 191t-78.5 191t-211.5 141t-286 52zM704 1280q191 0 353.5 -68.5t256.5 -186.5t94 -257t-94 -257 t-256.5 -186.5t-353.5 -68.5q-86 0 -176 16q-124 -88 -278 -128q-36 -9 -86 -16h-3q-11 0 -20.5 8t-11.5 21q-1 3 -1 6.5t0.5 6.5t2 6l2.5 5t3.5 5.5t4 5t4.5 5t4 4.5q5 6 23 25t26 29.5t22.5 29t25 38.5t20.5 44q-124 72 -195 177t-71 224q0 139 94 257t256.5 186.5 t353.5 68.5zM1526 111q10 -24 20.5 -44t25 -38.5t22.5 -29t26 -29.5t23 -25q1 -1 4 -4.5t4.5 -5t4 -5t3.5 -5.5l2.5 -5t2 -6t0.5 -6.5t-1 -6.5q-3 -14 -13 -22t-22 -7q-50 7 -86 16q-154 40 -278 128q-90 -16 -176 -16q-271 0 -472 132q58 -4 88 -4q161 0 309 45t264 129 q125 92 192 212t67 254q0 77 -23 152q129 -71 204 -178t75 -230q0 -120 -71 -224.5t-195 -176.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="896" d="M885 970q18 -20 7 -44l-540 -1157q-13 -25 -42 -25q-4 0 -14 2q-17 5 -25.5 19t-4.5 30l197 808l-406 -101q-4 -1 -12 -1q-18 0 -31 11q-18 15 -13 39l201 825q4 14 16 23t28 9h328q19 0 32 -12.5t13 -29.5q0 -8 -5 -18l-171 -463l396 98q8 2 12 2q19 0 34 -15z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 288v-320q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h96v192h-512v-192h96q40 0 68 -28t28 -68v-320q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h96v192h-512v-192h96q40 0 68 -28t28 -68v-320 q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h96v192q0 52 38 90t90 38h512v192h-96q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h320q40 0 68 -28t28 -68v-320q0 -40 -28 -68t-68 -28h-96v-192h512q52 0 90 -38t38 -90v-192h96q40 0 68 -28t28 -68 z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M896 708v-580q0 -104 -76 -180t-180 -76t-180 76t-76 180q0 26 19 45t45 19t45 -19t19 -45q0 -50 39 -89t89 -39t89 39t39 89v580q33 11 64 11t64 -11zM1664 681q0 -13 -9.5 -22.5t-22.5 -9.5q-11 0 -23 10q-49 46 -93 69t-102 23q-68 0 -128 -37t-103 -97 q-7 -10 -17.5 -28t-14.5 -24q-11 -17 -28 -17q-18 0 -29 17q-4 6 -14.5 24t-17.5 28q-43 60 -102.5 97t-127.5 37t-127.5 -37t-102.5 -97q-7 -10 -17.5 -28t-14.5 -24q-11 -17 -29 -17q-17 0 -28 17q-4 6 -14.5 24t-17.5 28q-43 60 -103 97t-128 37q-58 0 -102 -23t-93 -69 q-12 -10 -23 -10q-13 0 -22.5 9.5t-9.5 22.5q0 5 1 7q45 183 172.5 319.5t298 204.5t360.5 68q140 0 274.5 -40t246.5 -113.5t194.5 -187t115.5 -251.5q1 -2 1 -7zM896 1408v-98q-42 2 -64 2t-64 -2v98q0 26 19 45t45 19t45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M768 -128h896v640h-416q-40 0 -68 28t-28 68v416h-384v-1152zM1024 1312v64q0 13 -9.5 22.5t-22.5 9.5h-704q-13 0 -22.5 -9.5t-9.5 -22.5v-64q0 -13 9.5 -22.5t22.5 -9.5h704q13 0 22.5 9.5t9.5 22.5zM1280 640h299l-299 299v-299zM1792 512v-672q0 -40 -28 -68t-68 -28 h-960q-40 0 -68 28t-28 68v160h-544q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h1088q40 0 68 -28t28 -68v-328q21 -13 36 -28l408 -408q28 -28 48 -76t20 -88z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M736 960q0 -13 -9.5 -22.5t-22.5 -9.5t-22.5 9.5t-9.5 22.5q0 46 -54 71t-106 25q-13 0 -22.5 9.5t-9.5 22.5t9.5 22.5t22.5 9.5q50 0 99.5 -16t87 -54t37.5 -90zM896 960q0 72 -34.5 134t-90 101.5t-123 62t-136.5 22.5t-136.5 -22.5t-123 -62t-90 -101.5t-34.5 -134 q0 -101 68 -180q10 -11 30.5 -33t30.5 -33q128 -153 141 -298h228q13 145 141 298q10 11 30.5 33t30.5 33q68 79 68 180zM1024 960q0 -155 -103 -268q-45 -49 -74.5 -87t-59.5 -95.5t-34 -107.5q47 -28 47 -82q0 -37 -25 -64q25 -27 25 -64q0 -52 -45 -81q13 -23 13 -47 q0 -46 -31.5 -71t-77.5 -25q-20 -44 -60 -70t-87 -26t-87 26t-60 70q-46 0 -77.5 25t-31.5 71q0 24 13 47q-45 29 -45 81q0 37 25 64q-25 27 -25 64q0 54 47 82q-4 50 -34 107.5t-59.5 95.5t-74.5 87q-103 113 -103 268q0 99 44.5 184.5t117 142t164 89t186.5 32.5 t186.5 -32.5t164 -89t117 -142t44.5 -184.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 352v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5q-12 0 -24 10l-319 320q-9 9 -9 22q0 14 9 23l320 320q9 9 23 9q13 0 22.5 -9.5t9.5 -22.5v-192h1376q13 0 22.5 -9.5t9.5 -22.5zM1792 896q0 -14 -9 -23l-320 -320q-9 -9 -23 -9 q-13 0 -22.5 9.5t-9.5 22.5v192h-1376q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1376v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1280 608q0 14 -9 23t-23 9h-224v352q0 13 -9.5 22.5t-22.5 9.5h-192q-13 0 -22.5 -9.5t-9.5 -22.5v-352h-224q-13 0 -22.5 -9.5t-9.5 -22.5q0 -14 9 -23l352 -352q9 -9 23 -9t23 9l351 351q10 12 10 24zM1920 384q0 -159 -112.5 -271.5t-271.5 -112.5h-1088 q-185 0 -316.5 131.5t-131.5 316.5q0 130 70 240t188 165q-2 30 -2 43q0 212 150 362t362 150q156 0 285.5 -87t188.5 -231q71 62 166 62q106 0 181 -75t75 -181q0 -76 -41 -138q130 -31 213.5 -135.5t83.5 -238.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1280 672q0 14 -9 23l-352 352q-9 9 -23 9t-23 -9l-351 -351q-10 -12 -10 -24q0 -14 9 -23t23 -9h224v-352q0 -13 9.5 -22.5t22.5 -9.5h192q13 0 22.5 9.5t9.5 22.5v352h224q13 0 22.5 9.5t9.5 22.5zM1920 384q0 -159 -112.5 -271.5t-271.5 -112.5h-1088 q-185 0 -316.5 131.5t-131.5 316.5q0 130 70 240t188 165q-2 30 -2 43q0 212 150 362t362 150q156 0 285.5 -87t188.5 -231q71 62 166 62q106 0 181 -75t75 -181q0 -76 -41 -138q130 -31 213.5 -135.5t83.5 -238.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M384 192q0 -26 -19 -45t-45 -19t-45 19t-19 45t19 45t45 19t45 -19t19 -45zM1408 131q0 -121 -73 -190t-194 -69h-874q-121 0 -194 69t-73 190q0 68 5.5 131t24 138t47.5 132.5t81 103t120 60.5q-22 -52 -22 -120v-203q-58 -20 -93 -70t-35 -111q0 -80 56 -136t136 -56 t136 56t56 136q0 61 -35.5 111t-92.5 70v203q0 62 25 93q132 -104 295 -104t295 104q25 -31 25 -93v-64q-106 0 -181 -75t-75 -181v-89q-32 -29 -32 -71q0 -40 28 -68t68 -28t68 28t28 68q0 42 -32 71v89q0 52 38 90t90 38t90 -38t38 -90v-89q-32 -29 -32 -71q0 -40 28 -68 t68 -28t68 28t28 68q0 42 -32 71v89q0 68 -34.5 127.5t-93.5 93.5q0 10 0.5 42.5t0 48t-2.5 41.5t-7 47t-13 40q68 -15 120 -60.5t81 -103t47.5 -132.5t24 -138t5.5 -131zM1088 1024q0 -159 -112.5 -271.5t-271.5 -112.5t-271.5 112.5t-112.5 271.5t112.5 271.5t271.5 112.5 t271.5 -112.5t112.5 -271.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1280 832q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 832q0 -62 -35.5 -111t-92.5 -70v-395q0 -159 -131.5 -271.5t-316.5 -112.5t-316.5 112.5t-131.5 271.5v132q-164 20 -274 128t-110 252v512q0 26 19 45t45 19q6 0 16 -2q17 30 47 48 t65 18q53 0 90.5 -37.5t37.5 -90.5t-37.5 -90.5t-90.5 -37.5q-33 0 -64 18v-402q0 -106 94 -181t226 -75t226 75t94 181v402q-31 -18 -64 -18q-53 0 -90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5q35 0 65 -18t47 -48q10 2 16 2q26 0 45 -19t19 -45v-512q0 -144 -110 -252 t-274 -128v-132q0 -106 94 -181t226 -75t226 75t94 181v395q-57 21 -92.5 70t-35.5 111q0 80 56 136t136 56t136 -56t56 -136z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M640 1152h512v128h-512v-128zM288 1152v-1280h-64q-92 0 -158 66t-66 158v832q0 92 66 158t158 66h64zM1408 1152v-1280h-1024v1280h128v160q0 40 28 68t68 28h576q40 0 68 -28t28 -68v-160h128zM1792 928v-832q0 -92 -66 -158t-158 -66h-64v1280h64q92 0 158 -66 t66 -158z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M848 -160q0 16 -16 16q-59 0 -101.5 42.5t-42.5 101.5q0 16 -16 16t-16 -16q0 -73 51.5 -124.5t124.5 -51.5q16 0 16 16zM1664 128q0 -52 -38 -90t-90 -38h-448q0 -106 -75 -181t-181 -75t-181 75t-75 181h-448q-52 0 -90 38t-38 90q190 161 287 397.5t97 498.5 q0 165 96 262t264 117q-8 18 -8 37q0 40 28 68t68 28t68 -28t28 -68q0 -19 -8 -37q168 -20 264 -117t96 -262q0 -262 97 -498.5t287 -397.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1664 896q0 80 -56 136t-136 56h-64v-384h64q80 0 136 56t56 136zM0 128h1792q0 -106 -75 -181t-181 -75h-1280q-106 0 -181 75t-75 181zM1856 896q0 -159 -112.5 -271.5t-271.5 -112.5h-64v-32q0 -92 -66 -158t-158 -66h-704q-92 0 -158 66t-66 158v736q0 26 19 45 t45 19h1152q159 0 271.5 -112.5t112.5 -271.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M640 1472v-640q0 -61 -35.5 -111t-92.5 -70v-779q0 -52 -38 -90t-90 -38h-128q-52 0 -90 38t-38 90v779q-57 20 -92.5 70t-35.5 111v640q0 26 19 45t45 19t45 -19t19 -45v-416q0 -26 19 -45t45 -19t45 19t19 45v416q0 26 19 45t45 19t45 -19t19 -45v-416q0 -26 19 -45 t45 -19t45 19t19 45v416q0 26 19 45t45 19t45 -19t19 -45zM1408 1472v-1600q0 -52 -38 -90t-90 -38h-128q-52 0 -90 38t-38 90v512h-224q-13 0 -22.5 9.5t-9.5 22.5v800q0 132 94 226t226 94h256q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1280" d="M1024 352v-64q0 -14 -9 -23t-23 -9h-704q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h704q14 0 23 -9t9 -23zM1024 608v-64q0 -14 -9 -23t-23 -9h-704q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h704q14 0 23 -9t9 -23zM128 0h1024v768h-416q-40 0 -68 28t-28 68v416h-512v-1280z M768 896h376q-10 29 -22 41l-313 313q-12 12 -41 22v-376zM1280 864v-896q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h640q40 0 88 -20t76 -48l312 -312q28 -28 48 -76t20 -88z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M384 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 -128h384v1536h-1152v-1536h384v224q0 13 9.5 22.5t22.5 9.5h320q13 0 22.5 -9.5t9.5 -22.5v-224zM1408 1472v-1664q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v1664q0 26 19 45t45 19h1280q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M384 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 -128h384v1152h-256v-32q0 -40 -28 -68t-68 -28h-448q-40 0 -68 28t-28 68v32h-256v-1152h384v224q0 13 9.5 22.5t22.5 9.5h320q13 0 22.5 -9.5t9.5 -22.5v-224zM896 1056v320q0 13 -9.5 22.5t-22.5 9.5h-64q-13 0 -22.5 -9.5t-9.5 -22.5v-96h-128v96q0 13 -9.5 22.5 t-22.5 9.5h-64q-13 0 -22.5 -9.5t-9.5 -22.5v-320q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5v96h128v-96q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5zM1408 1088v-1280q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v1280q0 26 19 45t45 19h320 v288q0 40 28 68t68 28h448q40 0 68 -28t28 -68v-288h320q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M640 128q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM256 640h384v256h-158q-14 -2 -22 -9l-195 -195q-7 -12 -9 -22v-30zM1536 128q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5 t90.5 37.5t37.5 90.5zM1664 800v192q0 14 -9 23t-23 9h-224v224q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-224h-224q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h224v-224q0 -14 9 -23t23 -9h192q14 0 23 9t9 23v224h224q14 0 23 9t9 23zM1920 1344v-1152 q0 -26 -19 -45t-45 -19h-192q0 -106 -75 -181t-181 -75t-181 75t-75 181h-384q0 -106 -75 -181t-181 -75t-181 75t-75 181h-128q-26 0 -45 19t-19 45t19 45t45 19v416q0 26 13 58t32 51l198 198q19 19 51 32t58 13h160v320q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1280 416v192q0 14 -9 23t-23 9h-224v224q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-224h-224q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h224v-224q0 -14 9 -23t23 -9h192q14 0 23 9t9 23v224h224q14 0 23 9t9 23zM640 1152h512v128h-512v-128zM256 1152v-1280h-32 q-92 0 -158 66t-66 158v832q0 92 66 158t158 66h32zM1440 1152v-1280h-1088v1280h160v160q0 40 28 68t68 28h576q40 0 68 -28t28 -68v-160h160zM1792 928v-832q0 -92 -66 -158t-158 -66h-32v1280h32q92 0 158 -66t66 -158z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1920 576q-1 -32 -288 -96l-352 -32l-224 -64h-64l-293 -352h69q26 0 45 -4.5t19 -11.5t-19 -11.5t-45 -4.5h-96h-160h-64v32h64v416h-160l-192 -224h-96l-32 32v192h32v32h128v8l-192 24v128l192 24v8h-128v32h-32v192l32 32h96l192 -224h160v416h-64v32h64h160h96 q26 0 45 -4.5t19 -11.5t-19 -11.5t-45 -4.5h-69l293 -352h64l224 -64l352 -32q261 -58 287 -93z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M640 640v384h-256v-256q0 -53 37.5 -90.5t90.5 -37.5h128zM1664 192v-192h-1152v192l128 192h-128q-159 0 -271.5 112.5t-112.5 271.5v320l-64 64l32 128h480l32 128h960l32 -192l-64 -32v-800z" /> | ||||||
|  | <glyph unicode="" d="M1280 192v896q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-320h-512v320q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-896q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v320h512v-320q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1536 1120v-960 q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M1280 576v128q0 26 -19 45t-45 19h-320v320q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-320h-320q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h320v-320q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v320h320q26 0 45 19t19 45zM1536 1120v-960 q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M627 160q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23t-10 -23l-393 -393l393 -393q10 -10 10 -23zM1011 160q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23 t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23t-10 -23l-393 -393l393 -393q10 -10 10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M595 576q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23zM979 576q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23 l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M1075 224q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-393 393l-393 -393q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l466 -466q10 -10 10 -23zM1075 608q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-393 393l-393 -393 q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M1075 672q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l393 -393l393 393q10 10 23 10t23 -10l50 -50q10 -10 10 -23zM1075 1056q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23 t10 23l50 50q10 10 23 10t23 -10l393 -393l393 393q10 10 23 10t23 -10l50 -50q10 -10 10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="640" d="M627 992q0 -13 -10 -23l-393 -393l393 -393q10 -10 10 -23t-10 -23l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="640" d="M595 576q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M1075 352q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-393 393l-393 -393q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M1075 800q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l393 -393l393 393q10 10 23 10t23 -10l50 -50q10 -10 10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1792 544v832q0 13 -9.5 22.5t-22.5 9.5h-1600q-13 0 -22.5 -9.5t-9.5 -22.5v-832q0 -13 9.5 -22.5t22.5 -9.5h1600q13 0 22.5 9.5t9.5 22.5zM1920 1376v-1088q0 -66 -47 -113t-113 -47h-544q0 -37 16 -77.5t32 -71t16 -43.5q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19 t-19 45q0 14 16 44t32 70t16 78h-544q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M416 256q-66 0 -113 47t-47 113v704q0 66 47 113t113 47h1088q66 0 113 -47t47 -113v-704q0 -66 -47 -113t-113 -47h-1088zM384 1120v-704q0 -13 9.5 -22.5t22.5 -9.5h1088q13 0 22.5 9.5t9.5 22.5v704q0 13 -9.5 22.5t-22.5 9.5h-1088q-13 0 -22.5 -9.5t-9.5 -22.5z M1760 192h160v-96q0 -40 -47 -68t-113 -28h-1600q-66 0 -113 28t-47 68v96h160h1600zM1040 96q16 0 16 16t-16 16h-160q-16 0 -16 -16t16 -16h160z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M640 128q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1024 288v960q0 13 -9.5 22.5t-22.5 9.5h-832q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h832q13 0 22.5 9.5t9.5 22.5zM1152 1248v-1088q0 -66 -47 -113t-113 -47h-832 q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h832q66 0 113 -47t47 -113z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="768" d="M464 128q0 33 -23.5 56.5t-56.5 23.5t-56.5 -23.5t-23.5 -56.5t23.5 -56.5t56.5 -23.5t56.5 23.5t23.5 56.5zM672 288v704q0 13 -9.5 22.5t-22.5 9.5h-512q-13 0 -22.5 -9.5t-9.5 -22.5v-704q0 -13 9.5 -22.5t22.5 -9.5h512q13 0 22.5 9.5t9.5 22.5zM480 1136 q0 16 -16 16h-160q-16 0 -16 -16t16 -16h160q16 0 16 16zM768 1152v-1024q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v1024q0 52 38 90t90 38h512q52 0 90 -38t38 -90z" /> | ||||||
|  | <glyph unicode="" d="M768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103 t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M768 576v-384q0 -80 -56 -136t-136 -56h-384q-80 0 -136 56t-56 136v704q0 104 40.5 198.5t109.5 163.5t163.5 109.5t198.5 40.5h64q26 0 45 -19t19 -45v-128q0 -26 -19 -45t-45 -19h-64q-106 0 -181 -75t-75 -181v-32q0 -40 28 -68t68 -28h224q80 0 136 -56t56 -136z M1664 576v-384q0 -80 -56 -136t-136 -56h-384q-80 0 -136 56t-56 136v704q0 104 40.5 198.5t109.5 163.5t163.5 109.5t198.5 40.5h64q26 0 45 -19t19 -45v-128q0 -26 -19 -45t-45 -19h-64q-106 0 -181 -75t-75 -181v-32q0 -40 28 -68t68 -28h224q80 0 136 -56t56 -136z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M768 1216v-704q0 -104 -40.5 -198.5t-109.5 -163.5t-163.5 -109.5t-198.5 -40.5h-64q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h64q106 0 181 75t75 181v32q0 40 -28 68t-68 28h-224q-80 0 -136 56t-56 136v384q0 80 56 136t136 56h384q80 0 136 -56t56 -136zM1664 1216 v-704q0 -104 -40.5 -198.5t-109.5 -163.5t-163.5 -109.5t-198.5 -40.5h-64q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h64q106 0 181 75t75 181v32q0 40 -28 68t-68 28h-224q-80 0 -136 56t-56 136v384q0 80 56 136t136 56h384q80 0 136 -56t56 -136z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1568" d="M496 192q0 -60 -42.5 -102t-101.5 -42q-60 0 -102 42t-42 102t42 102t102 42q59 0 101.5 -42t42.5 -102zM928 0q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM320 640q0 -66 -47 -113t-113 -47t-113 47t-47 113 t47 113t113 47t113 -47t47 -113zM1360 192q0 -46 -33 -79t-79 -33t-79 33t-33 79t33 79t79 33t79 -33t33 -79zM528 1088q0 -73 -51.5 -124.5t-124.5 -51.5t-124.5 51.5t-51.5 124.5t51.5 124.5t124.5 51.5t124.5 -51.5t51.5 -124.5zM992 1280q0 -80 -56 -136t-136 -56 t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1536 640q0 -40 -28 -68t-68 -28t-68 28t-28 68t28 68t68 28t68 -28t28 -68zM1328 1088q0 -33 -23.5 -56.5t-56.5 -23.5t-56.5 23.5t-23.5 56.5t23.5 56.5t56.5 23.5t56.5 -23.5t23.5 -56.5z" /> | ||||||
|  | <glyph unicode="" d="M1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 416q0 -166 -127 -451q-3 -7 -10.5 -24t-13.5 -30t-13 -22q-12 -17 -28 -17q-15 0 -23.5 10t-8.5 25q0 9 2.5 26.5t2.5 23.5q5 68 5 123q0 101 -17.5 181t-48.5 138.5t-80 101t-105.5 69.5t-133 42.5t-154 21.5t-175.5 6h-224v-256q0 -26 -19 -45t-45 -19t-45 19 l-512 512q-19 19 -19 45t19 45l512 512q19 19 45 19t45 -19t19 -45v-256h224q713 0 875 -403q53 -134 53 -333z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M640 320q0 -40 -12.5 -82t-43 -76t-72.5 -34t-72.5 34t-43 76t-12.5 82t12.5 82t43 76t72.5 34t72.5 -34t43 -76t12.5 -82zM1280 320q0 -40 -12.5 -82t-43 -76t-72.5 -34t-72.5 34t-43 76t-12.5 82t12.5 82t43 76t72.5 34t72.5 -34t43 -76t12.5 -82zM1440 320 q0 120 -69 204t-187 84q-41 0 -195 -21q-71 -11 -157 -11t-157 11q-152 21 -195 21q-118 0 -187 -84t-69 -204q0 -88 32 -153.5t81 -103t122 -60t140 -29.5t149 -7h168q82 0 149 7t140 29.5t122 60t81 103t32 153.5zM1664 496q0 -207 -61 -331q-38 -77 -105.5 -133t-141 -86 t-170 -47.5t-171.5 -22t-167 -4.5q-78 0 -142 3t-147.5 12.5t-152.5 30t-137 51.5t-121 81t-86 115q-62 123 -62 331q0 237 136 396q-27 82 -27 170q0 116 51 218q108 0 190 -39.5t189 -123.5q147 35 309 35q148 0 280 -32q105 82 187 121t189 39q51 -102 51 -218 q0 -87 -27 -168q136 -160 136 -398z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1536 224v704q0 40 -28 68t-68 28h-704q-40 0 -68 28t-28 68v64q0 40 -28 68t-68 28h-320q-40 0 -68 -28t-28 -68v-960q0 -40 28 -68t68 -28h1216q40 0 68 28t28 68zM1664 928v-704q0 -92 -66 -158t-158 -66h-1216q-92 0 -158 66t-66 158v960q0 92 66 158t158 66h320 q92 0 158 -66t66 -158v-32h672q92 0 158 -66t66 -158z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1781 605q0 35 -53 35h-1088q-40 0 -85.5 -21.5t-71.5 -52.5l-294 -363q-18 -24 -18 -40q0 -35 53 -35h1088q40 0 86 22t71 53l294 363q18 22 18 39zM640 768h768v160q0 40 -28 68t-68 28h-576q-40 0 -68 28t-28 68v64q0 40 -28 68t-68 28h-320q-40 0 -68 -28t-28 -68 v-853l256 315q44 53 116 87.5t140 34.5zM1909 605q0 -62 -46 -120l-295 -363q-43 -53 -116 -87.5t-140 -34.5h-1088q-92 0 -158 66t-66 158v960q0 92 66 158t158 66h320q92 0 158 -66t66 -158v-32h544q92 0 158 -66t66 -158v-160h192q54 0 99 -24.5t67 -70.5q15 -32 15 -68z " /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M896 608v-64q0 -14 -9 -23t-23 -9h-224v-224q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v224h-224q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h224v224q0 14 9 23t23 9h64q14 0 23 -9t9 -23v-224h224q14 0 23 -9t9 -23zM1024 224v704q0 40 -28 68t-68 28h-704q-40 0 -68 -28 t-28 -68v-704q0 -40 28 -68t68 -28h704q40 0 68 28t28 68zM1152 928v-704q0 -92 -65.5 -158t-158.5 -66h-704q-93 0 -158.5 66t-65.5 158v704q0 93 65.5 158.5t158.5 65.5h704q93 0 158.5 -65.5t65.5 -158.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M928 1152q93 0 158.5 -65.5t65.5 -158.5v-704q0 -92 -65.5 -158t-158.5 -66h-704q-93 0 -158.5 66t-65.5 158v704q0 93 65.5 158.5t158.5 65.5h704zM1024 224v704q0 40 -28 68t-68 28h-704q-40 0 -68 -28t-28 -68v-704q0 -40 28 -68t68 -28h704q40 0 68 28t28 68z M864 640q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-576q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h576z" /> | ||||||
|  | <glyph unicode="" d="M1134 461q-37 -121 -138 -195t-228 -74t-228 74t-138 195q-8 25 4 48.5t38 31.5q25 8 48.5 -4t31.5 -38q25 -80 92.5 -129.5t151.5 -49.5t151.5 49.5t92.5 129.5q8 26 32 38t49 4t37 -31.5t4 -48.5zM640 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5 t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1152 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5 t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1134 307q8 -25 -4 -48.5t-37 -31.5t-49 4t-32 38q-25 80 -92.5 129.5t-151.5 49.5t-151.5 -49.5t-92.5 -129.5q-8 -26 -31.5 -38t-48.5 -4q-26 8 -38 31.5t-4 48.5q37 121 138 195t228 74t228 -74t138 -195zM640 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5 t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1152 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204 t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1152 448q0 -26 -19 -45t-45 -19h-640q-26 0 -45 19t-19 45t19 45t45 19h640q26 0 45 -19t19 -45zM640 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1152 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5 t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M832 448v128q0 14 -9 23t-23 9h-192v192q0 14 -9 23t-23 9h-128q-14 0 -23 -9t-9 -23v-192h-192q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h192v-192q0 -14 9 -23t23 -9h128q14 0 23 9t9 23v192h192q14 0 23 9t9 23zM1408 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5 t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1664 640q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1920 512q0 -212 -150 -362t-362 -150q-192 0 -338 128h-220q-146 -128 -338 -128q-212 0 -362 150 t-150 362t150 362t362 150h896q212 0 362 -150t150 -362z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M384 368v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM512 624v-96q0 -16 -16 -16h-224q-16 0 -16 16v96q0 16 16 16h224q16 0 16 -16zM384 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1408 368v-96q0 -16 -16 -16 h-864q-16 0 -16 16v96q0 16 16 16h864q16 0 16 -16zM768 624v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM640 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1024 624v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16 h96q16 0 16 -16zM896 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1280 624v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1664 368v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1152 880v-96 q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1408 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1664 880v-352q0 -16 -16 -16h-224q-16 0 -16 16v96q0 16 16 16h112v240q0 16 16 16h96q16 0 16 -16zM1792 128v896h-1664v-896 h1664zM1920 1024v-896q0 -53 -37.5 -90.5t-90.5 -37.5h-1664q-53 0 -90.5 37.5t-37.5 90.5v896q0 53 37.5 90.5t90.5 37.5h1664q53 0 90.5 -37.5t37.5 -90.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1664 491v616q-169 -91 -306 -91q-82 0 -145 32q-100 49 -184 76.5t-178 27.5q-173 0 -403 -127v-599q245 113 433 113q55 0 103.5 -7.5t98 -26t77 -31t82.5 -39.5l28 -14q44 -22 101 -22q120 0 293 92zM320 1280q0 -35 -17.5 -64t-46.5 -46v-1266q0 -14 -9 -23t-23 -9 h-64q-14 0 -23 9t-9 23v1266q-29 17 -46.5 46t-17.5 64q0 53 37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1792 1216v-763q0 -39 -35 -57q-10 -5 -17 -9q-218 -116 -369 -116q-88 0 -158 35l-28 14q-64 33 -99 48t-91 29t-114 14q-102 0 -235.5 -44t-228.5 -102 q-15 -9 -33 -9q-16 0 -32 8q-32 19 -32 56v742q0 35 31 55q35 21 78.5 42.5t114 52t152.5 49.5t155 19q112 0 209 -31t209 -86q38 -19 89 -19q122 0 310 112q22 12 31 17q31 16 62 -2q31 -20 31 -55z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M832 536v192q-181 -16 -384 -117v-185q205 96 384 110zM832 954v197q-172 -8 -384 -126v-189q215 111 384 118zM1664 491v184q-235 -116 -384 -71v224q-20 6 -39 15q-5 3 -33 17t-34.5 17t-31.5 15t-34.5 15.5t-32.5 13t-36 12.5t-35 8.5t-39.5 7.5t-39.5 4t-44 2 q-23 0 -49 -3v-222h19q102 0 192.5 -29t197.5 -82q19 -9 39 -15v-188q42 -17 91 -17q120 0 293 92zM1664 918v189q-169 -91 -306 -91q-45 0 -78 8v-196q148 -42 384 90zM320 1280q0 -35 -17.5 -64t-46.5 -46v-1266q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v1266 q-29 17 -46.5 46t-17.5 64q0 53 37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1792 1216v-763q0 -39 -35 -57q-10 -5 -17 -9q-218 -116 -369 -116q-88 0 -158 35l-28 14q-64 33 -99 48t-91 29t-114 14q-102 0 -235.5 -44t-228.5 -102q-15 -9 -33 -9q-16 0 -32 8 q-32 19 -32 56v742q0 35 31 55q35 21 78.5 42.5t114 52t152.5 49.5t155 19q112 0 209 -31t209 -86q38 -19 89 -19q122 0 310 112q22 12 31 17q31 16 62 -2q31 -20 31 -55z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M585 553l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23t-10 -23zM1664 96v-64q0 -14 -9 -23t-23 -9h-960q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h960q14 0 23 -9 t9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M617 137l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23t-10 -23l-393 -393l393 -393q10 -10 10 -23t-10 -23zM1208 1204l-373 -1291q-4 -13 -15.5 -19.5t-23.5 -2.5l-62 17q-13 4 -19.5 15.5t-2.5 24.5 l373 1291q4 13 15.5 19.5t23.5 2.5l62 -17q13 -4 19.5 -15.5t2.5 -24.5zM1865 553l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23t-10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M640 454v-70q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-512 512q-19 19 -19 45t19 45l512 512q29 31 70 14q39 -17 39 -59v-69l-397 -398q-19 -19 -19 -45t19 -45zM1792 416q0 -58 -17 -133.5t-38.5 -138t-48 -125t-40.5 -90.5l-20 -40q-8 -17 -28 -17q-6 0 -9 1 q-25 8 -23 34q43 400 -106 565q-64 71 -170.5 110.5t-267.5 52.5v-251q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-512 512q-19 19 -19 45t19 45l512 512q29 31 70 14q39 -17 39 -59v-262q411 -28 599 -221q169 -173 169 -509z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1186 579l257 250l-356 52l-66 10l-30 60l-159 322v-963l59 -31l318 -168l-60 355l-12 66zM1638 841l-363 -354l86 -500q5 -33 -6 -51.5t-34 -18.5q-17 0 -40 12l-449 236l-449 -236q-23 -12 -40 -12q-23 0 -34 18.5t-6 51.5l86 500l-364 354q-32 32 -23 59.5t54 34.5 l502 73l225 455q20 41 49 41q28 0 49 -41l225 -455l502 -73q45 -7 54 -34.5t-24 -59.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1401 1187l-640 -1280q-17 -35 -57 -35q-5 0 -15 2q-22 5 -35.5 22.5t-13.5 39.5v576h-576q-22 0 -39.5 13.5t-22.5 35.5t4 42t29 30l1280 640q13 7 29 7q27 0 45 -19q15 -14 18.5 -34.5t-6.5 -39.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M557 256h595v595zM512 301l595 595h-595v-595zM1664 224v-192q0 -14 -9 -23t-23 -9h-224v-224q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v224h-864q-14 0 -23 9t-9 23v864h-224q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h224v224q0 14 9 23t23 9h192q14 0 23 -9t9 -23 v-224h851l246 247q10 9 23 9t23 -9q9 -10 9 -23t-9 -23l-247 -246v-851h224q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M288 64q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM288 1216q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM928 1088q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM1024 1088q0 -52 -26 -96.5t-70 -69.5 q-2 -287 -226 -414q-68 -38 -203 -81q-128 -40 -169.5 -71t-41.5 -100v-26q44 -25 70 -69.5t26 -96.5q0 -80 -56 -136t-136 -56t-136 56t-56 136q0 52 26 96.5t70 69.5v820q-44 25 -70 69.5t-26 96.5q0 80 56 136t136 56t136 -56t56 -136q0 -52 -26 -96.5t-70 -69.5v-497 q54 26 154 57q55 17 87.5 29.5t70.5 31t59 39.5t40.5 51t28 69.5t8.5 91.5q-44 25 -70 69.5t-26 96.5q0 80 56 136t136 56t136 -56t56 -136z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M439 265l-256 -256q-10 -9 -23 -9q-12 0 -23 9q-9 10 -9 23t9 23l256 256q10 9 23 9t23 -9q9 -10 9 -23t-9 -23zM608 224v-320q0 -14 -9 -23t-23 -9t-23 9t-9 23v320q0 14 9 23t23 9t23 -9t9 -23zM384 448q0 -14 -9 -23t-23 -9h-320q-14 0 -23 9t-9 23t9 23t23 9h320 q14 0 23 -9t9 -23zM1648 320q0 -120 -85 -203l-147 -146q-83 -83 -203 -83q-121 0 -204 85l-334 335q-21 21 -42 56l239 18l273 -274q27 -27 68 -27.5t68 26.5l147 146q28 28 28 67q0 40 -28 68l-274 275l18 239q35 -21 56 -42l336 -336q84 -86 84 -204zM1031 1044l-239 -18 l-273 274q-28 28 -68 28q-39 0 -68 -27l-147 -146q-28 -28 -28 -67q0 -40 28 -68l274 -274l-18 -240q-35 21 -56 42l-336 336q-84 86 -84 204q0 120 85 203l147 146q83 83 203 83q121 0 204 -85l334 -335q21 -21 42 -56zM1664 960q0 -14 -9 -23t-23 -9h-320q-14 0 -23 9 t-9 23t9 23t23 9h320q14 0 23 -9t9 -23zM1120 1504v-320q0 -14 -9 -23t-23 -9t-23 9t-9 23v320q0 14 9 23t23 9t23 -9t9 -23zM1527 1353l-256 -256q-11 -9 -23 -9t-23 9q-9 10 -9 23t9 23l256 256q10 9 23 9t23 -9q9 -10 9 -23t-9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M704 280v-240q0 -16 -12 -28t-28 -12h-240q-16 0 -28 12t-12 28v240q0 16 12 28t28 12h240q16 0 28 -12t12 -28zM1020 880q0 -54 -15.5 -101t-35 -76.5t-55 -59.5t-57.5 -43.5t-61 -35.5q-41 -23 -68.5 -65t-27.5 -67q0 -17 -12 -32.5t-28 -15.5h-240q-15 0 -25.5 18.5 t-10.5 37.5v45q0 83 65 156.5t143 108.5q59 27 84 56t25 76q0 42 -46.5 74t-107.5 32q-65 0 -108 -29q-35 -25 -107 -115q-13 -16 -31 -16q-12 0 -25 8l-164 125q-13 10 -15.5 25t5.5 28q160 266 464 266q80 0 161 -31t146 -83t106 -127.5t41 -158.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="640" d="M640 192v-128q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h64v384h-64q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h384q26 0 45 -19t19 -45v-576h64q26 0 45 -19t19 -45zM512 1344v-192q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v192 q0 26 19 45t45 19h256q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="640" d="M512 288v-224q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v224q0 26 19 45t45 19h256q26 0 45 -19t19 -45zM542 1344l-28 -768q-1 -26 -20.5 -45t-45.5 -19h-256q-26 0 -45.5 19t-20.5 45l-28 768q-1 26 17.5 45t44.5 19h320q26 0 44.5 -19t17.5 -45z" /> | ||||||
|  | <glyph unicode="" d="M897 167v-167h-248l-159 252l-24 42q-8 9 -11 21h-3l-9 -21q-10 -20 -25 -44l-155 -250h-258v167h128l197 291l-185 272h-137v168h276l139 -228q2 -4 23 -42q8 -9 11 -21h3q3 9 11 21l25 42l140 228h257v-168h-125l-184 -267l204 -296h109zM1534 846v-206h-514l-3 27 q-4 28 -4 46q0 64 26 117t65 86.5t84 65t84 54.5t65 54t26 64q0 38 -29.5 62.5t-70.5 24.5q-51 0 -97 -39q-14 -11 -36 -38l-105 92q26 37 63 66q83 65 188 65q110 0 178 -59.5t68 -158.5q0 -56 -24.5 -103t-62 -76.5t-81.5 -58.5t-82 -50.5t-65.5 -51.5t-30.5 -63h232v80 h126z" /> | ||||||
|  | <glyph unicode="" d="M897 167v-167h-248l-159 252l-24 42q-8 9 -11 21h-3l-9 -21q-10 -20 -25 -44l-155 -250h-258v167h128l197 291l-185 272h-137v168h276l139 -228q2 -4 23 -42q8 -9 11 -21h3q3 9 11 21l25 42l140 228h257v-168h-125l-184 -267l204 -296h109zM1536 -50v-206h-514l-4 27 q-3 45 -3 46q0 64 26 117t65 86.5t84 65t84 54.5t65 54t26 64q0 38 -29.5 62.5t-70.5 24.5q-51 0 -97 -39q-14 -11 -36 -38l-105 92q26 37 63 66q80 65 188 65q110 0 178 -59.5t68 -158.5q0 -66 -34.5 -118.5t-84 -86t-99.5 -62.5t-87 -63t-41 -73h232v80h126z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M896 128l336 384h-768l-336 -384h768zM1909 1205q15 -34 9.5 -71.5t-30.5 -65.5l-896 -1024q-38 -44 -96 -44h-768q-38 0 -69.5 20.5t-47.5 54.5q-15 34 -9.5 71.5t30.5 65.5l896 1024q38 44 96 44h768q38 0 69.5 -20.5t47.5 -54.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1664 438q0 -81 -44.5 -135t-123.5 -54q-41 0 -77.5 17.5t-59 38t-56.5 38t-71 17.5q-110 0 -110 -124q0 -39 16 -115t15 -115v-5q-22 0 -33 -1q-34 -3 -97.5 -11.5t-115.5 -13.5t-98 -5q-61 0 -103 26.5t-42 83.5q0 37 17.5 71t38 56.5t38 59t17.5 77.5q0 79 -54 123.5 t-135 44.5q-84 0 -143 -45.5t-59 -127.5q0 -43 15 -83t33.5 -64.5t33.5 -53t15 -50.5q0 -45 -46 -89q-37 -35 -117 -35q-95 0 -245 24q-9 2 -27.5 4t-27.5 4l-13 2q-1 0 -3 1q-2 0 -2 1v1024q2 -1 17.5 -3.5t34 -5t21.5 -3.5q150 -24 245 -24q80 0 117 35q46 44 46 89 q0 22 -15 50.5t-33.5 53t-33.5 64.5t-15 83q0 82 59 127.5t144 45.5q80 0 134 -44.5t54 -123.5q0 -41 -17.5 -77.5t-38 -59t-38 -56.5t-17.5 -71q0 -57 42 -83.5t103 -26.5q64 0 180 15t163 17v-2q-1 -2 -3.5 -17.5t-5 -34t-3.5 -21.5q-24 -150 -24 -245q0 -80 35 -117 q44 -46 89 -46q22 0 50.5 15t53 33.5t64.5 33.5t83 15q82 0 127.5 -59t45.5 -143z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M1152 832v-128q0 -221 -147.5 -384.5t-364.5 -187.5v-132h256q26 0 45 -19t19 -45t-19 -45t-45 -19h-640q-26 0 -45 19t-19 45t19 45t45 19h256v132q-217 24 -364.5 187.5t-147.5 384.5v128q0 26 19 45t45 19t45 -19t19 -45v-128q0 -185 131.5 -316.5t316.5 -131.5 t316.5 131.5t131.5 316.5v128q0 26 19 45t45 19t45 -19t19 -45zM896 1216v-512q0 -132 -94 -226t-226 -94t-226 94t-94 226v512q0 132 94 226t226 94t226 -94t94 -226z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M271 591l-101 -101q-42 103 -42 214v128q0 26 19 45t45 19t45 -19t19 -45v-128q0 -53 15 -113zM1385 1193l-361 -361v-128q0 -132 -94 -226t-226 -94q-55 0 -109 19l-96 -96q97 -51 205 -51q185 0 316.5 131.5t131.5 316.5v128q0 26 19 45t45 19t45 -19t19 -45v-128 q0 -221 -147.5 -384.5t-364.5 -187.5v-132h256q26 0 45 -19t19 -45t-19 -45t-45 -19h-640q-26 0 -45 19t-19 45t19 45t45 19h256v132q-125 13 -235 81l-254 -254q-10 -10 -23 -10t-23 10l-82 82q-10 10 -10 23t10 23l1234 1234q10 10 23 10t23 -10l82 -82q10 -10 10 -23 t-10 -23zM1005 1325l-621 -621v512q0 132 94 226t226 94q102 0 184.5 -59t116.5 -152z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1280" d="M1088 576v640h-448v-1137q119 63 213 137q235 184 235 360zM1280 1344v-768q0 -86 -33.5 -170.5t-83 -150t-118 -127.5t-126.5 -103t-121 -77.5t-89.5 -49.5t-42.5 -20q-12 -6 -26 -6t-26 6q-16 7 -42.5 20t-89.5 49.5t-121 77.5t-126.5 103t-118 127.5t-83 150 t-33.5 170.5v768q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M128 -128h1408v1024h-1408v-1024zM512 1088v288q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-288q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1280 1088v288q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-288q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1664 1152v-1280 q0 -52 -38 -90t-90 -38h-1408q-52 0 -90 38t-38 90v1280q0 52 38 90t90 38h128v96q0 66 47 113t113 47h64q66 0 113 -47t47 -113v-96h384v96q0 66 47 113t113 47h64q66 0 113 -47t47 -113v-96h128q52 0 90 -38t38 -90z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M512 1344q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 1376v-320q0 -16 -12 -25q-8 -7 -20 -7q-4 0 -7 1l-448 96q-11 2 -18 11t-7 20h-256v-102q111 -23 183.5 -111t72.5 -203v-800q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v800 q0 106 62.5 190.5t161.5 114.5v111h-32q-59 0 -115 -23.5t-91.5 -53t-66 -66.5t-40.5 -53.5t-14 -24.5q-17 -35 -57 -35q-16 0 -29 7q-23 12 -31.5 37t3.5 49q5 10 14.5 26t37.5 53.5t60.5 70t85 67t108.5 52.5q-25 42 -25 86q0 66 47 113t113 47t113 -47t47 -113 q0 -33 -14 -64h302q0 11 7 20t18 11l448 96q3 1 7 1q12 0 20 -7q12 -9 12 -25z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1440 1088q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM1664 1376q0 -249 -75.5 -430.5t-253.5 -360.5q-81 -80 -195 -176l-20 -379q-2 -16 -16 -26l-384 -224q-7 -4 -16 -4q-12 0 -23 9l-64 64q-13 14 -8 32l85 276l-281 281l-276 -85q-3 -1 -9 -1 q-14 0 -23 9l-64 64q-17 19 -5 39l224 384q10 14 26 16l379 20q96 114 176 195q188 187 358 258t431 71q14 0 24 -9.5t10 -22.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1708 881l-188 -881h-304l181 849q4 21 1 43q-4 20 -16 35q-10 14 -28 24q-18 9 -40 9h-197l-205 -960h-303l204 960h-304l-205 -960h-304l272 1280h1139q157 0 245 -118q86 -116 52 -281z" /> | ||||||
|  | <glyph unicode="" d="M909 141l102 102q19 19 19 45t-19 45l-307 307l307 307q19 19 19 45t-19 45l-102 102q-19 19 -45 19t-45 -19l-454 -454q-19 -19 -19 -45t19 -45l454 -454q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M717 141l454 454q19 19 19 45t-19 45l-454 454q-19 19 -45 19t-45 -19l-102 -102q-19 -19 -19 -45t19 -45l307 -307l-307 -307q-19 -19 -19 -45t19 -45l102 -102q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1165 397l102 102q19 19 19 45t-19 45l-454 454q-19 19 -45 19t-45 -19l-454 -454q-19 -19 -19 -45t19 -45l102 -102q19 -19 45 -19t45 19l307 307l307 -307q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M813 237l454 454q19 19 19 45t-19 45l-102 102q-19 19 -45 19t-45 -19l-307 -307l-307 307q-19 19 -45 19t-45 -19l-102 -102q-19 -19 -19 -45t19 -45l454 -454q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1130 939l16 175h-884l47 -534h612l-22 -228l-197 -53l-196 53l-13 140h-175l22 -278l362 -100h4v1l359 99l50 544h-644l-15 181h674zM0 1408h1408l-128 -1438l-578 -162l-574 162z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M275 1408h1505l-266 -1333l-804 -267l-698 267l71 356h297l-29 -147l422 -161l486 161l68 339h-1208l58 297h1209l38 191h-1208z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M960 1280q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1792 352v-352q0 -22 -20 -30q-8 -2 -12 -2q-13 0 -23 9l-93 93q-119 -143 -318.5 -226.5t-429.5 -83.5t-429.5 83.5t-318.5 226.5l-93 -93q-9 -9 -23 -9q-4 0 -12 2q-20 8 -20 30v352 q0 14 9 23t23 9h352q22 0 30 -20q8 -19 -7 -35l-100 -100q67 -91 189.5 -153.5t271.5 -82.5v647h-192q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h192v163q-58 34 -93 92.5t-35 128.5q0 106 75 181t181 75t181 -75t75 -181q0 -70 -35 -128.5t-93 -92.5v-163h192q26 0 45 -19 t19 -45v-128q0 -26 -19 -45t-45 -19h-192v-647q149 20 271.5 82.5t189.5 153.5l-100 100q-15 16 -7 35q8 20 30 20h352q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1152" d="M1056 768q40 0 68 -28t28 -68v-576q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v576q0 40 28 68t68 28h32v320q0 185 131.5 316.5t316.5 131.5t316.5 -131.5t131.5 -316.5q0 -26 -19 -45t-45 -19h-64q-26 0 -45 19t-19 45q0 106 -75 181t-181 75t-181 -75t-75 -181 v-320h736z" /> | ||||||
|  | <glyph unicode="" d="M1024 640q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181zM1152 640q0 159 -112.5 271.5t-271.5 112.5t-271.5 -112.5t-112.5 -271.5t112.5 -271.5t271.5 -112.5t271.5 112.5t112.5 271.5zM1280 640q0 -212 -150 -362t-362 -150t-362 150 t-150 362t150 362t362 150t362 -150t150 -362zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M384 800v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM896 800v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM1408 800v-192q0 -40 -28 -68t-68 -28h-192 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="384" d="M384 288v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM384 800v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM384 1312v-192q0 -40 -28 -68t-68 -28h-192 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68z" /> | ||||||
|  | <glyph unicode="" d="M512 256q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM863 162q-13 232 -177 396t-396 177q-14 1 -24 -9t-10 -23v-128q0 -13 8.5 -22t21.5 -10q154 -11 264 -121t121 -264q1 -13 10 -21.5t22 -8.5h128q13 0 23 10 t9 24zM1247 161q-5 154 -56 297.5t-139.5 260t-205 205t-260 139.5t-297.5 56q-14 1 -23 -9q-10 -10 -10 -23v-128q0 -13 9 -22t22 -10q204 -7 378 -111.5t278.5 -278.5t111.5 -378q1 -13 10 -22t22 -9h128q13 0 23 10q11 9 9 23zM1536 1120v-960q0 -119 -84.5 -203.5 t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M768 1408q209 0 385.5 -103t279.5 -279.5t103 -385.5t-103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103zM1152 585q32 18 32 55t-32 55l-544 320q-31 19 -64 1q-32 -19 -32 -56v-640q0 -37 32 -56 q16 -8 32 -8q17 0 32 9z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1024 1084l316 -316l-572 -572l-316 316zM813 105l618 618q19 19 19 45t-19 45l-362 362q-18 18 -45 18t-45 -18l-618 -618q-19 -19 -19 -45t19 -45l362 -362q18 -18 45 -18t45 18zM1702 742l-907 -908q-37 -37 -90.5 -37t-90.5 37l-126 126q56 56 56 136t-56 136 t-136 56t-136 -56l-125 126q-37 37 -37 90.5t37 90.5l907 906q37 37 90.5 37t90.5 -37l125 -125q-56 -56 -56 -136t56 -136t136 -56t136 56l126 -125q37 -37 37 -90.5t-37 -90.5z" /> | ||||||
|  | <glyph unicode="" d="M1280 576v128q0 26 -19 45t-45 19h-896q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h896q26 0 45 19t19 45zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5 t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1152 736v-64q0 -14 -9 -23t-23 -9h-832q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h832q14 0 23 -9t9 -23zM1280 288v832q0 66 -47 113t-113 47h-832q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113zM1408 1120v-832q0 -119 -84.5 -203.5 t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M1018 933q-18 -37 -58 -37h-192v-864q0 -14 -9 -23t-23 -9h-704q-21 0 -29 18q-8 20 4 35l160 192q9 11 25 11h320v640h-192q-40 0 -58 37q-17 37 9 68l320 384q18 22 49 22t49 -22l320 -384q27 -32 9 -68z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M32 1280h704q13 0 22.5 -9.5t9.5 -23.5v-863h192q40 0 58 -37t-9 -69l-320 -384q-18 -22 -49 -22t-49 22l-320 384q-26 31 -9 69q18 37 58 37h192v640h-320q-14 0 -25 11l-160 192q-13 14 -4 34q9 19 29 19z" /> | ||||||
|  | <glyph unicode="" d="M685 237l614 614q19 19 19 45t-19 45l-102 102q-19 19 -45 19t-45 -19l-467 -467l-211 211q-19 19 -45 19t-45 -19l-102 -102q-19 -19 -19 -45t19 -45l358 -358q19 -19 45 -19t45 19zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5 t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M404 428l152 -152l-52 -52h-56v96h-96v56zM818 818q14 -13 -3 -30l-291 -291q-17 -17 -30 -3q-14 13 3 30l291 291q17 17 30 3zM544 128l544 544l-288 288l-544 -544v-288h288zM1152 736l92 92q28 28 28 68t-28 68l-152 152q-28 28 -68 28t-68 -28l-92 -92zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M1280 608v480q0 26 -19 45t-45 19h-480q-42 0 -59 -39q-17 -41 14 -70l144 -144l-534 -534q-19 -19 -19 -45t19 -45l102 -102q19 -19 45 -19t45 19l534 534l144 -144q18 -19 45 -19q12 0 25 5q39 17 39 59zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960 q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M1005 435l352 352q19 19 19 45t-19 45l-352 352q-30 31 -69 14q-40 -17 -40 -59v-160q-119 0 -216 -19.5t-162.5 -51t-114 -79t-76.5 -95.5t-44.5 -109t-21.5 -111.5t-5 -110.5q0 -181 167 -404q10 -12 25 -12q7 0 13 3q22 9 19 33q-44 354 62 473q46 52 130 75.5 t224 23.5v-160q0 -42 40 -59q12 -5 24 -5q26 0 45 19zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M640 448l256 128l-256 128v-256zM1024 1039v-542l-512 -256v542zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103 t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1145 861q18 -35 -5 -66l-320 -448q-19 -27 -52 -27t-52 27l-320 448q-23 31 -5 66q17 35 57 35h640q40 0 57 -35zM1280 160v960q0 13 -9.5 22.5t-22.5 9.5h-960q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h960q13 0 22.5 9.5t9.5 22.5zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M1145 419q-17 -35 -57 -35h-640q-40 0 -57 35q-18 35 5 66l320 448q19 27 52 27t52 -27l320 -448q23 -31 5 -66zM1280 160v960q0 13 -9.5 22.5t-22.5 9.5h-960q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h960q13 0 22.5 9.5t9.5 22.5zM1536 1120v-960 q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M1088 640q0 -33 -27 -52l-448 -320q-31 -23 -66 -5q-35 17 -35 57v640q0 40 35 57q35 18 66 -5l448 -320q27 -19 27 -52zM1280 160v960q0 14 -9 23t-23 9h-960q-14 0 -23 -9t-9 -23v-960q0 -14 9 -23t23 -9h960q14 0 23 9t9 23zM1536 1120v-960q0 -119 -84.5 -203.5 t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M976 229l35 -159q3 -12 -3 -22.5t-17 -14.5l-5 -1q-4 -2 -10.5 -3.5t-16 -4.5t-21.5 -5.5t-25.5 -5t-30 -5t-33.5 -4.5t-36.5 -3t-38.5 -1q-234 0 -409 130.5t-238 351.5h-95q-13 0 -22.5 9.5t-9.5 22.5v113q0 13 9.5 22.5t22.5 9.5h66q-2 57 1 105h-67q-14 0 -23 9 t-9 23v114q0 14 9 23t23 9h98q67 210 243.5 338t400.5 128q102 0 194 -23q11 -3 20 -15q6 -11 3 -24l-43 -159q-3 -13 -14 -19.5t-24 -2.5l-4 1q-4 1 -11.5 2.5l-17.5 3.5t-22.5 3.5t-26 3t-29 2.5t-29.5 1q-126 0 -226 -64t-150 -176h468q16 0 25 -12q10 -12 7 -26 l-24 -114q-5 -26 -32 -26h-488q-3 -37 0 -105h459q15 0 25 -12q9 -12 6 -27l-24 -112q-2 -11 -11 -18.5t-20 -7.5h-387q48 -117 149.5 -185.5t228.5 -68.5q18 0 36 1.5t33.5 3.5t29.5 4.5t24.5 5t18.5 4.5l12 3l5 2q13 5 26 -2q12 -7 15 -21z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M1020 399v-367q0 -14 -9 -23t-23 -9h-956q-14 0 -23 9t-9 23v150q0 13 9.5 22.5t22.5 9.5h97v383h-95q-14 0 -23 9.5t-9 22.5v131q0 14 9 23t23 9h95v223q0 171 123.5 282t314.5 111q185 0 335 -125q9 -8 10 -20.5t-7 -22.5l-103 -127q-9 -11 -22 -12q-13 -2 -23 7 q-5 5 -26 19t-69 32t-93 18q-85 0 -137 -47t-52 -123v-215h305q13 0 22.5 -9t9.5 -23v-131q0 -13 -9.5 -22.5t-22.5 -9.5h-305v-379h414v181q0 13 9 22.5t23 9.5h162q14 0 23 -9.5t9 -22.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M978 351q0 -153 -99.5 -263.5t-258.5 -136.5v-175q0 -14 -9 -23t-23 -9h-135q-13 0 -22.5 9.5t-9.5 22.5v175q-66 9 -127.5 31t-101.5 44.5t-74 48t-46.5 37.5t-17.5 18q-17 21 -2 41l103 135q7 10 23 12q15 2 24 -9l2 -2q113 -99 243 -125q37 -8 74 -8q81 0 142.5 43 t61.5 122q0 28 -15 53t-33.5 42t-58.5 37.5t-66 32t-80 32.5q-39 16 -61.5 25t-61.5 26.5t-62.5 31t-56.5 35.5t-53.5 42.5t-43.5 49t-35.5 58t-21 66.5t-8.5 78q0 138 98 242t255 134v180q0 13 9.5 22.5t22.5 9.5h135q14 0 23 -9t9 -23v-176q57 -6 110.5 -23t87 -33.5 t63.5 -37.5t39 -29t15 -14q17 -18 5 -38l-81 -146q-8 -15 -23 -16q-14 -3 -27 7q-3 3 -14.5 12t-39 26.5t-58.5 32t-74.5 26t-85.5 11.5q-95 0 -155 -43t-60 -111q0 -26 8.5 -48t29.5 -41.5t39.5 -33t56 -31t60.5 -27t70 -27.5q53 -20 81 -31.5t76 -35t75.5 -42.5t62 -50 t53 -63.5t31.5 -76.5t13 -94z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="898" d="M898 1066v-102q0 -14 -9 -23t-23 -9h-168q-23 -144 -129 -234t-276 -110q167 -178 459 -536q14 -16 4 -34q-8 -18 -29 -18h-195q-16 0 -25 12q-306 367 -498 571q-9 9 -9 22v127q0 13 9.5 22.5t22.5 9.5h112q132 0 212.5 43t102.5 125h-427q-14 0 -23 9t-9 23v102 q0 14 9 23t23 9h413q-57 113 -268 113h-145q-13 0 -22.5 9.5t-9.5 22.5v133q0 14 9 23t23 9h832q14 0 23 -9t9 -23v-102q0 -14 -9 -23t-23 -9h-233q47 -61 64 -144h171q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1027" d="M603 0h-172q-13 0 -22.5 9t-9.5 23v330h-288q-13 0 -22.5 9t-9.5 23v103q0 13 9.5 22.5t22.5 9.5h288v85h-288q-13 0 -22.5 9t-9.5 23v104q0 13 9.5 22.5t22.5 9.5h214l-321 578q-8 16 0 32q10 16 28 16h194q19 0 29 -18l215 -425q19 -38 56 -125q10 24 30.5 68t27.5 61 l191 420q8 19 29 19h191q17 0 27 -16q9 -14 1 -31l-313 -579h215q13 0 22.5 -9.5t9.5 -22.5v-104q0 -14 -9.5 -23t-22.5 -9h-290v-85h290q13 0 22.5 -9.5t9.5 -22.5v-103q0 -14 -9.5 -23t-22.5 -9h-290v-330q0 -13 -9.5 -22.5t-22.5 -9.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1664 352v-32q0 -132 -94 -226t-226 -94h-128q-132 0 -226 94t-94 226v480h-224q-2 -102 -14.5 -190.5t-30.5 -156t-48.5 -126.5t-57 -99.5t-67.5 -77.5t-69.5 -58.5t-74 -44t-69 -32t-65.5 -25.5q-4 -2 -32 -13q-8 -2 -12 -2q-22 0 -30 20l-71 178q-5 13 0 25t17 17 q7 3 20 7.5t18 6.5q31 12 46.5 18.5t44.5 20t45.5 26t42 32.5t40.5 42.5t34.5 53.5t30.5 68.5t22.5 83.5t17 103t6.5 123h-256q-14 0 -23 9t-9 23v160q0 14 9 23t23 9h1216q14 0 23 -9t9 -23v-160q0 -14 -9 -23t-23 -9h-224v-512q0 -26 19 -45t45 -19h128q26 0 45 19t19 45 v64q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1280 1376v-160q0 -14 -9 -23t-23 -9h-960q-14 0 -23 9t-9 23v160q0 14 9 23t23 9h960q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M514 341l81 299h-159l75 -300q1 -1 1 -3t1 -3q0 1 0.5 3.5t0.5 3.5zM630 768l35 128h-292l32 -128h225zM822 768h139l-35 128h-70zM1271 340l78 300h-162l81 -299q0 -1 0.5 -3.5t1.5 -3.5q0 1 0.5 3t0.5 3zM1382 768l33 128h-297l34 -128h230zM1792 736v-64q0 -14 -9 -23 t-23 -9h-213l-164 -616q-7 -24 -31 -24h-159q-24 0 -31 24l-166 616h-209l-167 -616q-7 -24 -31 -24h-159q-11 0 -19.5 7t-10.5 17l-160 616h-208q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h175l-33 128h-142q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h109l-89 344q-5 15 5 28 q10 12 26 12h137q26 0 31 -24l90 -360h359l97 360q7 24 31 24h126q24 0 31 -24l98 -360h365l93 360q5 24 31 24h137q16 0 26 -12q10 -13 5 -28l-91 -344h111q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-145l-34 -128h179q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1280" d="M1167 896q18 -182 -131 -258q117 -28 175 -103t45 -214q-7 -71 -32.5 -125t-64.5 -89t-97 -58.5t-121.5 -34.5t-145.5 -15v-255h-154v251q-80 0 -122 1v-252h-154v255q-18 0 -54 0.5t-55 0.5h-200l31 183h111q50 0 58 51v402h16q-6 1 -16 1v287q-13 68 -89 68h-111v164 l212 -1q64 0 97 1v252h154v-247q82 2 122 2v245h154v-252q79 -7 140 -22.5t113 -45t82.5 -78t36.5 -114.5zM952 351q0 36 -15 64t-37 46t-57.5 30.5t-65.5 18.5t-74 9t-69 3t-64.5 -1t-47.5 -1v-338q8 0 37 -0.5t48 -0.5t53 1.5t58.5 4t57 8.5t55.5 14t47.5 21t39.5 30 t24.5 40t9.5 51zM881 827q0 33 -12.5 58.5t-30.5 42t-48 28t-55 16.5t-61.5 8t-58 2.5t-54 -1t-39.5 -0.5v-307q5 0 34.5 -0.5t46.5 0t50 2t55 5.5t51.5 11t48.5 18.5t37 27t27 38.5t9 51z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1280" d="M1280 768v-800q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h544v-544q0 -40 28 -68t68 -28h544zM1277 896h-509v509q82 -15 132 -65l312 -312q50 -50 65 -132z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1280" d="M1024 160v64q0 14 -9 23t-23 9h-704q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h704q14 0 23 9t9 23zM1024 416v64q0 14 -9 23t-23 9h-704q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h704q14 0 23 9t9 23zM1280 768v-800q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28 t-28 68v1344q0 40 28 68t68 28h544v-544q0 -40 28 -68t68 -28h544zM1277 896h-509v509q82 -15 132 -65l312 -312q50 -50 65 -132z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1191 1128h177l-72 218l-12 47q-2 16 -2 20h-4l-3 -20q0 -1 -3.5 -18t-7.5 -29zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23zM1572 -23 v-233h-584v90l369 529q12 18 21 27l11 9v3q-2 0 -6.5 -0.5t-7.5 -0.5q-12 -3 -30 -3h-232v-115h-120v229h567v-89l-369 -530q-6 -8 -21 -26l-11 -11v-2l14 2q9 2 30 2h248v119h121zM1661 874v-106h-288v106h75l-47 144h-243l-47 -144h75v-106h-287v106h70l230 662h162 l230 -662h70z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1191 104h177l-72 218l-12 47q-2 16 -2 20h-4l-3 -20q0 -1 -3.5 -18t-7.5 -29zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23zM1661 -150 v-106h-288v106h75l-47 144h-243l-47 -144h75v-106h-287v106h70l230 662h162l230 -662h70zM1572 1001v-233h-584v90l369 529q12 18 21 27l11 9v3q-2 0 -6.5 -0.5t-7.5 -0.5q-12 -3 -30 -3h-232v-115h-120v229h567v-89l-369 -530q-6 -8 -21 -26l-11 -10v-3l14 3q9 1 30 1h248 v119h121z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23zM1792 -32v-192q0 -14 -9 -23t-23 -9h-832q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h832 q14 0 23 -9t9 -23zM1600 480v-192q0 -14 -9 -23t-23 -9h-640q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h640q14 0 23 -9t9 -23zM1408 992v-192q0 -14 -9 -23t-23 -9h-448q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h448q14 0 23 -9t9 -23zM1216 1504v-192q0 -14 -9 -23t-23 -9h-256 q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h256q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1216 -32v-192q0 -14 -9 -23t-23 -9h-256q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h256q14 0 23 -9t9 -23zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192 q14 0 23 -9t9 -23zM1408 480v-192q0 -14 -9 -23t-23 -9h-448q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h448q14 0 23 -9t9 -23zM1600 992v-192q0 -14 -9 -23t-23 -9h-640q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h640q14 0 23 -9t9 -23zM1792 1504v-192q0 -14 -9 -23t-23 -9h-832 q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h832q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" d="M1346 223q0 63 -44 116t-103 53q-52 0 -83 -37t-31 -94t36.5 -95t104.5 -38q50 0 85 27t35 68zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23 zM1486 165q0 -62 -13 -121.5t-41 -114t-68 -95.5t-98.5 -65.5t-127.5 -24.5q-62 0 -108 16q-24 8 -42 15l39 113q15 -7 31 -11q37 -13 75 -13q84 0 134.5 58.5t66.5 145.5h-2q-21 -23 -61.5 -37t-84.5 -14q-106 0 -173 71.5t-67 172.5q0 105 72 178t181 73q123 0 205 -94.5 t82 -252.5zM1456 882v-114h-469v114h167v432q0 7 0.5 19t0.5 17v16h-2l-7 -12q-8 -13 -26 -31l-62 -58l-82 86l192 185h123v-654h165z" /> | ||||||
|  | <glyph unicode="" d="M1346 1247q0 63 -44 116t-103 53q-52 0 -83 -37t-31 -94t36.5 -95t104.5 -38q50 0 85 27t35 68zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9 t9 -23zM1456 -142v-114h-469v114h167v432q0 7 0.5 19t0.5 17v16h-2l-7 -12q-8 -13 -26 -31l-62 -58l-82 86l192 185h123v-654h165zM1486 1189q0 -62 -13 -121.5t-41 -114t-68 -95.5t-98.5 -65.5t-127.5 -24.5q-62 0 -108 16q-24 8 -42 15l39 113q15 -7 31 -11q37 -13 75 -13 q84 0 134.5 58.5t66.5 145.5h-2q-21 -23 -61.5 -37t-84.5 -14q-106 0 -173 71.5t-67 172.5q0 105 72 178t181 73q123 0 205 -94.5t82 -252.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M256 192q0 26 -19 45t-45 19q-27 0 -45.5 -19t-18.5 -45q0 -27 18.5 -45.5t45.5 -18.5q26 0 45 18.5t19 45.5zM416 704v-640q0 -26 -19 -45t-45 -19h-288q-26 0 -45 19t-19 45v640q0 26 19 45t45 19h288q26 0 45 -19t19 -45zM1600 704q0 -86 -55 -149q15 -44 15 -76 q3 -76 -43 -137q17 -56 0 -117q-15 -57 -54 -94q9 -112 -49 -181q-64 -76 -197 -78h-36h-76h-17q-66 0 -144 15.5t-121.5 29t-120.5 39.5q-123 43 -158 44q-26 1 -45 19.5t-19 44.5v641q0 25 18 43.5t43 20.5q24 2 76 59t101 121q68 87 101 120q18 18 31 48t17.5 48.5 t13.5 60.5q7 39 12.5 61t19.5 52t34 50q19 19 45 19q46 0 82.5 -10.5t60 -26t40 -40.5t24 -45t12 -50t5 -45t0.5 -39q0 -38 -9.5 -76t-19 -60t-27.5 -56q-3 -6 -10 -18t-11 -22t-8 -24h277q78 0 135 -57t57 -135z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M256 960q0 -26 -19 -45t-45 -19q-27 0 -45.5 19t-18.5 45q0 27 18.5 45.5t45.5 18.5q26 0 45 -18.5t19 -45.5zM416 448v640q0 26 -19 45t-45 19h-288q-26 0 -45 -19t-19 -45v-640q0 -26 19 -45t45 -19h288q26 0 45 19t19 45zM1545 597q55 -61 55 -149q-1 -78 -57.5 -135 t-134.5 -57h-277q4 -14 8 -24t11 -22t10 -18q18 -37 27 -57t19 -58.5t10 -76.5q0 -24 -0.5 -39t-5 -45t-12 -50t-24 -45t-40 -40.5t-60 -26t-82.5 -10.5q-26 0 -45 19q-20 20 -34 50t-19.5 52t-12.5 61q-9 42 -13.5 60.5t-17.5 48.5t-31 48q-33 33 -101 120q-49 64 -101 121 t-76 59q-25 2 -43 20.5t-18 43.5v641q0 26 19 44.5t45 19.5q35 1 158 44q77 26 120.5 39.5t121.5 29t144 15.5h17h76h36q133 -2 197 -78q58 -69 49 -181q39 -37 54 -94q17 -61 0 -117q46 -61 43 -137q0 -32 -15 -76z" /> | ||||||
|  | <glyph unicode="" d="M919 233v157q0 50 -29 50q-17 0 -33 -16v-224q16 -16 33 -16q29 0 29 49zM1103 355h66v34q0 51 -33 51t-33 -51v-34zM532 621v-70h-80v-423h-74v423h-78v70h232zM733 495v-367h-67v40q-39 -45 -76 -45q-33 0 -42 28q-6 16 -6 54v290h66v-270q0 -24 1 -26q1 -15 15 -15 q20 0 42 31v280h67zM985 384v-146q0 -52 -7 -73q-12 -42 -53 -42q-35 0 -68 41v-36h-67v493h67v-161q32 40 68 40q41 0 53 -42q7 -21 7 -74zM1236 255v-9q0 -29 -2 -43q-3 -22 -15 -40q-27 -40 -80 -40q-52 0 -81 38q-21 27 -21 86v129q0 59 20 86q29 38 80 38t78 -38 q21 -28 21 -86v-76h-133v-65q0 -51 34 -51q24 0 30 26q0 1 0.5 7t0.5 16.5v21.5h68zM785 1079v-156q0 -51 -32 -51t-32 51v156q0 52 32 52t32 -52zM1318 366q0 177 -19 260q-10 44 -43 73.5t-76 34.5q-136 15 -412 15q-275 0 -411 -15q-44 -5 -76.5 -34.5t-42.5 -73.5 q-20 -87 -20 -260q0 -176 20 -260q10 -43 42.5 -73t75.5 -35q137 -15 412 -15t412 15q43 5 75.5 35t42.5 73q20 84 20 260zM563 1017l90 296h-75l-51 -195l-53 195h-78l24 -69t23 -69q35 -103 46 -158v-201h74v201zM852 936v130q0 58 -21 87q-29 38 -78 38q-51 0 -78 -38 q-21 -29 -21 -87v-130q0 -58 21 -87q27 -38 78 -38q49 0 78 38q21 27 21 87zM1033 816h67v370h-67v-283q-22 -31 -42 -31q-15 0 -16 16q-1 2 -1 26v272h-67v-293q0 -37 6 -55q11 -27 43 -27q36 0 77 45v-40zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960 q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" d="M971 292v-211q0 -67 -39 -67q-23 0 -45 22v301q22 22 45 22q39 0 39 -67zM1309 291v-46h-90v46q0 68 45 68t45 -68zM343 509h107v94h-312v-94h105v-569h100v569zM631 -60h89v494h-89v-378q-30 -42 -57 -42q-18 0 -21 21q-1 3 -1 35v364h-89v-391q0 -49 8 -73 q12 -37 58 -37q48 0 102 61v-54zM1060 88v197q0 73 -9 99q-17 56 -71 56q-50 0 -93 -54v217h-89v-663h89v48q45 -55 93 -55q54 0 71 55q9 27 9 100zM1398 98v13h-91q0 -51 -2 -61q-7 -36 -40 -36q-46 0 -46 69v87h179v103q0 79 -27 116q-39 51 -106 51q-68 0 -107 -51 q-28 -37 -28 -116v-173q0 -79 29 -116q39 -51 108 -51q72 0 108 53q18 27 21 54q2 9 2 58zM790 1011v210q0 69 -43 69t-43 -69v-210q0 -70 43 -70t43 70zM1509 260q0 -234 -26 -350q-14 -59 -58 -99t-102 -46q-184 -21 -555 -21t-555 21q-58 6 -102.5 46t-57.5 99 q-26 112 -26 350q0 234 26 350q14 59 58 99t103 47q183 20 554 20t555 -20q58 -7 102.5 -47t57.5 -99q26 -112 26 -350zM511 1536h102l-121 -399v-271h-100v271q-14 74 -61 212q-37 103 -65 187h106l71 -263zM881 1203v-175q0 -81 -28 -118q-37 -51 -106 -51q-67 0 -105 51 q-28 38 -28 118v175q0 80 28 117q38 51 105 51q69 0 106 -51q28 -37 28 -117zM1216 1365v-499h-91v55q-53 -62 -103 -62q-46 0 -59 37q-8 24 -8 75v394h91v-367q0 -33 1 -35q3 -22 21 -22q27 0 57 43v381h91z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M597 869q-10 -18 -257 -456q-27 -46 -65 -46h-239q-21 0 -31 17t0 36l253 448q1 0 0 1l-161 279q-12 22 -1 37q9 15 32 15h239q40 0 66 -45zM1403 1511q11 -16 0 -37l-528 -934v-1l336 -615q11 -20 1 -37q-10 -15 -32 -15h-239q-42 0 -66 45l-339 622q18 32 531 942 q25 45 64 45h241q22 0 31 -15z" /> | ||||||
|  | <glyph unicode="" d="M685 771q0 1 -126 222q-21 34 -52 34h-184q-18 0 -26 -11q-7 -12 1 -29l125 -216v-1l-196 -346q-9 -14 0 -28q8 -13 24 -13h185q31 0 50 36zM1309 1268q-7 12 -24 12h-187q-30 0 -49 -35l-411 -729q1 -2 262 -481q20 -35 52 -35h184q18 0 25 12q8 13 -1 28l-260 476v1 l409 723q8 16 0 28zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1280 640q0 37 -30 54l-512 320q-31 20 -65 2q-33 -18 -33 -56v-640q0 -38 33 -56q16 -8 31 -8q20 0 34 10l512 320q30 17 30 54zM1792 640q0 -96 -1 -150t-8.5 -136.5t-22.5 -147.5q-16 -73 -69 -123t-124 -58q-222 -25 -671 -25t-671 25q-71 8 -124.5 58t-69.5 123 q-14 65 -21.5 147.5t-8.5 136.5t-1 150t1 150t8.5 136.5t22.5 147.5q16 73 69 123t124 58q222 25 671 25t671 -25q71 -8 124.5 -58t69.5 -123q14 -65 21.5 -147.5t8.5 -136.5t1 -150z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M402 829l494 -305l-342 -285l-490 319zM1388 274v-108l-490 -293v-1l-1 1l-1 -1v1l-489 293v108l147 -96l342 284v2l1 -1l1 1v-2l343 -284zM554 1418l342 -285l-494 -304l-338 270zM1390 829l338 -271l-489 -319l-343 285zM1239 1418l489 -319l-338 -270l-494 304z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M928 135v-151l-707 -1v151zM1169 481v-701l-1 -35v-1h-1132l-35 1h-1v736h121v-618h928v618h120zM241 393l704 -65l-13 -150l-705 65zM309 709l683 -183l-39 -146l-683 183zM472 1058l609 -360l-77 -130l-609 360zM832 1389l398 -585l-124 -85l-399 584zM1285 1536 l121 -697l-149 -26l-121 697z" /> | ||||||
|  | <glyph unicode="" d="M1362 110v648h-135q20 -63 20 -131q0 -126 -64 -232.5t-174 -168.5t-240 -62q-197 0 -337 135.5t-140 327.5q0 68 20 131h-141v-648q0 -26 17.5 -43.5t43.5 -17.5h1069q25 0 43 17.5t18 43.5zM1078 643q0 124 -90.5 211.5t-218.5 87.5q-127 0 -217.5 -87.5t-90.5 -211.5 t90.5 -211.5t217.5 -87.5q128 0 218.5 87.5t90.5 211.5zM1362 1003v165q0 28 -20 48.5t-49 20.5h-174q-29 0 -49 -20.5t-20 -48.5v-165q0 -29 20 -49t49 -20h174q29 0 49 20t20 49zM1536 1211v-1142q0 -81 -58 -139t-139 -58h-1142q-81 0 -139 58t-58 139v1142q0 81 58 139 t139 58h1142q81 0 139 -58t58 -139z" /> | ||||||
|  | <glyph unicode="" d="M1248 1408q119 0 203.5 -84.5t84.5 -203.5v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960zM698 640q0 88 -62 150t-150 62t-150 -62t-62 -150t62 -150t150 -62t150 62t62 150zM1262 640q0 88 -62 150 t-150 62t-150 -62t-62 -150t62 -150t150 -62t150 62t62 150z" /> | ||||||
|  | <glyph unicode="" d="M768 914l201 -306h-402zM1133 384h94l-459 691l-459 -691h94l104 160h522zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M815 677q8 -63 -50.5 -101t-111.5 -6q-39 17 -53.5 58t-0.5 82t52 58q36 18 72.5 12t64 -35.5t27.5 -67.5zM926 698q-14 107 -113 164t-197 13q-63 -28 -100.5 -88.5t-34.5 -129.5q4 -91 77.5 -155t165.5 -56q91 8 152 84t50 168zM1165 1240q-20 27 -56 44.5t-58 22 t-71 12.5q-291 47 -566 -2q-43 -7 -66 -12t-55 -22t-50 -43q30 -28 76 -45.5t73.5 -22t87.5 -11.5q228 -29 448 -1q63 8 89.5 12t72.5 21.5t75 46.5zM1222 205q-8 -26 -15.5 -76.5t-14 -84t-28.5 -70t-58 -56.5q-86 -48 -189.5 -71.5t-202 -22t-201.5 18.5q-46 8 -81.5 18 t-76.5 27t-73 43.5t-52 61.5q-25 96 -57 292l6 16l18 9q223 -148 506.5 -148t507.5 148q21 -6 24 -23t-5 -45t-8 -37zM1403 1166q-26 -167 -111 -655q-5 -30 -27 -56t-43.5 -40t-54.5 -31q-252 -126 -610 -88q-248 27 -394 139q-15 12 -25.5 26.5t-17 35t-9 34t-6 39.5 t-5.5 35q-9 50 -26.5 150t-28 161.5t-23.5 147.5t-22 158q3 26 17.5 48.5t31.5 37.5t45 30t46 22.5t48 18.5q125 46 313 64q379 37 676 -50q155 -46 215 -122q16 -20 16.5 -51t-5.5 -54z" /> | ||||||
|  | <glyph unicode="" d="M848 666q0 43 -41 66t-77 1q-43 -20 -42.5 -72.5t43.5 -70.5q39 -23 81 4t36 72zM928 682q8 -66 -36 -121t-110 -61t-119 40t-56 113q-2 49 25.5 93t72.5 64q70 31 141.5 -10t81.5 -118zM1100 1073q-20 -21 -53.5 -34t-53 -16t-63.5 -8q-155 -20 -324 0q-44 6 -63 9.5 t-52.5 16t-54.5 32.5q13 19 36 31t40 15.5t47 8.5q198 35 408 1q33 -5 51 -8.5t43 -16t39 -31.5zM1142 327q0 7 5.5 26.5t3 32t-17.5 16.5q-161 -106 -365 -106t-366 106l-12 -6l-5 -12q26 -154 41 -210q47 -81 204 -108q249 -46 428 53q34 19 49 51.5t22.5 85.5t12.5 71z M1272 1020q9 53 -8 75q-43 55 -155 88q-216 63 -487 36q-132 -12 -226 -46q-38 -15 -59.5 -25t-47 -34t-29.5 -54q8 -68 19 -138t29 -171t24 -137q1 -5 5 -31t7 -36t12 -27t22 -28q105 -80 284 -100q259 -28 440 63q24 13 39.5 23t31 29t19.5 40q48 267 80 473zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M390 1408h219v-388h364v-241h-364v-394q0 -136 14 -172q13 -37 52 -60q50 -31 117 -31q117 0 232 76v-242q-102 -48 -178 -65q-77 -19 -173 -19q-105 0 -186 27q-78 25 -138 75q-58 51 -79 105q-22 54 -22 161v539h-170v217q91 30 155 84q64 55 103 132q39 78 54 196z " /> | ||||||
|  | <glyph unicode="" d="M1123 127v181q-88 -56 -174 -56q-51 0 -88 23q-29 17 -39 45q-11 30 -11 129v295h274v181h-274v291h-164q-11 -90 -40 -147t-78 -99q-48 -40 -116 -63v-163h127v-404q0 -78 17 -121q17 -42 59 -78q43 -37 104 -57q62 -20 140 -20q67 0 129 14q57 13 134 49zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="768" d="M765 237q8 -19 -5 -35l-350 -384q-10 -10 -23 -10q-14 0 -24 10l-355 384q-13 16 -5 35q9 19 29 19h224v1248q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1248h224q21 0 29 -19z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="768" d="M765 1043q-9 -19 -29 -19h-224v-1248q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v1248h-224q-21 0 -29 19t5 35l350 384q10 10 23 10q14 0 24 -10l355 -384q13 -16 5 -35z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1792 736v-192q0 -14 -9 -23t-23 -9h-1248v-224q0 -21 -19 -29t-35 5l-384 350q-10 10 -10 23q0 14 10 24l384 354q16 14 35 6q19 -9 19 -29v-224h1248q14 0 23 -9t9 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1728 643q0 -14 -10 -24l-384 -354q-16 -14 -35 -6q-19 9 -19 29v224h-1248q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h1248v224q0 21 19 29t35 -5l384 -350q10 -10 10 -23z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M1393 321q-39 -125 -123 -250q-129 -196 -257 -196q-49 0 -140 32q-86 32 -151 32q-61 0 -142 -33q-81 -34 -132 -34q-152 0 -301 259q-147 261 -147 503q0 228 113 374q112 144 284 144q72 0 177 -30q104 -30 138 -30q45 0 143 34q102 34 173 34q119 0 213 -65 q52 -36 104 -100q-79 -67 -114 -118q-65 -94 -65 -207q0 -124 69 -223t158 -126zM1017 1494q0 -61 -29 -136q-30 -75 -93 -138q-54 -54 -108 -72q-37 -11 -104 -17q3 149 78 257q74 107 250 148q1 -3 2.5 -11t2.5 -11q0 -4 0.5 -10t0.5 -10z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M682 530v-651l-682 94v557h682zM682 1273v-659h-682v565zM1664 530v-786l-907 125v661h907zM1664 1408v-794h-907v669z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1408" d="M493 1053q16 0 27.5 11.5t11.5 27.5t-11.5 27.5t-27.5 11.5t-27 -11.5t-11 -27.5t11 -27.5t27 -11.5zM915 1053q16 0 27 11.5t11 27.5t-11 27.5t-27 11.5t-27.5 -11.5t-11.5 -27.5t11.5 -27.5t27.5 -11.5zM103 869q42 0 72 -30t30 -72v-430q0 -43 -29.5 -73t-72.5 -30 t-73 30t-30 73v430q0 42 30 72t73 30zM1163 850v-666q0 -46 -32 -78t-77 -32h-75v-227q0 -43 -30 -73t-73 -30t-73 30t-30 73v227h-138v-227q0 -43 -30 -73t-73 -30q-42 0 -72 30t-30 73l-1 227h-74q-46 0 -78 32t-32 78v666h918zM931 1255q107 -55 171 -153.5t64 -215.5 h-925q0 117 64 215.5t172 153.5l-71 131q-7 13 5 20q13 6 20 -6l72 -132q95 42 201 42t201 -42l72 132q7 12 20 6q12 -7 5 -20zM1408 767v-430q0 -43 -30 -73t-73 -30q-42 0 -72 30t-30 73v430q0 43 30 72.5t72 29.5q43 0 73 -29.5t30 -72.5z" /> | ||||||
|  | <glyph unicode="" d="M663 1125q-11 -1 -15.5 -10.5t-8.5 -9.5q-5 -1 -5 5q0 12 19 15h10zM750 1111q-4 -1 -11.5 6.5t-17.5 4.5q24 11 32 -2q3 -6 -3 -9zM399 684q-4 1 -6 -3t-4.5 -12.5t-5.5 -13.5t-10 -13q-7 -10 -1 -12q4 -1 12.5 7t12.5 18q1 3 2 7t2 6t1.5 4.5t0.5 4v3t-1 2.5t-3 2z M1254 325q0 18 -55 42q4 15 7.5 27.5t5 26t3 21.5t0.5 22.5t-1 19.5t-3.5 22t-4 20.5t-5 25t-5.5 26.5q-10 48 -47 103t-72 75q24 -20 57 -83q87 -162 54 -278q-11 -40 -50 -42q-31 -4 -38.5 18.5t-8 83.5t-11.5 107q-9 39 -19.5 69t-19.5 45.5t-15.5 24.5t-13 15t-7.5 7 q-14 62 -31 103t-29.5 56t-23.5 33t-15 40q-4 21 6 53.5t4.5 49.5t-44.5 25q-15 3 -44.5 18t-35.5 16q-8 1 -11 26t8 51t36 27q37 3 51 -30t4 -58q-11 -19 -2 -26.5t30 -0.5q13 4 13 36v37q-5 30 -13.5 50t-21 30.5t-23.5 15t-27 7.5q-107 -8 -89 -134q0 -15 -1 -15 q-9 9 -29.5 10.5t-33 -0.5t-15.5 5q1 57 -16 90t-45 34q-27 1 -41.5 -27.5t-16.5 -59.5q-1 -15 3.5 -37t13 -37.5t15.5 -13.5q10 3 16 14q4 9 -7 8q-7 0 -15.5 14.5t-9.5 33.5q-1 22 9 37t34 14q17 0 27 -21t9.5 -39t-1.5 -22q-22 -15 -31 -29q-8 -12 -27.5 -23.5 t-20.5 -12.5q-13 -14 -15.5 -27t7.5 -18q14 -8 25 -19.5t16 -19t18.5 -13t35.5 -6.5q47 -2 102 15q2 1 23 7t34.5 10.5t29.5 13t21 17.5q9 14 20 8q5 -3 6.5 -8.5t-3 -12t-16.5 -9.5q-20 -6 -56.5 -21.5t-45.5 -19.5q-44 -19 -70 -23q-25 -5 -79 2q-10 2 -9 -2t17 -19 q25 -23 67 -22q17 1 36 7t36 14t33.5 17.5t30 17t24.5 12t17.5 2.5t8.5 -11q0 -2 -1 -4.5t-4 -5t-6 -4.5t-8.5 -5t-9 -4.5t-10 -5t-9.5 -4.5q-28 -14 -67.5 -44t-66.5 -43t-49 -1q-21 11 -63 73q-22 31 -25 22q-1 -3 -1 -10q0 -25 -15 -56.5t-29.5 -55.5t-21 -58t11.5 -63 q-23 -6 -62.5 -90t-47.5 -141q-2 -18 -1.5 -69t-5.5 -59q-8 -24 -29 -3q-32 31 -36 94q-2 28 4 56q4 19 -1 18l-4 -5q-36 -65 10 -166q5 -12 25 -28t24 -20q20 -23 104 -90.5t93 -76.5q16 -15 17.5 -38t-14 -43t-45.5 -23q8 -15 29 -44.5t28 -54t7 -70.5q46 24 7 92 q-4 8 -10.5 16t-9.5 12t-2 6q3 5 13 9.5t20 -2.5q46 -52 166 -36q133 15 177 87q23 38 34 30q12 -6 10 -52q-1 -25 -23 -92q-9 -23 -6 -37.5t24 -15.5q3 19 14.5 77t13.5 90q2 21 -6.5 73.5t-7.5 97t23 70.5q15 18 51 18q1 37 34.5 53t72.5 10.5t60 -22.5zM626 1152 q3 17 -2.5 30t-11.5 15q-9 2 -9 -7q2 -5 5 -6q10 0 7 -15q-3 -20 8 -20q3 0 3 3zM1045 955q-2 8 -6.5 11.5t-13 5t-14.5 5.5q-5 3 -9.5 8t-7 8t-5.5 6.5t-4 4t-4 -1.5q-14 -16 7 -43.5t39 -31.5q9 -1 14.5 8t3.5 20zM867 1168q0 11 -5 19.5t-11 12.5t-9 3q-14 -1 -7 -7l4 -2 q14 -4 18 -31q0 -3 8 2zM921 1401q0 2 -2.5 5t-9 7t-9.5 6q-15 15 -24 15q-9 -1 -11.5 -7.5t-1 -13t-0.5 -12.5q-1 -4 -6 -10.5t-6 -9t3 -8.5q4 -3 8 0t11 9t15 9q1 1 9 1t15 2t9 7zM1486 60q20 -12 31 -24.5t12 -24t-2.5 -22.5t-15.5 -22t-23.5 -19.5t-30 -18.5 t-31.5 -16.5t-32 -15.5t-27 -13q-38 -19 -85.5 -56t-75.5 -64q-17 -16 -68 -19.5t-89 14.5q-18 9 -29.5 23.5t-16.5 25.5t-22 19.5t-47 9.5q-44 1 -130 1q-19 0 -57 -1.5t-58 -2.5q-44 -1 -79.5 -15t-53.5 -30t-43.5 -28.5t-53.5 -11.5q-29 1 -111 31t-146 43q-19 4 -51 9.5 t-50 9t-39.5 9.5t-33.5 14.5t-17 19.5q-10 23 7 66.5t18 54.5q1 16 -4 40t-10 42.5t-4.5 36.5t10.5 27q14 12 57 14t60 12q30 18 42 35t12 51q21 -73 -32 -106q-32 -20 -83 -15q-34 3 -43 -10q-13 -15 5 -57q2 -6 8 -18t8.5 -18t4.5 -17t1 -22q0 -15 -17 -49t-14 -48 q3 -17 37 -26q20 -6 84.5 -18.5t99.5 -20.5q24 -6 74 -22t82.5 -23t55.5 -4q43 6 64.5 28t23 48t-7.5 58.5t-19 52t-20 36.5q-121 190 -169 242q-68 74 -113 40q-11 -9 -15 15q-3 16 -2 38q1 29 10 52t24 47t22 42q8 21 26.5 72t29.5 78t30 61t39 54q110 143 124 195 q-12 112 -16 310q-2 90 24 151.5t106 104.5q39 21 104 21q53 1 106 -13.5t89 -41.5q57 -42 91.5 -121.5t29.5 -147.5q-5 -95 30 -214q34 -113 133 -218q55 -59 99.5 -163t59.5 -191q8 -49 5 -84.5t-12 -55.5t-20 -22q-10 -2 -23.5 -19t-27 -35.5t-40.5 -33.5t-61 -14 q-18 1 -31.5 5t-22.5 13.5t-13.5 15.5t-11.5 20.5t-9 19.5q-22 37 -41 30t-28 -49t7 -97q20 -70 1 -195q-10 -65 18 -100.5t73 -33t85 35.5q59 49 89.5 66.5t103.5 42.5q53 18 77 36.5t18.5 34.5t-25 28.5t-51.5 23.5q-33 11 -49.5 48t-15 72.5t15.5 47.5q1 -31 8 -56.5 t14.5 -40.5t20.5 -28.5t21 -19t21.5 -13t16.5 -9.5z" /> | ||||||
|  | <glyph unicode="" d="M1024 36q-42 241 -140 498h-2l-2 -1q-16 -6 -43 -16.5t-101 -49t-137 -82t-131 -114.5t-103 -148l-15 11q184 -150 418 -150q132 0 256 52zM839 643q-21 49 -53 111q-311 -93 -673 -93q-1 -7 -1 -21q0 -124 44 -236.5t124 -201.5q50 89 123.5 166.5t142.5 124.5t130.5 81 t99.5 48l37 13q4 1 13 3.5t13 4.5zM732 855q-120 213 -244 378q-138 -65 -234 -186t-128 -272q302 0 606 80zM1416 536q-210 60 -409 29q87 -239 128 -469q111 75 185 189.5t96 250.5zM611 1277q-1 0 -2 -1q1 1 2 1zM1201 1132q-185 164 -433 164q-76 0 -155 -19 q131 -170 246 -382q69 26 130 60.5t96.5 61.5t65.5 57t37.5 40.5zM1424 647q-3 232 -149 410l-1 -1q-9 -12 -19 -24.5t-43.5 -44.5t-71 -60.5t-100 -65t-131.5 -64.5q25 -53 44 -95q2 -6 6.5 -17.5t7.5 -16.5q36 5 74.5 7t73.5 2t69 -1.5t64 -4t56.5 -5.5t48 -6.5t36.5 -6 t25 -4.5zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" d="M1173 473q0 50 -19.5 91.5t-48.5 68.5t-73 49t-82.5 34t-87.5 23l-104 24q-30 7 -44 10.5t-35 11.5t-30 16t-16.5 21t-7.5 30q0 77 144 77q43 0 77 -12t54 -28.5t38 -33.5t40 -29t48 -12q47 0 75.5 32t28.5 77q0 55 -56 99.5t-142 67.5t-182 23q-68 0 -132 -15.5 t-119.5 -47t-89 -87t-33.5 -128.5q0 -61 19 -106.5t56 -75.5t80 -48.5t103 -32.5l146 -36q90 -22 112 -36q32 -20 32 -60q0 -39 -40 -64.5t-105 -25.5q-51 0 -91.5 16t-65 38.5t-45.5 45t-46 38.5t-54 16q-50 0 -75.5 -30t-25.5 -75q0 -92 122 -157.5t291 -65.5 q73 0 140 18.5t122.5 53.5t88.5 93.5t33 131.5zM1536 256q0 -159 -112.5 -271.5t-271.5 -112.5q-130 0 -234 80q-77 -16 -150 -16q-143 0 -273.5 55.5t-225 150t-150 225t-55.5 273.5q0 73 16 150q-80 104 -80 234q0 159 112.5 271.5t271.5 112.5q130 0 234 -80 q77 16 150 16q143 0 273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -73 -16 -150q80 -104 80 -234z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1483 512l-587 -587q-52 -53 -127.5 -53t-128.5 53l-587 587q-53 53 -53 128t53 128l587 587q53 53 128 53t128 -53l265 -265l-398 -399l-188 188q-42 42 -99 42q-59 0 -100 -41l-120 -121q-42 -40 -42 -99q0 -58 42 -100l406 -408q30 -28 67 -37l6 -4h28q60 0 99 41 l619 619l2 -3q53 -53 53 -128t-53 -128zM1406 1138l120 -120q14 -15 14 -36t-14 -36l-730 -730q-17 -15 -37 -15v0q-4 0 -6 1q-18 2 -30 14l-407 408q-14 15 -14 36t14 35l121 120q13 15 35 15t36 -15l252 -252l574 575q15 15 36 15t36 -15z" /> | ||||||
|  | <glyph unicode="" d="M704 192v1024q0 14 -9 23t-23 9h-480q-14 0 -23 -9t-9 -23v-1024q0 -14 9 -23t23 -9h480q14 0 23 9t9 23zM1376 576v640q0 14 -9 23t-23 9h-480q-14 0 -23 -9t-9 -23v-640q0 -14 9 -23t23 -9h480q14 0 23 9t9 23zM1536 1344v-1408q0 -26 -19 -45t-45 -19h-1408 q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h1408q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1280" d="M1280 480q0 -40 -28 -68t-68 -28q-51 0 -80 43l-227 341h-45v-132l247 -411q9 -15 9 -33q0 -26 -19 -45t-45 -19h-192v-272q0 -46 -33 -79t-79 -33h-160q-46 0 -79 33t-33 79v272h-192q-26 0 -45 19t-19 45q0 18 9 33l247 411v132h-45l-227 -341q-29 -43 -80 -43 q-40 0 -68 28t-28 68q0 29 16 53l256 384q73 107 176 107h384q103 0 176 -107l256 -384q16 -24 16 -53zM864 1280q0 -93 -65.5 -158.5t-158.5 -65.5t-158.5 65.5t-65.5 158.5t65.5 158.5t158.5 65.5t158.5 -65.5t65.5 -158.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1024" d="M1024 832v-416q0 -40 -28 -68t-68 -28t-68 28t-28 68v352h-64v-912q0 -46 -33 -79t-79 -33t-79 33t-33 79v464h-64v-464q0 -46 -33 -79t-79 -33t-79 33t-33 79v912h-64v-352q0 -40 -28 -68t-68 -28t-68 28t-28 68v416q0 80 56 136t136 56h640q80 0 136 -56t56 -136z M736 1280q0 -93 -65.5 -158.5t-158.5 -65.5t-158.5 65.5t-65.5 158.5t65.5 158.5t158.5 65.5t158.5 -65.5t65.5 -158.5z" /> | ||||||
|  | <glyph unicode="" d="M773 234l350 473q16 22 24.5 59t-6 85t-61.5 79q-40 26 -83 25.5t-73.5 -17.5t-54.5 -45q-36 -40 -96 -40q-59 0 -95 40q-24 28 -54.5 45t-73.5 17.5t-84 -25.5q-46 -31 -60.5 -79t-6 -85t24.5 -59zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103 t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1472 640q0 117 -45.5 223.5t-123 184t-184 123t-223.5 45.5t-223.5 -45.5t-184 -123t-123 -184t-45.5 -223.5t45.5 -223.5t123 -184t184 -123t223.5 -45.5t223.5 45.5t184 123t123 184t45.5 223.5zM1748 363q-4 -15 -20 -20l-292 -96v-306q0 -16 -13 -26q-15 -10 -29 -4 l-292 94l-180 -248q-10 -13 -26 -13t-26 13l-180 248l-292 -94q-14 -6 -29 4q-13 10 -13 26v306l-292 96q-16 5 -20 20q-5 17 4 29l180 248l-180 248q-9 13 -4 29q4 15 20 20l292 96v306q0 16 13 26q15 10 29 4l292 -94l180 248q9 12 26 12t26 -12l180 -248l292 94 q14 6 29 -4q13 -10 13 -26v-306l292 -96q16 -5 20 -20q5 -16 -4 -29l-180 -248l180 -248q9 -12 4 -29z" /> | ||||||
|  | <glyph unicode="" d="M1262 233q-54 -9 -110 -9q-182 0 -337 90t-245 245t-90 337q0 192 104 357q-201 -60 -328.5 -229t-127.5 -384q0 -130 51 -248.5t136.5 -204t204 -136.5t248.5 -51q144 0 273.5 61.5t220.5 171.5zM1465 318q-94 -203 -283.5 -324.5t-413.5 -121.5q-156 0 -298 61 t-245 164t-164 245t-61 298q0 153 57.5 292.5t156 241.5t235.5 164.5t290 68.5q44 2 61 -39q18 -41 -15 -72q-86 -78 -131.5 -181.5t-45.5 -218.5q0 -148 73 -273t198 -198t273 -73q118 0 228 51q41 18 72 -13q14 -14 17.5 -34t-4.5 -38z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M1088 704q0 26 -19 45t-45 19h-256q-26 0 -45 -19t-19 -45t19 -45t45 -19h256q26 0 45 19t19 45zM1664 896v-960q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v960q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1728 1344v-256q0 -26 -19 -45t-45 -19h-1536 q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1536q26 0 45 -19t19 -45z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1664" d="M1632 576q0 -26 -19 -45t-45 -19h-224q0 -171 -67 -290l208 -209q19 -19 19 -45t-19 -45q-18 -19 -45 -19t-45 19l-198 197q-5 -5 -15 -13t-42 -28.5t-65 -36.5t-82 -29t-97 -13v896h-128v-896q-51 0 -101.5 13.5t-87 33t-66 39t-43.5 32.5l-15 14l-183 -207 q-20 -21 -48 -21q-24 0 -43 16q-19 18 -20.5 44.5t15.5 46.5l202 227q-58 114 -58 274h-224q-26 0 -45 19t-19 45t19 45t45 19h224v294l-173 173q-19 19 -19 45t19 45t45 19t45 -19l173 -173h844l173 173q19 19 45 19t45 -19t19 -45t-19 -45l-173 -173v-294h224q26 0 45 -19 t19 -45zM1152 1152h-640q0 133 93.5 226.5t226.5 93.5t226.5 -93.5t93.5 -226.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M1917 1016q23 -64 -150 -294q-24 -32 -65 -85q-78 -100 -90 -131q-17 -41 14 -81q17 -21 81 -82h1l1 -1l1 -1l2 -2q141 -131 191 -221q3 -5 6.5 -12.5t7 -26.5t-0.5 -34t-25 -27.5t-59 -12.5l-256 -4q-24 -5 -56 5t-52 22l-20 12q-30 21 -70 64t-68.5 77.5t-61 58 t-56.5 15.5q-3 -1 -8 -3.5t-17 -14.5t-21.5 -29.5t-17 -52t-6.5 -77.5q0 -15 -3.5 -27.5t-7.5 -18.5l-4 -5q-18 -19 -53 -22h-115q-71 -4 -146 16.5t-131.5 53t-103 66t-70.5 57.5l-25 24q-10 10 -27.5 30t-71.5 91t-106 151t-122.5 211t-130.5 272q-6 16 -6 27t3 16l4 6 q15 19 57 19l274 2q12 -2 23 -6.5t16 -8.5l5 -3q16 -11 24 -32q20 -50 46 -103.5t41 -81.5l16 -29q29 -60 56 -104t48.5 -68.5t41.5 -38.5t34 -14t27 5q2 1 5 5t12 22t13.5 47t9.5 81t0 125q-2 40 -9 73t-14 46l-6 12q-25 34 -85 43q-13 2 5 24q17 19 38 30q53 26 239 24 q82 -1 135 -13q20 -5 33.5 -13.5t20.5 -24t10.5 -32t3.5 -45.5t-1 -55t-2.5 -70.5t-1.5 -82.5q0 -11 -1 -42t-0.5 -48t3.5 -40.5t11.5 -39t22.5 -24.5q8 -2 17 -4t26 11t38 34.5t52 67t68 107.5q60 104 107 225q4 10 10 17.5t11 10.5l4 3l5 2.5t13 3t20 0.5l288 2 q39 5 64 -2.5t31 -16.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" d="M675 252q21 34 11 69t-45 50q-34 14 -73 1t-60 -46q-22 -34 -13 -68.5t43 -50.5t74.5 -2.5t62.5 47.5zM769 373q8 13 3.5 26.5t-17.5 18.5q-14 5 -28.5 -0.5t-21.5 -18.5q-17 -31 13 -45q14 -5 29 0.5t22 18.5zM943 266q-45 -102 -158 -150t-224 -12 q-107 34 -147.5 126.5t6.5 187.5q47 93 151.5 139t210.5 19q111 -29 158.5 -119.5t2.5 -190.5zM1255 426q-9 96 -89 170t-208.5 109t-274.5 21q-223 -23 -369.5 -141.5t-132.5 -264.5q9 -96 89 -170t208.5 -109t274.5 -21q223 23 369.5 141.5t132.5 264.5zM1563 422 q0 -68 -37 -139.5t-109 -137t-168.5 -117.5t-226 -83t-270.5 -31t-275 33.5t-240.5 93t-171.5 151t-65 199.5q0 115 69.5 245t197.5 258q169 169 341.5 236t246.5 -7q65 -64 20 -209q-4 -14 -1 -20t10 -7t14.5 0.5t13.5 3.5l6 2q139 59 246 59t153 -61q45 -63 0 -178 q-2 -13 -4.5 -20t4.5 -12.5t12 -7.5t17 -6q57 -18 103 -47t80 -81.5t34 -116.5zM1489 1046q42 -47 54.5 -108.5t-6.5 -117.5q-8 -23 -29.5 -34t-44.5 -4q-23 8 -34 29.5t-4 44.5q20 63 -24 111t-107 35q-24 -5 -45 8t-25 37q-5 24 8 44.5t37 25.5q60 13 119 -5.5t101 -65.5z M1670 1209q87 -96 112.5 -222.5t-13.5 -241.5q-9 -27 -34 -40t-52 -4t-40 34t-5 52q28 82 10 172t-80 158q-62 69 -148 95.5t-173 8.5q-28 -6 -52 9.5t-30 43.5t9.5 51.5t43.5 29.5q123 26 244 -11.5t208 -134.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1920" d="M805 163q-122 -67 -261 -67q-141 0 -261 67q98 61 167 149t94 191q25 -103 94 -191t167 -149zM453 1176v-344q0 -179 -89.5 -326t-234.5 -217q-129 152 -129 351q0 200 129.5 352t323.5 184zM958 991q-128 -152 -128 -351q0 -201 128 -351q-145 70 -234.5 218t-89.5 328 v341q196 -33 324 -185zM1638 163q-122 -67 -261 -67q-141 0 -261 67q98 61 167 149t94 191q25 -103 94 -191t167 -149zM1286 1176v-344q0 -179 -91 -326t-237 -217v0q133 154 133 351q0 195 -133 351q129 151 328 185zM1920 640q0 -201 -129 -351q-145 70 -234.5 218 t-89.5 328v341q194 -32 323.5 -184t129.5 -352z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1792" /> | ||||||
|  | </font> | ||||||
|  | </defs></svg>  | ||||||
| After Width: | Height: | Size: 193 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/_themes/sphinx_rtd_theme/static/font/fontawesome_webfont.ttf
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/_themes/sphinx_rtd_theme/static/font/fontawesome_webfont.ttf
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								docs/_themes/sphinx_rtd_theme/static/font/fontawesome_webfont.woff
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/_themes/sphinx_rtd_theme/static/font/fontawesome_webfont.woff
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										16
									
								
								docs/_themes/sphinx_rtd_theme/static/js/theme.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										16
									
								
								docs/_themes/sphinx_rtd_theme/static/js/theme.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | $( document ).ready(function() { | ||||||
|  |   // Shift nav in mobile when clicking the menu. | ||||||
|  |   $("[data-toggle='wy-nav-top']").click(function() { | ||||||
|  |     $("[data-toggle='wy-nav-shift']").toggleClass("shift"); | ||||||
|  |     $("[data-toggle='rst-versions']").toggleClass("shift"); | ||||||
|  |   }); | ||||||
|  |   // Close menu when you click a link. | ||||||
|  |   $(".wy-menu-vertical .current ul li a").click(function() { | ||||||
|  |     $("[data-toggle='wy-nav-shift']").removeClass("shift"); | ||||||
|  |     $("[data-toggle='rst-versions']").toggleClass("shift"); | ||||||
|  |   }); | ||||||
|  |   $("[data-toggle='rst-current-version']").click(function() { | ||||||
|  |     $("[data-toggle='rst-versions']").toggleClass("shift-up"); | ||||||
|  |   }); | ||||||
|  |   $("table.docutils:not(.field-list").wrap("<div class='wy-table-responsive'></div>"); | ||||||
|  | }); | ||||||
							
								
								
									
										8
									
								
								docs/_themes/sphinx_rtd_theme/theme.conf
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										8
									
								
								docs/_themes/sphinx_rtd_theme/theme.conf
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | [theme] | ||||||
|  | inherit = basic | ||||||
|  | stylesheet = css/theme.css | ||||||
|  |  | ||||||
|  | [options] | ||||||
|  | typekit_id = hiw1hhg | ||||||
|  | analytics_id = | ||||||
|  | canonical_url = | ||||||
							
								
								
									
										37
									
								
								docs/_themes/sphinx_rtd_theme/versions.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										37
									
								
								docs/_themes/sphinx_rtd_theme/versions.html
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | {% if READTHEDOCS %} | ||||||
|  | {# Add rst-badge after rst-versions for small badge style. #} | ||||||
|  |   <div class="rst-versions" data-toggle="rst-versions"> | ||||||
|  |     <span class="rst-current-version" data-toggle="rst-current-version"> | ||||||
|  |       <span class="icon icon-book"> Read the Docs</span> | ||||||
|  |       v: {{ current_version }}  | ||||||
|  |       <span class="icon icon-caret-down"></span> | ||||||
|  |     </span> | ||||||
|  |     <div class="rst-other-versions"> | ||||||
|  |       <dl> | ||||||
|  |         <dt>Versions</dt> | ||||||
|  |         {% for slug, url in versions %} | ||||||
|  |           <dd><a href="{{ url }}">{{ slug }}</a></dd> | ||||||
|  |         {% endfor %} | ||||||
|  |       </dl> | ||||||
|  |       <dl> | ||||||
|  |         <dt>Downloads</dt> | ||||||
|  |         {% for type, url in downloads %} | ||||||
|  |           <dd><a href="{{ url }}">{{ type }}</a></dd> | ||||||
|  |         {% endfor %} | ||||||
|  |       </dl> | ||||||
|  |       <dl> | ||||||
|  |         <dt>On Read the Docs</dt> | ||||||
|  |           <dd> | ||||||
|  |             <a href="//{{ PRODUCTION_DOMAIN }}/projects/{{ slug }}/?fromdocs={{ slug }}">Project Home</a> | ||||||
|  |           </dd> | ||||||
|  |           <dd> | ||||||
|  |             <a href="//{{ PRODUCTION_DOMAIN }}/builds/{{ slug }}/?fromdocs={{ slug }}">Builds</a> | ||||||
|  |           </dd> | ||||||
|  |       </dl> | ||||||
|  |       <hr/> | ||||||
|  |       Free document hosting provided by <a href="http://www.readthedocs.org">Read the Docs</a>. | ||||||
|  |  | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {% endif %} | ||||||
|  |  | ||||||
| @@ -38,16 +38,21 @@ Context Managers | |||||||
| ================ | ================ | ||||||
|  |  | ||||||
| .. autoclass:: mongoengine.context_managers.switch_db | .. autoclass:: mongoengine.context_managers.switch_db | ||||||
|  | .. autoclass:: mongoengine.context_managers.switch_collection | ||||||
| .. autoclass:: mongoengine.context_managers.no_dereference | .. autoclass:: mongoengine.context_managers.no_dereference | ||||||
| .. autoclass:: mongoengine.context_managers.query_counter | .. autoclass:: mongoengine.context_managers.query_counter | ||||||
|  |  | ||||||
| Querying | Querying | ||||||
| ======== | ======== | ||||||
|  |  | ||||||
|  | .. automodule:: mongoengine.queryset | ||||||
|  |     :synopsis: Queryset level operations | ||||||
|  |  | ||||||
|     .. autoclass:: mongoengine.queryset.QuerySet |     .. autoclass:: mongoengine.queryset.QuerySet | ||||||
|       :members: |       :members: | ||||||
|  |       :inherited-members: | ||||||
|  |  | ||||||
|    .. automethod:: mongoengine.queryset.QuerySet.__call__ |       .. automethod:: QuerySet.__call__ | ||||||
|  |  | ||||||
|     .. autoclass:: mongoengine.queryset.QuerySetNoCache |     .. autoclass:: mongoengine.queryset.QuerySetNoCache | ||||||
|       :members: |       :members: | ||||||
| @@ -79,6 +84,7 @@ Fields | |||||||
| .. autoclass:: mongoengine.fields.MapField | .. autoclass:: mongoengine.fields.MapField | ||||||
| .. autoclass:: mongoengine.fields.ReferenceField | .. autoclass:: mongoengine.fields.ReferenceField | ||||||
| .. autoclass:: mongoengine.fields.GenericReferenceField | .. autoclass:: mongoengine.fields.GenericReferenceField | ||||||
|  | .. autoclass:: mongoengine.fields.CachedReferenceField | ||||||
| .. autoclass:: mongoengine.fields.BinaryField | .. autoclass:: mongoengine.fields.BinaryField | ||||||
| .. autoclass:: mongoengine.fields.FileField | .. autoclass:: mongoengine.fields.FileField | ||||||
| .. autoclass:: mongoengine.fields.ImageField | .. autoclass:: mongoengine.fields.ImageField | ||||||
| @@ -89,6 +95,9 @@ Fields | |||||||
| .. autoclass:: mongoengine.fields.PointField | .. autoclass:: mongoengine.fields.PointField | ||||||
| .. autoclass:: mongoengine.fields.LineStringField | .. autoclass:: mongoengine.fields.LineStringField | ||||||
| .. autoclass:: mongoengine.fields.PolygonField | .. autoclass:: mongoengine.fields.PolygonField | ||||||
|  | .. autoclass:: mongoengine.fields.MultiPointField | ||||||
|  | .. autoclass:: mongoengine.fields.MultiLineStringField | ||||||
|  | .. autoclass:: mongoengine.fields.MultiPolygonField | ||||||
| .. autoclass:: mongoengine.fields.GridFSError | .. autoclass:: mongoengine.fields.GridFSError | ||||||
| .. autoclass:: mongoengine.fields.GridFSProxy | .. autoclass:: mongoengine.fields.GridFSProxy | ||||||
| .. autoclass:: mongoengine.fields.ImageGridFsProxy | .. autoclass:: mongoengine.fields.ImageGridFsProxy | ||||||
|   | |||||||
| @@ -2,6 +2,107 @@ | |||||||
| Changelog | Changelog | ||||||
| ========= | ========= | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Changes in 0.9.X - DEV | ||||||
|  | ====================== | ||||||
|  | - Allow specifying the '_cls' as a field for indexes #397 | ||||||
|  | - Stop ensure_indexes running on a secondaries unless connection is through mongos #746 | ||||||
|  | - Not overriding default values when loading a subset of fields #399 | ||||||
|  | - Saving document doesn't create new fields in existing collection #620 | ||||||
|  | - Added `Queryset.aggregate` wrapper to aggregation framework #703 | ||||||
|  | - Added support to show original model fields on to_json calls instead of db_field #697 | ||||||
|  | - Added Queryset.search_text to Text indexes searchs #700 | ||||||
|  | - Fixed tests for Django 1.7 #696 | ||||||
|  | - Follow ReferenceFields in EmbeddedDocuments with select_related #690 | ||||||
|  | - Added preliminary support for text indexes #680 | ||||||
|  | - Added `elemMatch` operator as well - `match` is too obscure #653 | ||||||
|  | - Added support for progressive JPEG #486 #548 | ||||||
|  | - Allow strings to be used in index creation #675 | ||||||
|  | - Fixed EmbeddedDoc weakref proxy issue #592 | ||||||
|  | - Fixed nested reference field distinct error #583 | ||||||
|  | - Fixed change tracking on nested MapFields #539 | ||||||
|  | - Dynamic fields in embedded documents now visible to queryset.only() / qs.exclude() #425 #507 | ||||||
|  | - Add authentication_source option to register_connection #178 #464 #573 #580 #590 | ||||||
|  | - Implemented equality between Documents and DBRefs #597 | ||||||
|  | - Fixed ReferenceField inside nested ListFields dereferencing problem #368 | ||||||
|  | - Added the ability to reload specific document fields #100 | ||||||
|  | - Added db_alias support and fixes for custom map/reduce output #586 | ||||||
|  | - post_save signal now has access to delta information about field changes #594 #589 | ||||||
|  | - Don't query with $orderby for qs.get() #600 | ||||||
|  | - Fix id shard key save issue #636 | ||||||
|  | - Fixes issue with recursive embedded document errors #557 | ||||||
|  | - Fix clear_changed_fields() clearing unsaved documents bug #602 | ||||||
|  | - Removing support for Django 1.4.x, pymongo 2.5.x, pymongo 2.6.x. | ||||||
|  | - Removing support for Python < 2.6.6 | ||||||
|  | - Fixed $maxDistance location for geoJSON $near queries with MongoDB 2.6+ #664 | ||||||
|  | - QuerySet.modify() method to provide find_and_modify() like behaviour #677 | ||||||
|  | - Added support for the using() method on a queryset #676 | ||||||
|  | - PYPY support #673 | ||||||
|  | - Connection pooling #674 | ||||||
|  | - Avoid to open all documents from cursors in an if stmt #655 | ||||||
|  | - Ability to clear the ordering #657 | ||||||
|  | - Raise NotUniqueError in Document.update() on pymongo.errors.DuplicateKeyError #626 | ||||||
|  | - Slots - memory improvements #625 | ||||||
|  | - Fixed incorrectly split a query key when it ends with "_" #619 | ||||||
|  | - Geo docs updates #613 | ||||||
|  | - Workaround a dateutil bug #608 | ||||||
|  | - Conditional save for atomic-style operations #511 | ||||||
|  | - Allow dynamic dictionary-style field access #559 | ||||||
|  | - Increase email field length to accommodate new TLDs #726 | ||||||
|  | - index_cls is ignored when deciding to set _cls as index prefix #733 | ||||||
|  | - Make 'db' argument to connection optional #737 | ||||||
|  | - Allow atomic update for the entire `DictField` #742 | ||||||
|  | - Added MultiPointField, MultiLineField, MultiPolygonField | ||||||
|  | - Fix multiple connections aliases being rewritten #748 | ||||||
|  |  | ||||||
|  | Changes in 0.8.7 | ||||||
|  | ================ | ||||||
|  | - Calling reload on deleted / nonexistant documents raises DoesNotExist (#538) | ||||||
|  | - Stop ensure_indexes running on a secondaries (#555) | ||||||
|  | - Fix circular import issue with django auth (#531) (#545) | ||||||
|  |  | ||||||
|  | Changes in 0.8.6 | ||||||
|  | ================ | ||||||
|  | - Fix django auth import (#531) | ||||||
|  |  | ||||||
|  | Changes in 0.8.5 | ||||||
|  | ================ | ||||||
|  | - Fix multi level nested fields getting marked as changed (#523) | ||||||
|  | - Django 1.6 login fix (#522) (#527) | ||||||
|  | - Django 1.6 session fix (#509) | ||||||
|  | - EmbeddedDocument._instance is now set when settng the attribute (#506) | ||||||
|  | - Fixed EmbeddedDocument with ReferenceField equality issue (#502) | ||||||
|  | - Fixed GenericReferenceField serialization order (#499) | ||||||
|  | - Fixed count and none bug (#498) | ||||||
|  | - Fixed bug with .only() and DictField with digit keys (#496) | ||||||
|  | - Added user_permissions to Django User object (#491, #492) | ||||||
|  | - Fix updating Geo Location fields (#488) | ||||||
|  | - Fix handling invalid dict field value (#485) | ||||||
|  | - Added app_label to MongoUser (#484) | ||||||
|  | - Use defaults when host and port are passed as None (#483) | ||||||
|  | - Fixed distinct casting issue with ListField of EmbeddedDocuments (#470) | ||||||
|  | - Fixed Django 1.6 sessions (#454, #480) | ||||||
|  |  | ||||||
|  | Changes in 0.8.4 | ||||||
|  | ================ | ||||||
|  | - Remove database name necessity in uri connection schema (#452) | ||||||
|  | - Fixed "$pull" semantics for nested ListFields (#447) | ||||||
|  | - Allow fields to be named the same as query operators (#445) | ||||||
|  | - Updated field filter logic - can now exclude subclass fields (#443) | ||||||
|  | - Fixed dereference issue with embedded listfield referencefields (#439) | ||||||
|  | - Fixed slice when using inheritance causing fields to be excluded (#437) | ||||||
|  | - Fixed ._get_db() attribute after a Document.switch_db() (#441) | ||||||
|  | - Dynamic Fields store and recompose Embedded Documents / Documents correctly (#449) | ||||||
|  | - Handle dynamic fieldnames that look like digits (#434) | ||||||
|  | - Added get_user_document and improve mongo_auth module (#423) | ||||||
|  | - Added str representation of GridFSProxy (#424) | ||||||
|  | - Update transform to handle docs erroneously passed to unset (#416) | ||||||
|  | - Fixed indexing - turn off _cls (#414) | ||||||
|  | - Fixed dereference threading issue in ComplexField.__get__ (#412) | ||||||
|  | - Fixed QuerySetNoCache.count() caching (#410) | ||||||
|  | - Don't follow references in _get_changed_fields (#422, #417) | ||||||
|  | - Allow args and kwargs to be passed through to_json (#420) | ||||||
|  |  | ||||||
| Changes in 0.8.3 | Changes in 0.8.3 | ||||||
| ================ | ================ | ||||||
| - Fixed EmbeddedDocuments with `id` also storing `_id` (#402) | - Fixed EmbeddedDocuments with `id` also storing `_id` (#402) | ||||||
| @@ -12,6 +113,9 @@ Changes in 0.8.3 | |||||||
| - Document.select_related() now respects `db_alias` (#377) | - Document.select_related() now respects `db_alias` (#377) | ||||||
| - Reload uses shard_key if applicable (#384) | - Reload uses shard_key if applicable (#384) | ||||||
| - Dynamic fields are ordered based on creation and stored in _fields_ordered (#396) | - Dynamic fields are ordered based on creation and stored in _fields_ordered (#396) | ||||||
|  |  | ||||||
|  |   **Potential breaking change:** http://docs.mongoengine.org/en/latest/upgrade.html#to-0-8-3 | ||||||
|  |  | ||||||
| - Fixed pickling dynamic documents `_dynamic_fields` (#387) | - Fixed pickling dynamic documents `_dynamic_fields` (#387) | ||||||
| - Fixed ListField setslice and delslice dirty tracking (#390) | - Fixed ListField setslice and delslice dirty tracking (#390) | ||||||
| - Added Django 1.5 PY3 support (#392) | - Added Django 1.5 PY3 support (#392) | ||||||
| @@ -20,6 +124,8 @@ Changes in 0.8.3 | |||||||
| - Fixed queryset.get() respecting no_dereference (#373) | - Fixed queryset.get() respecting no_dereference (#373) | ||||||
| - Added full_result kwarg to update (#380) | - Added full_result kwarg to update (#380) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Changes in 0.8.2 | Changes in 0.8.2 | ||||||
| ================ | ================ | ||||||
| - Added compare_indexes helper (#361) | - Added compare_indexes helper (#361) | ||||||
|   | |||||||
| @@ -92,7 +92,7 @@ pygments_style = 'sphinx' | |||||||
|  |  | ||||||
| # The theme to use for HTML and HTML Help pages.  Major themes that come with | # The theme to use for HTML and HTML Help pages.  Major themes that come with | ||||||
| # Sphinx are currently 'default' and 'sphinxdoc'. | # Sphinx are currently 'default' and 'sphinxdoc'. | ||||||
| html_theme = 'nature' | html_theme = 'sphinx_rtd_theme' | ||||||
|  |  | ||||||
| # Theme options are theme-specific and customize the look and feel of a theme | # Theme options are theme-specific and customize the look and feel of a theme | ||||||
| # further.  For a list of options available for each theme, see the | # further.  For a list of options available for each theme, see the | ||||||
| @@ -116,7 +116,7 @@ html_theme_path = ['_themes'] | |||||||
| # The name of an image file (within the static path) to use as favicon of the | # The name of an image file (within the static path) to use as favicon of the | ||||||
| # docs.  This file should be a Windows icon file (.ico) being 16x16 or 32x32 | # docs.  This file should be a Windows icon file (.ico) being 16x16 or 32x32 | ||||||
| # pixels large. | # pixels large. | ||||||
| #html_favicon = None | html_favicon = "favicon.ico" | ||||||
|  |  | ||||||
| # 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, | ||||||
| @@ -200,3 +200,6 @@ latex_documents = [ | |||||||
|  |  | ||||||
| autoclass_content = 'both' | autoclass_content = 'both' | ||||||
|  |  | ||||||
|  | html_theme_options = dict( | ||||||
|  |     canonical_url='http://docs.mongoengine.org/en/latest/' | ||||||
|  | ) | ||||||
|   | |||||||
| @@ -45,7 +45,7 @@ The :mod:`~mongoengine.django.auth` module also contains a | |||||||
| Custom User model | Custom User model | ||||||
| ================= | ================= | ||||||
| Django 1.5 introduced `Custom user Models | Django 1.5 introduced `Custom user Models | ||||||
| <https://docs.djangoproject.com/en/dev/topics/auth/customizing/#auth-custom-user>` | <https://docs.djangoproject.com/en/dev/topics/auth/customizing/#auth-custom-user>`_ | ||||||
| which can be used as an alternative to the MongoEngine authentication backend. | which can be used as an alternative to the MongoEngine authentication backend. | ||||||
|  |  | ||||||
| The main advantage of this option is that other components relying on | The main advantage of this option is that other components relying on | ||||||
| @@ -74,7 +74,7 @@ An additional ``MONGOENGINE_USER_DOCUMENT`` setting enables you to replace the | |||||||
| The custom :class:`User` must be a :class:`~mongoengine.Document` class, but | The custom :class:`User` must be a :class:`~mongoengine.Document` class, but | ||||||
| otherwise has the same requirements as a standard custom user model, | otherwise has the same requirements as a standard custom user model, | ||||||
| as specified in the `Django Documentation | as specified in the `Django Documentation | ||||||
| <https://docs.djangoproject.com/en/dev/topics/auth/customizing/>`. | <https://docs.djangoproject.com/en/dev/topics/auth/customizing/>`_. | ||||||
| In particular, the custom class must define :attr:`USERNAME_FIELD` and | In particular, the custom class must define :attr:`USERNAME_FIELD` and | ||||||
| :attr:`REQUIRED_FIELDS` attributes. | :attr:`REQUIRED_FIELDS` attributes. | ||||||
|  |  | ||||||
| @@ -90,10 +90,15 @@ session backend, ensure that your settings module has | |||||||
| into your settings module:: | into your settings module:: | ||||||
|  |  | ||||||
|     SESSION_ENGINE = 'mongoengine.django.sessions' |     SESSION_ENGINE = 'mongoengine.django.sessions' | ||||||
|  |     SESSION_SERIALIZER = 'mongoengine.django.sessions.BSONSerializer' | ||||||
|  |  | ||||||
| Django provides session cookie, which expires after ```SESSION_COOKIE_AGE``` seconds, but doesn't delete cookie at sessions backend, so ``'mongoengine.django.sessions'`` supports  `mongodb TTL | Django provides session cookie, which expires after ```SESSION_COOKIE_AGE``` seconds, but doesn't delete cookie at sessions backend, so ``'mongoengine.django.sessions'`` supports  `mongodb TTL | ||||||
| <http://docs.mongodb.org/manual/tutorial/expire-data/>`_. | <http://docs.mongodb.org/manual/tutorial/expire-data/>`_. | ||||||
|  |  | ||||||
|  | .. note:: ``SESSION_SERIALIZER`` is only necessary in Django 1.6 as the default | ||||||
|  |    serializer is based around JSON and doesn't know how to convert | ||||||
|  |    ``bson.objectid.ObjectId`` instances to strings. | ||||||
|  |  | ||||||
| .. versionadded:: 0.2.1 | .. versionadded:: 0.2.1 | ||||||
|  |  | ||||||
| Storage | Storage | ||||||
| @@ -128,7 +133,7 @@ appended to the filename until the generated filename doesn't exist. The | |||||||
|     >>> fs.listdir() |     >>> fs.listdir() | ||||||
|     ([], [u'hello.txt']) |     ([], [u'hello.txt']) | ||||||
|  |  | ||||||
| All files will be saved and retrieved in GridFS via the :class::`FileDocument` | All files will be saved and retrieved in GridFS via the :class:`FileDocument` | ||||||
| document, allowing easy access to the files without the GridFSStorage | document, allowing easy access to the files without the GridFSStorage | ||||||
| backend.:: | backend.:: | ||||||
|  |  | ||||||
| @@ -137,3 +142,36 @@ backend.:: | |||||||
|     [<FileDocument: FileDocument object>] |     [<FileDocument: FileDocument object>] | ||||||
|  |  | ||||||
| .. versionadded:: 0.4 | .. versionadded:: 0.4 | ||||||
|  |  | ||||||
|  | Shortcuts | ||||||
|  | ========= | ||||||
|  | Inspired by the `Django shortcut get_object_or_404 | ||||||
|  | <https://docs.djangoproject.com/en/dev/topics/http/shortcuts/#get-object-or-404>`_, | ||||||
|  | the :func:`~mongoengine.django.shortcuts.get_document_or_404` method returns  | ||||||
|  | a document or raises an Http404 exception if the document does not exist:: | ||||||
|  |  | ||||||
|  |     from mongoengine.django.shortcuts import get_document_or_404 | ||||||
|  |      | ||||||
|  |     admin_user = get_document_or_404(User, username='root') | ||||||
|  |  | ||||||
|  | The first argument may be a Document or QuerySet object. All other passed arguments | ||||||
|  | and keyword arguments are used in the query:: | ||||||
|  |  | ||||||
|  |     foo_email = get_document_or_404(User.objects.only('email'), username='foo', is_active=True).email | ||||||
|  |  | ||||||
|  | .. note:: Like with :func:`get`, a MultipleObjectsReturned will be raised if more than one | ||||||
|  |     object is found. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Also inspired by the `Django shortcut get_list_or_404 | ||||||
|  | <https://docs.djangoproject.com/en/dev/topics/http/shortcuts/#get-list-or-404>`_, | ||||||
|  | the :func:`~mongoengine.django.shortcuts.get_list_or_404` method returns a list of | ||||||
|  | documents or raises an Http404 exception if the list is empty:: | ||||||
|  |  | ||||||
|  |     from mongoengine.django.shortcuts import get_list_or_404 | ||||||
|  |      | ||||||
|  |     active_users = get_list_or_404(User, is_active=True) | ||||||
|  |  | ||||||
|  | The first argument may be a Document or QuerySet object. All other passed | ||||||
|  | arguments and keyword arguments are used to filter the query. | ||||||
|  |  | ||||||
|   | |||||||
| @@ -23,17 +23,20 @@ arguments should be provided:: | |||||||
|  |  | ||||||
|     connect('project1', username='webapp', password='pwd123') |     connect('project1', username='webapp', password='pwd123') | ||||||
|  |  | ||||||
| Uri style connections are also supported as long as you include the database | Uri style connections are also supported - just supply the uri as | ||||||
| name - just supply the uri as the :attr:`host` to | the :attr:`host` to | ||||||
| :func:`~mongoengine.connect`:: | :func:`~mongoengine.connect`:: | ||||||
|  |  | ||||||
|     connect('project1', host='mongodb://localhost/database_name') |     connect('project1', host='mongodb://localhost/database_name') | ||||||
|  |  | ||||||
|  | Note that database name from uri has priority over name | ||||||
|  | in ::func:`~mongoengine.connect` | ||||||
|  |  | ||||||
| ReplicaSets | ReplicaSets | ||||||
| =========== | =========== | ||||||
|  |  | ||||||
| MongoEngine supports :class:`~pymongo.mongo_replica_set_client.MongoReplicaSetClient` | MongoEngine supports :class:`~pymongo.mongo_replica_set_client.MongoReplicaSetClient`. | ||||||
| to use them please use a URI style connection and provide the `replicaSet` name in the | To use them, please use a URI style connection and provide the `replicaSet` name in the | ||||||
| connection kwargs. | connection kwargs. | ||||||
|  |  | ||||||
| Read preferences are supported through the connection or via individual | Read preferences are supported through the connection or via individual | ||||||
| @@ -97,3 +100,18 @@ access to the same User document across databases:: | |||||||
|  |  | ||||||
| .. note:: Make sure any aliases have been registered with | .. note:: Make sure any aliases have been registered with | ||||||
|     :func:`~mongoengine.register_connection` before using the context manager. |     :func:`~mongoengine.register_connection` before using the context manager. | ||||||
|  |  | ||||||
|  | There is also a switch collection context manager as well.  The | ||||||
|  | :class:`~mongoengine.context_managers.switch_collection` context manager allows | ||||||
|  | you to change the collection for a given class allowing quick and easy | ||||||
|  | access to the same Group document across collection:: | ||||||
|  |  | ||||||
|  |         from mongoengine.context_managers import switch_db | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         Group(name="test").save()  # Saves in the default db | ||||||
|  |  | ||||||
|  |         with switch_collection(Group, 'group2000') as Group: | ||||||
|  |             Group(name="hello Group 2000 collection!").save()  # Saves in group2000 collection | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ Defining documents | |||||||
| In MongoDB, a **document** is roughly equivalent to a **row** in an RDBMS. When | 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 | working with relational databases, rows are stored in **tables**, which have a | ||||||
| strict **schema** that the rows follow. MongoDB stores documents in | strict **schema** that the rows follow. MongoDB stores documents in | ||||||
| **collections** rather than tables - the principle difference is that no schema | **collections** rather than tables - the principal difference is that no schema | ||||||
| is enforced at a database level. | is enforced at a database level. | ||||||
|  |  | ||||||
| Defining a document's schema | Defining a document's schema | ||||||
| @@ -91,6 +91,12 @@ are as follows: | |||||||
| * :class:`~mongoengine.fields.StringField` | * :class:`~mongoengine.fields.StringField` | ||||||
| * :class:`~mongoengine.fields.URLField` | * :class:`~mongoengine.fields.URLField` | ||||||
| * :class:`~mongoengine.fields.UUIDField` | * :class:`~mongoengine.fields.UUIDField` | ||||||
|  | * :class:`~mongoengine.fields.PointField` | ||||||
|  | * :class:`~mongoengine.fields.LineStringField` | ||||||
|  | * :class:`~mongoengine.fields.PolygonField` | ||||||
|  | * :class:`~mongoengine.fields.MultiPointField` | ||||||
|  | * :class:`~mongoengine.fields.MultiLineStringField` | ||||||
|  | * :class:`~mongoengine.fields.MultiPolygonField` | ||||||
|  |  | ||||||
| Field arguments | Field arguments | ||||||
| --------------- | --------------- | ||||||
| @@ -290,6 +296,12 @@ instance of the object to the query:: | |||||||
|     # Find all pages that both Bob and John have authored |     # Find all pages that both Bob and John have authored | ||||||
|     Page.objects(authors__all=[bob, john]) |     Page.objects(authors__all=[bob, john]) | ||||||
|  |  | ||||||
|  |     # Remove Bob from the authors for a page. | ||||||
|  |     Page.objects(id='...').update_one(pull__authors=bob) | ||||||
|  |  | ||||||
|  |     # Add John to the authors for a page. | ||||||
|  |     Page.objects(id='...').update_one(push__authors=john) | ||||||
|  |  | ||||||
|  |  | ||||||
| Dealing with deletion of referred documents | Dealing with deletion of referred documents | ||||||
| ''''''''''''''''''''''''''''''''''''''''''' | ''''''''''''''''''''''''''''''''''''''''''' | ||||||
| @@ -442,6 +454,8 @@ The following example shows a :class:`Log` document that will be limited to | |||||||
|         ip_address = StringField() |         ip_address = StringField() | ||||||
|         meta = {'max_documents': 1000, 'max_size': 2000000} |         meta = {'max_documents': 1000, 'max_size': 2000000} | ||||||
|  |  | ||||||
|  | .. defining-indexes_ | ||||||
|  |  | ||||||
| Indexes | Indexes | ||||||
| ======= | ======= | ||||||
|  |  | ||||||
| @@ -451,7 +465,8 @@ by creating a list of index specifications called :attr:`indexes` in the | |||||||
| either be a single field name, a tuple containing multiple field names, or a | 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 | dictionary containing a full index definition. A direction may be specified on | ||||||
| fields by prefixing the field name with a **+** (for ascending) or a **-** sign | fields by prefixing the field name with a **+** (for ascending) or a **-** sign | ||||||
| (for descending). Note that direction only matters on multi-field indexes. :: | (for descending). Note that direction only matters on multi-field indexes. | ||||||
|  | Text indexes may be specified by prefixing the field name with a **$**. :: | ||||||
|  |  | ||||||
|     class Page(Document): |     class Page(Document): | ||||||
|         title = StringField() |         title = StringField() | ||||||
| @@ -485,6 +500,35 @@ If a dictionary is passed then the following options are available: | |||||||
|  |  | ||||||
|     Inheritance adds extra fields indices see: :ref:`document-inheritance`. |     Inheritance adds extra fields indices see: :ref:`document-inheritance`. | ||||||
|  |  | ||||||
|  | Global index default options | ||||||
|  | ---------------------------- | ||||||
|  |  | ||||||
|  | There are a few top level defaults for all indexes that can be set:: | ||||||
|  |  | ||||||
|  |     class Page(Document): | ||||||
|  |         title = StringField() | ||||||
|  |         rating = StringField() | ||||||
|  |         meta = { | ||||||
|  |             'index_options': {}, | ||||||
|  |             'index_background': True, | ||||||
|  |             'index_drop_dups': True, | ||||||
|  |             'index_cls': False | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | :attr:`index_options` (Optional) | ||||||
|  |     Set any default index options - see the `full options list <http://docs.mongodb.org/manual/reference/method/db.collection.ensureIndex/#db.collection.ensureIndex>`_ | ||||||
|  |  | ||||||
|  | :attr:`index_background` (Optional) | ||||||
|  |     Set the default value for if an index should be indexed in the background | ||||||
|  |  | ||||||
|  | :attr:`index_drop_dups` (Optional) | ||||||
|  |     Set the default value for if an index should drop duplicates | ||||||
|  |  | ||||||
|  | :attr:`index_cls` (Optional) | ||||||
|  |     A way to turn off a specific index for _cls. | ||||||
|  |  | ||||||
|  |  | ||||||
| Compound Indexes and Indexing sub documents | Compound Indexes and Indexing sub documents | ||||||
| ------------------------------------------- | ------------------------------------------- | ||||||
|  |  | ||||||
| @@ -494,6 +538,8 @@ field name to the index definition. | |||||||
| Sometimes its more efficient to index parts of Embedded / dictionary fields, | Sometimes its more efficient to index parts of Embedded / dictionary fields, | ||||||
| in this case use 'dot' notation to identify the value to index eg: `rank.title` | in this case use 'dot' notation to identify the value to index eg: `rank.title` | ||||||
|  |  | ||||||
|  | .. _geospatial-indexes: | ||||||
|  |  | ||||||
| Geospatial indexes | Geospatial indexes | ||||||
| ------------------ | ------------------ | ||||||
|  |  | ||||||
| @@ -504,6 +550,9 @@ The following fields will explicitly add a "2dsphere" index: | |||||||
|     - :class:`~mongoengine.fields.PointField` |     - :class:`~mongoengine.fields.PointField` | ||||||
|     - :class:`~mongoengine.fields.LineStringField` |     - :class:`~mongoengine.fields.LineStringField` | ||||||
|     - :class:`~mongoengine.fields.PolygonField` |     - :class:`~mongoengine.fields.PolygonField` | ||||||
|  |     - :class:`~mongoengine.fields.MultiPointField` | ||||||
|  |     - :class:`~mongoengine.fields.MultiLineStringField` | ||||||
|  |     - :class:`~mongoengine.fields.MultiPolygonField` | ||||||
|  |  | ||||||
| As "2dsphere" indexes can be part of a compound index, you may not want the | As "2dsphere" indexes can be part of a compound index, you may not want the | ||||||
| automatic index but would prefer a compound index.  In this example we turn off | automatic index but would prefer a compound index.  In this example we turn off | ||||||
| @@ -558,6 +607,11 @@ documentation for more information.  A common usecase might be session data:: | |||||||
|             ] |             ] | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | .. warning:: TTL indexes happen on the MongoDB server and not in the application | ||||||
|  |     code, therefore no signals will be fired on document deletion. | ||||||
|  |     If you need signals to be fired on deletion, then you must handle the | ||||||
|  |     deletion of Documents in your application code. | ||||||
|  |  | ||||||
| Comparing Indexes | Comparing Indexes | ||||||
| ----------------- | ----------------- | ||||||
|  |  | ||||||
| @@ -653,7 +707,6 @@ document.:: | |||||||
| .. note:: From 0.8 onwards you must declare :attr:`allow_inheritance` defaults | .. note:: From 0.8 onwards you must declare :attr:`allow_inheritance` defaults | ||||||
|           to False, meaning you must set it to True to use inheritance. |           to False, meaning you must set it to True to use inheritance. | ||||||
|  |  | ||||||
|  |  | ||||||
| Working with existing data | Working with existing data | ||||||
| -------------------------- | -------------------------- | ||||||
| As MongoEngine no longer defaults to needing :attr:`_cls` you can quickly and | As MongoEngine no longer defaults to needing :attr:`_cls` you can quickly and | ||||||
| @@ -673,3 +726,25 @@ defining all possible field types. | |||||||
|  |  | ||||||
| If you use :class:`~mongoengine.Document` and the database contains data that | If you use :class:`~mongoengine.Document` and the database contains data that | ||||||
| isn't defined then that data will be stored in the `document._data` dictionary. | isn't defined then that data will be stored in the `document._data` dictionary. | ||||||
|  |  | ||||||
|  | Abstract classes | ||||||
|  | ================ | ||||||
|  |  | ||||||
|  | If you want to add some extra functionality to a group of Document classes but | ||||||
|  | you don't need or want the overhead of inheritance you can use the | ||||||
|  | :attr:`abstract` attribute of :attr:`-mongoengine.Document.meta`. | ||||||
|  | This won't turn on :ref:`document-inheritance` but will allow you to keep your | ||||||
|  | code DRY:: | ||||||
|  |  | ||||||
|  |         class BaseDocument(Document): | ||||||
|  |             meta = { | ||||||
|  |                 'abstract': True, | ||||||
|  |             } | ||||||
|  |             def check_permissions(self): | ||||||
|  |                 ... | ||||||
|  |  | ||||||
|  |         class User(BaseDocument): | ||||||
|  |            ... | ||||||
|  |  | ||||||
|  | Now the User class will have access to the inherited `check_permissions` method | ||||||
|  | and won't store any of the extra `_cls` information. | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ a document is created to store details about animals, including a photo:: | |||||||
|  |  | ||||||
|     marmot = Animal(genus='Marmota', family='Sciuridae') |     marmot = Animal(genus='Marmota', family='Sciuridae') | ||||||
|  |  | ||||||
|     marmot_photo = open('marmot.jpg', 'r') |     marmot_photo = open('marmot.jpg', 'rb') | ||||||
|     marmot.photo.put(marmot_photo, content_type = 'image/jpeg') |     marmot.photo.put(marmot_photo, content_type = 'image/jpeg') | ||||||
|     marmot.save() |     marmot.save() | ||||||
|  |  | ||||||
| @@ -46,7 +46,7 @@ slightly different manner.  First, a new file must be created by calling the | |||||||
|     marmot.photo.write('some_more_image_data') |     marmot.photo.write('some_more_image_data') | ||||||
|     marmot.photo.close() |     marmot.photo.close() | ||||||
|  |  | ||||||
|     marmot.photo.save() |     marmot.save() | ||||||
|  |  | ||||||
| Deletion | Deletion | ||||||
| -------- | -------- | ||||||
| @@ -70,5 +70,5 @@ Replacing files | |||||||
| Files can be replaced with the :func:`replace` method. This works just like | 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:: | the :func:`put` method so even metadata can (and should) be replaced:: | ||||||
|  |  | ||||||
|     another_marmot = open('another_marmot.png', 'r') |     another_marmot = open('another_marmot.png', 'rb') | ||||||
|     marmot.photo.replace(another_marmot, content_type='image/png') |     marmot.photo.replace(another_marmot, content_type='image/png') | ||||||
|   | |||||||
| @@ -12,3 +12,4 @@ User Guide | |||||||
|    querying |    querying | ||||||
|    gridfs |    gridfs | ||||||
|    signals |    signals | ||||||
|  |    text-indexes | ||||||
|   | |||||||
| @@ -17,8 +17,8 @@ fetch documents from the database:: | |||||||
|  |  | ||||||
|     As of MongoEngine 0.8 the querysets utilise a local cache.  So iterating |     As of MongoEngine 0.8 the querysets utilise a local cache.  So iterating | ||||||
|     it multiple times will only cause a single query.  If this is not the |     it multiple times will only cause a single query.  If this is not the | ||||||
|     desired behavour you can call :class:`~mongoengine.QuerySet.no_cache` to |     desired behavour you can call :class:`~mongoengine.QuerySet.no_cache` | ||||||
|     return a non-caching queryset. |     (version **0.8.3+**) to return a non-caching queryset. | ||||||
|  |  | ||||||
| Filtering queries | Filtering queries | ||||||
| ================= | ================= | ||||||
| @@ -92,8 +92,8 @@ were added in 0.8 for:  :class:`~mongoengine.fields.PointField`, | |||||||
| * ``geo_within`` -- Check if a geometry is within a polygon.  For ease of use | * ``geo_within`` -- Check if a geometry is within a polygon.  For ease of use | ||||||
|     it accepts either a geojson geometry or just the polygon coordinates eg:: |     it accepts either a geojson geometry or just the polygon coordinates eg:: | ||||||
|  |  | ||||||
|         loc.objects(point__geo_with=[[[40, 5], [40, 6], [41, 6], [40, 5]]]) |         loc.objects(point__geo_within=[[[40, 5], [40, 6], [41, 6], [40, 5]]]) | ||||||
|         loc.objects(point__geo_with={"type": "Polygon", |         loc.objects(point__geo_within={"type": "Polygon", | ||||||
|                                  "coordinates": [[[40, 5], [40, 6], [41, 6], [40, 5]]]}) |                                  "coordinates": [[[40, 5], [40, 6], [41, 6], [40, 5]]]}) | ||||||
|  |  | ||||||
| * ``geo_within_box`` - simplified geo_within searching with a box eg:: | * ``geo_within_box`` - simplified geo_within searching with a box eg:: | ||||||
| @@ -488,8 +488,9 @@ calling it with keyword arguments:: | |||||||
| Atomic updates | Atomic updates | ||||||
| ============== | ============== | ||||||
| Documents may be updated atomically by using the | Documents may be updated atomically by using the | ||||||
| :meth:`~mongoengine.queryset.QuerySet.update_one` and | :meth:`~mongoengine.queryset.QuerySet.update_one`, | ||||||
| :meth:`~mongoengine.queryset.QuerySet.update` methods on a | :meth:`~mongoengine.queryset.QuerySet.update` and | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.modify` methods on a | ||||||
| :meth:`~mongoengine.queryset.QuerySet`. There are several different "modifiers" | :meth:`~mongoengine.queryset.QuerySet`. There are several different "modifiers" | ||||||
| that you may use with these methods: | that you may use with these methods: | ||||||
|  |  | ||||||
| @@ -497,14 +498,15 @@ that you may use with these methods: | |||||||
| * ``unset`` -- delete a particular value (since MongoDB v1.3+) | * ``unset`` -- delete a particular value (since MongoDB v1.3+) | ||||||
| * ``inc`` -- increment a value by a given amount | * ``inc`` -- increment a value by a given amount | ||||||
| * ``dec`` -- decrement 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`` -- append a value to a list | ||||||
| * ``push_all`` -- append several values to a list | * ``push_all`` -- append several values to a list | ||||||
| * ``pop`` -- remove the first or last element of a list | * ``pop`` -- remove the first or last element of a list `depending on the value`_ | ||||||
| * ``pull`` -- remove a value from a list | * ``pull`` -- remove a value from a list | ||||||
| * ``pull_all`` -- remove several values 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 | * ``add_to_set`` -- add value to a list only if its not in the list already | ||||||
|  |  | ||||||
|  | .. _depending on the value: http://docs.mongodb.org/manual/reference/operator/update/pop/ | ||||||
|  |  | ||||||
| The syntax for atomic updates is similar to the querying syntax, but the | The syntax for atomic updates is similar to the querying syntax, but the | ||||||
| modifier comes before the field, not after it:: | modifier comes before the field, not after it:: | ||||||
|  |  | ||||||
| @@ -523,6 +525,13 @@ modifier comes before the field, not after it:: | |||||||
|     >>> post.tags |     >>> post.tags | ||||||
|     ['database', 'nosql'] |     ['database', 'nosql'] | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |     If no modifier operator is specified the default will be ``$set``. So the following sentences are identical:: | ||||||
|  |  | ||||||
|  |         >>> BlogPost.objects(id=post.id).update(title='Example Post') | ||||||
|  |         >>> BlogPost.objects(id=post.id).update(set__title='Example Post') | ||||||
|  |  | ||||||
| .. note:: | .. note:: | ||||||
|  |  | ||||||
|     In version 0.5 the :meth:`~mongoengine.Document.save` runs atomic updates |     In version 0.5 the :meth:`~mongoengine.Document.save` runs atomic updates | ||||||
|   | |||||||
							
								
								
									
										49
									
								
								docs/guide/text-indexes.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								docs/guide/text-indexes.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | |||||||
|  | =========== | ||||||
|  | Text Search | ||||||
|  | =========== | ||||||
|  |  | ||||||
|  | After MongoDB 2.4 version, supports search documents by text indexes. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Defining a Document with text index | ||||||
|  | =================================== | ||||||
|  | Use the *$* prefix to set a text index, Look the declaration:: | ||||||
|  |    | ||||||
|  |   class News(Document): | ||||||
|  |       title = StringField() | ||||||
|  |       content = StringField() | ||||||
|  |       is_active = BooleanField() | ||||||
|  |  | ||||||
|  |       meta = {'indexes': [ | ||||||
|  |           {'fields': ['$title', "$content"], | ||||||
|  |            'default_language': 'english', | ||||||
|  |            'weight': {'title': 10, 'content': 2} | ||||||
|  |           } | ||||||
|  |       ]} | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Querying | ||||||
|  | ======== | ||||||
|  |  | ||||||
|  | Saving a document:: | ||||||
|  |  | ||||||
|  |   News(title="Using mongodb text search", | ||||||
|  |        content="Testing text search").save() | ||||||
|  |  | ||||||
|  |   News(title="MongoEngine 0.9 released", | ||||||
|  |        content="Various improvements").save() | ||||||
|  |  | ||||||
|  | Next, start a text search using :attr:`QuerySet.search_text` method:: | ||||||
|  |    | ||||||
|  |   document = News.objects.search_text('testing').first() | ||||||
|  |   document.title # may be: "Using mongodb text search" | ||||||
|  |    | ||||||
|  |   document = News.objects.search_text('released').first() | ||||||
|  |   document.title # may be: "MongoEngine 0.9 released" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Ordering by text score | ||||||
|  | ====================== | ||||||
|  |  | ||||||
|  |   objects = News.objects.search('mongo').order_by('$text_score') | ||||||
| @@ -2,8 +2,14 @@ | |||||||
| Upgrading | Upgrading | ||||||
| ######### | ######### | ||||||
|  |  | ||||||
|  | 0.8.7 | ||||||
|  | ***** | ||||||
|  |  | ||||||
| 0.8.2 to 0.8.2 | Calling reload on deleted / nonexistant documents now raises a DoesNotExist | ||||||
|  | exception. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 0.8.2 to 0.8.3 | ||||||
| ************** | ************** | ||||||
|  |  | ||||||
| Minor change that may impact users: | Minor change that may impact users: | ||||||
| @@ -16,8 +22,8 @@ fields.  Previously they were stored alphabetically. | |||||||
| ********** | ********** | ||||||
|  |  | ||||||
| There have been numerous backwards breaking changes in 0.8.  The reasons for | There have been numerous backwards breaking changes in 0.8.  The reasons for | ||||||
| these are ensure that MongoEngine has sane defaults going forward and | these are to ensure that MongoEngine has sane defaults going forward and that it | ||||||
| performs the best it can out the box.  Where possible there have been | performs the best it can out of the box.  Where possible there have been | ||||||
| FutureWarnings to help get you ready for the change, but that hasn't been | FutureWarnings to help get you ready for the change, but that hasn't been | ||||||
| possible for the whole of the release. | possible for the whole of the release. | ||||||
|  |  | ||||||
| @@ -71,7 +77,7 @@ inherited classes like so: :: | |||||||
| Document Definition | Document Definition | ||||||
| ------------------- | ------------------- | ||||||
|  |  | ||||||
| The default for inheritance has changed - its now off by default and | The default for inheritance has changed - it is now off by default and | ||||||
| :attr:`_cls` will not be stored automatically with the class.  So if you extend | :attr:`_cls` will not be stored automatically with the class.  So if you extend | ||||||
| your :class:`~mongoengine.Document` or :class:`~mongoengine.EmbeddedDocuments` | your :class:`~mongoengine.Document` or :class:`~mongoengine.EmbeddedDocuments` | ||||||
| you will need to declare :attr:`allow_inheritance` in the meta data like so: :: | you will need to declare :attr:`allow_inheritance` in the meta data like so: :: | ||||||
| @@ -81,7 +87,7 @@ you will need to declare :attr:`allow_inheritance` in the meta data like so: :: | |||||||
|  |  | ||||||
|         meta = {'allow_inheritance': True} |         meta = {'allow_inheritance': True} | ||||||
|  |  | ||||||
| Previously, if you had data the database that wasn't defined in the Document | Previously, if you had data in the database that wasn't defined in the Document | ||||||
| definition, it would set it as an attribute on the document.  This is no longer | definition, it would set it as an attribute on the document.  This is no longer | ||||||
| the case and the data is set only in the ``document._data`` dictionary: :: | the case and the data is set only in the ``document._data`` dictionary: :: | ||||||
|  |  | ||||||
| @@ -102,8 +108,8 @@ the case and the data is set only in the ``document._data`` dictionary: :: | |||||||
|     AttributeError: 'Animal' object has no attribute 'size' |     AttributeError: 'Animal' object has no attribute 'size' | ||||||
|  |  | ||||||
| The Document class has introduced a reserved function `clean()`, which will be | The Document class has introduced a reserved function `clean()`, which will be | ||||||
| called before saving the document. If your document class happen to have a method | called before saving the document. If your document class happens to have a method | ||||||
| with the same name, please try rename it. | with the same name, please try to rename it. | ||||||
|  |  | ||||||
|     def clean(self): |     def clean(self): | ||||||
|         pass |         pass | ||||||
| @@ -111,7 +117,7 @@ with the same name, please try rename it. | |||||||
| ReferenceField | ReferenceField | ||||||
| -------------- | -------------- | ||||||
|  |  | ||||||
| ReferenceFields now store ObjectId's by default - this is more efficient than | ReferenceFields now store ObjectIds by default - this is more efficient than | ||||||
| DBRefs as we already know what Document types they reference:: | DBRefs as we already know what Document types they reference:: | ||||||
|  |  | ||||||
|     # Old code |     # Old code | ||||||
| @@ -157,7 +163,7 @@ UUIDFields now default to storing binary values:: | |||||||
|     class Animal(Document): |     class Animal(Document): | ||||||
|         uuid = UUIDField(binary=False) |         uuid = UUIDField(binary=False) | ||||||
|  |  | ||||||
| To migrate all the uuid's you need to touch each object and mark it as dirty | To migrate all the uuids you need to touch each object and mark it as dirty | ||||||
| eg:: | eg:: | ||||||
|  |  | ||||||
|     # Doc definition |     # Doc definition | ||||||
| @@ -175,7 +181,7 @@ eg:: | |||||||
| DecimalField | DecimalField | ||||||
| ------------ | ------------ | ||||||
|  |  | ||||||
| DecimalField now store floats - previous it was storing strings and that | DecimalFields now store floats - previously it was storing strings and that | ||||||
| made it impossible to do comparisons when querying correctly.:: | made it impossible to do comparisons when querying correctly.:: | ||||||
|  |  | ||||||
|     # Old code |     # Old code | ||||||
| @@ -186,7 +192,7 @@ made it impossible to do comparisons when querying correctly.:: | |||||||
|     class Person(Document): |     class Person(Document): | ||||||
|         balance = DecimalField(force_string=True) |         balance = DecimalField(force_string=True) | ||||||
|  |  | ||||||
| To migrate all the uuid's you need to touch each object and mark it as dirty | To migrate all the DecimalFields you need to touch each object and mark it as dirty | ||||||
| eg:: | eg:: | ||||||
|  |  | ||||||
|     # Doc definition |     # Doc definition | ||||||
| @@ -198,7 +204,7 @@ eg:: | |||||||
|         p._mark_as_changed('balance') |         p._mark_as_changed('balance') | ||||||
|         p.save() |         p.save() | ||||||
|  |  | ||||||
| .. note:: DecimalField's have also been improved with the addition of precision | .. note:: DecimalFields have also been improved with the addition of precision | ||||||
|     and rounding.  See :class:`~mongoengine.fields.DecimalField` for more information. |     and rounding.  See :class:`~mongoengine.fields.DecimalField` for more information. | ||||||
|  |  | ||||||
| `An example test migration for DecimalFields is available on github | `An example test migration for DecimalFields is available on github | ||||||
| @@ -207,7 +213,7 @@ eg:: | |||||||
| Cascading Saves | Cascading Saves | ||||||
| --------------- | --------------- | ||||||
| To improve performance document saves will no longer automatically cascade. | To improve performance document saves will no longer automatically cascade. | ||||||
| Any changes to a Documents references will either have to be saved manually or | Any changes to a Document's references will either have to be saved manually or | ||||||
| you will have to explicitly tell it to cascade on save:: | you will have to explicitly tell it to cascade on save:: | ||||||
|  |  | ||||||
|     # At the class level: |     # At the class level: | ||||||
| @@ -249,7 +255,7 @@ update your code like so: :: | |||||||
|  |  | ||||||
|     # Update example a) assign queryset after a change: |     # Update example a) assign queryset after a change: | ||||||
|     mammals = Animal.objects(type="mammal") |     mammals = Animal.objects(type="mammal") | ||||||
|     carnivores = mammals.filter(order="Carnivora") # Reassign the new queryset so fitler can be applied |     carnivores = mammals.filter(order="Carnivora") # Reassign the new queryset so filter can be applied | ||||||
|     [m for m in carnivores]                        # This will return all carnivores |     [m for m in carnivores]                        # This will return all carnivores | ||||||
|  |  | ||||||
|     # Update example b) chain the queryset: |     # Update example b) chain the queryset: | ||||||
| @@ -270,13 +276,13 @@ queryset you should upgrade to use count:: | |||||||
|     len(Animal.objects(type="mammal")) |     len(Animal.objects(type="mammal")) | ||||||
|  |  | ||||||
|     # New code |     # New code | ||||||
|     Animal.objects(type="mammal").count()) |     Animal.objects(type="mammal").count() | ||||||
|  |  | ||||||
|  |  | ||||||
| .only() now inline with .exclude() | .only() now inline with .exclude() | ||||||
| ---------------------------------- | ---------------------------------- | ||||||
|  |  | ||||||
| The behaviour of `.only()` was highly ambious, now it works in the mirror fashion | The behaviour of `.only()` was highly ambiguous, now it works in mirror fashion | ||||||
| to `.exclude()`.  Chaining `.only()` calls will increase the fields required:: | to `.exclude()`.  Chaining `.only()` calls will increase the fields required:: | ||||||
|  |  | ||||||
|     # Old code |     # Old code | ||||||
| @@ -462,8 +468,8 @@ such the following have been changed: | |||||||
| Default collection naming | Default collection naming | ||||||
| ========================= | ========================= | ||||||
|  |  | ||||||
| Previously it was just lowercase, its now much more pythonic and readable as | Previously it was just lowercase, it's now much more pythonic and readable as | ||||||
| its lowercase and underscores, previously :: | it's lowercase and underscores, previously :: | ||||||
|  |  | ||||||
|     class MyAceDocument(Document): |     class MyAceDocument(Document): | ||||||
|         pass |         pass | ||||||
| @@ -530,5 +536,5 @@ Alternatively, you can rename your collections eg :: | |||||||
| mongodb 1.8 > 2.0 + | mongodb 1.8 > 2.0 + | ||||||
| =================== | =================== | ||||||
|  |  | ||||||
| Its been reported that indexes may need to be recreated to the newer version of indexes. | It's been reported that indexes may need to be recreated to the newer version of indexes. | ||||||
| To do this drop indexes and call ``ensure_indexes`` on each model. | To do this drop indexes and call ``ensure_indexes`` on each model. | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ import django | |||||||
| __all__ = (list(document.__all__) + fields.__all__ + connection.__all__ + | __all__ = (list(document.__all__) + fields.__all__ + connection.__all__ + | ||||||
|            list(queryset.__all__) + signals.__all__ + list(errors.__all__)) |            list(queryset.__all__) + signals.__all__ + list(errors.__all__)) | ||||||
|  |  | ||||||
| VERSION = (0, 8, 3) | VERSION = (0, 8, 7) | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_version(): | def get_version(): | ||||||
|   | |||||||
| @@ -1,12 +1,13 @@ | |||||||
| import weakref | import weakref | ||||||
|  | import functools | ||||||
|  | import itertools | ||||||
| from mongoengine.common import _import_class | from mongoengine.common import _import_class | ||||||
|  |  | ||||||
| __all__ = ("BaseDict", "BaseList") | __all__ = ("BaseDict", "BaseList") | ||||||
|  |  | ||||||
|  |  | ||||||
| class BaseDict(dict): | class BaseDict(dict): | ||||||
|     """A special dict so we can watch any changes |     """A special dict so we can watch any changes""" | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     _dereferenced = False |     _dereferenced = False | ||||||
|     _instance = None |     _instance = None | ||||||
| @@ -21,29 +22,37 @@ class BaseDict(dict): | |||||||
|         self._name = name |         self._name = name | ||||||
|         return super(BaseDict, self).__init__(dict_items) |         return super(BaseDict, self).__init__(dict_items) | ||||||
|  |  | ||||||
|     def __getitem__(self, *args, **kwargs): |     def __getitem__(self, key, *args, **kwargs): | ||||||
|         value = super(BaseDict, self).__getitem__(*args, **kwargs) |         value = super(BaseDict, self).__getitem__(key) | ||||||
|  |  | ||||||
|         EmbeddedDocument = _import_class('EmbeddedDocument') |         EmbeddedDocument = _import_class('EmbeddedDocument') | ||||||
|         if isinstance(value, EmbeddedDocument) and value._instance is None: |         if isinstance(value, EmbeddedDocument) and value._instance is None: | ||||||
|             value._instance = self._instance |             value._instance = self._instance | ||||||
|  |         elif not isinstance(value, BaseDict) and isinstance(value, dict): | ||||||
|  |             value = BaseDict(value, None, '%s.%s' % (self._name, key)) | ||||||
|  |             super(BaseDict, self).__setitem__(key, value) | ||||||
|  |             value._instance = self._instance | ||||||
|  |         elif not isinstance(value, BaseList) and isinstance(value, list): | ||||||
|  |             value = BaseList(value, None, '%s.%s' % (self._name, key)) | ||||||
|  |             super(BaseDict, self).__setitem__(key, value) | ||||||
|  |             value._instance = self._instance | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
|     def __setitem__(self, *args, **kwargs): |     def __setitem__(self, key, value, *args, **kwargs): | ||||||
|         self._mark_as_changed() |         self._mark_as_changed(key) | ||||||
|         return super(BaseDict, self).__setitem__(*args, **kwargs) |         return super(BaseDict, self).__setitem__(key, value) | ||||||
|  |  | ||||||
|     def __delete__(self, *args, **kwargs): |     def __delete__(self, *args, **kwargs): | ||||||
|         self._mark_as_changed() |         self._mark_as_changed() | ||||||
|         return super(BaseDict, self).__delete__(*args, **kwargs) |         return super(BaseDict, self).__delete__(*args, **kwargs) | ||||||
|  |  | ||||||
|     def __delitem__(self, *args, **kwargs): |     def __delitem__(self, key, *args, **kwargs): | ||||||
|         self._mark_as_changed() |         self._mark_as_changed(key) | ||||||
|         return super(BaseDict, self).__delitem__(*args, **kwargs) |         return super(BaseDict, self).__delitem__(key) | ||||||
|  |  | ||||||
|     def __delattr__(self, *args, **kwargs): |     def __delattr__(self, key, *args, **kwargs): | ||||||
|         self._mark_as_changed() |         self._mark_as_changed(key) | ||||||
|         return super(BaseDict, self).__delattr__(*args, **kwargs) |         return super(BaseDict, self).__delattr__(key) | ||||||
|  |  | ||||||
|     def __getstate__(self): |     def __getstate__(self): | ||||||
|         self.instance = None |         self.instance = None | ||||||
| @@ -70,8 +79,11 @@ class BaseDict(dict): | |||||||
|         self._mark_as_changed() |         self._mark_as_changed() | ||||||
|         return super(BaseDict, self).update(*args, **kwargs) |         return super(BaseDict, self).update(*args, **kwargs) | ||||||
|  |  | ||||||
|     def _mark_as_changed(self): |     def _mark_as_changed(self, key=None): | ||||||
|         if hasattr(self._instance, '_mark_as_changed'): |         if hasattr(self._instance, '_mark_as_changed'): | ||||||
|  |             if key: | ||||||
|  |                 self._instance._mark_as_changed('%s.%s' % (self._name, key)) | ||||||
|  |             else: | ||||||
|                 self._instance._mark_as_changed(self._name) |                 self._instance._mark_as_changed(self._name) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -92,21 +104,35 @@ class BaseList(list): | |||||||
|         self._name = name |         self._name = name | ||||||
|         return super(BaseList, self).__init__(list_items) |         return super(BaseList, self).__init__(list_items) | ||||||
|  |  | ||||||
|     def __getitem__(self, *args, **kwargs): |     def __getitem__(self, key, *args, **kwargs): | ||||||
|         value = super(BaseList, self).__getitem__(*args, **kwargs) |         value = super(BaseList, self).__getitem__(key) | ||||||
|  |  | ||||||
|         EmbeddedDocument = _import_class('EmbeddedDocument') |         EmbeddedDocument = _import_class('EmbeddedDocument') | ||||||
|         if isinstance(value, EmbeddedDocument) and value._instance is None: |         if isinstance(value, EmbeddedDocument) and value._instance is None: | ||||||
|             value._instance = self._instance |             value._instance = self._instance | ||||||
|  |         elif not isinstance(value, BaseDict) and isinstance(value, dict): | ||||||
|  |             value = BaseDict(value, None, '%s.%s' % (self._name, key)) | ||||||
|  |             super(BaseList, self).__setitem__(key, value) | ||||||
|  |             value._instance = self._instance | ||||||
|  |         elif not isinstance(value, BaseList) and isinstance(value, list): | ||||||
|  |             value = BaseList(value, None, '%s.%s' % (self._name, key)) | ||||||
|  |             super(BaseList, self).__setitem__(key, value) | ||||||
|  |             value._instance = self._instance | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
|     def __setitem__(self, *args, **kwargs): |     def __setitem__(self, key, value, *args, **kwargs): | ||||||
|  |         if isinstance(key, slice): | ||||||
|             self._mark_as_changed() |             self._mark_as_changed() | ||||||
|         return super(BaseList, self).__setitem__(*args, **kwargs) |         else: | ||||||
|  |             self._mark_as_changed(key) | ||||||
|  |         return super(BaseList, self).__setitem__(key, value) | ||||||
|  |  | ||||||
|     def __delitem__(self, *args, **kwargs): |     def __delitem__(self, key, *args, **kwargs): | ||||||
|  |         if isinstance(key, slice): | ||||||
|             self._mark_as_changed() |             self._mark_as_changed() | ||||||
|         return super(BaseList, self).__delitem__(*args, **kwargs) |         else: | ||||||
|  |             self._mark_as_changed(key) | ||||||
|  |         return super(BaseList, self).__delitem__(key) | ||||||
|  |  | ||||||
|     def __setslice__(self, *args, **kwargs): |     def __setslice__(self, *args, **kwargs): | ||||||
|         self._mark_as_changed() |         self._mark_as_changed() | ||||||
| @@ -153,6 +179,105 @@ class BaseList(list): | |||||||
|         self._mark_as_changed() |         self._mark_as_changed() | ||||||
|         return super(BaseList, self).sort(*args, **kwargs) |         return super(BaseList, self).sort(*args, **kwargs) | ||||||
|  |  | ||||||
|     def _mark_as_changed(self): |     def _mark_as_changed(self, key=None): | ||||||
|         if hasattr(self._instance, '_mark_as_changed'): |         if hasattr(self._instance, '_mark_as_changed'): | ||||||
|  |             if key: | ||||||
|  |                 self._instance._mark_as_changed('%s.%s' % (self._name, key)) | ||||||
|  |             else: | ||||||
|                 self._instance._mark_as_changed(self._name) |                 self._instance._mark_as_changed(self._name) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class StrictDict(object): | ||||||
|  |     __slots__ = () | ||||||
|  |     _special_fields = set(['get', 'pop', 'iteritems', 'items', 'keys', 'create']) | ||||||
|  |     _classes = {} | ||||||
|  |     def __init__(self, **kwargs): | ||||||
|  |         for k,v in kwargs.iteritems(): | ||||||
|  |             setattr(self, k, v) | ||||||
|  |     def __getitem__(self, key): | ||||||
|  |         key = '_reserved_' + key if key in self._special_fields else key | ||||||
|  |         try: | ||||||
|  |             return getattr(self, key) | ||||||
|  |         except AttributeError: | ||||||
|  |             raise KeyError(key) | ||||||
|  |     def __setitem__(self, key, value): | ||||||
|  |         key = '_reserved_' + key if key in self._special_fields else key | ||||||
|  |         return setattr(self, key, value) | ||||||
|  |     def __contains__(self, key): | ||||||
|  |         return hasattr(self, key) | ||||||
|  |     def get(self, key, default=None): | ||||||
|  |         try: | ||||||
|  |             return self[key] | ||||||
|  |         except KeyError: | ||||||
|  |             return default | ||||||
|  |     def pop(self, key, default=None): | ||||||
|  |         v = self.get(key, default) | ||||||
|  |         try: | ||||||
|  |             delattr(self, key) | ||||||
|  |         except AttributeError: | ||||||
|  |             pass | ||||||
|  |         return v | ||||||
|  |     def iteritems(self): | ||||||
|  |         for key in self: | ||||||
|  |             yield key, self[key] | ||||||
|  |     def items(self): | ||||||
|  |         return [(k, self[k]) for k in iter(self)] | ||||||
|  |     def keys(self): | ||||||
|  |         return list(iter(self)) | ||||||
|  |     def __iter__(self): | ||||||
|  |         return (key for key in self.__slots__ if hasattr(self, key)) | ||||||
|  |     def __len__(self): | ||||||
|  |         return len(list(self.iteritems())) | ||||||
|  |     def __eq__(self, other): | ||||||
|  |         return self.items() == other.items() | ||||||
|  |     def __neq__(self, other): | ||||||
|  |         return self.items() != other.items() | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def create(cls, allowed_keys): | ||||||
|  |         allowed_keys_tuple = tuple(('_reserved_' + k if k in cls._special_fields else k) for k in allowed_keys) | ||||||
|  |         allowed_keys = frozenset(allowed_keys_tuple) | ||||||
|  |         if allowed_keys not in cls._classes: | ||||||
|  |             class SpecificStrictDict(cls): | ||||||
|  |                 __slots__ = allowed_keys_tuple | ||||||
|  |                 def __repr__(self): | ||||||
|  |                     return "{%s}" % ', '.join('"{0!s}": {0!r}'.format(k,v) for (k,v) in self.iteritems()) | ||||||
|  |             cls._classes[allowed_keys] = SpecificStrictDict | ||||||
|  |         return cls._classes[allowed_keys] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class SemiStrictDict(StrictDict): | ||||||
|  |     __slots__ = ('_extras') | ||||||
|  |     _classes = {} | ||||||
|  |     def __getattr__(self, attr): | ||||||
|  |         try: | ||||||
|  |             super(SemiStrictDict, self).__getattr__(attr) | ||||||
|  |         except AttributeError: | ||||||
|  |             try: | ||||||
|  |                 return self.__getattribute__('_extras')[attr] | ||||||
|  |             except KeyError as e: | ||||||
|  |                 raise AttributeError(e) | ||||||
|  |     def __setattr__(self, attr, value): | ||||||
|  |         try: | ||||||
|  |             super(SemiStrictDict, self).__setattr__(attr, value) | ||||||
|  |         except AttributeError: | ||||||
|  |             try: | ||||||
|  |                 self._extras[attr] = value | ||||||
|  |             except AttributeError: | ||||||
|  |                 self._extras = {attr: value} | ||||||
|  |  | ||||||
|  |     def __delattr__(self, attr): | ||||||
|  |         try: | ||||||
|  |             super(SemiStrictDict, self).__delattr__(attr) | ||||||
|  |         except AttributeError: | ||||||
|  |             try: | ||||||
|  |                 del self._extras[attr] | ||||||
|  |             except KeyError as e: | ||||||
|  |                 raise AttributeError(e) | ||||||
|  |  | ||||||
|  |     def __iter__(self): | ||||||
|  |         try: | ||||||
|  |             extras_iter = iter(self.__getattribute__('_extras')) | ||||||
|  |         except AttributeError: | ||||||
|  |             extras_iter = () | ||||||
|  |         return itertools.chain(super(SemiStrictDict, self).__iter__(), extras_iter) | ||||||
|   | |||||||
| @@ -1,10 +1,11 @@ | |||||||
| import copy | import copy | ||||||
| import operator | import operator | ||||||
| import numbers | import numbers | ||||||
|  | from collections import Hashable | ||||||
| from functools import partial | from functools import partial | ||||||
|  |  | ||||||
| import pymongo | import pymongo | ||||||
| from bson import json_util | from bson import json_util, ObjectId | ||||||
| from bson.dbref import DBRef | from bson.dbref import DBRef | ||||||
| from bson.son import SON | from bson.son import SON | ||||||
|  |  | ||||||
| @@ -12,11 +13,10 @@ from mongoengine import signals | |||||||
| from mongoengine.common import _import_class | from mongoengine.common import _import_class | ||||||
| from mongoengine.errors import (ValidationError, InvalidDocumentError, | from mongoengine.errors import (ValidationError, InvalidDocumentError, | ||||||
|                                 LookUpError) |                                 LookUpError) | ||||||
| from mongoengine.python_support import (PY3, UNICODE_KWARGS, txt_type, | from mongoengine.python_support import PY3, txt_type | ||||||
|                                         to_str_keys_recursive) |  | ||||||
|  |  | ||||||
| from mongoengine.base.common import get_document, ALLOW_INHERITANCE | from mongoengine.base.common import get_document, ALLOW_INHERITANCE | ||||||
| from mongoengine.base.datastructures import BaseDict, BaseList | from mongoengine.base.datastructures import BaseDict, BaseList, StrictDict, SemiStrictDict | ||||||
| from mongoengine.base.fields import ComplexBaseField | from mongoengine.base.fields import ComplexBaseField | ||||||
|  |  | ||||||
| __all__ = ('BaseDocument', 'NON_FIELD_ERRORS') | __all__ = ('BaseDocument', 'NON_FIELD_ERRORS') | ||||||
| @@ -25,11 +25,12 @@ NON_FIELD_ERRORS = '__all__' | |||||||
|  |  | ||||||
|  |  | ||||||
| class BaseDocument(object): | class BaseDocument(object): | ||||||
|  |     __slots__ = ('_changed_fields', '_initialised', '_created', '_data', | ||||||
|  |                  '_dynamic_fields', '_auto_id_field', '_db_field_map', '__weakref__') | ||||||
|  |  | ||||||
|     _dynamic = False |     _dynamic = False | ||||||
|     _created = True |  | ||||||
|     _dynamic_lock = True |     _dynamic_lock = True | ||||||
|     _initialised = False |     STRICT = False | ||||||
|  |  | ||||||
|     def __init__(self, *args, **values): |     def __init__(self, *args, **values): | ||||||
|         """ |         """ | ||||||
| @@ -38,6 +39,8 @@ class BaseDocument(object): | |||||||
|         :param __auto_convert: Try and will cast python objects to Object types |         :param __auto_convert: Try and will cast python objects to Object types | ||||||
|         :param values: A dictionary of values for the document |         :param values: A dictionary of values for the document | ||||||
|         """ |         """ | ||||||
|  |         self._initialised = False | ||||||
|  |         self._created = True | ||||||
|         if args: |         if args: | ||||||
|             # Combine positional arguments with named arguments. |             # Combine positional arguments with named arguments. | ||||||
|             # We only want named arguments. |             # We only want named arguments. | ||||||
| @@ -48,21 +51,36 @@ class BaseDocument(object): | |||||||
|             for value in args: |             for value in args: | ||||||
|                 name = next(field) |                 name = next(field) | ||||||
|                 if name in values: |                 if name in values: | ||||||
|                     raise TypeError("Multiple values for keyword argument '" + name + "'") |                     raise TypeError( | ||||||
|  |                         "Multiple values for keyword argument '" + name + "'") | ||||||
|                 values[name] = value |                 values[name] = value | ||||||
|         __auto_convert = values.pop("__auto_convert", True) |         __auto_convert = values.pop("__auto_convert", True) | ||||||
|  |  | ||||||
|  |         # 399: set default values only to fields loaded from DB | ||||||
|  |         __only_fields = set(values.pop("__only_fields", values)) | ||||||
|  |  | ||||||
|         signals.pre_init.send(self.__class__, document=self, values=values) |         signals.pre_init.send(self.__class__, document=self, values=values) | ||||||
|  |  | ||||||
|  |         if self.STRICT and not self._dynamic: | ||||||
|  |             self._data = StrictDict.create(allowed_keys=self._fields_ordered)() | ||||||
|  |         else: | ||||||
|  |             self._data = SemiStrictDict.create( | ||||||
|  |                 allowed_keys=self._fields_ordered)() | ||||||
|  |  | ||||||
|  |         _created = values.pop("_created", True) | ||||||
|         self._data = {} |         self._data = {} | ||||||
|         self._dynamic_fields = SON() |         self._dynamic_fields = SON() | ||||||
|  |  | ||||||
|         # Assign default values to instance |         # Assign default values to instance | ||||||
|         for key, field in self._fields.iteritems(): |         for key, field in self._fields.iteritems(): | ||||||
|             if self._db_field_map.get(key, key) in values: |             if self._db_field_map.get(key, key) in __only_fields: | ||||||
|                 continue |                 continue | ||||||
|             value = getattr(self, key, None) |             value = getattr(self, key, None) | ||||||
|             setattr(self, key, value) |             setattr(self, key, value) | ||||||
|  |  | ||||||
|  |         if "_cls" not in values: | ||||||
|  |             self._cls = self._class_name | ||||||
|  |  | ||||||
|         # Set passed values after initialisation |         # Set passed values after initialisation | ||||||
|         if self._dynamic: |         if self._dynamic: | ||||||
|             dynamic_data = {} |             dynamic_data = {} | ||||||
| @@ -96,6 +114,7 @@ class BaseDocument(object): | |||||||
|  |  | ||||||
|         # Flag initialised |         # Flag initialised | ||||||
|         self._initialised = True |         self._initialised = True | ||||||
|  |         self._created = _created | ||||||
|         signals.post_init.send(self.__class__, document=self) |         signals.post_init.send(self.__class__, document=self) | ||||||
|  |  | ||||||
|     def __delattr__(self, *args, **kwargs): |     def __delattr__(self, *args, **kwargs): | ||||||
| @@ -129,17 +148,25 @@ class BaseDocument(object): | |||||||
|                 self._data[name] = value |                 self._data[name] = value | ||||||
|                 if hasattr(self, '_changed_fields'): |                 if hasattr(self, '_changed_fields'): | ||||||
|                     self._mark_as_changed(name) |                     self._mark_as_changed(name) | ||||||
|  |         try: | ||||||
|  |             self__created = self._created | ||||||
|  |         except AttributeError: | ||||||
|  |             self__created = True | ||||||
|  |  | ||||||
|         if (self._is_document and not self._created and |         if (self._is_document and not self__created and | ||||||
|                 name in self._meta.get('shard_key', tuple()) and |                 name in self._meta.get('shard_key', tuple()) and | ||||||
|                 self._data.get(name) != value): |                 self._data.get(name) != value): | ||||||
|             OperationError = _import_class('OperationError') |             OperationError = _import_class('OperationError') | ||||||
|             msg = "Shard Keys are immutable. Tried to update %s" % name |             msg = "Shard Keys are immutable. Tried to update %s" % name | ||||||
|             raise OperationError(msg) |             raise OperationError(msg) | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             self__initialised = self._initialised | ||||||
|  |         except AttributeError: | ||||||
|  |             self__initialised = False | ||||||
|         # Check if the user has created a new instance of a class |         # Check if the user has created a new instance of a class | ||||||
|         if (self._is_document and self._initialised |         if (self._is_document and self__initialised | ||||||
|            and self._created and name == self._meta['id_field']): |                 and self__created and name == self._meta['id_field']): | ||||||
|             super(BaseDocument, self).__setattr__('_created', False) |             super(BaseDocument, self).__setattr__('_created', False) | ||||||
|  |  | ||||||
|         super(BaseDocument, self).__setattr__(name, value) |         super(BaseDocument, self).__setattr__(name, value) | ||||||
| @@ -157,9 +184,11 @@ class BaseDocument(object): | |||||||
|         if isinstance(data["_data"], SON): |         if isinstance(data["_data"], SON): | ||||||
|             data["_data"] = self.__class__._from_son(data["_data"])._data |             data["_data"] = self.__class__._from_son(data["_data"])._data | ||||||
|         for k in ('_changed_fields', '_initialised', '_created', '_data', |         for k in ('_changed_fields', '_initialised', '_created', '_data', | ||||||
|                   '_fields_ordered', '_dynamic_fields'): |                   '_dynamic_fields'): | ||||||
|             if k in data: |             if k in data: | ||||||
|                 setattr(self, k, data[k]) |                 setattr(self, k, data[k]) | ||||||
|  |         if '_fields_ordered' in data: | ||||||
|  |             setattr(type(self), '_fields_ordered', data['_fields_ordered']) | ||||||
|         dynamic_fields = data.get('_dynamic_fields') or SON() |         dynamic_fields = data.get('_dynamic_fields') or SON() | ||||||
|         for k in dynamic_fields.keys(): |         for k in dynamic_fields.keys(): | ||||||
|             setattr(self, k, data["_data"].get(k)) |             setattr(self, k, data["_data"].get(k)) | ||||||
| @@ -181,7 +210,7 @@ class BaseDocument(object): | |||||||
|         """Dictionary-style field access, set a field's value. |         """Dictionary-style field access, set a field's value. | ||||||
|         """ |         """ | ||||||
|         # Ensure that the field exists before settings its value |         # Ensure that the field exists before settings its value | ||||||
|         if name not in self._fields: |         if not self._dynamic and name not in self._fields: | ||||||
|             raise KeyError(name) |             raise KeyError(name) | ||||||
|         return setattr(self, name, value) |         return setattr(self, name, value) | ||||||
|  |  | ||||||
| @@ -213,8 +242,9 @@ class BaseDocument(object): | |||||||
|  |  | ||||||
|     def __eq__(self, other): |     def __eq__(self, other): | ||||||
|         if isinstance(other, self.__class__) and hasattr(other, 'id'): |         if isinstance(other, self.__class__) and hasattr(other, 'id'): | ||||||
|             if self.id == other.id: |             return self.id == other.id | ||||||
|                 return True |         if isinstance(other, DBRef): | ||||||
|  |             return self._get_collection_name() == other.collection and self.id == other.id | ||||||
|         return False |         return False | ||||||
|  |  | ||||||
|     def __ne__(self, other): |     def __ne__(self, other): | ||||||
| @@ -237,20 +267,42 @@ class BaseDocument(object): | |||||||
|         """ |         """ | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     def to_mongo(self): |     def to_mongo(self, use_db_field=True, fields=[]): | ||||||
|         """Return as SON data ready for use with MongoDB. |         """ | ||||||
|  |         Return as SON data ready for use with MongoDB. | ||||||
|         """ |         """ | ||||||
|         data = SON() |         data = SON() | ||||||
|         data["_id"] = None |         data["_id"] = None | ||||||
|         data['_cls'] = self._class_name |         data['_cls'] = self._class_name | ||||||
|  |         EmbeddedDocumentField = _import_class("EmbeddedDocumentField") | ||||||
|  |         # only root fields ['test1.a', 'test2'] => ['test1', 'test2'] | ||||||
|  |         root_fields = set([f.split('.')[0] for f in fields]) | ||||||
|  |  | ||||||
|         for field_name in self: |         for field_name in self: | ||||||
|  |             if root_fields and field_name not in root_fields: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|             value = self._data.get(field_name, None) |             value = self._data.get(field_name, None) | ||||||
|             field = self._fields.get(field_name) |             field = self._fields.get(field_name) | ||||||
|  |  | ||||||
|             if field is None and self._dynamic: |             if field is None and self._dynamic: | ||||||
|                 field = self._dynamic_fields.get(field_name) |                 field = self._dynamic_fields.get(field_name) | ||||||
|  |  | ||||||
|             if value is not None: |             if value is not None: | ||||||
|  |  | ||||||
|  |                 if isinstance(field, (EmbeddedDocumentField)): | ||||||
|  |                     if fields: | ||||||
|  |                         key = '%s.' % field_name | ||||||
|  |                         embedded_fields = [ | ||||||
|  |                             i.replace(key, '') for i in fields | ||||||
|  |                             if i.startswith(key)] | ||||||
|  |  | ||||||
|  |                     else: | ||||||
|  |                         embedded_fields = [] | ||||||
|  |  | ||||||
|  |                     value = field.to_mongo(value, use_db_field=use_db_field, | ||||||
|  |                                            fields=embedded_fields) | ||||||
|  |                 else: | ||||||
|                     value = field.to_mongo(value) |                     value = field.to_mongo(value) | ||||||
|  |  | ||||||
|             # Handle self generating fields |             # Handle self generating fields | ||||||
| @@ -259,7 +311,10 @@ class BaseDocument(object): | |||||||
|                 self._data[field_name] = value |                 self._data[field_name] = value | ||||||
|  |  | ||||||
|             if value is not None: |             if value is not None: | ||||||
|  |                 if use_db_field: | ||||||
|                     data[field.db_field] = value |                     data[field.db_field] = value | ||||||
|  |                 else: | ||||||
|  |                     data[field.name] = value | ||||||
|  |  | ||||||
|         # If "_id" has not been set, then try and set it |         # If "_id" has not been set, then try and set it | ||||||
|         Document = _import_class("Document") |         Document = _import_class("Document") | ||||||
| @@ -294,7 +349,8 @@ class BaseDocument(object): | |||||||
|                    self._data.get(name)) for name in self._fields_ordered] |                    self._data.get(name)) for name in self._fields_ordered] | ||||||
|  |  | ||||||
|         EmbeddedDocumentField = _import_class("EmbeddedDocumentField") |         EmbeddedDocumentField = _import_class("EmbeddedDocumentField") | ||||||
|         GenericEmbeddedDocumentField = _import_class("GenericEmbeddedDocumentField") |         GenericEmbeddedDocumentField = _import_class( | ||||||
|  |             "GenericEmbeddedDocumentField") | ||||||
|  |  | ||||||
|         for field, value in fields: |         for field, value in fields: | ||||||
|             if value is not None: |             if value is not None: | ||||||
| @@ -316,14 +372,18 @@ class BaseDocument(object): | |||||||
|             pk = "None" |             pk = "None" | ||||||
|             if hasattr(self, 'pk'): |             if hasattr(self, 'pk'): | ||||||
|                 pk = self.pk |                 pk = self.pk | ||||||
|             elif self._instance: |             elif self._instance and hasattr(self._instance, 'pk'): | ||||||
|                 pk = self._instance.pk |                 pk = self._instance.pk | ||||||
|             message = "ValidationError (%s:%s) " % (self._class_name, pk) |             message = "ValidationError (%s:%s) " % (self._class_name, pk) | ||||||
|             raise ValidationError(message, errors=errors) |             raise ValidationError(message, errors=errors) | ||||||
|  |  | ||||||
|     def to_json(self): |     def to_json(self, *args, **kwargs): | ||||||
|         """Converts a document to JSON""" |         """Converts a document to JSON. | ||||||
|         return json_util.dumps(self.to_mongo()) |         :param use_db_field: Set to True by default but enables the output of the json structure with the field names and not the mongodb store db_names in case of set to False | ||||||
|  |         """ | ||||||
|  |         use_db_field = kwargs.pop('use_db_field') if kwargs.has_key( | ||||||
|  |             'use_db_field') else True | ||||||
|  |         return json_util.dumps(self.to_mongo(use_db_field),  *args, **kwargs) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def from_json(cls, json_data): |     def from_json(cls, json_data): | ||||||
| @@ -369,72 +429,104 @@ class BaseDocument(object): | |||||||
|         """ |         """ | ||||||
|         if not key: |         if not key: | ||||||
|             return |             return | ||||||
|  |  | ||||||
|  |         if not hasattr(self, '_changed_fields'): | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         if '.' in key: | ||||||
|  |             key, rest = key.split('.', 1) | ||||||
|             key = self._db_field_map.get(key, key) |             key = self._db_field_map.get(key, key) | ||||||
|         if (hasattr(self, '_changed_fields') and |             key = '%s.%s' % (key, rest) | ||||||
|            key not in self._changed_fields): |         else: | ||||||
|  |             key = self._db_field_map.get(key, key) | ||||||
|  |  | ||||||
|  |         if key not in self._changed_fields: | ||||||
|             self._changed_fields.append(key) |             self._changed_fields.append(key) | ||||||
|  |  | ||||||
|     def _clear_changed_fields(self): |     def _clear_changed_fields(self): | ||||||
|  |         """Using get_changed_fields iterate and remove any fields that are | ||||||
|  |         marked as changed""" | ||||||
|  |         for changed in self._get_changed_fields(): | ||||||
|  |             parts = changed.split(".") | ||||||
|  |             data = self | ||||||
|  |             for part in parts: | ||||||
|  |                 if isinstance(data, list): | ||||||
|  |                     try: | ||||||
|  |                         data = data[int(part)] | ||||||
|  |                     except IndexError: | ||||||
|  |                         data = None | ||||||
|  |                 elif isinstance(data, dict): | ||||||
|  |                     data = data.get(part, None) | ||||||
|  |                 else: | ||||||
|  |                     data = getattr(data, part, None) | ||||||
|  |                 if hasattr(data, "_changed_fields"): | ||||||
|  |                     if hasattr(data, "_is_document") and data._is_document: | ||||||
|  |                         continue | ||||||
|  |                     data._changed_fields = [] | ||||||
|         self._changed_fields = [] |         self._changed_fields = [] | ||||||
|         EmbeddedDocumentField = _import_class("EmbeddedDocumentField") |  | ||||||
|         for field_name, field in self._fields.iteritems(): |     def _nestable_types_changed_fields(self, changed_fields, key, data, inspected): | ||||||
|             if (isinstance(field, ComplexBaseField) and |         # Loop list / dict fields as they contain documents | ||||||
|                isinstance(field.field, EmbeddedDocumentField)): |         # Determine the iterator to use | ||||||
|                 field_value = getattr(self, field_name, None) |         if not hasattr(data, 'items'): | ||||||
|                 if field_value: |             iterator = enumerate(data) | ||||||
|                     for idx in (field_value if isinstance(field_value, dict) |         else: | ||||||
|                                 else xrange(len(field_value))): |             iterator = data.iteritems() | ||||||
|                         field_value[idx]._clear_changed_fields() |  | ||||||
|             elif isinstance(field, EmbeddedDocumentField): |         for index, value in iterator: | ||||||
|                 field_value = getattr(self, field_name, None) |             list_key = "%s%s." % (key, index) | ||||||
|                 if field_value: |             # don't check anything lower if this key is already marked | ||||||
|                     field_value._clear_changed_fields() |             # as changed. | ||||||
|  |             if list_key[:-1] in changed_fields: | ||||||
|  |                 continue | ||||||
|  |             if hasattr(value, '_get_changed_fields'): | ||||||
|  |                 changed = value._get_changed_fields(inspected) | ||||||
|  |                 changed_fields += ["%s%s" % (list_key, k) | ||||||
|  |                                    for k in changed if k] | ||||||
|  |             elif isinstance(value, (list, tuple, dict)): | ||||||
|  |                 self._nestable_types_changed_fields( | ||||||
|  |                     changed_fields, list_key, value, inspected) | ||||||
|  |  | ||||||
|     def _get_changed_fields(self, inspected=None): |     def _get_changed_fields(self, inspected=None): | ||||||
|         """Returns a list of all fields that have explicitly been changed. |         """Returns a list of all fields that have explicitly been changed. | ||||||
|         """ |         """ | ||||||
|         EmbeddedDocument = _import_class("EmbeddedDocument") |         EmbeddedDocument = _import_class("EmbeddedDocument") | ||||||
|         DynamicEmbeddedDocument = _import_class("DynamicEmbeddedDocument") |         DynamicEmbeddedDocument = _import_class("DynamicEmbeddedDocument") | ||||||
|         _changed_fields = [] |         ReferenceField = _import_class("ReferenceField") | ||||||
|         _changed_fields += getattr(self, '_changed_fields', []) |         changed_fields = [] | ||||||
|  |         changed_fields += getattr(self, '_changed_fields', []) | ||||||
|  |  | ||||||
|         inspected = inspected or set() |         inspected = inspected or set() | ||||||
|         if hasattr(self, 'id'): |         if hasattr(self, 'id') and isinstance(self.id, Hashable): | ||||||
|             if self.id in inspected: |             if self.id in inspected: | ||||||
|                 return _changed_fields |                 return changed_fields | ||||||
|             inspected.add(self.id) |             inspected.add(self.id) | ||||||
|  |  | ||||||
|         for field_name in self._fields_ordered: |         for field_name in self._fields_ordered: | ||||||
|  |  | ||||||
|             db_field_name = self._db_field_map.get(field_name, field_name) |             db_field_name = self._db_field_map.get(field_name, field_name) | ||||||
|             key = '%s.' % db_field_name |             key = '%s.' % db_field_name | ||||||
|             field = self._data.get(field_name, None) |             data = self._data.get(field_name, None) | ||||||
|             if hasattr(field, 'id'): |             field = self._fields.get(field_name) | ||||||
|                 if field.id in inspected: |  | ||||||
|                     continue |  | ||||||
|                 inspected.add(field.id) |  | ||||||
|  |  | ||||||
|             if (isinstance(field, (EmbeddedDocument, DynamicEmbeddedDocument)) |             if hasattr(data, 'id'): | ||||||
|                and db_field_name not in _changed_fields): |                 if data.id in inspected: | ||||||
|                  # Find all embedded fields that have been changed |  | ||||||
|                 changed = field._get_changed_fields(inspected) |  | ||||||
|                 _changed_fields += ["%s%s" % (key, k) for k in changed if k] |  | ||||||
|             elif (isinstance(field, (list, tuple, dict)) and |  | ||||||
|                     db_field_name not in _changed_fields): |  | ||||||
|                 # Loop list / dict fields as they contain documents |  | ||||||
|                 # Determine the iterator to use |  | ||||||
|                 if not hasattr(field, 'items'): |  | ||||||
|                     iterator = enumerate(field) |  | ||||||
|                 else: |  | ||||||
|                     iterator = field.iteritems() |  | ||||||
|                 for index, value in iterator: |  | ||||||
|                     if not hasattr(value, '_get_changed_fields'): |  | ||||||
|                     continue |                     continue | ||||||
|                     list_key = "%s%s." % (key, index) |                 inspected.add(data.id) | ||||||
|                     changed = value._get_changed_fields(inspected) |             if isinstance(field, ReferenceField): | ||||||
|                     _changed_fields += ["%s%s" % (list_key, k) |                 continue | ||||||
|                                         for k in changed if k] |             elif (isinstance(data, (EmbeddedDocument, DynamicEmbeddedDocument)) | ||||||
|         return _changed_fields |                   and db_field_name not in changed_fields): | ||||||
|  |                  # Find all embedded fields that have been changed | ||||||
|  |                 changed = data._get_changed_fields(inspected) | ||||||
|  |                 changed_fields += ["%s%s" % (key, k) for k in changed if k] | ||||||
|  |             elif (isinstance(data, (list, tuple, dict)) and | ||||||
|  |                     db_field_name not in changed_fields): | ||||||
|  |                 if (hasattr(field, 'field') and | ||||||
|  |                         isinstance(field.field, ReferenceField)): | ||||||
|  |                     continue | ||||||
|  |                 self._nestable_types_changed_fields( | ||||||
|  |                     changed_fields, key, data, inspected) | ||||||
|  |         return changed_fields | ||||||
|  |  | ||||||
|     def _delta(self): |     def _delta(self): | ||||||
|         """Returns the delta (set, unset) of the changes for a document. |         """Returns the delta (set, unset) of the changes for a document. | ||||||
| @@ -454,10 +546,13 @@ class BaseDocument(object): | |||||||
|                 d = doc |                 d = doc | ||||||
|                 new_path = [] |                 new_path = [] | ||||||
|                 for p in parts: |                 for p in parts: | ||||||
|                     if isinstance(d, DBRef): |                     if isinstance(d, (ObjectId, DBRef)): | ||||||
|                         break |                         break | ||||||
|                     elif isinstance(d, list) and p.isdigit(): |                     elif isinstance(d, list) and p.isdigit(): | ||||||
|  |                         try: | ||||||
|                             d = d[int(p)] |                             d = d[int(p)] | ||||||
|  |                         except IndexError: | ||||||
|  |                             d = None | ||||||
|                     elif hasattr(d, 'get'): |                     elif hasattr(d, 'get'): | ||||||
|                         d = d.get(p) |                         d = d.get(p) | ||||||
|                     new_path.append(p) |                     new_path.append(p) | ||||||
| @@ -522,7 +617,7 @@ class BaseDocument(object): | |||||||
|         return cls._meta.get('collection', None) |         return cls._meta.get('collection', None) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def _from_son(cls, son, _auto_dereference=True): |     def _from_son(cls, son, _auto_dereference=True, only_fields=[]): | ||||||
|         """Create an instance of a Document (subclass) from a PyMongo SON. |         """Create an instance of a Document (subclass) from a PyMongo SON. | ||||||
|         """ |         """ | ||||||
|  |  | ||||||
| @@ -530,10 +625,6 @@ class BaseDocument(object): | |||||||
|         # class if unavailable |         # class if unavailable | ||||||
|         class_name = son.get('_cls', cls._class_name) |         class_name = son.get('_cls', cls._class_name) | ||||||
|         data = dict(("%s" % key, value) for key, value in son.iteritems()) |         data = dict(("%s" % key, value) for key, value in son.iteritems()) | ||||||
|         if not UNICODE_KWARGS: |  | ||||||
|             # python 2.6.4 and lower cannot handle unicode keys |  | ||||||
|             # passed to class constructor example: cls(**data) |  | ||||||
|             to_str_keys_recursive(data) |  | ||||||
|  |  | ||||||
|         # Return correct subclass for document type |         # Return correct subclass for document type | ||||||
|         if class_name != cls._class_name: |         if class_name != cls._class_name: | ||||||
| @@ -563,6 +654,8 @@ class BaseDocument(object): | |||||||
|                     default = default() |                     default = default() | ||||||
|                 if isinstance(default, BaseDocument): |                 if isinstance(default, BaseDocument): | ||||||
|                     changed_fields.append(field_name) |                     changed_fields.append(field_name) | ||||||
|  |                 elif not only_fields or field_name in only_fields: | ||||||
|  |                     changed_fields.append(field_name) | ||||||
|  |  | ||||||
|         if errors_dict: |         if errors_dict: | ||||||
|             errors = "\n".join(["%s - %s" % (k, v) |             errors = "\n".join(["%s - %s" % (k, v) | ||||||
| @@ -571,11 +664,14 @@ class BaseDocument(object): | |||||||
|                    % (cls._class_name, errors)) |                    % (cls._class_name, errors)) | ||||||
|             raise InvalidDocumentError(msg) |             raise InvalidDocumentError(msg) | ||||||
|  |  | ||||||
|         obj = cls(__auto_convert=False, **data) |         if cls.STRICT: | ||||||
|  |             data = dict((k, v) | ||||||
|  |                         for k, v in data.iteritems() if k in cls._fields) | ||||||
|  |         obj = cls(__auto_convert=False, _created=False, __only_fields=only_fields, **data) | ||||||
|         obj._changed_fields = changed_fields |         obj._changed_fields = changed_fields | ||||||
|         obj._created = False |  | ||||||
|         if not _auto_dereference: |         if not _auto_dereference: | ||||||
|             obj._fields = fields |             obj._fields = fields | ||||||
|  |  | ||||||
|         return obj |         return obj | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
| @@ -623,22 +719,30 @@ class BaseDocument(object): | |||||||
|         # Check to see if we need to include _cls |         # Check to see if we need to include _cls | ||||||
|         allow_inheritance = cls._meta.get('allow_inheritance', |         allow_inheritance = cls._meta.get('allow_inheritance', | ||||||
|                                           ALLOW_INHERITANCE) |                                           ALLOW_INHERITANCE) | ||||||
|         include_cls = allow_inheritance and not spec.get('sparse', False) |         include_cls = (allow_inheritance and not spec.get('sparse', False) and | ||||||
|  |                        spec.get('cls',  True)) | ||||||
|  |  | ||||||
|  |         # 733: don't include cls if index_cls is False unless there is an explicit cls with the index | ||||||
|  |         include_cls = include_cls and (spec.get('cls', False) or cls._meta.get('index_cls', True)) | ||||||
|  |         if "cls" in spec: | ||||||
|  |             spec.pop('cls') | ||||||
|         for key in spec['fields']: |         for key in spec['fields']: | ||||||
|             # If inherited spec continue |             # If inherited spec continue | ||||||
|             if isinstance(key, (list, tuple)): |             if isinstance(key, (list, tuple)): | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             # ASCENDING from +, |             # ASCENDING from + | ||||||
|             # DESCENDING from - |             # DESCENDING from - | ||||||
|             # GEO2D from * |             # GEO2D from * | ||||||
|  |             # TEXT from $ | ||||||
|             direction = pymongo.ASCENDING |             direction = pymongo.ASCENDING | ||||||
|             if key.startswith("-"): |             if key.startswith("-"): | ||||||
|                 direction = pymongo.DESCENDING |                 direction = pymongo.DESCENDING | ||||||
|             elif key.startswith("*"): |             elif key.startswith("*"): | ||||||
|                 direction = pymongo.GEO2D |                 direction = pymongo.GEO2D | ||||||
|             if key.startswith(("+", "-", "*")): |             elif key.startswith("$"): | ||||||
|  |                 direction = pymongo.TEXT | ||||||
|  |             if key.startswith(("+", "-", "*", "$")): | ||||||
|                 key = key[1:] |                 key = key[1:] | ||||||
|  |  | ||||||
|             # Use real field name, do it manually because we need field |             # Use real field name, do it manually because we need field | ||||||
| @@ -649,8 +753,14 @@ class BaseDocument(object): | |||||||
|                 fields = [] |                 fields = [] | ||||||
|             else: |             else: | ||||||
|                 fields = cls._lookup_field(parts) |                 fields = cls._lookup_field(parts) | ||||||
|                 parts = [field if field == '_id' else field.db_field |                 parts = [] | ||||||
|                          for field in fields] |                 for field in fields: | ||||||
|  |                     try: | ||||||
|  |                         if field != "_id": | ||||||
|  |                             field = field.db_field | ||||||
|  |                     except AttributeError: | ||||||
|  |                         pass | ||||||
|  |                     parts.append(field) | ||||||
|                 key = '.'.join(parts) |                 key = '.'.join(parts) | ||||||
|             index_list.append((key, direction)) |             index_list.append((key, direction)) | ||||||
|  |  | ||||||
| @@ -723,7 +833,8 @@ class BaseDocument(object): | |||||||
|         geo_field_type_names = ["EmbeddedDocumentField", "GeoPointField", |         geo_field_type_names = ["EmbeddedDocumentField", "GeoPointField", | ||||||
|                                 "PointField", "LineStringField", "PolygonField"] |                                 "PointField", "LineStringField", "PolygonField"] | ||||||
|  |  | ||||||
|         geo_field_types = tuple([_import_class(field) for field in geo_field_type_names]) |         geo_field_types = tuple([_import_class(field) | ||||||
|  |                                  for field in geo_field_type_names]) | ||||||
|  |  | ||||||
|         for field in cls._fields.values(): |         for field in cls._fields.values(): | ||||||
|             if not isinstance(field, geo_field_types): |             if not isinstance(field, geo_field_types): | ||||||
| @@ -733,7 +844,8 @@ class BaseDocument(object): | |||||||
|                 if field_cls in inspected: |                 if field_cls in inspected: | ||||||
|                     continue |                     continue | ||||||
|                 if hasattr(field_cls, '_geo_indices'): |                 if hasattr(field_cls, '_geo_indices'): | ||||||
|                     geo_indices += field_cls._geo_indices(inspected, parent_field=field.db_field) |                     geo_indices += field_cls._geo_indices( | ||||||
|  |                         inspected, parent_field=field.db_field) | ||||||
|             elif field._geo_index: |             elif field._geo_index: | ||||||
|                 field_name = field.db_field |                 field_name = field.db_field | ||||||
|                 if parent_field: |                 if parent_field: | ||||||
| @@ -747,6 +859,9 @@ class BaseDocument(object): | |||||||
|         """Lookup a field based on its attribute and return a list containing |         """Lookup a field based on its attribute and return a list containing | ||||||
|         the field's parents and the field. |         the field's parents and the field. | ||||||
|         """ |         """ | ||||||
|  |  | ||||||
|  |         ListField = _import_class("ListField") | ||||||
|  |  | ||||||
|         if not isinstance(parts, (list, tuple)): |         if not isinstance(parts, (list, tuple)): | ||||||
|             parts = [parts] |             parts = [parts] | ||||||
|         fields = [] |         fields = [] | ||||||
| @@ -754,7 +869,7 @@ class BaseDocument(object): | |||||||
|  |  | ||||||
|         for field_name in parts: |         for field_name in parts: | ||||||
|             # Handle ListField indexing: |             # Handle ListField indexing: | ||||||
|             if field_name.isdigit(): |             if field_name.isdigit() and isinstance(field, ListField): | ||||||
|                 new_field = field.field |                 new_field = field.field | ||||||
|                 fields.append(field_name) |                 fields.append(field_name) | ||||||
|                 continue |                 continue | ||||||
| @@ -784,8 +899,17 @@ class BaseDocument(object): | |||||||
|                    # Look up subfield on the previous field |                    # Look up subfield on the previous field | ||||||
|                     new_field = field.lookup_member(field_name) |                     new_field = field.lookup_member(field_name) | ||||||
|                 if not new_field and isinstance(field, ComplexBaseField): |                 if not new_field and isinstance(field, ComplexBaseField): | ||||||
|  |                     if hasattr(field.field, 'document_type') and cls._dynamic \ | ||||||
|  |                             and field.field.document_type._dynamic: | ||||||
|  |                         DynamicField = _import_class('DynamicField') | ||||||
|  |                         new_field = DynamicField(db_field=field_name) | ||||||
|  |                     else: | ||||||
|                         fields.append(field_name) |                         fields.append(field_name) | ||||||
|                         continue |                         continue | ||||||
|  |                 elif not new_field and hasattr(field, 'document_type') and cls._dynamic \ | ||||||
|  |                         and field.document_type._dynamic: | ||||||
|  |                     DynamicField = _import_class('DynamicField') | ||||||
|  |                     new_field = DynamicField(db_field=field_name) | ||||||
|                 elif not new_field: |                 elif not new_field: | ||||||
|                     raise LookUpError('Cannot resolve field "%s"' |                     raise LookUpError('Cannot resolve field "%s"' | ||||||
|                                       % field_name) |                                       % field_name) | ||||||
| @@ -805,7 +929,11 @@ class BaseDocument(object): | |||||||
|         """Dynamically set the display value for a field with choices""" |         """Dynamically set the display value for a field with choices""" | ||||||
|         for attr_name, field in self._fields.items(): |         for attr_name, field in self._fields.items(): | ||||||
|             if field.choices: |             if field.choices: | ||||||
|                 setattr(self, |                 if self._dynamic: | ||||||
|  |                     obj = self | ||||||
|  |                 else: | ||||||
|  |                     obj = type(self) | ||||||
|  |                 setattr(obj, | ||||||
|                         'get_%s_display' % attr_name, |                         'get_%s_display' % attr_name, | ||||||
|                         partial(self.__get_field_display, field=field)) |                         partial(self.__get_field_display, field=field)) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -11,10 +11,12 @@ from mongoengine.errors import ValidationError | |||||||
| from mongoengine.base.common import ALLOW_INHERITANCE | from mongoengine.base.common import ALLOW_INHERITANCE | ||||||
| from mongoengine.base.datastructures import BaseDict, BaseList | from mongoengine.base.datastructures import BaseDict, BaseList | ||||||
|  |  | ||||||
| __all__ = ("BaseField", "ComplexBaseField", "ObjectIdField", "GeoJsonBaseField") | __all__ = ("BaseField", "ComplexBaseField", | ||||||
|  |            "ObjectIdField", "GeoJsonBaseField") | ||||||
|  |  | ||||||
|  |  | ||||||
| class BaseField(object): | class BaseField(object): | ||||||
|  |  | ||||||
|     """A base class for fields in a MongoDB document. Instances of this class |     """A base class for fields in a MongoDB document. Instances of this class | ||||||
|     may be added to subclasses of `Document` to define a document's schema. |     may be added to subclasses of `Document` to define a document's schema. | ||||||
|  |  | ||||||
| @@ -43,7 +45,7 @@ class BaseField(object): | |||||||
|         :param required: If the field is required. Whether it has to have a |         :param required: If the field is required. Whether it has to have a | ||||||
|             value or not. Defaults to False. |             value or not. Defaults to False. | ||||||
|         :param default: (optional) The default value for this field if no value |         :param default: (optional) The default value for this field if no value | ||||||
|             has been set (or if the value has been unset).  It Can be a |             has been set (or if the value has been unset).  It can be a | ||||||
|             callable. |             callable. | ||||||
|         :param unique: Is the field value unique or not.  Defaults to False. |         :param unique: Is the field value unique or not.  Defaults to False. | ||||||
|         :param unique_with: (optional) The other field this field should be |         :param unique_with: (optional) The other field this field should be | ||||||
| @@ -60,6 +62,7 @@ class BaseField(object): | |||||||
|             used when generating model forms from the document model. |             used when generating model forms from the document model. | ||||||
|         """ |         """ | ||||||
|         self.db_field = (db_field or name) if not primary_key else '_id' |         self.db_field = (db_field or name) if not primary_key else '_id' | ||||||
|  |  | ||||||
|         if name: |         if name: | ||||||
|             msg = "Fields' 'name' attribute deprecated in favour of 'db_field'" |             msg = "Fields' 'name' attribute deprecated in favour of 'db_field'" | ||||||
|             warnings.warn(msg, DeprecationWarning) |             warnings.warn(msg, DeprecationWarning) | ||||||
| @@ -89,12 +92,7 @@ class BaseField(object): | |||||||
|             return self |             return self | ||||||
|  |  | ||||||
|         # Get value from document instance if available |         # Get value from document instance if available | ||||||
|         value = instance._data.get(self.name) |         return instance._data.get(self.name) | ||||||
|  |  | ||||||
|         EmbeddedDocument = _import_class('EmbeddedDocument') |  | ||||||
|         if isinstance(value, EmbeddedDocument) and value._instance is None: |  | ||||||
|             value._instance = weakref.proxy(instance) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def __set__(self, instance, value): |     def __set__(self, instance, value): | ||||||
|         """Descriptor for assigning a value to a field in a document. |         """Descriptor for assigning a value to a field in a document. | ||||||
| @@ -116,6 +114,10 @@ class BaseField(object): | |||||||
|                 # Values cant be compared eg: naive and tz datetimes |                 # Values cant be compared eg: naive and tz datetimes | ||||||
|                 # So mark it as changed |                 # So mark it as changed | ||||||
|                 instance._mark_as_changed(self.name) |                 instance._mark_as_changed(self.name) | ||||||
|  |  | ||||||
|  |         EmbeddedDocument = _import_class('EmbeddedDocument') | ||||||
|  |         if isinstance(value, EmbeddedDocument): | ||||||
|  |             value._instance = weakref.proxy(instance) | ||||||
|         instance._data[self.name] = value |         instance._data[self.name] = value | ||||||
|  |  | ||||||
|     def error(self, message="", errors=None, field_name=None): |     def error(self, message="", errors=None, field_name=None): | ||||||
| @@ -176,6 +178,7 @@ class BaseField(object): | |||||||
|  |  | ||||||
|  |  | ||||||
| class ComplexBaseField(BaseField): | class ComplexBaseField(BaseField): | ||||||
|  |  | ||||||
|     """Handles complex fields, such as lists / dictionaries. |     """Handles complex fields, such as lists / dictionaries. | ||||||
|  |  | ||||||
|     Allows for nesting of embedded documents inside complex types. |     Allows for nesting of embedded documents inside complex types. | ||||||
| @@ -186,7 +189,6 @@ class ComplexBaseField(BaseField): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     field = None |     field = None | ||||||
|     __dereference = False |  | ||||||
|  |  | ||||||
|     def __get__(self, instance, owner): |     def __get__(self, instance, owner): | ||||||
|         """Descriptor to automatically dereference references. |         """Descriptor to automatically dereference references. | ||||||
| @@ -201,9 +203,11 @@ class ComplexBaseField(BaseField): | |||||||
|                        (self.field is None or isinstance(self.field, |                        (self.field is None or isinstance(self.field, | ||||||
|                                                          (GenericReferenceField, ReferenceField)))) |                                                          (GenericReferenceField, ReferenceField)))) | ||||||
|  |  | ||||||
|  |         _dereference = _import_class("DeReference")() | ||||||
|  |  | ||||||
|         self._auto_dereference = instance._fields[self.name]._auto_dereference |         self._auto_dereference = instance._fields[self.name]._auto_dereference | ||||||
|         if not self.__dereference and instance._initialised and dereference: |         if instance._initialised and dereference and instance._data.get(self.name): | ||||||
|             instance._data[self.name] = self._dereference( |             instance._data[self.name] = _dereference( | ||||||
|                 instance._data.get(self.name), max_depth=1, instance=instance, |                 instance._data.get(self.name), max_depth=1, instance=instance, | ||||||
|                 name=self.name |                 name=self.name | ||||||
|             ) |             ) | ||||||
| @@ -222,7 +226,7 @@ class ComplexBaseField(BaseField): | |||||||
|         if (self._auto_dereference and instance._initialised and |         if (self._auto_dereference and instance._initialised and | ||||||
|                 isinstance(value, (BaseList, BaseDict)) |                 isinstance(value, (BaseList, BaseDict)) | ||||||
|                 and not value._dereferenced): |                 and not value._dereferenced): | ||||||
|             value = self._dereference( |             value = _dereference( | ||||||
|                 value, max_depth=1, instance=instance, name=self.name |                 value, max_depth=1, instance=instance, name=self.name | ||||||
|             ) |             ) | ||||||
|             value._dereferenced = True |             value._dereferenced = True | ||||||
| @@ -382,15 +386,9 @@ class ComplexBaseField(BaseField): | |||||||
|  |  | ||||||
|     owner_document = property(_get_owner_document, _set_owner_document) |     owner_document = property(_get_owner_document, _set_owner_document) | ||||||
|  |  | ||||||
|     @property |  | ||||||
|     def _dereference(self,): |  | ||||||
|         if not self.__dereference: |  | ||||||
|             DeReference = _import_class("DeReference") |  | ||||||
|             self.__dereference = DeReference()  # Cached |  | ||||||
|         return self.__dereference |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ObjectIdField(BaseField): | class ObjectIdField(BaseField): | ||||||
|  |  | ||||||
|     """A field wrapper around MongoDB's ObjectIds. |     """A field wrapper around MongoDB's ObjectIds. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
| @@ -419,6 +417,7 @@ class ObjectIdField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class GeoJsonBaseField(BaseField): | class GeoJsonBaseField(BaseField): | ||||||
|  |  | ||||||
|     """A geo json field storing a geojson style object. |     """A geo json field storing a geojson style object. | ||||||
|     .. versionadded:: 0.8 |     .. versionadded:: 0.8 | ||||||
|     """ |     """ | ||||||
| @@ -442,7 +441,8 @@ class GeoJsonBaseField(BaseField): | |||||||
|         if isinstance(value, dict): |         if isinstance(value, dict): | ||||||
|             if set(value.keys()) == set(['type', 'coordinates']): |             if set(value.keys()) == set(['type', 'coordinates']): | ||||||
|                 if value['type'] != self._type: |                 if value['type'] != self._type: | ||||||
|                     self.error('%s type must be "%s"' % (self._name, self._type)) |                     self.error('%s type must be "%s"' % | ||||||
|  |                                (self._name, self._type)) | ||||||
|                 return self.validate(value['coordinates']) |                 return self.validate(value['coordinates']) | ||||||
|             else: |             else: | ||||||
|                 self.error('%s can only accept a valid GeoJson dictionary' |                 self.error('%s can only accept a valid GeoJson dictionary' | ||||||
| @@ -457,7 +457,7 @@ class GeoJsonBaseField(BaseField): | |||||||
|         if error: |         if error: | ||||||
|             self.error(error) |             self.error(error) | ||||||
|  |  | ||||||
|     def _validate_polygon(self, value): |     def _validate_polygon(self, value, top_level=True): | ||||||
|         if not isinstance(value, (list, tuple)): |         if not isinstance(value, (list, tuple)): | ||||||
|             return 'Polygons must contain list of linestrings' |             return 'Polygons must contain list of linestrings' | ||||||
|  |  | ||||||
| @@ -475,7 +475,10 @@ class GeoJsonBaseField(BaseField): | |||||||
|             if error and error not in errors: |             if error and error not in errors: | ||||||
|                 errors.append(error) |                 errors.append(error) | ||||||
|         if errors: |         if errors: | ||||||
|  |             if top_level: | ||||||
|                 return "Invalid Polygon:\n%s" % ", ".join(errors) |                 return "Invalid Polygon:\n%s" % ", ".join(errors) | ||||||
|  |             else: | ||||||
|  |                 return "%s" % ", ".join(errors) | ||||||
|  |  | ||||||
|     def _validate_linestring(self, value, top_level=True): |     def _validate_linestring(self, value, top_level=True): | ||||||
|         """Validates a linestring""" |         """Validates a linestring""" | ||||||
| @@ -509,6 +512,66 @@ class GeoJsonBaseField(BaseField): | |||||||
|               not isinstance(value[1], (float, int))): |               not isinstance(value[1], (float, int))): | ||||||
|             return "Both values (%s) in point must be float or int" % repr(value) |             return "Both values (%s) in point must be float or int" % repr(value) | ||||||
|  |  | ||||||
|  |     def _validate_multipoint(self, value): | ||||||
|  |         if not isinstance(value, (list, tuple)): | ||||||
|  |             return 'MultiPoint must be a list of Point' | ||||||
|  |  | ||||||
|  |         # Quick and dirty validator | ||||||
|  |         try: | ||||||
|  |             value[0][0] | ||||||
|  |         except: | ||||||
|  |             return "Invalid MultiPoint must contain at least one valid point" | ||||||
|  |  | ||||||
|  |         errors = [] | ||||||
|  |         for point in value: | ||||||
|  |             error = self._validate_point(point) | ||||||
|  |             if error and error not in errors: | ||||||
|  |                 errors.append(error) | ||||||
|  |  | ||||||
|  |         if errors: | ||||||
|  |             return "%s" % ", ".join(errors) | ||||||
|  |  | ||||||
|  |     def _validate_multilinestring(self, value, top_level=True): | ||||||
|  |         if not isinstance(value, (list, tuple)): | ||||||
|  |             return 'MultiLineString must be a list of LineString' | ||||||
|  |  | ||||||
|  |         # Quick and dirty validator | ||||||
|  |         try: | ||||||
|  |             value[0][0][0] | ||||||
|  |         except: | ||||||
|  |             return "Invalid MultiLineString must contain at least one valid linestring" | ||||||
|  |  | ||||||
|  |         errors = [] | ||||||
|  |         for linestring in value: | ||||||
|  |             error = self._validate_linestring(linestring, False) | ||||||
|  |             if error and error not in errors: | ||||||
|  |                 errors.append(error) | ||||||
|  |  | ||||||
|  |         if errors: | ||||||
|  |             if top_level: | ||||||
|  |                 return "Invalid MultiLineString:\n%s" % ", ".join(errors) | ||||||
|  |             else: | ||||||
|  |                 return "%s" % ", ".join(errors) | ||||||
|  |  | ||||||
|  |     def _validate_multipolygon(self, value): | ||||||
|  |         if not isinstance(value, (list, tuple)): | ||||||
|  |             return 'MultiPolygon must be a list of Polygon' | ||||||
|  |  | ||||||
|  |         # Quick and dirty validator | ||||||
|  |         try: | ||||||
|  |             value[0][0][0][0] | ||||||
|  |         except: | ||||||
|  |             return "Invalid MultiPolygon must contain at least one valid Polygon" | ||||||
|  |  | ||||||
|  |         errors = [] | ||||||
|  |         for polygon in value: | ||||||
|  |             error = self._validate_polygon(polygon, False) | ||||||
|  |             if error and error not in errors: | ||||||
|  |                 errors.append(error) | ||||||
|  |  | ||||||
|  |         if errors: | ||||||
|  |             return "Invalid MultiPolygon:\n%s" % ", ".join(errors) | ||||||
|  |  | ||||||
|     def to_mongo(self, value): |     def to_mongo(self, value): | ||||||
|         if isinstance(value, dict): |         if isinstance(value, dict): | ||||||
|             return value |             return value | ||||||
|   | |||||||
| @@ -16,6 +16,7 @@ __all__ = ('DocumentMetaclass', 'TopLevelDocumentMetaclass') | |||||||
|  |  | ||||||
|  |  | ||||||
| class DocumentMetaclass(type): | class DocumentMetaclass(type): | ||||||
|  |  | ||||||
|     """Metaclass for all documents. |     """Metaclass for all documents. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
| @@ -29,6 +30,7 @@ class DocumentMetaclass(type): | |||||||
|             return super_new(cls, name, bases, attrs) |             return super_new(cls, name, bases, attrs) | ||||||
|  |  | ||||||
|         attrs['_is_document'] = attrs.get('_is_document', False) |         attrs['_is_document'] = attrs.get('_is_document', False) | ||||||
|  |         attrs['_cached_reference_fields'] = [] | ||||||
|  |  | ||||||
|         # EmbeddedDocuments could have meta data for inheritance |         # EmbeddedDocuments could have meta data for inheritance | ||||||
|         if 'meta' in attrs: |         if 'meta' in attrs: | ||||||
| @@ -45,6 +47,10 @@ class DocumentMetaclass(type): | |||||||
|                     meta.merge(base._meta) |                     meta.merge(base._meta) | ||||||
|             attrs['_meta'] = meta |             attrs['_meta'] = meta | ||||||
|  |  | ||||||
|  |         if '_meta' in attrs and attrs['_meta'].get('allow_inheritance', ALLOW_INHERITANCE): | ||||||
|  |             StringField = _import_class('StringField') | ||||||
|  |             attrs['_cls'] = StringField() | ||||||
|  |  | ||||||
|         # Handle document Fields |         # Handle document Fields | ||||||
|  |  | ||||||
|         # Merge all fields from subclasses |         # Merge all fields from subclasses | ||||||
| @@ -141,7 +147,8 @@ class DocumentMetaclass(type): | |||||||
|                 base._subclasses += (_cls,) |                 base._subclasses += (_cls,) | ||||||
|             base._types = base._subclasses   # TODO depreciate _types |             base._types = base._subclasses   # TODO depreciate _types | ||||||
|  |  | ||||||
|         Document, EmbeddedDocument, DictField = cls._import_classes() |         (Document, EmbeddedDocument, DictField, | ||||||
|  |          CachedReferenceField) = cls._import_classes() | ||||||
|  |  | ||||||
|         if issubclass(new_class, Document): |         if issubclass(new_class, Document): | ||||||
|             new_class._collection = None |             new_class._collection = None | ||||||
| @@ -170,6 +177,20 @@ class DocumentMetaclass(type): | |||||||
|             f = field |             f = field | ||||||
|             f.owner_document = new_class |             f.owner_document = new_class | ||||||
|             delete_rule = getattr(f, 'reverse_delete_rule', DO_NOTHING) |             delete_rule = getattr(f, 'reverse_delete_rule', DO_NOTHING) | ||||||
|  |             if isinstance(f, CachedReferenceField): | ||||||
|  |  | ||||||
|  |                 if issubclass(new_class, EmbeddedDocument): | ||||||
|  |                     raise InvalidDocumentError( | ||||||
|  |                         "CachedReferenceFields is not allowed in EmbeddedDocuments") | ||||||
|  |                 if not f.document_type: | ||||||
|  |                     raise InvalidDocumentError( | ||||||
|  |                         "Document is not avaiable to sync") | ||||||
|  |  | ||||||
|  |                 if f.auto_sync: | ||||||
|  |                     f.start_listener() | ||||||
|  |  | ||||||
|  |                 f.document_type._cached_reference_fields.append(f) | ||||||
|  |  | ||||||
|             if isinstance(f, ComplexBaseField) and hasattr(f, 'field'): |             if isinstance(f, ComplexBaseField) and hasattr(f, 'field'): | ||||||
|                 delete_rule = getattr(f.field, |                 delete_rule = getattr(f.field, | ||||||
|                                       'reverse_delete_rule', |                                       'reverse_delete_rule', | ||||||
| @@ -224,10 +245,12 @@ class DocumentMetaclass(type): | |||||||
|         Document = _import_class('Document') |         Document = _import_class('Document') | ||||||
|         EmbeddedDocument = _import_class('EmbeddedDocument') |         EmbeddedDocument = _import_class('EmbeddedDocument') | ||||||
|         DictField = _import_class('DictField') |         DictField = _import_class('DictField') | ||||||
|         return (Document, EmbeddedDocument, DictField) |         CachedReferenceField = _import_class('CachedReferenceField') | ||||||
|  |         return (Document, EmbeddedDocument, DictField, CachedReferenceField) | ||||||
|  |  | ||||||
|  |  | ||||||
| class TopLevelDocumentMetaclass(DocumentMetaclass): | class TopLevelDocumentMetaclass(DocumentMetaclass): | ||||||
|  |  | ||||||
|     """Metaclass for top-level documents (i.e. documents that have their own |     """Metaclass for top-level documents (i.e. documents that have their own | ||||||
|     collection in the database. |     collection in the database. | ||||||
|     """ |     """ | ||||||
| @@ -359,7 +382,8 @@ class TopLevelDocumentMetaclass(DocumentMetaclass): | |||||||
|                     new_class.id = field |                     new_class.id = field | ||||||
|  |  | ||||||
|         # Set primary key if not defined by the document |         # Set primary key if not defined by the document | ||||||
|         new_class._auto_id_field = False |         new_class._auto_id_field = getattr(parent_doc_cls, | ||||||
|  |                                            '_auto_id_field', False) | ||||||
|         if not new_class._meta.get('id_field'): |         if not new_class._meta.get('id_field'): | ||||||
|             new_class._auto_id_field = True |             new_class._auto_id_field = True | ||||||
|             new_class._meta['id_field'] = 'id' |             new_class._meta['id_field'] = 'id' | ||||||
| @@ -386,6 +410,7 @@ class TopLevelDocumentMetaclass(DocumentMetaclass): | |||||||
|  |  | ||||||
|  |  | ||||||
| class MetaDict(dict): | class MetaDict(dict): | ||||||
|  |  | ||||||
|     """Custom dictionary for meta classes. |     """Custom dictionary for meta classes. | ||||||
|     Handles the merging of set indexes |     Handles the merging of set indexes | ||||||
|     """ |     """ | ||||||
| @@ -400,5 +425,6 @@ class MetaDict(dict): | |||||||
|  |  | ||||||
|  |  | ||||||
| class BasesTuple(tuple): | class BasesTuple(tuple): | ||||||
|  |  | ||||||
|     """Special class to handle introspection of bases tuple in __new__""" |     """Special class to handle introspection of bases tuple in __new__""" | ||||||
|     pass |     pass | ||||||
|   | |||||||
| @@ -23,8 +23,10 @@ def _import_class(cls_name): | |||||||
|     field_classes = ('DictField', 'DynamicField', 'EmbeddedDocumentField', |     field_classes = ('DictField', 'DynamicField', 'EmbeddedDocumentField', | ||||||
|                      'FileField', 'GenericReferenceField', |                      'FileField', 'GenericReferenceField', | ||||||
|                      'GenericEmbeddedDocumentField', 'GeoPointField', |                      'GenericEmbeddedDocumentField', 'GeoPointField', | ||||||
|                      'PointField', 'LineStringField', 'PolygonField', |                      'PointField', 'LineStringField', 'ListField', | ||||||
|                      'ReferenceField', 'StringField', 'ComplexBaseField') |                      'PolygonField', 'ReferenceField', 'StringField', | ||||||
|  |                      'CachedReferenceField', | ||||||
|  |                      'ComplexBaseField', 'GeoJsonBaseField') | ||||||
|     queryset_classes = ('OperationError',) |     queryset_classes = ('OperationError',) | ||||||
|     deref_classes = ('DeReference',) |     deref_classes = ('DeReference',) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,10 @@ | |||||||
| import pymongo | import pymongo | ||||||
| from pymongo import MongoClient, MongoReplicaSetClient, uri_parser | from pymongo import MongoClient, MongoReplicaSetClient, uri_parser | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     import motor | ||||||
|  | except ImportError: | ||||||
|  |     motor = None | ||||||
|  |  | ||||||
| __all__ = ['ConnectionError', 'connect', 'register_connection', | __all__ = ['ConnectionError', 'connect', 'register_connection', | ||||||
|            'DEFAULT_CONNECTION_NAME'] |            'DEFAULT_CONNECTION_NAME'] | ||||||
| @@ -18,9 +22,11 @@ _connections = {} | |||||||
| _dbs = {} | _dbs = {} | ||||||
|  |  | ||||||
|  |  | ||||||
| def register_connection(alias, name, host='localhost', port=27017, | def register_connection(alias, name=None, host=None, port=None, | ||||||
|                         is_slave=False, read_preference=False, slaves=None, |                         read_preference=False, | ||||||
|                         username=None, password=None, **kwargs): |                         username=None, password=None, authentication_source=None, | ||||||
|  |                         async=False, | ||||||
|  |                         **kwargs): | ||||||
|     """Add a connection. |     """Add a connection. | ||||||
|  |  | ||||||
|     :param alias: the name that will be used to refer to this connection |     :param alias: the name that will be used to refer to this connection | ||||||
| @@ -28,46 +34,42 @@ def register_connection(alias, name, host='localhost', port=27017, | |||||||
|     :param name: the name of the specific database to use |     :param name: the name of the specific database to use | ||||||
|     :param host: the host name of the :program:`mongod` instance to connect to |     :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 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 |     :param read_preference: The read preference for the collection | ||||||
|        ** Added pymongo 2.1 |        ** 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 username: username to authenticate with | ||||||
|     :param password: password to authenticate with |     :param password: password to authenticate with | ||||||
|  |     :param authentication_source: database to authenticate against | ||||||
|     :param kwargs: allow ad-hoc parameters to be passed into the pymongo driver |     :param kwargs: allow ad-hoc parameters to be passed into the pymongo driver | ||||||
|  |  | ||||||
|     """ |     """ | ||||||
|     global _connection_settings |     global _connection_settings | ||||||
|  |  | ||||||
|     conn_settings = { |     conn_settings = { | ||||||
|         'name': name, |         'name': name or 'test', | ||||||
|         'host': host, |         'host': host or 'localhost', | ||||||
|         'port': port, |         'port': port or 27017, | ||||||
|         'is_slave': is_slave, |         'read_preference': read_preference, | ||||||
|         'slaves': slaves or [], |  | ||||||
|         'username': username, |         'username': username, | ||||||
|         'password': password, |         'password': password, | ||||||
|         'read_preference': read_preference |         'authentication_source': authentication_source, | ||||||
|  |         'async': async | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     # Handle uri style connections |     # Handle uri style connections | ||||||
|     if "://" in host: |     if "://" in conn_settings['host']: | ||||||
|         uri_dict = uri_parser.parse_uri(host) |         uri_dict = uri_parser.parse_uri(conn_settings['host']) | ||||||
|         if uri_dict.get('database') is None: |  | ||||||
|             raise ConnectionError("If using URI style connection include "\ |  | ||||||
|                                   "database name in string") |  | ||||||
|         conn_settings.update({ |         conn_settings.update({ | ||||||
|             'host': host, |             'name': uri_dict.get('database') or name, | ||||||
|             'name': uri_dict.get('database'), |  | ||||||
|             'username': uri_dict.get('username'), |             'username': uri_dict.get('username'), | ||||||
|             'password': uri_dict.get('password'), |             'password': uri_dict.get('password'), | ||||||
|             'read_preference': read_preference, |             'read_preference': read_preference, | ||||||
|         }) |         }) | ||||||
|         if "replicaSet" in host: |         if "replicaSet" in conn_settings['host']: | ||||||
|             conn_settings['replicaSet'] = True |             conn_settings['replicaSet'] = True | ||||||
|  |  | ||||||
|  |     # Deprecated parameters that should not be passed on | ||||||
|  |     kwargs.pop('slaves', None) | ||||||
|  |     kwargs.pop('is_slave', None) | ||||||
|  |  | ||||||
|     conn_settings.update(kwargs) |     conn_settings.update(kwargs) | ||||||
|     _connection_settings[alias] = conn_settings |     _connection_settings[alias] = conn_settings | ||||||
|  |  | ||||||
| @@ -97,22 +99,21 @@ def get_connection(alias=DEFAULT_CONNECTION_NAME, reconnect=False): | |||||||
|             raise ConnectionError(msg) |             raise ConnectionError(msg) | ||||||
|         conn_settings = _connection_settings[alias].copy() |         conn_settings = _connection_settings[alias].copy() | ||||||
|  |  | ||||||
|         if hasattr(pymongo, 'version_tuple'):  # Support for 2.1+ |  | ||||||
|         conn_settings.pop('name', None) |         conn_settings.pop('name', None) | ||||||
|             conn_settings.pop('slaves', None) |  | ||||||
|             conn_settings.pop('is_slave', None) |  | ||||||
|         conn_settings.pop('username', None) |         conn_settings.pop('username', None) | ||||||
|         conn_settings.pop('password', None) |         conn_settings.pop('password', None) | ||||||
|         else: |         conn_settings.pop('authentication_source', None) | ||||||
|             # Get all the slave connections |         async = conn_settings.pop('async') | ||||||
|             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', None) |  | ||||||
|  |  | ||||||
|  |         if async: | ||||||
|  |             if not motor: | ||||||
|  |                 raise ImproperlyConfigured("Motor library was not found") | ||||||
|  |  | ||||||
|  |             connection_class = motor.MotorClient | ||||||
|  |  | ||||||
|  |         else: | ||||||
|             connection_class = MongoClient |             connection_class = MongoClient | ||||||
|  |  | ||||||
|         if 'replicaSet' in conn_settings: |         if 'replicaSet' in conn_settings: | ||||||
|             conn_settings['hosts_or_uri'] = conn_settings.pop('host', None) |             conn_settings['hosts_or_uri'] = conn_settings.pop('host', None) | ||||||
|             # Discard port since it can't be used on MongoReplicaSetClient |             # Discard port since it can't be used on MongoReplicaSetClient | ||||||
| @@ -120,12 +121,30 @@ def get_connection(alias=DEFAULT_CONNECTION_NAME, reconnect=False): | |||||||
|             # Discard replicaSet if not base string |             # Discard replicaSet if not base string | ||||||
|             if not isinstance(conn_settings['replicaSet'], basestring): |             if not isinstance(conn_settings['replicaSet'], basestring): | ||||||
|                 conn_settings.pop('replicaSet', None) |                 conn_settings.pop('replicaSet', None) | ||||||
|  |  | ||||||
|  |             if async: | ||||||
|  |                 connection_class = motor.MotorReplicaSetClient | ||||||
|  |             else: | ||||||
|                 connection_class = MongoReplicaSetClient |                 connection_class = MongoReplicaSetClient | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             _connections[alias] = connection_class(**conn_settings) |             connection = None | ||||||
|  |             # check for shared connections | ||||||
|  |             connection_settings_iterator = ( | ||||||
|  |                 (db_alias, settings.copy()) for db_alias, settings in _connection_settings.iteritems()) | ||||||
|  |             for db_alias, connection_settings in connection_settings_iterator: | ||||||
|  |                 connection_settings.pop('name', None) | ||||||
|  |                 connection_settings.pop('username', None) | ||||||
|  |                 connection_settings.pop('password', None) | ||||||
|  |                 if conn_settings == connection_settings and _connections.get(db_alias, None): | ||||||
|  |                     connection = _connections[db_alias] | ||||||
|  |                     break | ||||||
|  |  | ||||||
|  |             _connections[alias] = connection if connection else connection_class( | ||||||
|  |                 **conn_settings) | ||||||
|         except Exception, e: |         except Exception, e: | ||||||
|             raise ConnectionError("Cannot connect to database %s :\n%s" % (alias, e)) |             raise ConnectionError( | ||||||
|  |                 "Cannot connect to database %s :\n%s" % (alias, e)) | ||||||
|     return _connections[alias] |     return _connections[alias] | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -141,12 +160,13 @@ def get_db(alias=DEFAULT_CONNECTION_NAME, reconnect=False): | |||||||
|         # Authenticate if necessary |         # Authenticate if necessary | ||||||
|         if conn_settings['username'] and conn_settings['password']: |         if conn_settings['username'] and conn_settings['password']: | ||||||
|             db.authenticate(conn_settings['username'], |             db.authenticate(conn_settings['username'], | ||||||
|                             conn_settings['password']) |                             conn_settings['password'], | ||||||
|  |                             source=conn_settings['authentication_source']) | ||||||
|         _dbs[alias] = db |         _dbs[alias] = db | ||||||
|     return _dbs[alias] |     return _dbs[alias] | ||||||
|  |  | ||||||
|  |  | ||||||
| def connect(db, alias=DEFAULT_CONNECTION_NAME, **kwargs): | def connect(db=None, alias=DEFAULT_CONNECTION_NAME, **kwargs): | ||||||
|     """Connect to the database specified by the 'db' argument. |     """Connect to the database specified by the 'db' argument. | ||||||
|  |  | ||||||
|     Connection settings may be provided here as well if the database is not |     Connection settings may be provided here as well if the database is not | ||||||
|   | |||||||
| @@ -1,6 +1,5 @@ | |||||||
| from mongoengine.common import _import_class | from mongoengine.common import _import_class | ||||||
| from mongoengine.connection import DEFAULT_CONNECTION_NAME, get_db | from mongoengine.connection import DEFAULT_CONNECTION_NAME, get_db | ||||||
| from mongoengine.queryset import QuerySet |  | ||||||
|  |  | ||||||
|  |  | ||||||
| __all__ = ("switch_db", "switch_collection", "no_dereference", | __all__ = ("switch_db", "switch_collection", "no_dereference", | ||||||
| @@ -162,12 +161,6 @@ class no_sub_classes(object): | |||||||
|         return self.cls |         return self.cls | ||||||
|  |  | ||||||
|  |  | ||||||
| class QuerySetNoDeRef(QuerySet): |  | ||||||
|     """Special no_dereference QuerySet""" |  | ||||||
|     def __dereference(items, max_depth=1, instance=None, name=None): |  | ||||||
|             return items |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class query_counter(object): | class query_counter(object): | ||||||
|     """ Query_counter context manager to get the number of queries. """ |     """ Query_counter context manager to get the number of queries. """ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ from base import (BaseDict, BaseList, TopLevelDocumentMetaclass, get_document) | |||||||
| from fields import (ReferenceField, ListField, DictField, MapField) | from fields import (ReferenceField, ListField, DictField, MapField) | ||||||
| from connection import get_db | from connection import get_db | ||||||
| from queryset import QuerySet | from queryset import QuerySet | ||||||
| from document import Document | from document import Document, EmbeddedDocument | ||||||
|  |  | ||||||
|  |  | ||||||
| class DeReference(object): | class DeReference(object): | ||||||
| @@ -12,7 +12,7 @@ class DeReference(object): | |||||||
|     def __call__(self, items, max_depth=1, instance=None, name=None): |     def __call__(self, items, max_depth=1, instance=None, name=None): | ||||||
|         """ |         """ | ||||||
|         Cheaply dereferences the items to a set depth. |         Cheaply dereferences the items to a set depth. | ||||||
|         Also handles the convertion of complex data types. |         Also handles the conversion of complex data types. | ||||||
|  |  | ||||||
|         :param items: The iterable (dict, list, queryset) to be dereferenced. |         :param items: The iterable (dict, list, queryset) to be dereferenced. | ||||||
|         :param max_depth: The maximum depth to recurse to |         :param max_depth: The maximum depth to recurse to | ||||||
| @@ -33,9 +33,10 @@ class DeReference(object): | |||||||
|         self.max_depth = max_depth |         self.max_depth = max_depth | ||||||
|         doc_type = None |         doc_type = None | ||||||
|  |  | ||||||
|         if instance and isinstance(instance, (Document, TopLevelDocumentMetaclass)): |         if instance and isinstance(instance, (Document, EmbeddedDocument, | ||||||
|  |                                               TopLevelDocumentMetaclass)): | ||||||
|             doc_type = instance._fields.get(name) |             doc_type = instance._fields.get(name) | ||||||
|             if hasattr(doc_type, 'field'): |             while hasattr(doc_type, 'field'): | ||||||
|                 doc_type = doc_type.field |                 doc_type = doc_type.field | ||||||
|  |  | ||||||
|             if isinstance(doc_type, ReferenceField): |             if isinstance(doc_type, ReferenceField): | ||||||
| @@ -50,9 +51,19 @@ class DeReference(object): | |||||||
|                     return items |                     return items | ||||||
|                 elif not field.dbref: |                 elif not field.dbref: | ||||||
|                     if not hasattr(items, 'items'): |                     if not hasattr(items, 'items'): | ||||||
|                         items = [field.to_python(v) |  | ||||||
|                              if not isinstance(v, (DBRef, Document)) else v |                         def _get_items(items): | ||||||
|                              for v in items] |                             new_items = [] | ||||||
|  |                             for v in items: | ||||||
|  |                                 if isinstance(v, list): | ||||||
|  |                                     new_items.append(_get_items(v)) | ||||||
|  |                                 elif not isinstance(v, (DBRef, Document)): | ||||||
|  |                                     new_items.append(field.to_python(v)) | ||||||
|  |                                 else: | ||||||
|  |                                     new_items.append(v) | ||||||
|  |                             return new_items | ||||||
|  |  | ||||||
|  |                         items = _get_items(items) | ||||||
|                     else: |                     else: | ||||||
|                         items = dict([ |                         items = dict([ | ||||||
|                             (k, field.to_python(v)) |                             (k, field.to_python(v)) | ||||||
| @@ -84,7 +95,7 @@ class DeReference(object): | |||||||
|         # Recursively find dbreferences |         # Recursively find dbreferences | ||||||
|         depth += 1 |         depth += 1 | ||||||
|         for k, item in iterator: |         for k, item in iterator: | ||||||
|             if isinstance(item, Document): |             if isinstance(item, (Document, EmbeddedDocument)): | ||||||
|                 for field_name, field in item._fields.iteritems(): |                 for field_name, field in item._fields.iteritems(): | ||||||
|                     v = item._data.get(field_name, None) |                     v = item._data.get(field_name, None) | ||||||
|                     if isinstance(v, (DBRef)): |                     if isinstance(v, (DBRef)): | ||||||
| @@ -113,11 +124,11 @@ class DeReference(object): | |||||||
|         """Fetch all references and convert to their document objects |         """Fetch all references and convert to their document objects | ||||||
|         """ |         """ | ||||||
|         object_map = {} |         object_map = {} | ||||||
|         for col, dbrefs in self.reference_map.iteritems(): |         for collection, dbrefs in self.reference_map.iteritems(): | ||||||
|             keys = object_map.keys() |             keys = object_map.keys() | ||||||
|             refs = list(set([dbref for dbref in dbrefs if unicode(dbref).encode('utf-8') not in keys])) |             refs = list(set([dbref for dbref in dbrefs if unicode(dbref).encode('utf-8') not in keys])) | ||||||
|             if hasattr(col, 'objects'):  # We have a document class for the refs |             if hasattr(collection, 'objects'):  # We have a document class for the refs | ||||||
|                 references = col.objects.in_bulk(refs) |                 references = collection.objects.in_bulk(refs) | ||||||
|                 for key, doc in references.iteritems(): |                 for key, doc in references.iteritems(): | ||||||
|                     object_map[key] = doc |                     object_map[key] = doc | ||||||
|             else:  # Generic reference: use the refs data to convert to document |             else:  # Generic reference: use the refs data to convert to document | ||||||
| @@ -125,19 +136,19 @@ class DeReference(object): | |||||||
|                     continue |                     continue | ||||||
|  |  | ||||||
|                 if doc_type: |                 if doc_type: | ||||||
|                     references = doc_type._get_db()[col].find({'_id': {'$in': refs}}) |                     references = doc_type._get_db()[collection].find({'_id': {'$in': refs}}) | ||||||
|                     for ref in references: |                     for ref in references: | ||||||
|                         doc = doc_type._from_son(ref) |                         doc = doc_type._from_son(ref) | ||||||
|                         object_map[doc.id] = doc |                         object_map[doc.id] = doc | ||||||
|                 else: |                 else: | ||||||
|                     references = get_db()[col].find({'_id': {'$in': refs}}) |                     references = get_db()[collection].find({'_id': {'$in': refs}}) | ||||||
|                     for ref in references: |                     for ref in references: | ||||||
|                         if '_cls' in ref: |                         if '_cls' in ref: | ||||||
|                             doc = get_document(ref["_cls"])._from_son(ref) |                             doc = get_document(ref["_cls"])._from_son(ref) | ||||||
|                         elif doc_type is None: |                         elif doc_type is None: | ||||||
|                             doc = get_document( |                             doc = get_document( | ||||||
|                                 ''.join(x.capitalize() |                                 ''.join(x.capitalize() | ||||||
|                                     for x in col.split('_')))._from_son(ref) |                                     for x in collection.split('_')))._from_son(ref) | ||||||
|                         else: |                         else: | ||||||
|                             doc = doc_type._from_son(ref) |                             doc = doc_type._from_son(ref) | ||||||
|                         object_map[doc.id] = doc |                         object_map[doc.id] = doc | ||||||
| @@ -169,7 +180,11 @@ class DeReference(object): | |||||||
|                 return self.object_map.get(items['_ref'].id, items) |                 return self.object_map.get(items['_ref'].id, items) | ||||||
|             elif '_cls' in items: |             elif '_cls' in items: | ||||||
|                 doc = get_document(items['_cls'])._from_son(items) |                 doc = get_document(items['_cls'])._from_son(items) | ||||||
|  |                 _cls = doc._data.pop('_cls', None) | ||||||
|  |                 del items['_cls'] | ||||||
|                 doc._data = self._attach_objects(doc._data, depth, doc, None) |                 doc._data = self._attach_objects(doc._data, depth, doc, None) | ||||||
|  |                 if _cls is not None: | ||||||
|  |                     doc._data['_cls'] = _cls | ||||||
|                 return doc |                 return doc | ||||||
|  |  | ||||||
|         if not hasattr(items, 'items'): |         if not hasattr(items, 'items'): | ||||||
| @@ -191,7 +206,7 @@ class DeReference(object): | |||||||
|  |  | ||||||
|             if k in self.object_map and not is_list: |             if k in self.object_map and not is_list: | ||||||
|                 data[k] = self.object_map[k] |                 data[k] = self.object_map[k] | ||||||
|             elif isinstance(v, Document): |             elif isinstance(v, (Document, EmbeddedDocument)): | ||||||
|                 for field_name, field in v._fields.iteritems(): |                 for field_name, field in v._fields.iteritems(): | ||||||
|                     v = data[k]._data.get(field_name, None) |                     v = data[k]._data.get(field_name, None) | ||||||
|                     if isinstance(v, (DBRef)): |                     if isinstance(v, (DBRef)): | ||||||
| @@ -203,7 +218,8 @@ class DeReference(object): | |||||||
|                     elif isinstance(v, (list, tuple)) and depth <= self.max_depth: |                     elif isinstance(v, (list, tuple)) and depth <= self.max_depth: | ||||||
|                         data[k]._data[field_name] = self._attach_objects(v, depth, instance=instance, name=name) |                         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: |             elif isinstance(v, (dict, list, tuple)) and depth <= self.max_depth: | ||||||
|                 data[k] = self._attach_objects(v, depth - 1, instance=instance, name=name) |                 item_name = '%s.%s' % (name, k) if name else name | ||||||
|  |                 data[k] = self._attach_objects(v, depth - 1, instance=instance, name=item_name) | ||||||
|             elif hasattr(v, 'id'): |             elif hasattr(v, 'id'): | ||||||
|                 data[k] = self.object_map.get(v.id, v) |                 data[k] = self.object_map.get(v.id, v) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,6 +8,10 @@ from django.contrib import auth | |||||||
| from django.contrib.auth.models import AnonymousUser | from django.contrib.auth.models import AnonymousUser | ||||||
| from django.utils.translation import ugettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
|  | from .utils import datetime_now | ||||||
|  |  | ||||||
|  | REDIRECT_FIELD_NAME = 'next' | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     from django.contrib.auth.hashers import check_password, make_password |     from django.contrib.auth.hashers import check_password, make_password | ||||||
| except ImportError: | except ImportError: | ||||||
| @@ -33,10 +37,6 @@ except ImportError: | |||||||
|         hash = get_hexdigest(algo, salt, raw_password) |         hash = get_hexdigest(algo, salt, raw_password) | ||||||
|         return '%s$%s$%s' % (algo, salt, hash) |         return '%s$%s$%s' % (algo, salt, hash) | ||||||
|  |  | ||||||
| from .utils import datetime_now |  | ||||||
|  |  | ||||||
| REDIRECT_FIELD_NAME = 'next' |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ContentType(Document): | class ContentType(Document): | ||||||
|     name = StringField(max_length=100) |     name = StringField(max_length=100) | ||||||
| @@ -230,6 +230,9 @@ class User(Document): | |||||||
|     date_joined = DateTimeField(default=datetime_now, |     date_joined = DateTimeField(default=datetime_now, | ||||||
|                                 verbose_name=_('date joined')) |                                 verbose_name=_('date joined')) | ||||||
|  |  | ||||||
|  |     user_permissions = ListField(ReferenceField(Permission), verbose_name=_('user permissions'), | ||||||
|  |                                                 help_text=_('Permissions for the user.')) | ||||||
|  |  | ||||||
|     USERNAME_FIELD = 'username' |     USERNAME_FIELD = 'username' | ||||||
|     REQUIRED_FIELDS = ['email'] |     REQUIRED_FIELDS = ['email'] | ||||||
|  |  | ||||||
| @@ -378,9 +381,10 @@ class MongoEngineBackend(object): | |||||||
|     supports_object_permissions = False |     supports_object_permissions = False | ||||||
|     supports_anonymous_user = False |     supports_anonymous_user = False | ||||||
|     supports_inactive_user = False |     supports_inactive_user = False | ||||||
|  |     _user_doc = False | ||||||
|  |  | ||||||
|     def authenticate(self, username=None, password=None): |     def authenticate(self, username=None, password=None): | ||||||
|         user = User.objects(username=username).first() |         user = self.user_document.objects(username=username).first() | ||||||
|         if user: |         if user: | ||||||
|             if password and user.check_password(password): |             if password and user.check_password(password): | ||||||
|                 backend = auth.get_backends()[0] |                 backend = auth.get_backends()[0] | ||||||
| @@ -389,8 +393,14 @@ class MongoEngineBackend(object): | |||||||
|         return None |         return None | ||||||
|  |  | ||||||
|     def get_user(self, user_id): |     def get_user(self, user_id): | ||||||
|         return User.objects.with_id(user_id) |         return self.user_document.objects.with_id(user_id) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def user_document(self): | ||||||
|  |         if self._user_doc is False: | ||||||
|  |             from .mongo_auth.models import get_user_document | ||||||
|  |             self._user_doc = get_user_document() | ||||||
|  |         return self._user_doc | ||||||
|  |  | ||||||
| def get_user(userid): | def get_user(userid): | ||||||
|     """Returns a User object from an id (User.id). Django's equivalent takes |     """Returns a User object from an id (User.id). Django's equivalent takes | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  | from django.contrib.auth.hashers import make_password | ||||||
| from django.contrib.auth.models import UserManager | from django.contrib.auth.models import UserManager | ||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.db import models | from django.db import models | ||||||
| @@ -6,10 +7,29 @@ from django.utils.importlib import import_module | |||||||
| from django.utils.translation import ugettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | __all__ = ( | ||||||
|  |     'get_user_document', | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
| MONGOENGINE_USER_DOCUMENT = getattr( | MONGOENGINE_USER_DOCUMENT = getattr( | ||||||
|     settings, 'MONGOENGINE_USER_DOCUMENT', 'mongoengine.django.auth.User') |     settings, 'MONGOENGINE_USER_DOCUMENT', 'mongoengine.django.auth.User') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_user_document(): | ||||||
|  |     """Get the user document class used for authentication. | ||||||
|  |  | ||||||
|  |     This is the class defined in settings.MONGOENGINE_USER_DOCUMENT, which | ||||||
|  |     defaults to `mongoengine.django.auth.User`. | ||||||
|  |  | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     name = MONGOENGINE_USER_DOCUMENT | ||||||
|  |     dot = name.rindex('.') | ||||||
|  |     module = import_module(name[:dot]) | ||||||
|  |     return getattr(module, name[dot + 1:]) | ||||||
|  |  | ||||||
|  |  | ||||||
| class MongoUserManager(UserManager): | class MongoUserManager(UserManager): | ||||||
|     """A User manager wich allows the use of MongoEngine documents in Django. |     """A User manager wich allows the use of MongoEngine documents in Django. | ||||||
|  |  | ||||||
| @@ -44,7 +64,7 @@ class MongoUserManager(UserManager): | |||||||
|     def contribute_to_class(self, model, name): |     def contribute_to_class(self, model, name): | ||||||
|         super(MongoUserManager, self).contribute_to_class(model, name) |         super(MongoUserManager, self).contribute_to_class(model, name) | ||||||
|         self.dj_model = self.model |         self.dj_model = self.model | ||||||
|         self.model = self._get_user_document() |         self.model = get_user_document() | ||||||
|  |  | ||||||
|         self.dj_model.USERNAME_FIELD = self.model.USERNAME_FIELD |         self.dj_model.USERNAME_FIELD = self.model.USERNAME_FIELD | ||||||
|         username = models.CharField(_('username'), max_length=30, unique=True) |         username = models.CharField(_('username'), max_length=30, unique=True) | ||||||
| @@ -55,16 +75,6 @@ class MongoUserManager(UserManager): | |||||||
|             field = models.CharField(_(name), max_length=30) |             field = models.CharField(_(name), max_length=30) | ||||||
|             field.contribute_to_class(self.dj_model, name) |             field.contribute_to_class(self.dj_model, name) | ||||||
|  |  | ||||||
|     def _get_user_document(self): |  | ||||||
|         try: |  | ||||||
|             name = MONGOENGINE_USER_DOCUMENT |  | ||||||
|             dot = name.rindex('.') |  | ||||||
|             module = import_module(name[:dot]) |  | ||||||
|             return getattr(module, name[dot + 1:]) |  | ||||||
|         except ImportError: |  | ||||||
|             raise ImproperlyConfigured("Error importing %s, please check " |  | ||||||
|                                        "settings.MONGOENGINE_USER_DOCUMENT" |  | ||||||
|                                        % name) |  | ||||||
|  |  | ||||||
|     def get(self, *args, **kwargs): |     def get(self, *args, **kwargs): | ||||||
|         try: |         try: | ||||||
| @@ -85,5 +95,21 @@ class MongoUserManager(UserManager): | |||||||
|  |  | ||||||
|  |  | ||||||
| class MongoUser(models.Model): | class MongoUser(models.Model): | ||||||
|  |     """"Dummy user model for Django. | ||||||
|  |  | ||||||
|  |     MongoUser is used to replace Django's UserManager with MongoUserManager. | ||||||
|  |     The actual user document class is mongoengine.django.auth.User or any | ||||||
|  |     other document class specified in MONGOENGINE_USER_DOCUMENT. | ||||||
|  |  | ||||||
|  |     To get the user document class, use `get_user_document()`. | ||||||
|  |  | ||||||
|  |     """ | ||||||
|  |  | ||||||
|     objects = MongoUserManager() |     objects = MongoUserManager() | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         app_label = 'mongo_auth' | ||||||
|  |  | ||||||
|  |     def set_password(self, password): | ||||||
|  |         """Doesn't do anything, but works around the issue with Django 1.6.""" | ||||||
|  |         make_password(password) | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | from bson import json_util | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.contrib.sessions.backends.base import SessionBase, CreateError | from django.contrib.sessions.backends.base import SessionBase, CreateError | ||||||
| from django.core.exceptions import SuspiciousOperation | from django.core.exceptions import SuspiciousOperation | ||||||
| @@ -55,6 +56,12 @@ class SessionStore(SessionBase): | |||||||
|     """A MongoEngine-based session store for Django. |     """A MongoEngine-based session store for Django. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |     def _get_session(self, *args, **kwargs): | ||||||
|  |         sess = super(SessionStore, self)._get_session(*args, **kwargs) | ||||||
|  |         if sess.get('_auth_user_id', None): | ||||||
|  |             sess['_auth_user_id'] = str(sess.get('_auth_user_id')) | ||||||
|  |         return sess | ||||||
|  |  | ||||||
|     def load(self): |     def load(self): | ||||||
|         try: |         try: | ||||||
|             s = MongoSession.objects(session_key=self.session_key, |             s = MongoSession.objects(session_key=self.session_key, | ||||||
| @@ -103,3 +110,15 @@ class SessionStore(SessionBase): | |||||||
|                 return |                 return | ||||||
|             session_key = self.session_key |             session_key = self.session_key | ||||||
|         MongoSession.objects(session_key=session_key).delete() |         MongoSession.objects(session_key=session_key).delete() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BSONSerializer(object): | ||||||
|  |     """ | ||||||
|  |     Serializer that can handle BSON types (eg ObjectId). | ||||||
|  |     """ | ||||||
|  |     def dumps(self, obj): | ||||||
|  |         return json_util.dumps(obj, separators=(',', ':')).encode('ascii') | ||||||
|  |  | ||||||
|  |     def loads(self, data): | ||||||
|  |         return json_util.loads(data.decode('ascii')) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -76,7 +76,7 @@ class GridFSStorage(Storage): | |||||||
|         """Find the documents in the store with the given name |         """Find the documents in the store with the given name | ||||||
|         """ |         """ | ||||||
|         docs = self.document.objects |         docs = self.document.objects | ||||||
|         doc = [d for d in docs if getattr(d, self.field).name == name] |         doc = [d for d in docs if hasattr(getattr(d, self.field), 'name') and getattr(d, self.field).name == name] | ||||||
|         if doc: |         if doc: | ||||||
|             return doc[0] |             return doc[0] | ||||||
|         else: |         else: | ||||||
|   | |||||||
| @@ -1,39 +1,31 @@ | |||||||
| #coding: utf-8 | #coding: utf-8 | ||||||
| from nose.plugins.skip import SkipTest |  | ||||||
|  |  | ||||||
| from mongoengine.python_support import PY3 |  | ||||||
| from mongoengine import connect |  | ||||||
|  |  | ||||||
| try: |  | ||||||
|     from django.test import TestCase |  | ||||||
|     from django.conf import settings |  | ||||||
| except Exception as err: |  | ||||||
|     if PY3: |  | ||||||
| from unittest import TestCase | from unittest import TestCase | ||||||
|         # Dummy value so no error |  | ||||||
|         class settings: | from mongoengine import connect | ||||||
|             MONGO_DATABASE_NAME = 'dummy' | from mongoengine.connection import get_db | ||||||
|     else: |  | ||||||
|         raise err |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class MongoTestCase(TestCase): | class MongoTestCase(TestCase): | ||||||
|  |  | ||||||
|     def setUp(self): |  | ||||||
|         if PY3: |  | ||||||
|             raise SkipTest('django does not have Python 3 support') |  | ||||||
|  |  | ||||||
|     """ |     """ | ||||||
|     TestCase class that clear the collection between the tests |     TestCase class that clear the collection between the tests | ||||||
|     """ |     """ | ||||||
|     db_name = 'test_%s' % settings.MONGO_DATABASE_NAME |  | ||||||
|  |     @property | ||||||
|  |     def db_name(self): | ||||||
|  |         from django.conf import settings | ||||||
|  |         return 'test_%s' % getattr(settings, 'MONGO_DATABASE_NAME', 'dummy') | ||||||
|  |  | ||||||
|     def __init__(self, methodName='runtest'): |     def __init__(self, methodName='runtest'): | ||||||
|         self.db = connect(self.db_name).get_db() |         connect(self.db_name) | ||||||
|  |         self.db = get_db() | ||||||
|         super(MongoTestCase, self).__init__(methodName) |         super(MongoTestCase, self).__init__(methodName) | ||||||
|  |  | ||||||
|     def _post_teardown(self): |     def dropCollections(self): | ||||||
|         super(MongoTestCase, self)._post_teardown() |  | ||||||
|         for collection in self.db.collection_names(): |         for collection in self.db.collection_names(): | ||||||
|             if collection == 'system.indexes': |             if collection == 'system.indexes': | ||||||
|                 continue |                 continue | ||||||
|             self.db.drop_collection(collection) |             self.db.drop_collection(collection) | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         self.dropCollections() | ||||||
|   | |||||||
| @@ -12,7 +12,9 @@ from mongoengine.common import _import_class | |||||||
| from mongoengine.base import (DocumentMetaclass, TopLevelDocumentMetaclass, | from mongoengine.base import (DocumentMetaclass, TopLevelDocumentMetaclass, | ||||||
|                               BaseDocument, BaseDict, BaseList, |                               BaseDocument, BaseDict, BaseList, | ||||||
|                               ALLOW_INHERITANCE, get_document) |                               ALLOW_INHERITANCE, get_document) | ||||||
| from mongoengine.queryset import OperationError, NotUniqueError, QuerySet | from mongoengine.errors import ValidationError | ||||||
|  | from mongoengine.queryset import (OperationError, NotUniqueError, | ||||||
|  |                                   QuerySet, transform) | ||||||
| from mongoengine.connection import get_db, DEFAULT_CONNECTION_NAME | from mongoengine.connection import get_db, DEFAULT_CONNECTION_NAME | ||||||
| from mongoengine.context_managers import switch_db, switch_collection | from mongoengine.context_managers import switch_db, switch_collection | ||||||
|  |  | ||||||
| @@ -39,6 +41,7 @@ class InvalidCollectionError(Exception): | |||||||
|  |  | ||||||
|  |  | ||||||
| class EmbeddedDocument(BaseDocument): | class EmbeddedDocument(BaseDocument): | ||||||
|  |  | ||||||
|     """A :class:`~mongoengine.Document` that isn't stored in its own |     """A :class:`~mongoengine.Document` that isn't stored in its own | ||||||
|     collection.  :class:`~mongoengine.EmbeddedDocument`\ s should be used as |     collection.  :class:`~mongoengine.EmbeddedDocument`\ s should be used as | ||||||
|     fields on :class:`~mongoengine.Document`\ s through the |     fields on :class:`~mongoengine.Document`\ s through the | ||||||
| @@ -53,15 +56,16 @@ class EmbeddedDocument(BaseDocument): | |||||||
|     dictionary. |     dictionary. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |     __slots__ = ('_instance') | ||||||
|  |  | ||||||
|     # The __metaclass__ attribute is removed by 2to3 when running with Python3 |     # The __metaclass__ attribute is removed by 2to3 when running with Python3 | ||||||
|     # my_metaclass is defined so that metaclass can be queried in Python 2 & 3 |     # my_metaclass is defined so that metaclass can be queried in Python 2 & 3 | ||||||
|     my_metaclass = DocumentMetaclass |     my_metaclass = DocumentMetaclass | ||||||
|     __metaclass__ = DocumentMetaclass |     __metaclass__ = DocumentMetaclass | ||||||
|  |  | ||||||
|     _instance = None |  | ||||||
|  |  | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(EmbeddedDocument, self).__init__(*args, **kwargs) |         super(EmbeddedDocument, self).__init__(*args, **kwargs) | ||||||
|  |         self._instance = None | ||||||
|         self._changed_fields = [] |         self._changed_fields = [] | ||||||
|  |  | ||||||
|     def __eq__(self, other): |     def __eq__(self, other): | ||||||
| @@ -74,6 +78,7 @@ class EmbeddedDocument(BaseDocument): | |||||||
|  |  | ||||||
|  |  | ||||||
| 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 | ||||||
|     collections of documents stored in MongoDB. Inherit from this class, and |     collections of documents stored in MongoDB. Inherit from this class, and | ||||||
|     add fields as class attributes to define a document's structure. |     add fields as class attributes to define a document's structure. | ||||||
| @@ -124,9 +129,12 @@ class Document(BaseDocument): | |||||||
|     my_metaclass = TopLevelDocumentMetaclass |     my_metaclass = TopLevelDocumentMetaclass | ||||||
|     __metaclass__ = TopLevelDocumentMetaclass |     __metaclass__ = TopLevelDocumentMetaclass | ||||||
|  |  | ||||||
|  |     __slots__ = ('__objects') | ||||||
|  |  | ||||||
|     def pk(): |     def pk(): | ||||||
|         """Primary key alias |         """Primary key alias | ||||||
|         """ |         """ | ||||||
|  |  | ||||||
|         def fget(self): |         def fget(self): | ||||||
|             return getattr(self, self._meta['id_field']) |             return getattr(self, self._meta['id_field']) | ||||||
|  |  | ||||||
| @@ -135,6 +143,13 @@ class Document(BaseDocument): | |||||||
|         return property(fget, fset) |         return property(fget, fset) | ||||||
|     pk = pk() |     pk = pk() | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def text_score(self): | ||||||
|  |         """ | ||||||
|  |         Used for text searchs | ||||||
|  |         """ | ||||||
|  |         return self._data.get('text_score') | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def _get_db(cls): |     def _get_db(cls): | ||||||
|         """Some Model using other db_alias""" |         """Some Model using other db_alias""" | ||||||
| @@ -179,7 +194,7 @@ class Document(BaseDocument): | |||||||
|  |  | ||||||
|     def save(self, force_insert=False, validate=True, clean=True, |     def save(self, force_insert=False, validate=True, clean=True, | ||||||
|              write_concern=None,  cascade=None, cascade_kwargs=None, |              write_concern=None,  cascade=None, cascade_kwargs=None, | ||||||
|              _refs=None, **kwargs): |              _refs=None, save_condition=None, **kwargs): | ||||||
|         """Save the :class:`~mongoengine.Document` to the database. If the |         """Save the :class:`~mongoengine.Document` to the database. If the | ||||||
|         document already exists, it will be updated, otherwise it will be |         document already exists, it will be updated, otherwise it will be | ||||||
|         created. |         created. | ||||||
| @@ -202,7 +217,8 @@ class Document(BaseDocument): | |||||||
|         :param cascade_kwargs: (optional) kwargs dictionary to be passed throw |         :param cascade_kwargs: (optional) kwargs dictionary to be passed throw | ||||||
|             to cascading saves.  Implies ``cascade=True``. |             to cascading saves.  Implies ``cascade=True``. | ||||||
|         :param _refs: A list of processed references used in cascading saves |         :param _refs: A list of processed references used in cascading saves | ||||||
|  |         :param save_condition: only perform save if matching record in db | ||||||
|  |             satisfies condition(s) (e.g., version number) | ||||||
|         .. versionchanged:: 0.5 |         .. versionchanged:: 0.5 | ||||||
|             In existing documents it only saves changed fields using |             In existing documents it only saves changed fields using | ||||||
|             set / unset.  Saves are cascaded and any |             set / unset.  Saves are cascaded and any | ||||||
| @@ -216,6 +232,9 @@ class Document(BaseDocument): | |||||||
|             meta['cascade'] = True.  Also you can pass different kwargs to |             meta['cascade'] = True.  Also you can pass different kwargs to | ||||||
|             the cascade save using cascade_kwargs which overwrites the |             the cascade save using cascade_kwargs which overwrites the | ||||||
|             existing kwargs with custom values. |             existing kwargs with custom values. | ||||||
|  |         .. versionchanged:: 0.8.5 | ||||||
|  |             Optional save_condition that only overwrites existing documents | ||||||
|  |             if the condition is satisfied in the current db record. | ||||||
|         """ |         """ | ||||||
|         signals.pre_save.send(self.__class__, document=self) |         signals.pre_save.send(self.__class__, document=self) | ||||||
|  |  | ||||||
| @@ -229,7 +248,8 @@ class Document(BaseDocument): | |||||||
|  |  | ||||||
|         created = ('_id' not in doc or self._created or force_insert) |         created = ('_id' not in doc or self._created or force_insert) | ||||||
|  |  | ||||||
|         signals.pre_save_post_validation.send(self.__class__, document=self, created=created) |         signals.pre_save_post_validation.send(self.__class__, document=self, | ||||||
|  |                                               created=created) | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             collection = self._get_collection() |             collection = self._get_collection() | ||||||
| @@ -242,7 +262,12 @@ class Document(BaseDocument): | |||||||
|                 object_id = doc['_id'] |                 object_id = doc['_id'] | ||||||
|                 updates, removals = self._delta() |                 updates, removals = self._delta() | ||||||
|                 # Need to add shard key to query, or you get an error |                 # Need to add shard key to query, or you get an error | ||||||
|                 select_dict = {'_id': object_id} |                 if save_condition is not None: | ||||||
|  |                     select_dict = transform.query(self.__class__, | ||||||
|  |                                                   **save_condition) | ||||||
|  |                 else: | ||||||
|  |                     select_dict = {} | ||||||
|  |                 select_dict['_id'] = object_id | ||||||
|                 shard_key = self.__class__._meta.get('shard_key', tuple()) |                 shard_key = self.__class__._meta.get('shard_key', tuple()) | ||||||
|                 for k in shard_key: |                 for k in shard_key: | ||||||
|                     actual_key = self._db_field_map.get(k, k) |                     actual_key = self._db_field_map.get(k, k) | ||||||
| @@ -262,12 +287,14 @@ class Document(BaseDocument): | |||||||
|                 if removals: |                 if removals: | ||||||
|                     update_query["$unset"] = removals |                     update_query["$unset"] = removals | ||||||
|                 if updates or removals: |                 if updates or removals: | ||||||
|  |                     upsert = save_condition is None | ||||||
|                     last_error = collection.update(select_dict, update_query, |                     last_error = collection.update(select_dict, update_query, | ||||||
|                                                    upsert=True, **write_concern) |                                                    upsert=upsert, **write_concern) | ||||||
|                     created = is_new_object(last_error) |                     created = is_new_object(last_error) | ||||||
|  |  | ||||||
|             if cascade is None: |             if cascade is None: | ||||||
|                 cascade = self._meta.get('cascade', False) or cascade_kwargs is not None |                 cascade = self._meta.get( | ||||||
|  |                     'cascade', False) or cascade_kwargs is not None | ||||||
|  |  | ||||||
|             if cascade: |             if cascade: | ||||||
|                 kwargs = { |                 kwargs = { | ||||||
| @@ -280,7 +307,9 @@ class Document(BaseDocument): | |||||||
|                     kwargs.update(cascade_kwargs) |                     kwargs.update(cascade_kwargs) | ||||||
|                 kwargs['_refs'] = _refs |                 kwargs['_refs'] = _refs | ||||||
|                 self.cascade_save(**kwargs) |                 self.cascade_save(**kwargs) | ||||||
|  |         except pymongo.errors.DuplicateKeyError, err: | ||||||
|  |             message = u'Tried to save duplicate unique keys (%s)' | ||||||
|  |             raise NotUniqueError(message % unicode(err)) | ||||||
|         except pymongo.errors.OperationFailure, err: |         except pymongo.errors.OperationFailure, err: | ||||||
|             message = 'Could not save document (%s)' |             message = 'Could not save document (%s)' | ||||||
|             if re.match('^E1100[01] duplicate key', unicode(err)): |             if re.match('^E1100[01] duplicate key', unicode(err)): | ||||||
| @@ -290,12 +319,12 @@ class Document(BaseDocument): | |||||||
|                 raise NotUniqueError(message % unicode(err)) |                 raise NotUniqueError(message % unicode(err)) | ||||||
|             raise OperationError(message % unicode(err)) |             raise OperationError(message % unicode(err)) | ||||||
|         id_field = self._meta['id_field'] |         id_field = self._meta['id_field'] | ||||||
|         if id_field not in self._meta.get('shard_key', []): |         if created or id_field not in self._meta.get('shard_key', []): | ||||||
|             self[id_field] = self._fields[id_field].to_python(object_id) |             self[id_field] = self._fields[id_field].to_python(object_id) | ||||||
|  |  | ||||||
|  |         signals.post_save.send(self.__class__, document=self, created=created) | ||||||
|         self._clear_changed_fields() |         self._clear_changed_fields() | ||||||
|         self._created = False |         self._created = False | ||||||
|         signals.post_save.send(self.__class__, document=self, created=created) |  | ||||||
|         return self |         return self | ||||||
|  |  | ||||||
|     def cascade_save(self, *args, **kwargs): |     def cascade_save(self, *args, **kwargs): | ||||||
| @@ -358,7 +387,8 @@ class Document(BaseDocument): | |||||||
|                     del(query["_cls"]) |                     del(query["_cls"]) | ||||||
|                 return self._qs.filter(**query).update_one(**kwargs) |                 return self._qs.filter(**query).update_one(**kwargs) | ||||||
|             else: |             else: | ||||||
|                 raise OperationError('attempt to update a document not yet saved') |                 raise OperationError( | ||||||
|  |                     'attempt to update a document not yet saved') | ||||||
|  |  | ||||||
|         # Need to add shard key to query, or you get an error |         # Need to add shard key to query, or you get an error | ||||||
|         return self._qs.filter(**self._object_key).update_one(**kwargs) |         return self._qs.filter(**self._object_key).update_one(**kwargs) | ||||||
| @@ -377,7 +407,8 @@ class Document(BaseDocument): | |||||||
|         signals.pre_delete.send(self.__class__, document=self) |         signals.pre_delete.send(self.__class__, document=self) | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             self._qs.filter(**self._object_key).delete(write_concern=write_concern, _from_doc_delete=True) |             self._qs.filter( | ||||||
|  |                 **self._object_key).delete(write_concern=write_concern, _from_doc_delete=True) | ||||||
|         except pymongo.errors.OperationFailure, err: |         except pymongo.errors.OperationFailure, err: | ||||||
|             message = u'Could not delete document (%s)' % err.message |             message = u'Could not delete document (%s)' % err.message | ||||||
|             raise OperationError(message) |             raise OperationError(message) | ||||||
| @@ -400,7 +431,7 @@ class Document(BaseDocument): | |||||||
|         """ |         """ | ||||||
|         with switch_db(self.__class__, db_alias) as cls: |         with switch_db(self.__class__, db_alias) as cls: | ||||||
|             collection = cls._get_collection() |             collection = cls._get_collection() | ||||||
|             db = cls._get_db |             db = cls._get_db() | ||||||
|         self._get_collection = lambda: collection |         self._get_collection = lambda: collection | ||||||
|         self._get_db = lambda: db |         self._get_db = lambda: db | ||||||
|         self._collection = collection |         self._collection = collection | ||||||
| @@ -444,25 +475,41 @@ class Document(BaseDocument): | |||||||
|         DeReference()([self], max_depth + 1) |         DeReference()([self], max_depth + 1) | ||||||
|         return self |         return self | ||||||
|  |  | ||||||
|     def reload(self, max_depth=1): |     def reload(self, *fields, **kwargs): | ||||||
|         """Reloads all attributes from the database. |         """Reloads all attributes from the database. | ||||||
|  |  | ||||||
|  |         :param fields: (optional) args list of fields to reload | ||||||
|  |         :param max_depth: (optional) depth of dereferencing to follow | ||||||
|  |  | ||||||
|         .. versionadded:: 0.1.2 |         .. versionadded:: 0.1.2 | ||||||
|         .. versionchanged:: 0.6  Now chainable |         .. versionchanged:: 0.6  Now chainable | ||||||
|  |         .. versionchanged:: 0.9  Can provide specific fields to reload | ||||||
|         """ |         """ | ||||||
|  |         max_depth = 1 | ||||||
|  |         if fields and isinstance(fields[0], int): | ||||||
|  |             max_depth = fields[0] | ||||||
|  |             fields = fields[1:] | ||||||
|  |         elif "max_depth" in kwargs: | ||||||
|  |             max_depth = kwargs["max_depth"] | ||||||
|  |  | ||||||
|  |         if not self.pk: | ||||||
|  |             raise self.DoesNotExist("Document does not exist") | ||||||
|         obj = self._qs.read_preference(ReadPreference.PRIMARY).filter( |         obj = self._qs.read_preference(ReadPreference.PRIMARY).filter( | ||||||
|                 **self._object_key).limit(1).select_related(max_depth=max_depth) |             **self._object_key).only(*fields).limit(1 | ||||||
|  |                                                     ).select_related(max_depth=max_depth) | ||||||
|  |  | ||||||
|         if obj: |         if obj: | ||||||
|             obj = obj[0] |             obj = obj[0] | ||||||
|         else: |         else: | ||||||
|             msg = "Reloaded document has been deleted" |             raise self.DoesNotExist("Document does not exist") | ||||||
|             raise OperationError(msg) |  | ||||||
|         for field in self._fields_ordered: |         for field in self._fields_ordered: | ||||||
|  |             if not fields or field in fields: | ||||||
|                 setattr(self, field, self._reload(field, obj[field])) |                 setattr(self, field, self._reload(field, obj[field])) | ||||||
|  |  | ||||||
|         self._changed_fields = obj._changed_fields |         self._changed_fields = obj._changed_fields | ||||||
|         self._created = False |         self._created = False | ||||||
|         return obj |         return self | ||||||
|  |  | ||||||
|     def _reload(self, key, value): |     def _reload(self, key, value): | ||||||
|         """Used by :meth:`~mongoengine.Document.reload` to ensure the |         """Used by :meth:`~mongoengine.Document.reload` to ensure the | ||||||
| @@ -536,6 +583,8 @@ class Document(BaseDocument): | |||||||
|     def ensure_indexes(cls): |     def ensure_indexes(cls): | ||||||
|         """Checks the document meta data and ensures all the indexes exist. |         """Checks the document meta data and ensures all the indexes exist. | ||||||
|  |  | ||||||
|  |         Global defaults can be set in the meta - see :doc:`guide/defining-documents` | ||||||
|  |  | ||||||
|         .. note:: You can disable automatic index creation by setting |         .. note:: You can disable automatic index creation by setting | ||||||
|                   `auto_create_index` to False in the documents meta data |                   `auto_create_index` to False in the documents meta data | ||||||
|         """ |         """ | ||||||
| @@ -545,6 +594,10 @@ class Document(BaseDocument): | |||||||
|         index_cls = cls._meta.get('index_cls', True) |         index_cls = cls._meta.get('index_cls', True) | ||||||
|  |  | ||||||
|         collection = cls._get_collection() |         collection = cls._get_collection() | ||||||
|  |         # 746: when connection is via mongos, the read preference is not necessarily an indication that | ||||||
|  |         # this code runs on a secondary | ||||||
|  |         if not collection.is_mongos and collection.read_preference > 1: | ||||||
|  |             return | ||||||
|  |  | ||||||
|         # determine if an index which we are creating includes |         # determine if an index which we are creating includes | ||||||
|         # _cls as its first field; if so, we can avoid creating |         # _cls as its first field; if so, we can avoid creating | ||||||
| @@ -582,6 +635,7 @@ class Document(BaseDocument): | |||||||
|  |  | ||||||
|         # get all the base classes, subclasses and sieblings |         # get all the base classes, subclasses and sieblings | ||||||
|         classes = [] |         classes = [] | ||||||
|  |  | ||||||
|         def get_classes(cls): |         def get_classes(cls): | ||||||
|  |  | ||||||
|             if (cls not in classes and |             if (cls not in classes and | ||||||
| @@ -639,7 +693,8 @@ class Document(BaseDocument): | |||||||
|         """ |         """ | ||||||
|  |  | ||||||
|         required = cls.list_indexes() |         required = cls.list_indexes() | ||||||
|         existing = [info['key'] for info in cls._get_collection().index_information().values()] |         existing = [info['key'] | ||||||
|  |                     for info in cls._get_collection().index_information().values()] | ||||||
|         missing = [index for index in required if index not in existing] |         missing = [index for index in required if index not in existing] | ||||||
|         extra = [index for index in existing if index not in required] |         extra = [index for index in existing if index not in required] | ||||||
|  |  | ||||||
| @@ -657,6 +712,7 @@ class Document(BaseDocument): | |||||||
|  |  | ||||||
|  |  | ||||||
| class DynamicDocument(Document): | class DynamicDocument(Document): | ||||||
|  |  | ||||||
|     """A Dynamic Document class allowing flexible, expandable and uncontrolled |     """A Dynamic Document class allowing flexible, expandable and uncontrolled | ||||||
|     schemas.  As a :class:`~mongoengine.Document` subclass, acts in the same |     schemas.  As a :class:`~mongoengine.Document` subclass, acts in the same | ||||||
|     way as an ordinary document but has expando style properties.  Any data |     way as an ordinary document but has expando style properties.  Any data | ||||||
| @@ -688,6 +744,7 @@ class DynamicDocument(Document): | |||||||
|  |  | ||||||
|  |  | ||||||
| class DynamicEmbeddedDocument(EmbeddedDocument): | class DynamicEmbeddedDocument(EmbeddedDocument): | ||||||
|  |  | ||||||
|     """A Dynamic Embedded Document class allowing flexible, expandable and |     """A Dynamic Embedded Document class allowing flexible, expandable and | ||||||
|     uncontrolled schemas. See :class:`~mongoengine.DynamicDocument` for more |     uncontrolled schemas. See :class:`~mongoengine.DynamicDocument` for more | ||||||
|     information about dynamic documents. |     information about dynamic documents. | ||||||
| @@ -714,6 +771,7 @@ class DynamicEmbeddedDocument(EmbeddedDocument): | |||||||
|  |  | ||||||
|  |  | ||||||
| class MapReduceDocument(object): | class MapReduceDocument(object): | ||||||
|  |  | ||||||
|     """A document returned from a map/reduce query. |     """A document returned from a map/reduce query. | ||||||
|  |  | ||||||
|     :param collection: An instance of :class:`~pymongo.Collection` |     :param collection: An instance of :class:`~pymongo.Collection` | ||||||
| @@ -744,7 +802,7 @@ class MapReduceDocument(object): | |||||||
|             try: |             try: | ||||||
|                 self.key = id_field_type(self.key) |                 self.key = id_field_type(self.key) | ||||||
|             except: |             except: | ||||||
|                 raise Exception("Could not cast key as %s" % \ |                 raise Exception("Could not cast key as %s" % | ||||||
|                                 id_field_type.__name__) |                                 id_field_type.__name__) | ||||||
|  |  | ||||||
|         if not hasattr(self, "_key_object"): |         if not hasattr(self, "_key_object"): | ||||||
|   | |||||||
| @@ -34,21 +34,25 @@ except ImportError: | |||||||
|     Image = None |     Image = None | ||||||
|     ImageOps = None |     ImageOps = None | ||||||
|  |  | ||||||
| __all__ = ['StringField',  'URLField',  'EmailField',  'IntField',  'LongField', | __all__ = [ | ||||||
|  |     'StringField', 'URLField', 'EmailField', 'IntField', 'LongField', | ||||||
|     'FloatField', 'DecimalField', 'BooleanField', 'DateTimeField', |     'FloatField', 'DecimalField', 'BooleanField', 'DateTimeField', | ||||||
|     'ComplexDateTimeField', 'EmbeddedDocumentField', 'ObjectIdField', |     'ComplexDateTimeField', 'EmbeddedDocumentField', 'ObjectIdField', | ||||||
|     'GenericEmbeddedDocumentField', 'DynamicField', 'ListField', |     'GenericEmbeddedDocumentField', 'DynamicField', 'ListField', | ||||||
|     'SortedListField', 'DictField', 'MapField', 'ReferenceField', |     'SortedListField', 'DictField', 'MapField', 'ReferenceField', | ||||||
|            'GenericReferenceField',  'BinaryField',  'GridFSError', |     'CachedReferenceField', 'GenericReferenceField', 'BinaryField', | ||||||
|            'GridFSProxy',  'FileField',  'ImageGridFsProxy', |     'GridFSError', 'GridFSProxy', 'FileField', 'ImageGridFsProxy', | ||||||
|     'ImproperlyConfigured', 'ImageField', 'GeoPointField', 'PointField', |     'ImproperlyConfigured', 'ImageField', 'GeoPointField', 'PointField', | ||||||
|            'LineStringField', 'PolygonField', 'SequenceField',  'UUIDField'] |     'LineStringField', 'PolygonField', 'SequenceField', 'UUIDField', | ||||||
|  |     'MultiPointField', 'MultiLineStringField', 'MultiPolygonField', | ||||||
|  |     'GeoJsonBaseField'] | ||||||
|  |  | ||||||
|  |  | ||||||
| RECURSIVE_REFERENCE_CONSTANT = 'self' | RECURSIVE_REFERENCE_CONSTANT = 'self' | ||||||
|  |  | ||||||
|  |  | ||||||
| class StringField(BaseField): | class StringField(BaseField): | ||||||
|  |  | ||||||
|     """A unicode string field. |     """A unicode string field. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
| @@ -108,6 +112,7 @@ class StringField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class URLField(StringField): | class URLField(StringField): | ||||||
|  |  | ||||||
|     """A field that validates input as an URL. |     """A field that validates input as an URL. | ||||||
|  |  | ||||||
|     .. versionadded:: 0.3 |     .. versionadded:: 0.3 | ||||||
| @@ -115,7 +120,8 @@ class URLField(StringField): | |||||||
|  |  | ||||||
|     _URL_REGEX = re.compile( |     _URL_REGEX = re.compile( | ||||||
|         r'^(?:http|ftp)s?://'  # http:// or https:// |         r'^(?:http|ftp)s?://'  # http:// or https:// | ||||||
|         r'(?:(?:[A-Z0-9](?:[A-Z0-9-]{0,61}[A-Z0-9])?\.)+(?:[A-Z]{2,6}\.?|[A-Z0-9-]{2,}\.?)|'  # domain... |         # domain... | ||||||
|  |         r'(?:(?:[A-Z0-9](?:[A-Z0-9-]{0,61}[A-Z0-9])?\.)+(?:[A-Z]{2,6}\.?|[A-Z0-9-]{2,}\.?)|' | ||||||
|         r'localhost|'  # localhost... |         r'localhost|'  # localhost... | ||||||
|         r'\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})'  # ...or ip |         r'\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})'  # ...or ip | ||||||
|         r'(?::\d+)?'  # optional port |         r'(?::\d+)?'  # optional port | ||||||
| @@ -144,15 +150,19 @@ class URLField(StringField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class EmailField(StringField): | class EmailField(StringField): | ||||||
|  |  | ||||||
|     """A field that validates input as an E-Mail-Address. |     """A field that validates input as an E-Mail-Address. | ||||||
|  |  | ||||||
|     .. versionadded:: 0.4 |     .. versionadded:: 0.4 | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     EMAIL_REGEX = re.compile( |     EMAIL_REGEX = re.compile( | ||||||
|         r"(^[-!#$%&'*+/=?^_`{}|~0-9A-Z]+(\.[-!#$%&'*+/=?^_`{}|~0-9A-Z]+)*"  # dot-atom |         # dot-atom | ||||||
|         r'|^"([\001-\010\013\014\016-\037!#-\[\]-\177]|\\[\001-011\013\014\016-\177])*"'  # quoted-string |         r"(^[-!#$%&'*+/=?^_`{}|~0-9A-Z]+(\.[-!#$%&'*+/=?^_`{}|~0-9A-Z]+)*" | ||||||
|         r')@(?:[A-Z0-9](?:[A-Z0-9-]{0,253}[A-Z0-9])?\.)+[A-Z]{2,6}\.?$', re.IGNORECASE  # domain |         # quoted-string | ||||||
|  |         r'|^"([\001-\010\013\014\016-\037!#-\[\]-\177]|\\[\001-011\013\014\016-\177])*"' | ||||||
|  |         # domain (max length of an ICAAN TLD is 22 characters) | ||||||
|  |         r')@(?:[A-Z0-9](?:[A-Z0-9-]{0,253}[A-Z0-9])?\.)+[A-Z]{2,22}$', re.IGNORECASE | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     def validate(self, value): |     def validate(self, value): | ||||||
| @@ -162,6 +172,7 @@ class EmailField(StringField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class IntField(BaseField): | class IntField(BaseField): | ||||||
|  |  | ||||||
|     """An 32-bit integer field. |     """An 32-bit integer field. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
| @@ -196,6 +207,7 @@ class IntField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class LongField(BaseField): | class LongField(BaseField): | ||||||
|  |  | ||||||
|     """An 64-bit integer field. |     """An 64-bit integer field. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
| @@ -230,6 +242,7 @@ class LongField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class FloatField(BaseField): | class FloatField(BaseField): | ||||||
|  |  | ||||||
|     """An floating point number field. |     """An floating point number field. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
| @@ -264,6 +277,7 @@ class FloatField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class DecimalField(BaseField): | class DecimalField(BaseField): | ||||||
|  |  | ||||||
|     """A fixed-point decimal number field. |     """A fixed-point decimal number field. | ||||||
|  |  | ||||||
|     .. versionchanged:: 0.8 |     .. versionchanged:: 0.8 | ||||||
| @@ -304,10 +318,13 @@ class DecimalField(BaseField): | |||||||
|             return value |             return value | ||||||
|  |  | ||||||
|         # Convert to string for python 2.6 before casting to Decimal |         # Convert to string for python 2.6 before casting to Decimal | ||||||
|  |         try: | ||||||
|             value = decimal.Decimal("%s" % value) |             value = decimal.Decimal("%s" % value) | ||||||
|  |         except decimal.InvalidOperation: | ||||||
|  |             return value | ||||||
|         return value.quantize(self.precision, rounding=self.rounding) |         return value.quantize(self.precision, rounding=self.rounding) | ||||||
|  |  | ||||||
|     def to_mongo(self, value): |     def to_mongo(self, value, use_db_field=True): | ||||||
|         if value is None: |         if value is None: | ||||||
|             return value |             return value | ||||||
|         if self.force_string: |         if self.force_string: | ||||||
| @@ -334,6 +351,7 @@ class DecimalField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class BooleanField(BaseField): | class BooleanField(BaseField): | ||||||
|  |  | ||||||
|     """A boolean field type. |     """A boolean field type. | ||||||
|  |  | ||||||
|     .. versionadded:: 0.1.2 |     .. versionadded:: 0.1.2 | ||||||
| @@ -352,6 +370,7 @@ class BooleanField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class DateTimeField(BaseField): | class DateTimeField(BaseField): | ||||||
|  |  | ||||||
|     """A datetime field. |     """A datetime field. | ||||||
|  |  | ||||||
|     Uses the python-dateutil library if available alternatively use time.strptime |     Uses the python-dateutil library if available alternatively use time.strptime | ||||||
| @@ -387,7 +406,7 @@ class DateTimeField(BaseField): | |||||||
|         if dateutil: |         if dateutil: | ||||||
|             try: |             try: | ||||||
|                 return dateutil.parser.parse(value) |                 return dateutil.parser.parse(value) | ||||||
|             except ValueError: |             except (TypeError, ValueError): | ||||||
|                 return None |                 return None | ||||||
|  |  | ||||||
|         # split usecs, because they are not recognized by strptime. |         # split usecs, because they are not recognized by strptime. | ||||||
| @@ -419,6 +438,7 @@ class DateTimeField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class ComplexDateTimeField(StringField): | class ComplexDateTimeField(StringField): | ||||||
|  |  | ||||||
|     """ |     """ | ||||||
|     ComplexDateTimeField handles microseconds exactly instead of rounding |     ComplexDateTimeField handles microseconds exactly instead of rounding | ||||||
|     like DateTimeField does. |     like DateTimeField does. | ||||||
| @@ -521,6 +541,7 @@ class ComplexDateTimeField(StringField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class EmbeddedDocumentField(BaseField): | class EmbeddedDocumentField(BaseField): | ||||||
|  |  | ||||||
|     """An embedded document field - with a declared document_type. |     """An embedded document field - with a declared document_type. | ||||||
|     Only valid values are subclasses of :class:`~mongoengine.EmbeddedDocument`. |     Only valid values are subclasses of :class:`~mongoengine.EmbeddedDocument`. | ||||||
|     """ |     """ | ||||||
| @@ -547,10 +568,11 @@ class EmbeddedDocumentField(BaseField): | |||||||
|             return self.document_type._from_son(value) |             return self.document_type._from_son(value) | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
|     def to_mongo(self, value): |     def to_mongo(self, value, use_db_field=True, fields=[]): | ||||||
|         if not isinstance(value, self.document_type): |         if not isinstance(value, self.document_type): | ||||||
|             return value |             return value | ||||||
|         return self.document_type.to_mongo(value) |         return self.document_type.to_mongo(value, use_db_field, | ||||||
|  |                                            fields=fields) | ||||||
|  |  | ||||||
|     def validate(self, value, clean=True): |     def validate(self, value, clean=True): | ||||||
|         """Make sure that the document instance is an instance of the |         """Make sure that the document instance is an instance of the | ||||||
| @@ -570,6 +592,7 @@ class EmbeddedDocumentField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class GenericEmbeddedDocumentField(BaseField): | class GenericEmbeddedDocumentField(BaseField): | ||||||
|  |  | ||||||
|     """A generic embedded document field - allows any |     """A generic embedded document field - allows any | ||||||
|     :class:`~mongoengine.EmbeddedDocument` to be stored. |     :class:`~mongoengine.EmbeddedDocument` to be stored. | ||||||
|  |  | ||||||
| @@ -597,17 +620,18 @@ class GenericEmbeddedDocumentField(BaseField): | |||||||
|  |  | ||||||
|         value.validate(clean=clean) |         value.validate(clean=clean) | ||||||
|  |  | ||||||
|     def to_mongo(self, document): |     def to_mongo(self, document, use_db_field=True): | ||||||
|         if document is None: |         if document is None: | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|         data = document.to_mongo() |         data = document.to_mongo(use_db_field) | ||||||
|         if not '_cls' in data: |         if not '_cls' in data: | ||||||
|             data['_cls'] = document._class_name |             data['_cls'] = document._class_name | ||||||
|         return data |         return data | ||||||
|  |  | ||||||
|  |  | ||||||
| class DynamicField(BaseField): | class DynamicField(BaseField): | ||||||
|  |  | ||||||
|     """A truly dynamic field type capable of handling different and varying |     """A truly dynamic field type capable of handling different and varying | ||||||
|     types of data. |     types of data. | ||||||
|  |  | ||||||
| @@ -624,7 +648,9 @@ class DynamicField(BaseField): | |||||||
|             cls = value.__class__ |             cls = value.__class__ | ||||||
|             val = value.to_mongo() |             val = value.to_mongo() | ||||||
|             # If we its a document thats not inherited add _cls |             # If we its a document thats not inherited add _cls | ||||||
|             if (isinstance(value, (Document, EmbeddedDocument))): |             if (isinstance(value,   Document)): | ||||||
|  |                 val = {"_ref": value.to_dbref(), "_cls": cls.__name__} | ||||||
|  |             if (isinstance(value, EmbeddedDocument)): | ||||||
|                 val['_cls'] = cls.__name__ |                 val['_cls'] = cls.__name__ | ||||||
|             return val |             return val | ||||||
|  |  | ||||||
| @@ -645,6 +671,15 @@ class DynamicField(BaseField): | |||||||
|             value = [v for k, v in sorted(data.iteritems(), key=itemgetter(0))] |             value = [v for k, v in sorted(data.iteritems(), key=itemgetter(0))] | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
|  |     def to_python(self, value): | ||||||
|  |         if isinstance(value, dict) and '_cls' in value: | ||||||
|  |             doc_cls = get_document(value['_cls']) | ||||||
|  |             if '_ref' in value: | ||||||
|  |                 value = doc_cls._get_db().dereference(value['_ref']) | ||||||
|  |             return doc_cls._from_son(value) | ||||||
|  |  | ||||||
|  |         return super(DynamicField, self).to_python(value) | ||||||
|  |  | ||||||
|     def lookup_member(self, member_name): |     def lookup_member(self, member_name): | ||||||
|         return member_name |         return member_name | ||||||
|  |  | ||||||
| @@ -660,6 +695,7 @@ class DynamicField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class ListField(ComplexBaseField): | class ListField(ComplexBaseField): | ||||||
|  |  | ||||||
|     """A list field that wraps a standard field, allowing multiple instances |     """A list field that wraps a standard field, allowing multiple instances | ||||||
|     of the field to be used as a list in the database. |     of the field to be used as a list in the database. | ||||||
|  |  | ||||||
| @@ -693,6 +729,7 @@ class ListField(ComplexBaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class SortedListField(ListField): | class SortedListField(ListField): | ||||||
|  |  | ||||||
|     """A ListField that sorts the contents of its list before writing to |     """A ListField that sorts the contents of its list before writing to | ||||||
|     the database in order to ensure that a sorted list is always |     the database in order to ensure that a sorted list is always | ||||||
|     retrieved. |     retrieved. | ||||||
| @@ -725,12 +762,31 @@ class SortedListField(ListField): | |||||||
|         return sorted(value, reverse=self._order_reverse) |         return sorted(value, reverse=self._order_reverse) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def key_not_string(d): | ||||||
|  |     """ Helper function to recursively determine if any key in a dictionary is | ||||||
|  |     not a string. | ||||||
|  |     """ | ||||||
|  |     for k, v in d.items(): | ||||||
|  |         if not isinstance(k, basestring) or (isinstance(v, dict) and key_not_string(v)): | ||||||
|  |             return True | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def key_has_dot_or_dollar(d): | ||||||
|  |     """ Helper function to recursively determine if any key in a dictionary | ||||||
|  |     contains a dot or a dollar sign. | ||||||
|  |     """ | ||||||
|  |     for k, v in d.items(): | ||||||
|  |         if ('.' in k or '$' in k) or (isinstance(v, dict) and key_has_dot_or_dollar(v)): | ||||||
|  |             return True | ||||||
|  |  | ||||||
|  |  | ||||||
| class DictField(ComplexBaseField): | class DictField(ComplexBaseField): | ||||||
|  |  | ||||||
|     """A dictionary field that wraps a standard Python dictionary. This is |     """A dictionary field that wraps a standard Python dictionary. This is | ||||||
|     similar to an embedded document, but the structure is not defined. |     similar to an embedded document, but the structure is not defined. | ||||||
|  |  | ||||||
|     .. note:: |     .. note:: | ||||||
|         Required means it cannot be empty - as the default for ListFields is [] |         Required means it cannot be empty - as the default for DictFields is {} | ||||||
|  |  | ||||||
|     .. versionadded:: 0.3 |     .. versionadded:: 0.3 | ||||||
|     .. versionchanged:: 0.5 - Can now handle complex / varying types of data |     .. versionchanged:: 0.5 - Can now handle complex / varying types of data | ||||||
| @@ -750,11 +806,11 @@ class DictField(ComplexBaseField): | |||||||
|         if not isinstance(value, dict): |         if not isinstance(value, dict): | ||||||
|             self.error('Only dictionaries may be used in a DictField') |             self.error('Only dictionaries may be used in a DictField') | ||||||
|  |  | ||||||
|         if any(k for k in value.keys() if not isinstance(k, basestring)): |         if key_not_string(value): | ||||||
|             msg = ("Invalid dictionary key - documents must " |             msg = ("Invalid dictionary key - documents must " | ||||||
|                    "have only string keys") |                    "have only string keys") | ||||||
|             self.error(msg) |             self.error(msg) | ||||||
|         if any(('.' in k or '$' in k) for k in value.keys()): |         if key_has_dot_or_dollar(value): | ||||||
|             self.error('Invalid dictionary key name - keys may not contain "."' |             self.error('Invalid dictionary key name - keys may not contain "."' | ||||||
|                        ' or "$" characters') |                        ' or "$" characters') | ||||||
|         super(DictField, self).validate(value) |         super(DictField, self).validate(value) | ||||||
| @@ -769,10 +825,19 @@ class DictField(ComplexBaseField): | |||||||
|  |  | ||||||
|         if op in match_operators and isinstance(value, basestring): |         if op in match_operators and isinstance(value, basestring): | ||||||
|             return StringField().prepare_query_value(op, value) |             return StringField().prepare_query_value(op, value) | ||||||
|  |  | ||||||
|  |         if hasattr(self.field, 'field'): | ||||||
|  |             if op in ('set', 'unset') and isinstance(value, dict): | ||||||
|  |                 return dict( | ||||||
|  |                     (k, self.field.prepare_query_value(op, v)) | ||||||
|  |                     for k, v in value.items()) | ||||||
|  |             return self.field.prepare_query_value(op, value) | ||||||
|  |  | ||||||
|         return super(DictField, self).prepare_query_value(op, value) |         return super(DictField, self).prepare_query_value(op, value) | ||||||
|  |  | ||||||
|  |  | ||||||
| class MapField(DictField): | class MapField(DictField): | ||||||
|  |  | ||||||
|     """A field that maps a name to a specified field type. Similar to |     """A field that maps a name to a specified field type. Similar to | ||||||
|     a DictField, except the 'value' of each item must match the specified |     a DictField, except the 'value' of each item must match the specified | ||||||
|     field type. |     field type. | ||||||
| @@ -788,6 +853,7 @@ class MapField(DictField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class ReferenceField(BaseField): | class ReferenceField(BaseField): | ||||||
|  |  | ||||||
|     """A reference to a document that will be automatically dereferenced on |     """A reference to a document that will be automatically dereferenced on | ||||||
|     access (lazily). |     access (lazily). | ||||||
|  |  | ||||||
| @@ -921,7 +987,147 @@ class ReferenceField(BaseField): | |||||||
|         return self.document_type._fields.get(member_name) |         return self.document_type._fields.get(member_name) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CachedReferenceField(BaseField): | ||||||
|  |  | ||||||
|  |     """ | ||||||
|  |     A referencefield with cache fields to porpuse pseudo-joins | ||||||
|  |     .. versionadded:: 0.9 | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, document_type, fields=[], auto_sync=True, **kwargs): | ||||||
|  |         """Initialises the Cached Reference Field. | ||||||
|  |  | ||||||
|  |         :param fields:  A list of fields to be cached in document | ||||||
|  |         :param auto_sync: if True documents are auto updated. | ||||||
|  |         """ | ||||||
|  |         if not isinstance(document_type, basestring) and \ | ||||||
|  |                 not issubclass(document_type, (Document, basestring)): | ||||||
|  |  | ||||||
|  |             self.error('Argument to CachedReferenceField constructor must be a' | ||||||
|  |                        ' document class or a string') | ||||||
|  |  | ||||||
|  |         self.auto_sync = auto_sync | ||||||
|  |         self.document_type_obj = document_type | ||||||
|  |         self.fields = fields | ||||||
|  |         super(CachedReferenceField, self).__init__(**kwargs) | ||||||
|  |  | ||||||
|  |     def start_listener(self): | ||||||
|  |         from mongoengine import signals | ||||||
|  |         signals.post_save.connect(self.on_document_pre_save, | ||||||
|  |                                   sender=self.document_type) | ||||||
|  |  | ||||||
|  |     def on_document_pre_save(self, sender, document, created, **kwargs): | ||||||
|  |         if not created: | ||||||
|  |             update_kwargs = dict( | ||||||
|  |                 ('set__%s__%s' % (self.name, k), v) | ||||||
|  |                 for k, v in document._delta()[0].items() | ||||||
|  |                 if k in self.fields) | ||||||
|  |  | ||||||
|  |             if update_kwargs: | ||||||
|  |                 filter_kwargs = {} | ||||||
|  |                 filter_kwargs[self.name] = document | ||||||
|  |  | ||||||
|  |                 self.owner_document.objects( | ||||||
|  |                     **filter_kwargs).update(**update_kwargs) | ||||||
|  |  | ||||||
|  |     def to_python(self, value): | ||||||
|  |         if isinstance(value, dict): | ||||||
|  |             collection = self.document_type._get_collection_name() | ||||||
|  |             value = DBRef( | ||||||
|  |                 collection, self.document_type.id.to_python(value['_id'])) | ||||||
|  |  | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def document_type(self): | ||||||
|  |         if isinstance(self.document_type_obj, basestring): | ||||||
|  |             if self.document_type_obj == RECURSIVE_REFERENCE_CONSTANT: | ||||||
|  |                 self.document_type_obj = self.owner_document | ||||||
|  |             else: | ||||||
|  |                 self.document_type_obj = get_document(self.document_type_obj) | ||||||
|  |         return self.document_type_obj | ||||||
|  |  | ||||||
|  |     def __get__(self, instance, owner): | ||||||
|  |         if instance is None: | ||||||
|  |             # Document class being used rather than a document object | ||||||
|  |             return self | ||||||
|  |  | ||||||
|  |         # Get value from document instance if available | ||||||
|  |         value = instance._data.get(self.name) | ||||||
|  |         self._auto_dereference = instance._fields[self.name]._auto_dereference | ||||||
|  |         # Dereference DBRefs | ||||||
|  |         if self._auto_dereference and isinstance(value, DBRef): | ||||||
|  |             value = self.document_type._get_db().dereference(value) | ||||||
|  |             if value is not None: | ||||||
|  |                 instance._data[self.name] = self.document_type._from_son(value) | ||||||
|  |  | ||||||
|  |         return super(CachedReferenceField, self).__get__(instance, owner) | ||||||
|  |  | ||||||
|  |     def to_mongo(self, document): | ||||||
|  |         id_field_name = self.document_type._meta['id_field'] | ||||||
|  |         id_field = self.document_type._fields[id_field_name] | ||||||
|  |         doc_tipe = self.document_type | ||||||
|  |  | ||||||
|  |         if isinstance(document, Document): | ||||||
|  |             # We need the id from the saved object to create the DBRef | ||||||
|  |             id_ = document.pk | ||||||
|  |             if id_ is None: | ||||||
|  |                 self.error('You can only reference documents once they have' | ||||||
|  |                            ' been saved to the database') | ||||||
|  |         else: | ||||||
|  |             self.error('Only accept a document object') | ||||||
|  |  | ||||||
|  |         value = SON(( | ||||||
|  |             ("_id", id_field.to_mongo(id_)), | ||||||
|  |         )) | ||||||
|  |  | ||||||
|  |         value.update(dict(document.to_mongo(fields=self.fields))) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def prepare_query_value(self, op, value): | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |  | ||||||
|  |         if isinstance(value, Document): | ||||||
|  |             if value.pk is None: | ||||||
|  |                 self.error('You can only reference documents once they have' | ||||||
|  |                            ' been saved to the database') | ||||||
|  |             return {'_id': value.pk} | ||||||
|  |  | ||||||
|  |         raise NotImplementedError | ||||||
|  |  | ||||||
|  |     def validate(self, value): | ||||||
|  |  | ||||||
|  |         if not isinstance(value, (self.document_type)): | ||||||
|  |             self.error("A CachedReferenceField only accepts documents") | ||||||
|  |  | ||||||
|  |         if isinstance(value, Document) and value.id is None: | ||||||
|  |             self.error('You can only reference documents once they have been ' | ||||||
|  |                        'saved to the database') | ||||||
|  |  | ||||||
|  |     def lookup_member(self, member_name): | ||||||
|  |         return self.document_type._fields.get(member_name) | ||||||
|  |  | ||||||
|  |     def sync_all(self): | ||||||
|  |         """ | ||||||
|  |         Sync all cached fields on demand. | ||||||
|  |         Caution: this operation may be slower. | ||||||
|  |         """ | ||||||
|  |         update_key = 'set__%s' % self.name | ||||||
|  |  | ||||||
|  |         for doc in self.document_type.objects: | ||||||
|  |             filter_kwargs = {} | ||||||
|  |             filter_kwargs[self.name] = doc | ||||||
|  |  | ||||||
|  |             update_kwargs = {} | ||||||
|  |             update_kwargs[update_key] = doc | ||||||
|  |  | ||||||
|  |             self.owner_document.objects( | ||||||
|  |                 **filter_kwargs).update(**update_kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
| class GenericReferenceField(BaseField): | class GenericReferenceField(BaseField): | ||||||
|  |  | ||||||
|     """A reference to *any* :class:`~mongoengine.document.Document` subclass |     """A reference to *any* :class:`~mongoengine.document.Document` subclass | ||||||
|     that will be automatically dereferenced on access (lazily). |     that will be automatically dereferenced on access (lazily). | ||||||
|  |  | ||||||
| @@ -940,6 +1146,7 @@ class GenericReferenceField(BaseField): | |||||||
|             return self |             return self | ||||||
|  |  | ||||||
|         value = instance._data.get(self.name) |         value = instance._data.get(self.name) | ||||||
|  |  | ||||||
|         self._auto_dereference = instance._fields[self.name]._auto_dereference |         self._auto_dereference = instance._fields[self.name]._auto_dereference | ||||||
|         if self._auto_dereference and isinstance(value, (dict, SON)): |         if self._auto_dereference and isinstance(value, (dict, SON)): | ||||||
|             instance._data[self.name] = self.dereference(value) |             instance._data[self.name] = self.dereference(value) | ||||||
| @@ -967,7 +1174,7 @@ class GenericReferenceField(BaseField): | |||||||
|             doc = doc_cls._from_son(doc) |             doc = doc_cls._from_son(doc) | ||||||
|         return doc |         return doc | ||||||
|  |  | ||||||
|     def to_mongo(self, document): |     def to_mongo(self, document, use_db_field=True): | ||||||
|         if document is None: |         if document is None: | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
| @@ -989,7 +1196,10 @@ class GenericReferenceField(BaseField): | |||||||
|         id_ = id_field.to_mongo(id_) |         id_ = id_field.to_mongo(id_) | ||||||
|         collection = document._get_collection_name() |         collection = document._get_collection_name() | ||||||
|         ref = DBRef(collection, id_) |         ref = DBRef(collection, id_) | ||||||
|         return {'_cls': document._class_name, '_ref': ref} |         return SON(( | ||||||
|  |             ('_cls', document._class_name), | ||||||
|  |             ('_ref', ref) | ||||||
|  |         )) | ||||||
|  |  | ||||||
|     def prepare_query_value(self, op, value): |     def prepare_query_value(self, op, value): | ||||||
|         if value is None: |         if value is None: | ||||||
| @@ -999,6 +1209,7 @@ class GenericReferenceField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class BinaryField(BaseField): | class BinaryField(BaseField): | ||||||
|  |  | ||||||
|     """A binary data field. |     """A binary data field. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
| @@ -1030,6 +1241,7 @@ class GridFSError(Exception): | |||||||
|  |  | ||||||
|  |  | ||||||
| class GridFSProxy(object): | class GridFSProxy(object): | ||||||
|  |  | ||||||
|     """Proxy object to handle writing and reading of files to and from GridFS |     """Proxy object to handle writing and reading of files to and from GridFS | ||||||
|  |  | ||||||
|     .. versionadded:: 0.4 |     .. versionadded:: 0.4 | ||||||
| @@ -1083,6 +1295,11 @@ class GridFSProxy(object): | |||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return '<%s: %s>' % (self.__class__.__name__, self.grid_id) |         return '<%s: %s>' % (self.__class__.__name__, self.grid_id) | ||||||
|  |  | ||||||
|  |     def __str__(self): | ||||||
|  |         name = getattr( | ||||||
|  |             self.get(), 'filename', self.grid_id) if self.get() else '(no file)' | ||||||
|  |         return '<%s: %s>' % (self.__class__.__name__, name) | ||||||
|  |  | ||||||
|     def __eq__(self, other): |     def __eq__(self, other): | ||||||
|         if isinstance(other, GridFSProxy): |         if isinstance(other, GridFSProxy): | ||||||
|             return ((self.grid_id == other.grid_id) and |             return ((self.grid_id == other.grid_id) and | ||||||
| @@ -1094,7 +1311,8 @@ class GridFSProxy(object): | |||||||
|     @property |     @property | ||||||
|     def fs(self): |     def fs(self): | ||||||
|         if not self._fs: |         if not self._fs: | ||||||
|             self._fs = gridfs.GridFS(get_db(self.db_alias), self.collection_name) |             self._fs = gridfs.GridFS( | ||||||
|  |                 get_db(self.db_alias), self.collection_name) | ||||||
|         return self._fs |         return self._fs | ||||||
|  |  | ||||||
|     def get(self, id=None): |     def get(self, id=None): | ||||||
| @@ -1168,6 +1386,7 @@ class GridFSProxy(object): | |||||||
|  |  | ||||||
|  |  | ||||||
| class FileField(BaseField): | class FileField(BaseField): | ||||||
|  |  | ||||||
|     """A GridFS storage field. |     """A GridFS storage field. | ||||||
|  |  | ||||||
|     .. versionadded:: 0.4 |     .. versionadded:: 0.4 | ||||||
| @@ -1212,7 +1431,8 @@ class FileField(BaseField): | |||||||
|                     pass |                     pass | ||||||
|  |  | ||||||
|             # Create a new proxy object as we don't already have one |             # Create a new proxy object as we don't already have one | ||||||
|             instance._data[key] = self.get_proxy_obj(key=key, instance=instance) |             instance._data[key] = self.get_proxy_obj( | ||||||
|  |                 key=key, instance=instance) | ||||||
|             instance._data[key].put(value) |             instance._data[key].put(value) | ||||||
|         else: |         else: | ||||||
|             instance._data[key] = value |             instance._data[key] = value | ||||||
| @@ -1250,11 +1470,13 @@ class FileField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class ImageGridFsProxy(GridFSProxy): | class ImageGridFsProxy(GridFSProxy): | ||||||
|  |  | ||||||
|     """ |     """ | ||||||
|     Proxy for ImageField |     Proxy for ImageField | ||||||
|  |  | ||||||
|     versionadded: 0.6 |     versionadded: 0.6 | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def put(self, file_obj, **kwargs): |     def put(self, file_obj, **kwargs): | ||||||
|         """ |         """ | ||||||
|         Insert a image in database |         Insert a image in database | ||||||
| @@ -1271,6 +1493,16 @@ class ImageGridFsProxy(GridFSProxy): | |||||||
|         except Exception, e: |         except Exception, e: | ||||||
|             raise ValidationError('Invalid image: %s' % e) |             raise ValidationError('Invalid image: %s' % e) | ||||||
|  |  | ||||||
|  |         # Progressive JPEG | ||||||
|  |         progressive = img.info.get('progressive') or False | ||||||
|  |  | ||||||
|  |         if (kwargs.get('progressive') and | ||||||
|  |                 isinstance(kwargs.get('progressive'), bool) and | ||||||
|  |                 img_format == 'JPEG'): | ||||||
|  |             progressive = True | ||||||
|  |         else: | ||||||
|  |             progressive = False | ||||||
|  |  | ||||||
|         if (field.size and (img.size[0] > field.size['width'] or |         if (field.size and (img.size[0] > field.size['width'] or | ||||||
|                             img.size[1] > field.size['height'])): |                             img.size[1] > field.size['height'])): | ||||||
|             size = field.size |             size = field.size | ||||||
| @@ -1290,7 +1522,8 @@ class ImageGridFsProxy(GridFSProxy): | |||||||
|             size = field.thumbnail_size |             size = field.thumbnail_size | ||||||
|  |  | ||||||
|             if size['force']: |             if size['force']: | ||||||
|                 thumbnail = ImageOps.fit(img, (size['width'], size['height']), Image.ANTIALIAS) |                 thumbnail = ImageOps.fit( | ||||||
|  |                     img, (size['width'], size['height']), Image.ANTIALIAS) | ||||||
|             else: |             else: | ||||||
|                 thumbnail = img.copy() |                 thumbnail = img.copy() | ||||||
|                 thumbnail.thumbnail((size['width'], |                 thumbnail.thumbnail((size['width'], | ||||||
| @@ -1298,14 +1531,14 @@ class ImageGridFsProxy(GridFSProxy): | |||||||
|                                     Image.ANTIALIAS) |                                     Image.ANTIALIAS) | ||||||
|  |  | ||||||
|         if thumbnail: |         if thumbnail: | ||||||
|             thumb_id = self._put_thumbnail(thumbnail, img_format) |             thumb_id = self._put_thumbnail(thumbnail, img_format, progressive) | ||||||
|         else: |         else: | ||||||
|             thumb_id = None |             thumb_id = None | ||||||
|  |  | ||||||
|         w, h = img.size |         w, h = img.size | ||||||
|  |  | ||||||
|         io = StringIO() |         io = StringIO() | ||||||
|         img.save(io, img_format) |         img.save(io, img_format, progressive=progressive) | ||||||
|         io.seek(0) |         io.seek(0) | ||||||
|  |  | ||||||
|         return super(ImageGridFsProxy, self).put(io, |         return super(ImageGridFsProxy, self).put(io, | ||||||
| @@ -1323,11 +1556,11 @@ class ImageGridFsProxy(GridFSProxy): | |||||||
|  |  | ||||||
|         return super(ImageGridFsProxy, self).delete(*args, **kwargs) |         return super(ImageGridFsProxy, self).delete(*args, **kwargs) | ||||||
|  |  | ||||||
|     def _put_thumbnail(self, thumbnail, format, **kwargs): |     def _put_thumbnail(self, thumbnail, format, progressive, **kwargs): | ||||||
|         w, h = thumbnail.size |         w, h = thumbnail.size | ||||||
|  |  | ||||||
|         io = StringIO() |         io = StringIO() | ||||||
|         thumbnail.save(io, format) |         thumbnail.save(io, format, progressive=progressive) | ||||||
|         io.seek(0) |         io.seek(0) | ||||||
|  |  | ||||||
|         return self.fs.put(io, width=w, |         return self.fs.put(io, width=w, | ||||||
| @@ -1376,6 +1609,7 @@ class ImproperlyConfigured(Exception): | |||||||
|  |  | ||||||
|  |  | ||||||
| class ImageField(FileField): | class ImageField(FileField): | ||||||
|  |  | ||||||
|     """ |     """ | ||||||
|     A Image File storage field. |     A Image File storage field. | ||||||
|  |  | ||||||
| @@ -1414,6 +1648,7 @@ class ImageField(FileField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class SequenceField(BaseField): | class SequenceField(BaseField): | ||||||
|  |  | ||||||
|     """Provides a sequental counter see: |     """Provides a sequental counter see: | ||||||
|      http://www.mongodb.org/display/DOCS/Object+IDs#ObjectIDs-SequenceNumbers |      http://www.mongodb.org/display/DOCS/Object+IDs#ObjectIDs-SequenceNumbers | ||||||
|  |  | ||||||
| @@ -1513,6 +1748,14 @@ class SequenceField(BaseField): | |||||||
|  |  | ||||||
|         return super(SequenceField, self).__set__(instance, value) |         return super(SequenceField, self).__set__(instance, value) | ||||||
|  |  | ||||||
|  |     def prepare_query_value(self, op, value): | ||||||
|  |         """ | ||||||
|  |         This method is overriden in order to convert the query value into to required | ||||||
|  |         type. We need to do this in order to be able to successfully compare query | ||||||
|  |         values passed as string, the base implementation returns the value as is. | ||||||
|  |         """ | ||||||
|  |         return self.value_decorator(value) | ||||||
|  |  | ||||||
|     def to_python(self, value): |     def to_python(self, value): | ||||||
|         if value is None: |         if value is None: | ||||||
|             value = self.generate() |             value = self.generate() | ||||||
| @@ -1520,6 +1763,7 @@ class SequenceField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class UUIDField(BaseField): | class UUIDField(BaseField): | ||||||
|  |  | ||||||
|     """A UUID field. |     """A UUID field. | ||||||
|  |  | ||||||
|     .. versionadded:: 0.6 |     .. versionadded:: 0.6 | ||||||
| @@ -1572,7 +1816,13 @@ class UUIDField(BaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class GeoPointField(BaseField): | class GeoPointField(BaseField): | ||||||
|     """A list storing a latitude and longitude. |  | ||||||
|  |     """A list storing a longitude and latitude coordinate. | ||||||
|  |  | ||||||
|  |     .. note:: this represents a generic point in a 2D plane and a legacy way of | ||||||
|  |         representing a geo point. It admits 2d indexes but not "2dsphere" indexes | ||||||
|  |         in MongoDB > 2.4 which are more natural for modeling geospatial points. | ||||||
|  |         See :ref:`geospatial-indexes` | ||||||
|  |  | ||||||
|     .. versionadded:: 0.4 |     .. versionadded:: 0.4 | ||||||
|     """ |     """ | ||||||
| @@ -1587,14 +1837,17 @@ class GeoPointField(BaseField): | |||||||
|                        'of (x, y)') |                        'of (x, y)') | ||||||
|  |  | ||||||
|         if not len(value) == 2: |         if not len(value) == 2: | ||||||
|             self.error("Value (%s) must be a two-dimensional point" % repr(value)) |             self.error("Value (%s) must be a two-dimensional point" % | ||||||
|  |                        repr(value)) | ||||||
|         elif (not isinstance(value[0], (float, int)) or |         elif (not isinstance(value[0], (float, int)) or | ||||||
|               not isinstance(value[1], (float, int))): |               not isinstance(value[1], (float, int))): | ||||||
|             self.error("Both values (%s) in point must be float or int" % repr(value)) |             self.error( | ||||||
|  |                 "Both values (%s) in point must be float or int" % repr(value)) | ||||||
|  |  | ||||||
|  |  | ||||||
| class PointField(GeoJsonBaseField): | class PointField(GeoJsonBaseField): | ||||||
|     """A geo json field storing a latitude and longitude. |  | ||||||
|  |     """A GeoJSON field storing a longitude and latitude coordinate. | ||||||
|  |  | ||||||
|     The data is represented as: |     The data is represented as: | ||||||
|  |  | ||||||
| @@ -1613,7 +1866,8 @@ class PointField(GeoJsonBaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class LineStringField(GeoJsonBaseField): | class LineStringField(GeoJsonBaseField): | ||||||
|     """A geo json field storing a line of latitude and longitude coordinates. |  | ||||||
|  |     """A GeoJSON field storing a line of longitude and latitude coordinates. | ||||||
|  |  | ||||||
|     The data is represented as: |     The data is represented as: | ||||||
|  |  | ||||||
| @@ -1631,7 +1885,8 @@ class LineStringField(GeoJsonBaseField): | |||||||
|  |  | ||||||
|  |  | ||||||
| class PolygonField(GeoJsonBaseField): | class PolygonField(GeoJsonBaseField): | ||||||
|     """A geo json field storing a polygon of latitude and longitude coordinates. |  | ||||||
|  |     """A GeoJSON field storing a polygon of longitude and latitude coordinates. | ||||||
|  |  | ||||||
|     The data is represented as: |     The data is represented as: | ||||||
|  |  | ||||||
| @@ -1649,3 +1904,70 @@ class PolygonField(GeoJsonBaseField): | |||||||
|     .. versionadded:: 0.8 |     .. versionadded:: 0.8 | ||||||
|     """ |     """ | ||||||
|     _type = "Polygon" |     _type = "Polygon" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MultiPointField(GeoJsonBaseField): | ||||||
|  |  | ||||||
|  |     """A GeoJSON field storing a list of Points. | ||||||
|  |  | ||||||
|  |     The data is represented as: | ||||||
|  |  | ||||||
|  |     .. code-block:: js | ||||||
|  |  | ||||||
|  |         { "type" : "MultiPoint" , | ||||||
|  |           "coordinates" : [[x1, y1], [x2, y2]]} | ||||||
|  |  | ||||||
|  |     You can either pass a dict with the full information or a list | ||||||
|  |     to set the value. | ||||||
|  |  | ||||||
|  |     Requires mongodb >= 2.6 | ||||||
|  |     .. versionadded:: 0.9 | ||||||
|  |     """ | ||||||
|  |     _type = "MultiPoint" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MultiLineStringField(GeoJsonBaseField): | ||||||
|  |  | ||||||
|  |     """A GeoJSON field storing a list of LineStrings. | ||||||
|  |  | ||||||
|  |     The data is represented as: | ||||||
|  |  | ||||||
|  |     .. code-block:: js | ||||||
|  |  | ||||||
|  |         { "type" : "MultiLineString" , | ||||||
|  |           "coordinates" : [[[x1, y1], [x1, y1] ... [xn, yn]], | ||||||
|  |                            [[x1, y1], [x1, y1] ... [xn, yn]]]} | ||||||
|  |  | ||||||
|  |     You can either pass a dict with the full information or a list of points. | ||||||
|  |  | ||||||
|  |     Requires mongodb >= 2.6 | ||||||
|  |     .. versionadded:: 0.9 | ||||||
|  |     """ | ||||||
|  |     _type = "MultiLineString" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MultiPolygonField(GeoJsonBaseField): | ||||||
|  |  | ||||||
|  |     """A GeoJSON field storing  list of Polygons. | ||||||
|  |  | ||||||
|  |     The data is represented as: | ||||||
|  |  | ||||||
|  |     .. code-block:: js | ||||||
|  |  | ||||||
|  |         { "type" : "Polygon" , | ||||||
|  |           "coordinates" : [[ | ||||||
|  |                 [[x1, y1], [x1, y1] ... [xn, yn]], | ||||||
|  |                 [[x1, y1], [x1, y1] ... [xn, yn]] | ||||||
|  |             ], [ | ||||||
|  |                 [[x1, y1], [x1, y1] ... [xn, yn]], | ||||||
|  |                 [[x1, y1], [x1, y1] ... [xn, yn]] | ||||||
|  |             ] | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     You can either pass a dict with the full information or a list | ||||||
|  |     of Polygons. | ||||||
|  |  | ||||||
|  |     Requires mongodb >= 2.6 | ||||||
|  |     .. versionadded:: 0.9 | ||||||
|  |     """ | ||||||
|  |     _type = "MultiPolygon" | ||||||
|   | |||||||
| @@ -3,8 +3,6 @@ | |||||||
| import sys | import sys | ||||||
|  |  | ||||||
| PY3 = sys.version_info[0] == 3 | PY3 = sys.version_info[0] == 3 | ||||||
| PY25 = sys.version_info[:2] == (2, 5) |  | ||||||
| UNICODE_KWARGS = int(''.join([str(x) for x in sys.version_info[:3]])) > 264 |  | ||||||
|  |  | ||||||
| if PY3: | if PY3: | ||||||
|     import codecs |     import codecs | ||||||
| @@ -29,33 +27,3 @@ else: | |||||||
|     txt_type = unicode |     txt_type = unicode | ||||||
|  |  | ||||||
| str_types = (bin_type, txt_type) | str_types = (bin_type, txt_type) | ||||||
|  |  | ||||||
| if PY25: |  | ||||||
|     def product(*args, **kwds): |  | ||||||
|         pools = map(tuple, args) * kwds.get('repeat', 1) |  | ||||||
|         result = [[]] |  | ||||||
|         for pool in pools: |  | ||||||
|             result = [x + [y] for x in result for y in pool] |  | ||||||
|         for prod in result: |  | ||||||
|             yield tuple(prod) |  | ||||||
|     reduce = reduce |  | ||||||
| else: |  | ||||||
|     from itertools import product |  | ||||||
|     from functools import reduce |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # For use with Python 2.5 |  | ||||||
| # converts all keys from unicode to str for d and all nested dictionaries |  | ||||||
| def to_str_keys_recursive(d): |  | ||||||
|     if isinstance(d, list): |  | ||||||
|         for val in d: |  | ||||||
|             if isinstance(val, (dict, list)): |  | ||||||
|                 to_str_keys_recursive(val) |  | ||||||
|     elif isinstance(d, dict): |  | ||||||
|         for key, val in d.items(): |  | ||||||
|             if isinstance(val, (dict, list)): |  | ||||||
|                 to_str_keys_recursive(val) |  | ||||||
|             if isinstance(key, unicode): |  | ||||||
|                 d[str(key)] = d.pop(key) |  | ||||||
|     else: |  | ||||||
|         raise ValueError("non list/dict parameter not allowed") |  | ||||||
|   | |||||||
| @@ -7,16 +7,20 @@ import pprint | |||||||
| import re | import re | ||||||
| import warnings | import warnings | ||||||
|  |  | ||||||
|  | from bson import SON | ||||||
| from bson.code import Code | from bson.code import Code | ||||||
| from bson import json_util | from bson import json_util | ||||||
| import pymongo | import pymongo | ||||||
|  | import pymongo.errors | ||||||
| from pymongo.common import validate_read_preference | from pymongo.common import validate_read_preference | ||||||
|  |  | ||||||
| from mongoengine import signals | from mongoengine import signals | ||||||
|  | from mongoengine.connection import get_db | ||||||
|  | from mongoengine.context_managers import switch_db | ||||||
| from mongoengine.common import _import_class | from mongoengine.common import _import_class | ||||||
|  | from mongoengine.base.common import get_document | ||||||
| from mongoengine.errors import (OperationError, NotUniqueError, | from mongoengine.errors import (OperationError, NotUniqueError, | ||||||
|                                 InvalidQueryError) |                                 InvalidQueryError, LookUpError) | ||||||
|  |  | ||||||
| from mongoengine.queryset import transform | from mongoengine.queryset import transform | ||||||
| from mongoengine.queryset.field_list import QueryFieldList | from mongoengine.queryset.field_list import QueryFieldList | ||||||
| from mongoengine.queryset.visitor import Q, QNode | from mongoengine.queryset.visitor import Q, QNode | ||||||
| @@ -35,6 +39,7 @@ RE_TYPE = type(re.compile('')) | |||||||
|  |  | ||||||
|  |  | ||||||
| class BaseQuerySet(object): | class BaseQuerySet(object): | ||||||
|  |  | ||||||
|     """A set of results returned from a query. Wraps a MongoDB cursor, |     """A set of results returned from a query. Wraps a MongoDB cursor, | ||||||
|     providing :class:`~mongoengine.Document` objects as the results. |     providing :class:`~mongoengine.Document` objects as the results. | ||||||
|     """ |     """ | ||||||
| @@ -49,7 +54,7 @@ class BaseQuerySet(object): | |||||||
|         self._initial_query = {} |         self._initial_query = {} | ||||||
|         self._where_clause = None |         self._where_clause = None | ||||||
|         self._loaded_fields = QueryFieldList() |         self._loaded_fields = QueryFieldList() | ||||||
|         self._ordering = [] |         self._ordering = None | ||||||
|         self._snapshot = False |         self._snapshot = False | ||||||
|         self._timeout = True |         self._timeout = True | ||||||
|         self._class_check = True |         self._class_check = True | ||||||
| @@ -60,7 +65,8 @@ class BaseQuerySet(object): | |||||||
|         self._none = False |         self._none = False | ||||||
|         self._as_pymongo = False |         self._as_pymongo = False | ||||||
|         self._as_pymongo_coerce = False |         self._as_pymongo_coerce = False | ||||||
|         self._len = None |         self._search_text = None | ||||||
|  |         self._include_text_scores = False | ||||||
|  |  | ||||||
|         # If inheritance is allowed, only return instances and instances of |         # If inheritance is allowed, only return instances and instances of | ||||||
|         # subclasses of the class being used |         # subclasses of the class being used | ||||||
| @@ -68,12 +74,14 @@ class BaseQuerySet(object): | |||||||
|             if len(self._document._subclasses) == 1: |             if len(self._document._subclasses) == 1: | ||||||
|                 self._initial_query = {"_cls": self._document._subclasses[0]} |                 self._initial_query = {"_cls": self._document._subclasses[0]} | ||||||
|             else: |             else: | ||||||
|                 self._initial_query = {"_cls": {"$in": self._document._subclasses}} |                 self._initial_query = { | ||||||
|  |                     "_cls": {"$in": self._document._subclasses}} | ||||||
|             self._loaded_fields = QueryFieldList(always_include=['_cls']) |             self._loaded_fields = QueryFieldList(always_include=['_cls']) | ||||||
|         self._cursor_obj = None |         self._cursor_obj = None | ||||||
|         self._limit = None |         self._limit = None | ||||||
|         self._skip = None |         self._skip = None | ||||||
|         self._hint = -1  # Using -1 as None is a valid value for hint |         self._hint = -1  # Using -1 as None is a valid value for hint | ||||||
|  |         self.only_fields = [] | ||||||
|  |  | ||||||
|     def __call__(self, q_obj=None, class_check=True, slave_okay=False, |     def __call__(self, q_obj=None, class_check=True, slave_okay=False, | ||||||
|                  read_preference=None, **query): |                  read_preference=None, **query): | ||||||
| @@ -144,16 +152,34 @@ class BaseQuerySet(object): | |||||||
|             if queryset._scalar: |             if queryset._scalar: | ||||||
|                 return queryset._get_scalar( |                 return queryset._get_scalar( | ||||||
|                     queryset._document._from_son(queryset._cursor[key], |                     queryset._document._from_son(queryset._cursor[key], | ||||||
|                                                  _auto_dereference=self._auto_dereference)) |                                                  _auto_dereference=self._auto_dereference, | ||||||
|  |                                                  only_fields=self.only_fields)) | ||||||
|  |  | ||||||
|             if queryset._as_pymongo: |             if queryset._as_pymongo: | ||||||
|                 return queryset._get_as_pymongo(queryset._cursor.next()) |                 return queryset._get_as_pymongo(queryset._cursor[key]) | ||||||
|             return queryset._document._from_son(queryset._cursor[key], |             return queryset._document._from_son(queryset._cursor[key], | ||||||
|                                                 _auto_dereference=self._auto_dereference) |                                                 _auto_dereference=self._auto_dereference, only_fields=self.only_fields) | ||||||
|         raise AttributeError |         raise AttributeError | ||||||
|  |  | ||||||
|     def __iter__(self): |     def __iter__(self): | ||||||
|         raise NotImplementedError |         raise NotImplementedError | ||||||
|  |  | ||||||
|  |     def _has_data(self): | ||||||
|  |         """ Retrieves whether cursor has any data. """ | ||||||
|  |  | ||||||
|  |         queryset = self.order_by() | ||||||
|  |         return False if queryset.first() is None else True | ||||||
|  |  | ||||||
|  |     def __nonzero__(self): | ||||||
|  |         """ Avoid to open all records in an if stmt in Py2. """ | ||||||
|  |  | ||||||
|  |         return self._has_data() | ||||||
|  |  | ||||||
|  |     def __bool__(self): | ||||||
|  |         """ Avoid to open all records in an if stmt in Py3. """ | ||||||
|  |  | ||||||
|  |         return self._has_data() | ||||||
|  |  | ||||||
|     # Core functions |     # Core functions | ||||||
|  |  | ||||||
|     def all(self): |     def all(self): | ||||||
| @@ -165,6 +191,36 @@ class BaseQuerySet(object): | |||||||
|         """ |         """ | ||||||
|         return self.__call__(*q_objs, **query) |         return self.__call__(*q_objs, **query) | ||||||
|  |  | ||||||
|  |     def search_text(self, text, language=None, include_text_scores=False): | ||||||
|  |         """ | ||||||
|  |         Start a text search, using text indexes. | ||||||
|  |         Require: MongoDB server version 2.6+. | ||||||
|  |  | ||||||
|  |         :param language:  The language that determines the list of stop words | ||||||
|  |             for the search and the rules for the stemmer and tokenizer. | ||||||
|  |             If not specified, the search uses the default language of the index. | ||||||
|  |             For supported languages, see `Text Search Languages <http://docs.mongodb.org/manual/reference/text-search-languages/#text-search-languages>`. | ||||||
|  |  | ||||||
|  |         :param include_text_scores: If True, automaticaly add a text_score attribute to Document. | ||||||
|  |  | ||||||
|  |         """ | ||||||
|  |         queryset = self.clone() | ||||||
|  |         if queryset._search_text: | ||||||
|  |             raise OperationError( | ||||||
|  |                 "Is not possible to use search_text two times.") | ||||||
|  |  | ||||||
|  |         query_kwargs = SON({'$search': text}) | ||||||
|  |         if language: | ||||||
|  |             query_kwargs['$language'] = language | ||||||
|  |  | ||||||
|  |         queryset._query_obj &= Q(__raw__={'$text': query_kwargs}) | ||||||
|  |         queryset._mongo_query = None | ||||||
|  |         queryset._cursor_obj = None | ||||||
|  |         queryset._search_text = text | ||||||
|  |         queryset._include_text_scores = include_text_scores | ||||||
|  |  | ||||||
|  |         return queryset | ||||||
|  |  | ||||||
|     def get(self, *q_objs, **query): |     def get(self, *q_objs, **query): | ||||||
|         """Retrieve the the matching object raising |         """Retrieve the the matching object raising | ||||||
|         :class:`~mongoengine.queryset.MultipleObjectsReturned` or |         :class:`~mongoengine.queryset.MultipleObjectsReturned` or | ||||||
| @@ -175,7 +231,7 @@ class BaseQuerySet(object): | |||||||
|         .. versionadded:: 0.3 |         .. versionadded:: 0.3 | ||||||
|         """ |         """ | ||||||
|         queryset = self.clone() |         queryset = self.clone() | ||||||
|         queryset = queryset.limit(2) |         queryset = queryset.order_by().limit(2) | ||||||
|         queryset = queryset.filter(*q_objs, **query) |         queryset = queryset.filter(*q_objs, **query) | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
| @@ -302,6 +358,9 @@ class BaseQuerySet(object): | |||||||
|         signals.pre_bulk_insert.send(self._document, documents=docs) |         signals.pre_bulk_insert.send(self._document, documents=docs) | ||||||
|         try: |         try: | ||||||
|             ids = self._collection.insert(raw, **write_concern) |             ids = self._collection.insert(raw, **write_concern) | ||||||
|  |         except pymongo.errors.DuplicateKeyError, err: | ||||||
|  |             message = 'Could not save document (%s)' | ||||||
|  |             raise NotUniqueError(message % unicode(err)) | ||||||
|         except pymongo.errors.OperationFailure, err: |         except pymongo.errors.OperationFailure, err: | ||||||
|             message = 'Could not save document (%s)' |             message = 'Could not save document (%s)' | ||||||
|             if re.match('^E1100[01] duplicate key', unicode(err)): |             if re.match('^E1100[01] duplicate key', unicode(err)): | ||||||
| @@ -331,14 +390,9 @@ class BaseQuerySet(object): | |||||||
|             :meth:`skip` that has been applied to this cursor into account when |             :meth:`skip` that has been applied to this cursor into account when | ||||||
|             getting the count |             getting the count | ||||||
|         """ |         """ | ||||||
|         if self._limit == 0: |         if self._limit == 0 and with_limit_and_skip or self._none: | ||||||
|             return 0 |             return 0 | ||||||
|         if with_limit_and_skip and self._len is not None: |         return self._cursor.count(with_limit_and_skip=with_limit_and_skip) | ||||||
|             return self._len |  | ||||||
|         count = self._cursor.count(with_limit_and_skip=with_limit_and_skip) |  | ||||||
|         if with_limit_and_skip: |  | ||||||
|             self._len = count |  | ||||||
|         return count |  | ||||||
|  |  | ||||||
|     def delete(self, write_concern=None, _from_doc_delete=False): |     def delete(self, write_concern=None, _from_doc_delete=False): | ||||||
|         """Delete the documents matched by the query. |         """Delete the documents matched by the query. | ||||||
| @@ -351,6 +405,7 @@ class BaseQuerySet(object): | |||||||
|             will force an fsync on the primary server. |             will force an fsync on the primary server. | ||||||
|         :param _from_doc_delete: True when called from document delete therefore |         :param _from_doc_delete: True when called from document delete therefore | ||||||
|             signals will have been triggered so don't loop. |             signals will have been triggered so don't loop. | ||||||
|  |         :returns number of deleted documents | ||||||
|         """ |         """ | ||||||
|         queryset = self.clone() |         queryset = self.clone() | ||||||
|         doc = queryset._document |         doc = queryset._document | ||||||
| @@ -368,9 +423,11 @@ class BaseQuerySet(object): | |||||||
|                                 has_delete_signal) and not _from_doc_delete |                                 has_delete_signal) and not _from_doc_delete | ||||||
|  |  | ||||||
|         if call_document_delete: |         if call_document_delete: | ||||||
|  |             cnt = 0 | ||||||
|             for doc in queryset: |             for doc in queryset: | ||||||
|                 doc.delete(write_concern=write_concern) |                 doc.delete(write_concern=write_concern) | ||||||
|             return |                 cnt += 1 | ||||||
|  |             return cnt | ||||||
|  |  | ||||||
|         delete_rules = doc._meta.get('delete_rules') or {} |         delete_rules = doc._meta.get('delete_rules') or {} | ||||||
|         # Check for DENY rules before actually deleting/nullifying any other |         # Check for DENY rules before actually deleting/nullifying any other | ||||||
| @@ -401,7 +458,8 @@ class BaseQuerySet(object): | |||||||
|                     write_concern=write_concern, |                     write_concern=write_concern, | ||||||
|                     **{'pull_all__%s' % field_name: self}) |                     **{'pull_all__%s' % field_name: self}) | ||||||
|  |  | ||||||
|         queryset._collection.remove(queryset._query, write_concern=write_concern) |         result = queryset._collection.remove(queryset._query, write_concern=write_concern) | ||||||
|  |         return result["n"] | ||||||
|  |  | ||||||
|     def update(self, upsert=False, multi=True, write_concern=None, |     def update(self, upsert=False, multi=True, write_concern=None, | ||||||
|                full_result=False, **update): |                full_result=False, **update): | ||||||
| @@ -445,6 +503,8 @@ class BaseQuerySet(object): | |||||||
|                 return result |                 return result | ||||||
|             elif result: |             elif result: | ||||||
|                 return result['n'] |                 return result['n'] | ||||||
|  |         except pymongo.errors.DuplicateKeyError, err: | ||||||
|  |             raise NotUniqueError(u'Update failed (%s)' % unicode(err)) | ||||||
|         except pymongo.errors.OperationFailure, err: |         except pymongo.errors.OperationFailure, err: | ||||||
|             if unicode(err) == u'multi not coded yet': |             if unicode(err) == u'multi not coded yet': | ||||||
|                 message = u'update() method requires MongoDB 1.1.3+' |                 message = u'update() method requires MongoDB 1.1.3+' | ||||||
| @@ -468,6 +528,60 @@ class BaseQuerySet(object): | |||||||
|         return self.update( |         return self.update( | ||||||
|             upsert=upsert, multi=False, write_concern=write_concern, **update) |             upsert=upsert, multi=False, write_concern=write_concern, **update) | ||||||
|  |  | ||||||
|  |     def modify(self, upsert=False, full_response=False, remove=False, new=False, **update): | ||||||
|  |         """Update and return the updated document. | ||||||
|  |  | ||||||
|  |         Returns either the document before or after modification based on `new` | ||||||
|  |         parameter. If no documents match the query and `upsert` is false, | ||||||
|  |         returns ``None``. If upserting and `new` is false, returns ``None``. | ||||||
|  |  | ||||||
|  |         If the full_response parameter is ``True``, the return value will be | ||||||
|  |         the entire response object from the server, including the 'ok' and | ||||||
|  |         'lastErrorObject' fields, rather than just the modified document. | ||||||
|  |         This is useful mainly because the 'lastErrorObject' document holds | ||||||
|  |         information about the command's execution. | ||||||
|  |  | ||||||
|  |         :param upsert: insert if document doesn't exist (default ``False``) | ||||||
|  |         :param full_response: return the entire response object from the | ||||||
|  |             server (default ``False``) | ||||||
|  |         :param remove: remove rather than updating (default ``False``) | ||||||
|  |         :param new: return updated rather than original document | ||||||
|  |             (default ``False``) | ||||||
|  |         :param update: Django-style update keyword arguments | ||||||
|  |  | ||||||
|  |         .. versionadded:: 0.9 | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         if remove and new: | ||||||
|  |             raise OperationError("Conflicting parameters: remove and new") | ||||||
|  |  | ||||||
|  |         if not update and not upsert and not remove: | ||||||
|  |             raise OperationError( | ||||||
|  |                 "No update parameters, must either update or remove") | ||||||
|  |  | ||||||
|  |         queryset = self.clone() | ||||||
|  |         query = queryset._query | ||||||
|  |         update = transform.update(queryset._document, **update) | ||||||
|  |         sort = queryset._ordering | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             result = queryset._collection.find_and_modify( | ||||||
|  |                 query, update, upsert=upsert, sort=sort, remove=remove, new=new, | ||||||
|  |                 full_response=full_response, **self._cursor_args) | ||||||
|  |         except pymongo.errors.DuplicateKeyError, err: | ||||||
|  |             raise NotUniqueError(u"Update failed (%s)" % err) | ||||||
|  |         except pymongo.errors.OperationFailure, err: | ||||||
|  |             raise OperationError(u"Update failed (%s)" % err) | ||||||
|  |  | ||||||
|  |         if full_response: | ||||||
|  |             if result["value"] is not None: | ||||||
|  |                 result["value"] = self._document._from_son(result["value"], only_fields=self.only_fields) | ||||||
|  |         else: | ||||||
|  |             if result is not None: | ||||||
|  |                 result = self._document._from_son(result, only_fields=self.only_fields) | ||||||
|  |  | ||||||
|  |         return result | ||||||
|  |  | ||||||
|     def with_id(self, object_id): |     def with_id(self, object_id): | ||||||
|         """Retrieve the object matching the id provided.  Uses `object_id` only |         """Retrieve the object matching the id provided.  Uses `object_id` only | ||||||
|         and raises InvalidQueryError if a filter has been applied. Returns |         and raises InvalidQueryError if a filter has been applied. Returns | ||||||
| @@ -499,13 +613,13 @@ class BaseQuerySet(object): | |||||||
|         if self._scalar: |         if self._scalar: | ||||||
|             for doc in docs: |             for doc in docs: | ||||||
|                 doc_map[doc['_id']] = self._get_scalar( |                 doc_map[doc['_id']] = self._get_scalar( | ||||||
|                     self._document._from_son(doc)) |                     self._document._from_son(doc, only_fields=self.only_fields)) | ||||||
|         elif self._as_pymongo: |         elif self._as_pymongo: | ||||||
|             for doc in docs: |             for doc in docs: | ||||||
|                 doc_map[doc['_id']] = self._get_as_pymongo(doc) |                 doc_map[doc['_id']] = self._get_as_pymongo(doc) | ||||||
|         else: |         else: | ||||||
|             for doc in docs: |             for doc in docs: | ||||||
|                 doc_map[doc['_id']] = self._document._from_son(doc) |                 doc_map[doc['_id']] = self._document._from_son(doc, only_fields=self.only_fields) | ||||||
|  |  | ||||||
|         return doc_map |         return doc_map | ||||||
|  |  | ||||||
| @@ -524,6 +638,19 @@ class BaseQuerySet(object): | |||||||
|  |  | ||||||
|         return self |         return self | ||||||
|  |  | ||||||
|  |     def using(self, alias): | ||||||
|  |         """This method is for controlling which database the QuerySet will be evaluated against if you are using more than one database. | ||||||
|  |  | ||||||
|  |         :param alias: The database alias | ||||||
|  |  | ||||||
|  |         .. versionadded:: 0.9 | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         with switch_db(self._document, alias) as cls: | ||||||
|  |             collection = cls._get_collection() | ||||||
|  |  | ||||||
|  |         return self.clone_into(self.__class__(self._document, collection)) | ||||||
|  |  | ||||||
|     def clone(self): |     def clone(self): | ||||||
|         """Creates a copy of the current |         """Creates a copy of the current | ||||||
|           :class:`~mongoengine.queryset.QuerySet` |           :class:`~mongoengine.queryset.QuerySet` | ||||||
| @@ -537,13 +664,15 @@ class BaseQuerySet(object): | |||||||
|           :class:`~mongoengine.queryset.base.BaseQuerySet` into another child class |           :class:`~mongoengine.queryset.base.BaseQuerySet` into another child class | ||||||
|         """ |         """ | ||||||
|         if not isinstance(cls, BaseQuerySet): |         if not isinstance(cls, BaseQuerySet): | ||||||
|             raise OperationError('%s is not a subclass of BaseQuerySet' % cls.__name__) |             raise OperationError( | ||||||
|  |                 '%s is not a subclass of BaseQuerySet' % cls.__name__) | ||||||
|  |  | ||||||
|         copy_props = ('_mongo_query', '_initial_query', '_none', '_query_obj', |         copy_props = ('_mongo_query', '_initial_query', '_none', '_query_obj', | ||||||
|                       '_where_clause', '_loaded_fields', '_ordering', '_snapshot', |                       '_where_clause', '_loaded_fields', '_ordering', '_snapshot', | ||||||
|                       '_timeout', '_class_check', '_slave_okay', '_read_preference', |                       '_timeout', '_class_check', '_slave_okay', '_read_preference', | ||||||
|                       '_iter', '_scalar', '_as_pymongo', '_as_pymongo_coerce', |                       '_iter', '_scalar', '_as_pymongo', '_as_pymongo_coerce', | ||||||
|                       '_limit', '_skip', '_hint', '_auto_dereference') |                       '_limit', '_skip', '_hint', '_auto_dereference', | ||||||
|  |                       '_search_text', '_include_text_scores', 'only_fields') | ||||||
|  |  | ||||||
|         for prop in copy_props: |         for prop in copy_props: | ||||||
|             val = getattr(self, prop) |             val = getattr(self, prop) | ||||||
| @@ -626,9 +755,22 @@ class BaseQuerySet(object): | |||||||
|         try: |         try: | ||||||
|             field = self._fields_to_dbfields([field]).pop() |             field = self._fields_to_dbfields([field]).pop() | ||||||
|         finally: |         finally: | ||||||
|             return self._dereference(queryset._cursor.distinct(field), 1, |             distinct = self._dereference(queryset._cursor.distinct(field), 1, | ||||||
|                                          name=field, instance=self._document) |                                          name=field, instance=self._document) | ||||||
|  |  | ||||||
|  |             # We may need to cast to the correct type eg. | ||||||
|  |             # ListField(EmbeddedDocumentField) | ||||||
|  |             doc_field = getattr( | ||||||
|  |                 self._document._fields.get(field), "field", None) | ||||||
|  |             instance = getattr(doc_field, "document_type", False) | ||||||
|  |             EmbeddedDocumentField = _import_class('EmbeddedDocumentField') | ||||||
|  |             GenericEmbeddedDocumentField = _import_class( | ||||||
|  |                 'GenericEmbeddedDocumentField') | ||||||
|  |             if instance and isinstance(doc_field, (EmbeddedDocumentField, | ||||||
|  |                                                    GenericEmbeddedDocumentField)): | ||||||
|  |                 distinct = [instance(**doc) for doc in distinct] | ||||||
|  |             return distinct | ||||||
|  |  | ||||||
|     def only(self, *fields): |     def only(self, *fields): | ||||||
|         """Load only a subset of this document's fields. :: |         """Load only a subset of this document's fields. :: | ||||||
|  |  | ||||||
| @@ -648,6 +790,7 @@ class BaseQuerySet(object): | |||||||
|         .. versionchanged:: 0.5 - Added subfield support |         .. versionchanged:: 0.5 - Added subfield support | ||||||
|         """ |         """ | ||||||
|         fields = dict([(f, QueryFieldList.ONLY) for f in fields]) |         fields = dict([(f, QueryFieldList.ONLY) for f in fields]) | ||||||
|  |         self.only_fields = fields.keys() | ||||||
|         return self.fields(True, **fields) |         return self.fields(True, **fields) | ||||||
|  |  | ||||||
|     def exclude(self, *fields): |     def exclude(self, *fields): | ||||||
| @@ -704,7 +847,8 @@ class BaseQuerySet(object): | |||||||
|         for value, group in itertools.groupby(fields, lambda x: x[1]): |         for value, group in itertools.groupby(fields, lambda x: x[1]): | ||||||
|             fields = [field for field, value in group] |             fields = [field for field, value in group] | ||||||
|             fields = queryset._fields_to_dbfields(fields) |             fields = queryset._fields_to_dbfields(fields) | ||||||
|             queryset._loaded_fields += QueryFieldList(fields, value=value, _only_called=_only_called) |             queryset._loaded_fields += QueryFieldList( | ||||||
|  |                 fields, value=value, _only_called=_only_called) | ||||||
|  |  | ||||||
|         return queryset |         return queryset | ||||||
|  |  | ||||||
| @@ -827,17 +971,42 @@ class BaseQuerySet(object): | |||||||
|  |  | ||||||
|     # JSON Helpers |     # JSON Helpers | ||||||
|  |  | ||||||
|     def to_json(self): |     def to_json(self, *args, **kwargs): | ||||||
|         """Converts a queryset to JSON""" |         """Converts a queryset to JSON""" | ||||||
|         return json_util.dumps(self.as_pymongo()) |         return json_util.dumps(self.as_pymongo(), *args, **kwargs) | ||||||
|  |  | ||||||
|     def from_json(self, json_data): |     def from_json(self, json_data): | ||||||
|         """Converts json data to unsaved objects""" |         """Converts json data to unsaved objects""" | ||||||
|         son_data = json_util.loads(json_data) |         son_data = json_util.loads(json_data) | ||||||
|         return [self._document._from_son(data) for data in son_data] |         return [self._document._from_son(data, only_fields=self.only_fields) for data in son_data] | ||||||
|  |  | ||||||
|  |     def aggregate(self, *pipeline, **kwargs): | ||||||
|  |         """ | ||||||
|  |         Perform a aggreggate function based in your queryset params | ||||||
|  |         :param pipeline: list of agreggation commands, | ||||||
|  |             see: http://docs.mongodb.org/manual/core/aggregation-pipeline/ | ||||||
|  |  | ||||||
|  |         .. versionadded:: 0.9 | ||||||
|  |         """ | ||||||
|  |         initial_pipeline = [] | ||||||
|  |  | ||||||
|  |         if self._query: | ||||||
|  |             initial_pipeline.append({'$match': self._query}) | ||||||
|  |  | ||||||
|  |         if self._ordering: | ||||||
|  |             initial_pipeline.append({'$sort': dict(self._ordering)}) | ||||||
|  |  | ||||||
|  |         if self._limit is not None: | ||||||
|  |             initial_pipeline.append({'$limit': self._limit}) | ||||||
|  |  | ||||||
|  |         if self._skip is not None: | ||||||
|  |             initial_pipeline.append({'$skip': self._skip}) | ||||||
|  |  | ||||||
|  |         pipeline = initial_pipeline + list(pipeline) | ||||||
|  |  | ||||||
|  |         return self._collection.aggregate(pipeline, cursor={}, **kwargs) | ||||||
|  |  | ||||||
|     # JS functionality |     # JS functionality | ||||||
|  |  | ||||||
|     def map_reduce(self, map_f, reduce_f, output, finalize_f=None, limit=None, |     def map_reduce(self, map_f, reduce_f, output, finalize_f=None, limit=None, | ||||||
|                    scope=None): |                    scope=None): | ||||||
|         """Perform a map/reduce query using the current query spec |         """Perform a map/reduce query using the current query spec | ||||||
| @@ -855,7 +1024,7 @@ class BaseQuerySet(object): | |||||||
|         :param output: output collection name, if set to 'inline' will try to |         :param output: output collection name, if set to 'inline' will try to | ||||||
|            use :class:`~pymongo.collection.Collection.inline_map_reduce` |            use :class:`~pymongo.collection.Collection.inline_map_reduce` | ||||||
|            This can also be a dictionary containing output options |            This can also be a dictionary containing output options | ||||||
|            see: http://docs.mongodb.org/manual/reference/commands/#mapReduce |            see: http://docs.mongodb.org/manual/reference/command/mapReduce/#dbcmd.mapReduce | ||||||
|         :param finalize_f: finalize function, an optional function that |         :param finalize_f: finalize function, an optional function that | ||||||
|                            performs any post-reduction processing. |                            performs any post-reduction processing. | ||||||
|         :param scope: values to insert into map/reduce global scope. Optional. |         :param scope: values to insert into map/reduce global scope. Optional. | ||||||
| @@ -918,8 +1087,36 @@ class BaseQuerySet(object): | |||||||
|             map_reduce_function = 'inline_map_reduce' |             map_reduce_function = 'inline_map_reduce' | ||||||
|         else: |         else: | ||||||
|             map_reduce_function = 'map_reduce' |             map_reduce_function = 'map_reduce' | ||||||
|  |  | ||||||
|  |             if isinstance(output, basestring): | ||||||
|                 mr_args['out'] = output |                 mr_args['out'] = output | ||||||
|  |  | ||||||
|  |             elif isinstance(output, dict): | ||||||
|  |                 ordered_output = [] | ||||||
|  |  | ||||||
|  |                 for part in ('replace', 'merge', 'reduce'): | ||||||
|  |                     value = output.get(part) | ||||||
|  |                     if value: | ||||||
|  |                         ordered_output.append((part, value)) | ||||||
|  |                         break | ||||||
|  |  | ||||||
|  |                 else: | ||||||
|  |                     raise OperationError("actionData not specified for output") | ||||||
|  |  | ||||||
|  |                 db_alias = output.get('db_alias') | ||||||
|  |                 remaing_args = ['db', 'sharded', 'nonAtomic'] | ||||||
|  |  | ||||||
|  |                 if db_alias: | ||||||
|  |                     ordered_output.append(('db', get_db(db_alias).name)) | ||||||
|  |                     del remaing_args[0] | ||||||
|  |  | ||||||
|  |                 for part in remaing_args: | ||||||
|  |                     value = output.get(part) | ||||||
|  |                     if value: | ||||||
|  |                         ordered_output.append((part, value)) | ||||||
|  |  | ||||||
|  |                 mr_args['out'] = SON(ordered_output) | ||||||
|  |  | ||||||
|         results = getattr(queryset._collection, map_reduce_function)( |         results = getattr(queryset._collection, map_reduce_function)( | ||||||
|             map_f, reduce_f, **mr_args) |             map_f, reduce_f, **mr_args) | ||||||
|  |  | ||||||
| @@ -1133,7 +1330,7 @@ class BaseQuerySet(object): | |||||||
|         if self._as_pymongo: |         if self._as_pymongo: | ||||||
|             return self._get_as_pymongo(raw_doc) |             return self._get_as_pymongo(raw_doc) | ||||||
|         doc = self._document._from_son(raw_doc, |         doc = self._document._from_son(raw_doc, | ||||||
|                                        _auto_dereference=self._auto_dereference) |                                        _auto_dereference=self._auto_dereference, only_fields=self.only_fields) | ||||||
|         if self._scalar: |         if self._scalar: | ||||||
|             return self._get_scalar(doc) |             return self._get_scalar(doc) | ||||||
|  |  | ||||||
| @@ -1168,6 +1365,13 @@ class BaseQuerySet(object): | |||||||
|             cursor_args['slave_okay'] = self._slave_okay |             cursor_args['slave_okay'] = self._slave_okay | ||||||
|         if self._loaded_fields: |         if self._loaded_fields: | ||||||
|             cursor_args['fields'] = self._loaded_fields.as_dict() |             cursor_args['fields'] = self._loaded_fields.as_dict() | ||||||
|  |  | ||||||
|  |         if self._include_text_scores: | ||||||
|  |             if 'fields' not in cursor_args: | ||||||
|  |                 cursor_args['fields'] = {} | ||||||
|  |  | ||||||
|  |             cursor_args['fields']['text_score'] = {'$meta': "textScore"} | ||||||
|  |  | ||||||
|         return cursor_args |         return cursor_args | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
| @@ -1184,8 +1388,9 @@ class BaseQuerySet(object): | |||||||
|             if self._ordering: |             if self._ordering: | ||||||
|                 # Apply query ordering |                 # Apply query ordering | ||||||
|                 self._cursor_obj.sort(self._ordering) |                 self._cursor_obj.sort(self._ordering) | ||||||
|             elif self._document._meta['ordering']: |             elif self._ordering is None and self._document._meta['ordering']: | ||||||
|                 # Otherwise, apply the ordering from the document model |                 # Otherwise, apply the ordering from the document model, unless | ||||||
|  |                 # it's been explicitly cleared via order_by with no arguments | ||||||
|                 order = self._get_order_by(self._document._meta['ordering']) |                 order = self._get_order_by(self._document._meta['ordering']) | ||||||
|                 self._cursor_obj.sort(order) |                 self._cursor_obj.sort(order) | ||||||
|  |  | ||||||
| @@ -1208,7 +1413,10 @@ class BaseQuerySet(object): | |||||||
|     def _query(self): |     def _query(self): | ||||||
|         if self._mongo_query is None: |         if self._mongo_query is None: | ||||||
|             self._mongo_query = self._query_obj.to_query(self._document) |             self._mongo_query = self._query_obj.to_query(self._document) | ||||||
|             if self._class_check: |             if self._class_check and self._initial_query: | ||||||
|  |                 if "_cls" in self._mongo_query: | ||||||
|  |                     self._mongo_query = {"$and": [self._initial_query, self._mongo_query]} | ||||||
|  |                 else: | ||||||
|                     self._mongo_query.update(self._initial_query) |                     self._mongo_query.update(self._initial_query) | ||||||
|         return self._mongo_query |         return self._mongo_query | ||||||
|  |  | ||||||
| @@ -1339,13 +1547,33 @@ class BaseQuerySet(object): | |||||||
|  |  | ||||||
|         return frequencies |         return frequencies | ||||||
|  |  | ||||||
|     def _fields_to_dbfields(self, fields): |     def _fields_to_dbfields(self, fields, subdoc=False): | ||||||
|         """Translate fields paths to its db equivalents""" |         """Translate fields paths to its db equivalents""" | ||||||
|         ret = [] |         ret = [] | ||||||
|  |         subclasses = [] | ||||||
|  |         document = self._document | ||||||
|  |         if document._meta['allow_inheritance']: | ||||||
|  |             subclasses = [get_document(x) | ||||||
|  |                           for x in document._subclasses][1:] | ||||||
|         for field in fields: |         for field in fields: | ||||||
|  |             try: | ||||||
|                 field = ".".join(f.db_field for f in |                 field = ".".join(f.db_field for f in | ||||||
|                              self._document._lookup_field(field.split('.'))) |                                  document._lookup_field(field.split('.'))) | ||||||
|                 ret.append(field) |                 ret.append(field) | ||||||
|  |             except LookUpError, err: | ||||||
|  |                 found = False | ||||||
|  |                 for subdoc in subclasses: | ||||||
|  |                     try: | ||||||
|  |                         subfield = ".".join(f.db_field for f in | ||||||
|  |                                             subdoc._lookup_field(field.split('.'))) | ||||||
|  |                         ret.append(subfield) | ||||||
|  |                         found = True | ||||||
|  |                         break | ||||||
|  |                     except LookUpError, e: | ||||||
|  |                         pass | ||||||
|  |  | ||||||
|  |                 if not found: | ||||||
|  |                     raise err | ||||||
|         return ret |         return ret | ||||||
|  |  | ||||||
|     def _get_order_by(self, keys): |     def _get_order_by(self, keys): | ||||||
| @@ -1355,6 +1583,13 @@ class BaseQuerySet(object): | |||||||
|         for key in keys: |         for key in keys: | ||||||
|             if not key: |             if not key: | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|  |             if key == '$text_score': | ||||||
|  |                 # automatically set to include text scores | ||||||
|  |                 self._include_text_scores = True | ||||||
|  |                 key_list.append(('text_score', {'$meta': "textScore"})) | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|             direction = pymongo.ASCENDING |             direction = pymongo.ASCENDING | ||||||
|             if key[0] == '-': |             if key[0] == '-': | ||||||
|                 direction = pymongo.DESCENDING |                 direction = pymongo.DESCENDING | ||||||
| @@ -1367,7 +1602,7 @@ class BaseQuerySet(object): | |||||||
|                 pass |                 pass | ||||||
|             key_list.append((key, direction)) |             key_list.append((key, direction)) | ||||||
|  |  | ||||||
|         if self._cursor_obj: |         if self._cursor_obj and key_list: | ||||||
|             self._cursor_obj.sort(key_list) |             self._cursor_obj.sort(key_list) | ||||||
|         return key_list |         return key_list | ||||||
|  |  | ||||||
| @@ -1425,6 +1660,7 @@ class BaseQuerySet(object): | |||||||
|                     # type of this field and use the corresponding |                     # type of this field and use the corresponding | ||||||
|                     # .to_python(...) |                     # .to_python(...) | ||||||
|                     from mongoengine.fields import EmbeddedDocumentField |                     from mongoengine.fields import EmbeddedDocumentField | ||||||
|  |  | ||||||
|                     obj = self._document |                     obj = self._document | ||||||
|                     for chunk in path.split('.'): |                     for chunk in path.split('.'): | ||||||
|                         obj = getattr(obj, chunk, None) |                         obj = getattr(obj, chunk, None) | ||||||
| @@ -1435,6 +1671,7 @@ class BaseQuerySet(object): | |||||||
|                     if obj and data is not None: |                     if obj and data is not None: | ||||||
|                         data = obj.to_python(data) |                         data = obj.to_python(data) | ||||||
|             return data |             return data | ||||||
|  |  | ||||||
|         return clean(row) |         return clean(row) | ||||||
|  |  | ||||||
|     def _sub_js_fields(self, code): |     def _sub_js_fields(self, code): | ||||||
| @@ -1443,6 +1680,7 @@ class BaseQuerySet(object): | |||||||
|         substituted for the MongoDB name of the field (specified using the |         substituted for the MongoDB name of the field (specified using the | ||||||
|         :attr:`name` keyword argument in a field's constructor). |         :attr:`name` keyword argument in a field's constructor). | ||||||
|         """ |         """ | ||||||
|  |  | ||||||
|         def field_sub(match): |         def field_sub(match): | ||||||
|             # Extract just the field name, and look up the field objects |             # Extract just the field name, and look up the field objects | ||||||
|             field_name = match.group(1).split('.') |             field_name = match.group(1).split('.') | ||||||
|   | |||||||
| @@ -55,6 +55,7 @@ class QueryFieldList(object): | |||||||
|  |  | ||||||
|         if self.always_include: |         if self.always_include: | ||||||
|             if self.value is self.ONLY and self.fields: |             if self.value is self.ONLY and self.fields: | ||||||
|  |                 if sorted(self.slice.keys()) != sorted(self.fields): | ||||||
|                     self.fields = self.fields.union(self.always_include) |                     self.fields = self.fields.union(self.always_include) | ||||||
|             else: |             else: | ||||||
|                 self.fields -= self.always_include |                 self.fields -= self.always_include | ||||||
|   | |||||||
| @@ -94,8 +94,26 @@ class QuerySet(BaseQuerySet): | |||||||
|             except StopIteration: |             except StopIteration: | ||||||
|                 self._has_more = False |                 self._has_more = False | ||||||
|  |  | ||||||
|  |     def count(self, with_limit_and_skip=True): | ||||||
|  |         """Count the selected elements in the query. | ||||||
|  |  | ||||||
|  |         :param with_limit_and_skip (optional): take any :meth:`limit` or | ||||||
|  |             :meth:`skip` that has been applied to this cursor into account when | ||||||
|  |             getting the count | ||||||
|  |         """ | ||||||
|  |         if with_limit_and_skip is False: | ||||||
|  |             return super(QuerySet, self).count(with_limit_and_skip) | ||||||
|  |  | ||||||
|  |         if self._len is None: | ||||||
|  |             self._len = super(QuerySet, self).count(with_limit_and_skip) | ||||||
|  |  | ||||||
|  |         return self._len | ||||||
|  |  | ||||||
|     def no_cache(self): |     def no_cache(self): | ||||||
|         """Convert to a non_caching queryset""" |         """Convert to a non_caching queryset | ||||||
|  |  | ||||||
|  |         .. versionadded:: 0.8.3 Convert to non caching queryset | ||||||
|  |         """ | ||||||
|         if self._result_cache is not None: |         if self._result_cache is not None: | ||||||
|             raise OperationError("QuerySet already cached") |             raise OperationError("QuerySet already cached") | ||||||
|         return self.clone_into(QuerySetNoCache(self._document, self._collection)) |         return self.clone_into(QuerySetNoCache(self._document, self._collection)) | ||||||
| @@ -105,7 +123,10 @@ class QuerySetNoCache(BaseQuerySet): | |||||||
|     """A non caching QuerySet""" |     """A non caching QuerySet""" | ||||||
|  |  | ||||||
|     def cache(self): |     def cache(self): | ||||||
|         """Convert to a caching queryset""" |         """Convert to a caching queryset | ||||||
|  |  | ||||||
|  |         .. versionadded:: 0.8.3 Convert to caching queryset | ||||||
|  |         """ | ||||||
|         return self.clone_into(QuerySet(self._document, self._collection)) |         return self.clone_into(QuerySet(self._document, self._collection)) | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
| @@ -134,3 +155,10 @@ class QuerySetNoCache(BaseQuerySet): | |||||||
|             queryset = self.clone() |             queryset = self.clone() | ||||||
|         queryset.rewind() |         queryset.rewind() | ||||||
|         return queryset |         return queryset | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class QuerySetNoDeRef(QuerySet): | ||||||
|  |     """Special no_dereference QuerySet""" | ||||||
|  |  | ||||||
|  |     def __dereference(items, max_depth=1, instance=None, name=None): | ||||||
|  |         return items | ||||||
| @@ -3,6 +3,7 @@ from collections import defaultdict | |||||||
| import pymongo | import pymongo | ||||||
| from bson import SON | from bson import SON | ||||||
|  |  | ||||||
|  | from mongoengine.connection import get_connection | ||||||
| from mongoengine.common import _import_class | from mongoengine.common import _import_class | ||||||
| from mongoengine.errors import InvalidQueryError, LookUpError | from mongoengine.errors import InvalidQueryError, LookUpError | ||||||
|  |  | ||||||
| @@ -10,7 +11,7 @@ __all__ = ('query', 'update') | |||||||
|  |  | ||||||
|  |  | ||||||
| COMPARISON_OPERATORS = ('ne', 'gt', 'gte', 'lt', 'lte', 'in', 'nin', 'mod', | COMPARISON_OPERATORS = ('ne', 'gt', 'gte', 'lt', 'lte', 'in', 'nin', 'mod', | ||||||
|                         'all', 'size', 'exists', 'not') |                         'all', 'size', 'exists', 'not', 'elemMatch') | ||||||
| GEO_OPERATORS = ('within_distance', 'within_spherical_distance', | GEO_OPERATORS = ('within_distance', 'within_spherical_distance', | ||||||
|                  'within_box', 'within_polygon', 'near', 'near_sphere', |                  'within_box', 'within_polygon', 'near', 'near_sphere', | ||||||
|                  'max_distance', 'geo_within', 'geo_within_box', |                  'max_distance', 'geo_within', 'geo_within_box', | ||||||
| @@ -38,16 +39,16 @@ def query(_doc_cls=None, _field_operation=False, **query): | |||||||
|             mongo_query.update(value) |             mongo_query.update(value) | ||||||
|             continue |             continue | ||||||
|  |  | ||||||
|         parts = key.split('__') |         parts = key.rsplit('__') | ||||||
|         indices = [(i, p) for i, p in enumerate(parts) if p.isdigit()] |         indices = [(i, p) for i, p in enumerate(parts) if p.isdigit()] | ||||||
|         parts = [part for part in parts if not part.isdigit()] |         parts = [part for part in parts if not part.isdigit()] | ||||||
|         # Check for an operator and transform to mongo-style if there is |         # Check for an operator and transform to mongo-style if there is | ||||||
|         op = None |         op = None | ||||||
|         if parts[-1] in MATCH_OPERATORS: |         if len(parts) > 1 and parts[-1] in MATCH_OPERATORS: | ||||||
|             op = parts.pop() |             op = parts.pop() | ||||||
|  |  | ||||||
|         negate = False |         negate = False | ||||||
|         if parts[-1] == 'not': |         if len(parts) > 1 and parts[-1] == 'not': | ||||||
|             parts.pop() |             parts.pop() | ||||||
|             negate = True |             negate = True | ||||||
|  |  | ||||||
| @@ -59,14 +60,20 @@ def query(_doc_cls=None, _field_operation=False, **query): | |||||||
|                 raise InvalidQueryError(e) |                 raise InvalidQueryError(e) | ||||||
|             parts = [] |             parts = [] | ||||||
|  |  | ||||||
|  |             CachedReferenceField = _import_class('CachedReferenceField') | ||||||
|  |  | ||||||
|             cleaned_fields = [] |             cleaned_fields = [] | ||||||
|             for field in fields: |             for field in fields: | ||||||
|                 append_field = True |                 append_field = True | ||||||
|                 if isinstance(field, basestring): |                 if isinstance(field, basestring): | ||||||
|                     parts.append(field) |                     parts.append(field) | ||||||
|                     append_field = False |                     append_field = False | ||||||
|  |                 # is last and CachedReferenceField | ||||||
|  |                 elif isinstance(field, CachedReferenceField) and fields[-1] == field: | ||||||
|  |                     parts.append('%s._id' % field.db_field) | ||||||
|                 else: |                 else: | ||||||
|                     parts.append(field.db_field) |                     parts.append(field.db_field) | ||||||
|  |  | ||||||
|                 if append_field: |                 if append_field: | ||||||
|                     cleaned_fields.append(field) |                     cleaned_fields.append(field) | ||||||
|  |  | ||||||
| @@ -85,6 +92,10 @@ def query(_doc_cls=None, _field_operation=False, **query): | |||||||
|                         value = field |                         value = field | ||||||
|                 else: |                 else: | ||||||
|                     value = field.prepare_query_value(op, value) |                     value = field.prepare_query_value(op, value) | ||||||
|  |  | ||||||
|  |                     if isinstance(field, CachedReferenceField) and value: | ||||||
|  |                         value = value['_id'] | ||||||
|  |  | ||||||
|             elif op in ('in', 'nin', 'all', 'near') and not isinstance(value, dict): |             elif op in ('in', 'nin', 'all', 'near') and not isinstance(value, dict): | ||||||
|                 # 'in', 'nin' and 'all' require a list of values |                 # 'in', 'nin' and 'all' require a list of values | ||||||
|                 value = [field.prepare_query_value(op, v) for v in value] |                 value = [field.prepare_query_value(op, v) for v in value] | ||||||
| @@ -94,7 +105,7 @@ def query(_doc_cls=None, _field_operation=False, **query): | |||||||
|             if op in GEO_OPERATORS: |             if op in GEO_OPERATORS: | ||||||
|                 value = _geo_operator(field, op, value) |                 value = _geo_operator(field, op, value) | ||||||
|             elif op in CUSTOM_OPERATORS: |             elif op in CUSTOM_OPERATORS: | ||||||
|                 if op == 'match': |                 if op in ('elem_match', 'match'): | ||||||
|                     value = field.prepare_query_value(op, value) |                     value = field.prepare_query_value(op, value) | ||||||
|                     value = {"$elemMatch": value} |                     value = {"$elemMatch": value} | ||||||
|                 else: |                 else: | ||||||
| @@ -115,14 +126,28 @@ def query(_doc_cls=None, _field_operation=False, **query): | |||||||
|             if key in mongo_query and isinstance(mongo_query[key], dict): |             if key in mongo_query and isinstance(mongo_query[key], dict): | ||||||
|                 mongo_query[key].update(value) |                 mongo_query[key].update(value) | ||||||
|                 # $maxDistance needs to come last - convert to SON |                 # $maxDistance needs to come last - convert to SON | ||||||
|                 if '$maxDistance' in mongo_query[key]: |  | ||||||
|                 value_dict = mongo_query[key] |                 value_dict = mongo_query[key] | ||||||
|  |                 if ('$maxDistance' in value_dict and '$near' in value_dict): | ||||||
|                     value_son = SON() |                     value_son = SON() | ||||||
|  |                     if isinstance(value_dict['$near'], dict): | ||||||
|  |                         for k, v in value_dict.iteritems(): | ||||||
|  |                             if k == '$maxDistance': | ||||||
|  |                                 continue | ||||||
|  |                             value_son[k] = v | ||||||
|  |                         if (get_connection().max_wire_version <= 1): | ||||||
|  |                             value_son['$maxDistance'] = value_dict[ | ||||||
|  |                                 '$maxDistance'] | ||||||
|  |                         else: | ||||||
|  |                             value_son['$near'] = SON(value_son['$near']) | ||||||
|  |                             value_son['$near'][ | ||||||
|  |                                 '$maxDistance'] = value_dict['$maxDistance'] | ||||||
|  |                     else: | ||||||
|                         for k, v in value_dict.iteritems(): |                         for k, v in value_dict.iteritems(): | ||||||
|                             if k == '$maxDistance': |                             if k == '$maxDistance': | ||||||
|                                 continue |                                 continue | ||||||
|                             value_son[k] = v |                             value_son[k] = v | ||||||
|                         value_son['$maxDistance'] = value_dict['$maxDistance'] |                         value_son['$maxDistance'] = value_dict['$maxDistance'] | ||||||
|  |  | ||||||
|                     mongo_query[key] = value_son |                     mongo_query[key] = value_son | ||||||
|             else: |             else: | ||||||
|                 # Store for manually merging later |                 # Store for manually merging later | ||||||
| @@ -151,6 +176,9 @@ def update(_doc_cls=None, **update): | |||||||
|             mongo_update.update(value) |             mongo_update.update(value) | ||||||
|             continue |             continue | ||||||
|         parts = key.split('__') |         parts = key.split('__') | ||||||
|  |         # if there is no operator, default to "set" | ||||||
|  |         if len(parts) < 3 and parts[0] not in UPDATE_OPERATORS: | ||||||
|  |             parts.insert(0, 'set') | ||||||
|         # Check for an operator and transform to mongo-style if there is |         # Check for an operator and transform to mongo-style if there is | ||||||
|         op = None |         op = None | ||||||
|         if parts[0] in UPDATE_OPERATORS: |         if parts[0] in UPDATE_OPERATORS: | ||||||
| @@ -182,6 +210,7 @@ def update(_doc_cls=None, **update): | |||||||
|             parts = [] |             parts = [] | ||||||
|  |  | ||||||
|             cleaned_fields = [] |             cleaned_fields = [] | ||||||
|  |             appended_sub_field = False | ||||||
|             for field in fields: |             for field in fields: | ||||||
|                 append_field = True |                 append_field = True | ||||||
|                 if isinstance(field, basestring): |                 if isinstance(field, basestring): | ||||||
| @@ -193,21 +222,34 @@ def update(_doc_cls=None, **update): | |||||||
|                 else: |                 else: | ||||||
|                     parts.append(field.db_field) |                     parts.append(field.db_field) | ||||||
|                 if append_field: |                 if append_field: | ||||||
|  |                     appended_sub_field = False | ||||||
|                     cleaned_fields.append(field) |                     cleaned_fields.append(field) | ||||||
|  |                     if hasattr(field, 'field'): | ||||||
|  |                         cleaned_fields.append(field.field) | ||||||
|  |                         appended_sub_field = True | ||||||
|  |  | ||||||
|             # Convert value to proper value |             # Convert value to proper value | ||||||
|  |             if appended_sub_field: | ||||||
|  |                 field = cleaned_fields[-2] | ||||||
|  |             else: | ||||||
|                 field = cleaned_fields[-1] |                 field = cleaned_fields[-1] | ||||||
|  |  | ||||||
|  |             GeoJsonBaseField = _import_class("GeoJsonBaseField") | ||||||
|  |             if isinstance(field, GeoJsonBaseField): | ||||||
|  |                 value = field.to_mongo(value) | ||||||
|  |  | ||||||
|             if op in (None, 'set', 'push', 'pull'): |             if op in (None, 'set', 'push', 'pull'): | ||||||
|                 if field.required or value is not None: |                 if field.required or value is not None: | ||||||
|                     value = field.prepare_query_value(op, value) |                     value = field.prepare_query_value(op, value) | ||||||
|             elif op in ('pushAll', 'pullAll'): |             elif op in ('pushAll', 'pullAll'): | ||||||
|                 value = [field.prepare_query_value(op, v) for v in value] |                 value = [field.prepare_query_value(op, v) for v in value] | ||||||
|             elif op == 'addToSet': |             elif op in ('addToSet', 'setOnInsert'): | ||||||
|                 if isinstance(value, (list, tuple, set)): |                 if isinstance(value, (list, tuple, set)): | ||||||
|                     value = [field.prepare_query_value(op, v) for v in value] |                     value = [field.prepare_query_value(op, v) for v in value] | ||||||
|                 elif field.required or value is not None: |                 elif field.required or value is not None: | ||||||
|                     value = field.prepare_query_value(op, value) |                     value = field.prepare_query_value(op, value) | ||||||
|  |             elif op == "unset": | ||||||
|  |                 value = 1 | ||||||
|  |  | ||||||
|         if match: |         if match: | ||||||
|             match = '$' + match |             match = '$' + match | ||||||
| @@ -221,11 +263,25 @@ def update(_doc_cls=None, **update): | |||||||
|  |  | ||||||
|         if 'pull' in op and '.' in key: |         if 'pull' in op and '.' in key: | ||||||
|             # Dot operators don't work on pull operations |             # Dot operators don't work on pull operations | ||||||
|             # it uses nested dict syntax |             # unless they point to a list field | ||||||
|  |             # Otherwise it uses nested dict syntax | ||||||
|             if op == 'pullAll': |             if op == 'pullAll': | ||||||
|                 raise InvalidQueryError("pullAll operations only support " |                 raise InvalidQueryError("pullAll operations only support " | ||||||
|                                         "a single field depth") |                                         "a single field depth") | ||||||
|  |  | ||||||
|  |             # Look for the last list field and use dot notation until there | ||||||
|  |             field_classes = [c.__class__ for c in cleaned_fields] | ||||||
|  |             field_classes.reverse() | ||||||
|  |             ListField = _import_class('ListField') | ||||||
|  |             if ListField in field_classes: | ||||||
|  |                 # Join all fields via dot notation to the last ListField | ||||||
|  |                 # Then process as normal | ||||||
|  |                 last_listField = len( | ||||||
|  |                     cleaned_fields) - field_classes.index(ListField) | ||||||
|  |                 key = ".".join(parts[:last_listField]) | ||||||
|  |                 parts = parts[last_listField:] | ||||||
|  |                 parts.insert(0, key) | ||||||
|  |  | ||||||
|             parts.reverse() |             parts.reverse() | ||||||
|             for key in parts: |             for key in parts: | ||||||
|                 value = {key: value} |                 value = {key: value} | ||||||
|   | |||||||
| @@ -1,8 +1,9 @@ | |||||||
| import copy | import copy | ||||||
|  |  | ||||||
| from mongoengine.errors import InvalidQueryError | from itertools import product | ||||||
| from mongoengine.python_support import product, reduce | from functools import reduce | ||||||
|  |  | ||||||
|  | from mongoengine.errors import InvalidQueryError | ||||||
| from mongoengine.queryset import transform | from mongoengine.queryset import transform | ||||||
|  |  | ||||||
| __all__ = ('Q',) | __all__ = ('Q',) | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
| %define srcname mongoengine | %define srcname mongoengine | ||||||
|  |  | ||||||
| Name:           python-%{srcname} | Name:           python-%{srcname} | ||||||
| Version:        0.8.3 | Version:        0.8.7 | ||||||
| Release:        1%{?dist} | Release:        1%{?dist} | ||||||
| Summary:        A Python Document-Object Mapper for working with MongoDB | Summary:        A Python Document-Object Mapper for working with MongoDB | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1 +1 @@ | |||||||
| pymongo | pymongo>=2.7.1 | ||||||
|   | |||||||
							
								
								
									
										21
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								setup.py
									
									
									
									
									
								
							| @@ -38,27 +38,30 @@ CLASSIFIERS = [ | |||||||
|     'Operating System :: OS Independent', |     'Operating System :: OS Independent', | ||||||
|     'Programming Language :: Python', |     'Programming Language :: Python', | ||||||
|     "Programming Language :: Python :: 2", |     "Programming Language :: Python :: 2", | ||||||
|     "Programming Language :: Python :: 2.6", |     "Programming Language :: Python :: 2.6.6", | ||||||
|     "Programming Language :: Python :: 2.7", |     "Programming Language :: Python :: 2.7", | ||||||
|     "Programming Language :: Python :: 3", |     "Programming Language :: Python :: 3", | ||||||
|     "Programming Language :: Python :: 3.1", |  | ||||||
|     "Programming Language :: Python :: 3.2", |     "Programming Language :: Python :: 3.2", | ||||||
|  |     "Programming Language :: Python :: 3.3", | ||||||
|  |     "Programming Language :: Python :: 3.4", | ||||||
|     "Programming Language :: Python :: Implementation :: CPython", |     "Programming Language :: Python :: Implementation :: CPython", | ||||||
|  |     "Programming Language :: Python :: Implementation :: PyPy", | ||||||
|     'Topic :: Database', |     'Topic :: Database', | ||||||
|     'Topic :: Software Development :: Libraries :: Python Modules', |     'Topic :: Software Development :: Libraries :: Python Modules', | ||||||
| ] | ] | ||||||
|  |  | ||||||
| extra_opts = {} | extra_opts = {"packages": find_packages(exclude=["tests", "tests.*"])} | ||||||
| if sys.version_info[0] == 3: | if sys.version_info[0] == 3: | ||||||
|     extra_opts['use_2to3'] = True |     extra_opts['use_2to3'] = True | ||||||
|     extra_opts['tests_require'] = ['nose', 'coverage', 'blinker', 'jinja2==2.6', 'django>=1.5.1'] |     extra_opts['tests_require'] = ['nose', 'coverage', 'blinker', 'jinja2==2.6', 'Pillow>=2.0.0', 'django>=1.5.1'] | ||||||
|     extra_opts['packages'] = find_packages(exclude=('tests',)) |  | ||||||
|     if "test" in sys.argv or "nosetests" in sys.argv: |     if "test" in sys.argv or "nosetests" in sys.argv: | ||||||
|         extra_opts['packages'].append("tests") |         extra_opts['packages'] = find_packages() | ||||||
|         extra_opts['package_data'] = {"tests": ["fields/mongoengine.png", "fields/mongodb_leaf.png"]} |         extra_opts['package_data'] = {"tests": ["fields/mongoengine.png", "fields/mongodb_leaf.png"]} | ||||||
| else: | else: | ||||||
|     extra_opts['tests_require'] = ['nose', 'coverage', 'blinker', 'django>=1.4.2', 'PIL', 'jinja2==2.6', 'python-dateutil'] |     extra_opts['tests_require'] = ['nose', 'coverage', 'blinker', 'django>=1.4.2', 'Pillow>=2.0.0', 'jinja2>=2.6', 'python-dateutil'] | ||||||
|     extra_opts['packages'] = find_packages(exclude=('tests',)) |  | ||||||
|  |     if sys.version_info[0] == 2 and sys.version_info[1] == 6: | ||||||
|  |         extra_opts['tests_require'].append('unittest2') | ||||||
|  |  | ||||||
| setup(name='mongoengine', | setup(name='mongoengine', | ||||||
|       version=VERSION, |       version=VERSION, | ||||||
| @@ -74,7 +77,7 @@ setup(name='mongoengine', | |||||||
|       long_description=LONG_DESCRIPTION, |       long_description=LONG_DESCRIPTION, | ||||||
|       platforms=['any'], |       platforms=['any'], | ||||||
|       classifiers=CLASSIFIERS, |       classifiers=CLASSIFIERS, | ||||||
|       install_requires=['pymongo>=2.5'], |       install_requires=['pymongo>=2.7.1'], | ||||||
|       test_suite='nose.collector', |       test_suite='nose.collector', | ||||||
|       **extra_opts |       **extra_opts | ||||||
| ) | ) | ||||||
|   | |||||||
							
								
								
									
										0
									
								
								tests/async/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								tests/async/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										36
									
								
								tests/async/test_connection.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								tests/async/test_connection.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | |||||||
|  | from mongoengine import * | ||||||
|  | import motor | ||||||
|  | import mongoengine.connection | ||||||
|  | from mongoengine.connection import get_db, get_connection, ConnectionError | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     import unittest2 as unittest | ||||||
|  | except ImportError: | ||||||
|  |     import unittest | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ConnectionTest(unittest.TestCase): | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         mongoengine.connection._connection_settings = {} | ||||||
|  |         mongoengine.connection._connections = {} | ||||||
|  |         mongoengine.connection._dbs = {} | ||||||
|  |  | ||||||
|  |     def test_register_connection(self): | ||||||
|  |         """ | ||||||
|  |         Ensure that the connect() method works properly. | ||||||
|  |         """ | ||||||
|  |         register_connection('asyncdb', 'mongoengineasynctest', async=True) | ||||||
|  |  | ||||||
|  |         self.assertEqual( | ||||||
|  |             mongoengine.connection._connection_settings['asyncdb']['name'], | ||||||
|  |             'mongoengineasynctest') | ||||||
|  |  | ||||||
|  |         self.assertTrue( | ||||||
|  |             mongoengine.connection._connection_settings['asyncdb']['async']) | ||||||
|  |         conn = get_connection('asyncdb') | ||||||
|  |         self.assertTrue(isinstance(conn, motor.MotorClient)) | ||||||
|  |  | ||||||
|  |         db = get_db('asyncdb') | ||||||
|  |         self.assertTrue(isinstance(db, motor.MotorDatabase)) | ||||||
|  |         self.assertEqual(db.name, 'mongoengineasynctest') | ||||||
| @@ -36,9 +36,9 @@ class ClassMethodsTest(unittest.TestCase): | |||||||
|     def test_definition(self): |     def test_definition(self): | ||||||
|         """Ensure that document may be defined using fields. |         """Ensure that document may be defined using fields. | ||||||
|         """ |         """ | ||||||
|         self.assertEqual(['age', 'id', 'name'], |         self.assertEqual(['_cls', 'age', 'id', 'name'], | ||||||
|                          sorted(self.Person._fields.keys())) |                          sorted(self.Person._fields.keys())) | ||||||
|         self.assertEqual(["IntField", "ObjectIdField", "StringField"], |         self.assertEqual(["IntField", "ObjectIdField", "StringField", "StringField"], | ||||||
|                         sorted([x.__class__.__name__ for x in |                         sorted([x.__class__.__name__ for x in | ||||||
|                                 self.Person._fields.values()])) |                                 self.Person._fields.values()])) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -207,22 +207,21 @@ class DeltaTest(unittest.TestCase): | |||||||
|         doc.embedded_field.list_field[2].string_field = 'hello world' |         doc.embedded_field.list_field[2].string_field = 'hello world' | ||||||
|         doc.embedded_field.list_field[2] = doc.embedded_field.list_field[2] |         doc.embedded_field.list_field[2] = doc.embedded_field.list_field[2] | ||||||
|         self.assertEqual(doc._get_changed_fields(), |         self.assertEqual(doc._get_changed_fields(), | ||||||
|                          ['embedded_field.list_field']) |                          ['embedded_field.list_field.2']) | ||||||
|         self.assertEqual(doc.embedded_field._delta(), ({ |         self.assertEqual(doc.embedded_field._delta(), ({'list_field.2': { | ||||||
|             'list_field': ['1', 2, { |  | ||||||
|             '_cls': 'Embedded', |  | ||||||
|             'string_field': 'hello world', |  | ||||||
|             'int_field': 1, |  | ||||||
|             'list_field': ['1', 2, {'hello': 'world'}], |  | ||||||
|             'dict_field': {'hello': 'world'}}]}, {})) |  | ||||||
|         self.assertEqual(doc._delta(), ({ |  | ||||||
|             'embedded_field.list_field': ['1', 2, { |  | ||||||
|                 '_cls': 'Embedded', |                 '_cls': 'Embedded', | ||||||
|                 'string_field': 'hello world', |                 'string_field': 'hello world', | ||||||
|                 'int_field': 1, |                 'int_field': 1, | ||||||
|                 'list_field': ['1', 2, {'hello': 'world'}], |                 'list_field': ['1', 2, {'hello': 'world'}], | ||||||
|                 'dict_field': {'hello': 'world'}} |                 'dict_field': {'hello': 'world'}} | ||||||
|             ]}, {})) |             }, {})) | ||||||
|  |         self.assertEqual(doc._delta(), ({'embedded_field.list_field.2': { | ||||||
|  |                 '_cls': 'Embedded', | ||||||
|  |                 'string_field': 'hello world', | ||||||
|  |                 'int_field': 1, | ||||||
|  |                 'list_field': ['1', 2, {'hello': 'world'}], | ||||||
|  |                 'dict_field': {'hello': 'world'}} | ||||||
|  |             }, {})) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc = doc.reload(10) |         doc = doc.reload(10) | ||||||
|         self.assertEqual(doc.embedded_field.list_field[2].string_field, |         self.assertEqual(doc.embedded_field.list_field[2].string_field, | ||||||
| @@ -253,7 +252,7 @@ class DeltaTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         del(doc.embedded_field.list_field[2].list_field[2]['hello']) |         del(doc.embedded_field.list_field[2].list_field[2]['hello']) | ||||||
|         self.assertEqual(doc._delta(), |         self.assertEqual(doc._delta(), | ||||||
|                          ({'embedded_field.list_field.2.list_field': [1, 2, {}]}, {})) |                          ({}, {'embedded_field.list_field.2.list_field.2.hello': 1})) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc = doc.reload(10) |         doc = doc.reload(10) | ||||||
|  |  | ||||||
| @@ -313,29 +312,24 @@ class DeltaTest(unittest.TestCase): | |||||||
|         self.circular_reference_deltas_2(DynamicDocument, Document) |         self.circular_reference_deltas_2(DynamicDocument, Document) | ||||||
|         self.circular_reference_deltas_2(DynamicDocument, DynamicDocument) |         self.circular_reference_deltas_2(DynamicDocument, DynamicDocument) | ||||||
|  |  | ||||||
|     def circular_reference_deltas_2(self, DocClass1, DocClass2): |     def circular_reference_deltas_2(self, DocClass1, DocClass2, dbref=True): | ||||||
|  |  | ||||||
|         class Person(DocClass1): |         class Person(DocClass1): | ||||||
|             name = StringField() |             name = StringField() | ||||||
|             owns = ListField(ReferenceField('Organization')) |             owns = ListField(ReferenceField('Organization', dbref=dbref)) | ||||||
|             employer = ReferenceField('Organization') |             employer = ReferenceField('Organization', dbref=dbref) | ||||||
|  |  | ||||||
|         class Organization(DocClass2): |         class Organization(DocClass2): | ||||||
|             name = StringField() |             name = StringField() | ||||||
|             owner = ReferenceField('Person') |             owner = ReferenceField('Person', dbref=dbref) | ||||||
|             employees = ListField(ReferenceField('Person')) |             employees = ListField(ReferenceField('Person', dbref=dbref)) | ||||||
|  |  | ||||||
|         Person.drop_collection() |         Person.drop_collection() | ||||||
|         Organization.drop_collection() |         Organization.drop_collection() | ||||||
|  |  | ||||||
|         person = Person(name="owner") |         person = Person(name="owner").save() | ||||||
|         person.save() |         employee = Person(name="employee").save() | ||||||
|  |         organization = Organization(name="company").save() | ||||||
|         employee = Person(name="employee") |  | ||||||
|         employee.save() |  | ||||||
|  |  | ||||||
|         organization = Organization(name="company") |  | ||||||
|         organization.save() |  | ||||||
|  |  | ||||||
|         person.owns.append(organization) |         person.owns.append(organization) | ||||||
|         organization.owner = person |         organization.owner = person | ||||||
| @@ -355,6 +349,8 @@ class DeltaTest(unittest.TestCase): | |||||||
|         self.assertEqual(o.owner, p) |         self.assertEqual(o.owner, p) | ||||||
|         self.assertEqual(e.employer, o) |         self.assertEqual(e.employer, o) | ||||||
|  |  | ||||||
|  |         return person, organization, employee | ||||||
|  |  | ||||||
|     def test_delta_db_field(self): |     def test_delta_db_field(self): | ||||||
|         self.delta_db_field(Document) |         self.delta_db_field(Document) | ||||||
|         self.delta_db_field(DynamicDocument) |         self.delta_db_field(DynamicDocument) | ||||||
| @@ -551,22 +547,21 @@ class DeltaTest(unittest.TestCase): | |||||||
|         doc.embedded_field.list_field[2].string_field = 'hello world' |         doc.embedded_field.list_field[2].string_field = 'hello world' | ||||||
|         doc.embedded_field.list_field[2] = doc.embedded_field.list_field[2] |         doc.embedded_field.list_field[2] = doc.embedded_field.list_field[2] | ||||||
|         self.assertEqual(doc._get_changed_fields(), |         self.assertEqual(doc._get_changed_fields(), | ||||||
|             ['db_embedded_field.db_list_field']) |             ['db_embedded_field.db_list_field.2']) | ||||||
|         self.assertEqual(doc.embedded_field._delta(), ({ |         self.assertEqual(doc.embedded_field._delta(), ({'db_list_field.2': { | ||||||
|             'db_list_field': ['1', 2, { |  | ||||||
|             '_cls': 'Embedded', |             '_cls': 'Embedded', | ||||||
|             'db_string_field': 'hello world', |             'db_string_field': 'hello world', | ||||||
|             'db_int_field': 1, |             'db_int_field': 1, | ||||||
|             'db_list_field': ['1', 2, {'hello': 'world'}], |             'db_list_field': ['1', 2, {'hello': 'world'}], | ||||||
|             'db_dict_field': {'hello': 'world'}}]}, {})) |             'db_dict_field': {'hello': 'world'}}}, {})) | ||||||
|         self.assertEqual(doc._delta(), ({ |         self.assertEqual(doc._delta(), ({ | ||||||
|             'db_embedded_field.db_list_field': ['1', 2, { |             'db_embedded_field.db_list_field.2': { | ||||||
|                 '_cls': 'Embedded', |                 '_cls': 'Embedded', | ||||||
|                 'db_string_field': 'hello world', |                 'db_string_field': 'hello world', | ||||||
|                 'db_int_field': 1, |                 'db_int_field': 1, | ||||||
|                 'db_list_field': ['1', 2, {'hello': 'world'}], |                 'db_list_field': ['1', 2, {'hello': 'world'}], | ||||||
|                 'db_dict_field': {'hello': 'world'}} |                 'db_dict_field': {'hello': 'world'}} | ||||||
|             ]}, {})) |             }, {})) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc = doc.reload(10) |         doc = doc.reload(10) | ||||||
|         self.assertEqual(doc.embedded_field.list_field[2].string_field, |         self.assertEqual(doc.embedded_field.list_field[2].string_field, | ||||||
| @@ -597,8 +592,7 @@ class DeltaTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         del(doc.embedded_field.list_field[2].list_field[2]['hello']) |         del(doc.embedded_field.list_field[2].list_field[2]['hello']) | ||||||
|         self.assertEqual(doc._delta(), |         self.assertEqual(doc._delta(), | ||||||
|             ({'db_embedded_field.db_list_field.2.db_list_field': |             ({}, {'db_embedded_field.db_list_field.2.db_list_field.2.hello': 1})) | ||||||
|                 [1, 2, {}]}, {})) |  | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc = doc.reload(10) |         doc = doc.reload(10) | ||||||
|  |  | ||||||
| @@ -686,6 +680,99 @@ class DeltaTest(unittest.TestCase): | |||||||
|         self.assertEqual(doc._get_changed_fields(), ['list_field']) |         self.assertEqual(doc._get_changed_fields(), ['list_field']) | ||||||
|         self.assertEqual(doc._delta(), ({}, {'list_field': 1})) |         self.assertEqual(doc._delta(), ({}, {'list_field': 1})) | ||||||
|  |  | ||||||
|  |     def test_delta_with_dbref_true(self): | ||||||
|  |         person, organization, employee = self.circular_reference_deltas_2(Document, Document, True) | ||||||
|  |         employee.name = 'test' | ||||||
|  |  | ||||||
|  |         self.assertEqual(organization._get_changed_fields(), []) | ||||||
|  |  | ||||||
|  |         updates, removals = organization._delta() | ||||||
|  |         self.assertEqual({}, removals) | ||||||
|  |         self.assertEqual({}, updates) | ||||||
|  |  | ||||||
|  |         organization.employees.append(person) | ||||||
|  |         updates, removals = organization._delta() | ||||||
|  |         self.assertEqual({}, removals) | ||||||
|  |         self.assertTrue('employees' in updates) | ||||||
|  |  | ||||||
|  |     def test_delta_with_dbref_false(self): | ||||||
|  |         person, organization, employee = self.circular_reference_deltas_2(Document, Document, False) | ||||||
|  |         employee.name = 'test' | ||||||
|  |  | ||||||
|  |         self.assertEqual(organization._get_changed_fields(), []) | ||||||
|  |  | ||||||
|  |         updates, removals = organization._delta() | ||||||
|  |         self.assertEqual({}, removals) | ||||||
|  |         self.assertEqual({}, updates) | ||||||
|  |  | ||||||
|  |         organization.employees.append(person) | ||||||
|  |         updates, removals = organization._delta() | ||||||
|  |         self.assertEqual({}, removals) | ||||||
|  |         self.assertTrue('employees' in updates) | ||||||
|  |  | ||||||
|  |     def test_nested_nested_fields_mark_as_changed(self): | ||||||
|  |         class EmbeddedDoc(EmbeddedDocument): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class MyDoc(Document): | ||||||
|  |             subs = MapField(MapField(EmbeddedDocumentField(EmbeddedDoc))) | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         MyDoc.drop_collection() | ||||||
|  |  | ||||||
|  |         mydoc = MyDoc(name='testcase1', subs={'a': {'b': EmbeddedDoc(name='foo')}}).save() | ||||||
|  |  | ||||||
|  |         mydoc = MyDoc.objects.first() | ||||||
|  |         subdoc = mydoc.subs['a']['b'] | ||||||
|  |         subdoc.name = 'bar' | ||||||
|  |  | ||||||
|  |         self.assertEqual(["name"], subdoc._get_changed_fields()) | ||||||
|  |         self.assertEqual(["subs.a.b.name"], mydoc._get_changed_fields()) | ||||||
|  |  | ||||||
|  |         mydoc._clear_changed_fields() | ||||||
|  |         self.assertEqual([], mydoc._get_changed_fields()) | ||||||
|  |  | ||||||
|  |     def test_referenced_object_changed_attributes(self): | ||||||
|  |         """Ensures that when you save a new reference to a field, the referenced object isn't altered""" | ||||||
|  |  | ||||||
|  |         class Organization(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class User(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             org = ReferenceField('Organization', required=True) | ||||||
|  |  | ||||||
|  |         Organization.drop_collection() | ||||||
|  |         User.drop_collection() | ||||||
|  |  | ||||||
|  |         org1 = Organization(name='Org 1') | ||||||
|  |         org1.save() | ||||||
|  |  | ||||||
|  |         org2 = Organization(name='Org 2') | ||||||
|  |         org2.save() | ||||||
|  |  | ||||||
|  |         user = User(name='Fred', org=org1) | ||||||
|  |         user.save() | ||||||
|  |  | ||||||
|  |         org1.reload() | ||||||
|  |         org2.reload() | ||||||
|  |         user.reload() | ||||||
|  |         self.assertEqual(org1.name, 'Org 1') | ||||||
|  |         self.assertEqual(org2.name, 'Org 2') | ||||||
|  |         self.assertEqual(user.name, 'Fred') | ||||||
|  |  | ||||||
|  |         user.name = 'Harold' | ||||||
|  |         user.org = org2 | ||||||
|  |  | ||||||
|  |         org2.name = 'New Org 2' | ||||||
|  |         self.assertEqual(org2.name, 'New Org 2') | ||||||
|  |  | ||||||
|  |         user.save() | ||||||
|  |         org2.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(org2.name, 'New Org 2') | ||||||
|  |         org2.reload() | ||||||
|  |         self.assertEqual(org2.name, 'New Org 2') | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -292,6 +292,59 @@ class DynamicTest(unittest.TestCase): | |||||||
|         person.save() |         person.save() | ||||||
|         self.assertEqual(Person.objects.first().age, 35) |         self.assertEqual(Person.objects.first().age, 35) | ||||||
|  |  | ||||||
|  |     def test_dynamic_embedded_works_with_only(self): | ||||||
|  |         """Ensure custom fieldnames on a dynamic embedded document are found by qs.only()""" | ||||||
|  |  | ||||||
|  |         class Address(DynamicEmbeddedDocument): | ||||||
|  |             city = StringField() | ||||||
|  |  | ||||||
|  |         class Person(DynamicDocument): | ||||||
|  |             address = EmbeddedDocumentField(Address) | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         Person(name="Eric", address=Address(city="San Francisco", street_number="1337")).save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(Person.objects.first().address.street_number, '1337') | ||||||
|  |         self.assertEqual(Person.objects.only('address__street_number').first().address.street_number, '1337') | ||||||
|  |  | ||||||
|  |     def test_dynamic_and_embedded_dict_access(self): | ||||||
|  |         """Ensure embedded dynamic documents work with dict[] style access""" | ||||||
|  |  | ||||||
|  |         class Address(EmbeddedDocument): | ||||||
|  |             city = StringField() | ||||||
|  |  | ||||||
|  |         class Person(DynamicDocument): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         Person(name="Ross", address=Address(city="London")).save() | ||||||
|  |  | ||||||
|  |         person = Person.objects.first() | ||||||
|  |         person.attrval = "This works" | ||||||
|  |  | ||||||
|  |         person["phone"] = "555-1212" # but this should too | ||||||
|  |  | ||||||
|  |         # Same thing two levels deep | ||||||
|  |         person["address"]["city"] = "Lundenne" | ||||||
|  |         person.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(Person.objects.first().address.city, "Lundenne") | ||||||
|  |  | ||||||
|  |         self.assertEqual(Person.objects.first().phone, "555-1212") | ||||||
|  |  | ||||||
|  |         person = Person.objects.first() | ||||||
|  |         person.address = Address(city="Londinium") | ||||||
|  |         person.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(Person.objects.first().address.city, "Londinium") | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         person = Person.objects.first() | ||||||
|  |         person["age"] = 35 | ||||||
|  |         person.save() | ||||||
|  |         self.assertEqual(Person.objects.first().age, 35) | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -156,6 +156,35 @@ class IndexesTest(unittest.TestCase): | |||||||
|         self.assertEqual([{'fields': [('_cls', 1), ('title', 1)]}], |         self.assertEqual([{'fields': [('_cls', 1), ('title', 1)]}], | ||||||
|                          A._meta['index_specs']) |                          A._meta['index_specs']) | ||||||
|  |  | ||||||
|  |     def test_index_no_cls(self): | ||||||
|  |         """Ensure index specs are inhertited correctly""" | ||||||
|  |  | ||||||
|  |         class A(Document): | ||||||
|  |             title = StringField() | ||||||
|  |             meta = { | ||||||
|  |                 'indexes': [ | ||||||
|  |                         {'fields': ('title',), 'cls': False}, | ||||||
|  |                 ], | ||||||
|  |                 'allow_inheritance': True, | ||||||
|  |                 'index_cls': False | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |         self.assertEqual([('title', 1)], A._meta['index_specs'][0]['fields']) | ||||||
|  |         A._get_collection().drop_indexes() | ||||||
|  |         A.ensure_indexes() | ||||||
|  |         info = A._get_collection().index_information() | ||||||
|  |         self.assertEqual(len(info.keys()), 2) | ||||||
|  |  | ||||||
|  |         class B(A): | ||||||
|  |             c = StringField() | ||||||
|  |             d = StringField() | ||||||
|  |             meta = { | ||||||
|  |                 'indexes': [{'fields': ['c']}, {'fields': ['d'], 'cls': True}], | ||||||
|  |                 'allow_inheritance': True | ||||||
|  |             } | ||||||
|  |         self.assertEqual([('c', 1)], B._meta['index_specs'][1]['fields']) | ||||||
|  |         self.assertEqual([('_cls', 1), ('d', 1)], B._meta['index_specs'][2]['fields']) | ||||||
|  |  | ||||||
|     def test_build_index_spec_is_not_destructive(self): |     def test_build_index_spec_is_not_destructive(self): | ||||||
|  |  | ||||||
|         class MyDoc(Document): |         class MyDoc(Document): | ||||||
| @@ -472,7 +501,7 @@ class IndexesTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         def invalid_index_2(): |         def invalid_index_2(): | ||||||
|             return BlogPost.objects.hint(('tags', 1)) |             return BlogPost.objects.hint(('tags', 1)) | ||||||
|         self.assertRaises(TypeError, invalid_index_2) |         self.assertRaises(Exception, invalid_index_2) | ||||||
|  |  | ||||||
|     def test_unique(self): |     def test_unique(self): | ||||||
|         """Ensure that uniqueness constraints are applied to fields. |         """Ensure that uniqueness constraints are applied to fields. | ||||||
| @@ -708,5 +737,32 @@ class IndexesTest(unittest.TestCase): | |||||||
|                          report.to_mongo()) |                          report.to_mongo()) | ||||||
|         self.assertEqual(report, Report.objects.get(pk=my_key)) |         self.assertEqual(report, Report.objects.get(pk=my_key)) | ||||||
|  |  | ||||||
|  |     def test_string_indexes(self): | ||||||
|  |  | ||||||
|  |         class MyDoc(Document): | ||||||
|  |             provider_ids = DictField() | ||||||
|  |             meta = { | ||||||
|  |                 "indexes": ["provider_ids.foo", "provider_ids.bar"], | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         info = MyDoc.objects._collection.index_information() | ||||||
|  |         info = [value['key'] for key, value in info.iteritems()] | ||||||
|  |         self.assertTrue([('provider_ids.foo', 1)] in info) | ||||||
|  |         self.assertTrue([('provider_ids.bar', 1)] in info) | ||||||
|  |  | ||||||
|  |     def test_text_indexes(self): | ||||||
|  |  | ||||||
|  |         class Book(Document): | ||||||
|  |             title = DictField() | ||||||
|  |             meta = { | ||||||
|  |                 "indexes": ["$title"], | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         indexes = Book.objects._collection.index_information() | ||||||
|  |         self.assertTrue("title_text" in indexes) | ||||||
|  |         key = indexes["title_text"]["key"] | ||||||
|  |         self.assertTrue(('_fts', 'text') in key) | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -163,7 +163,7 @@ class InheritanceTest(unittest.TestCase): | |||||||
|         class Employee(Person): |         class Employee(Person): | ||||||
|             salary = IntField() |             salary = IntField() | ||||||
|  |  | ||||||
|         self.assertEqual(['age', 'id', 'name', 'salary'], |         self.assertEqual(['_cls', 'age', 'id', 'name', 'salary'], | ||||||
|                          sorted(Employee._fields.keys())) |                          sorted(Employee._fields.keys())) | ||||||
|         self.assertEqual(Employee._get_collection_name(), |         self.assertEqual(Employee._get_collection_name(), | ||||||
|                          Person._get_collection_name()) |                          Person._get_collection_name()) | ||||||
| @@ -180,7 +180,7 @@ class InheritanceTest(unittest.TestCase): | |||||||
|         class Employee(Person): |         class Employee(Person): | ||||||
|             salary = IntField() |             salary = IntField() | ||||||
|  |  | ||||||
|         self.assertEqual(['age', 'id', 'name', 'salary'], |         self.assertEqual(['_cls', 'age', 'id', 'name', 'salary'], | ||||||
|                          sorted(Employee._fields.keys())) |                          sorted(Employee._fields.keys())) | ||||||
|         self.assertEqual(Person(name="Bob", age=35).to_mongo().keys(), |         self.assertEqual(Person(name="Bob", age=35).to_mongo().keys(), | ||||||
|                          ['_cls', 'name', 'age']) |                          ['_cls', 'name', 'age']) | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ from tests.fixtures import (PickleEmbedded, PickleTest, PickleSignalsTest, | |||||||
|  |  | ||||||
| from mongoengine import * | from mongoengine import * | ||||||
| from mongoengine.errors import (NotRegistered, InvalidDocumentError, | from mongoengine.errors import (NotRegistered, InvalidDocumentError, | ||||||
|                                 InvalidQueryError) |                                 InvalidQueryError, NotUniqueError) | ||||||
| from mongoengine.queryset import NULLIFY, Q | from mongoengine.queryset import NULLIFY, Q | ||||||
| from mongoengine.connection import get_db | from mongoengine.connection import get_db | ||||||
| from mongoengine.base import get_document | from mongoengine.base import get_document | ||||||
| @@ -57,7 +57,7 @@ class InstanceTest(unittest.TestCase): | |||||||
|             date = DateTimeField(default=datetime.now) |             date = DateTimeField(default=datetime.now) | ||||||
|             meta = { |             meta = { | ||||||
|                 'max_documents': 10, |                 'max_documents': 10, | ||||||
|                 'max_size': 90000, |                 'max_size': 4096, | ||||||
|             } |             } | ||||||
|  |  | ||||||
|         Log.drop_collection() |         Log.drop_collection() | ||||||
| @@ -75,7 +75,7 @@ class InstanceTest(unittest.TestCase): | |||||||
|         options = Log.objects._collection.options() |         options = Log.objects._collection.options() | ||||||
|         self.assertEqual(options['capped'], True) |         self.assertEqual(options['capped'], True) | ||||||
|         self.assertEqual(options['max'], 10) |         self.assertEqual(options['max'], 10) | ||||||
|         self.assertEqual(options['size'], 90000) |         self.assertTrue(options['size'] >= 4096) | ||||||
|  |  | ||||||
|         # Check that the document cannot be redefined with different options |         # Check that the document cannot be redefined with different options | ||||||
|         def recreate_log_document(): |         def recreate_log_document(): | ||||||
| @@ -353,6 +353,14 @@ class InstanceTest(unittest.TestCase): | |||||||
|         self.assertEqual(person.name, "Test User") |         self.assertEqual(person.name, "Test User") | ||||||
|         self.assertEqual(person.age, 20) |         self.assertEqual(person.age, 20) | ||||||
|  |  | ||||||
|  |         person.reload('age') | ||||||
|  |         self.assertEqual(person.name, "Test User") | ||||||
|  |         self.assertEqual(person.age, 21) | ||||||
|  |  | ||||||
|  |         person.reload() | ||||||
|  |         self.assertEqual(person.name, "Mr Test User") | ||||||
|  |         self.assertEqual(person.age, 21) | ||||||
|  |  | ||||||
|         person.reload() |         person.reload() | ||||||
|         self.assertEqual(person.name, "Mr Test User") |         self.assertEqual(person.name, "Mr Test User") | ||||||
|         self.assertEqual(person.age, 21) |         self.assertEqual(person.age, 21) | ||||||
| @@ -398,10 +406,11 @@ class InstanceTest(unittest.TestCase): | |||||||
|         doc.embedded_field.dict_field['woot'] = "woot" |         doc.embedded_field.dict_field['woot'] = "woot" | ||||||
|  |  | ||||||
|         self.assertEqual(doc._get_changed_fields(), [ |         self.assertEqual(doc._get_changed_fields(), [ | ||||||
|             'list_field', 'dict_field', 'embedded_field.list_field', |             'list_field', 'dict_field.woot', 'embedded_field.list_field', | ||||||
|             'embedded_field.dict_field']) |             'embedded_field.dict_field.woot']) | ||||||
|         doc.save() |         doc.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(len(doc.list_field), 4) | ||||||
|         doc = doc.reload(10) |         doc = doc.reload(10) | ||||||
|         self.assertEqual(doc._get_changed_fields(), []) |         self.assertEqual(doc._get_changed_fields(), []) | ||||||
|         self.assertEqual(len(doc.list_field), 4) |         self.assertEqual(len(doc.list_field), 4) | ||||||
| @@ -409,6 +418,37 @@ class InstanceTest(unittest.TestCase): | |||||||
|         self.assertEqual(len(doc.embedded_field.list_field), 4) |         self.assertEqual(len(doc.embedded_field.list_field), 4) | ||||||
|         self.assertEqual(len(doc.embedded_field.dict_field), 2) |         self.assertEqual(len(doc.embedded_field.dict_field), 2) | ||||||
|  |  | ||||||
|  |         doc.list_field.append(1) | ||||||
|  |         doc.save() | ||||||
|  |         doc.dict_field['extra'] = 1 | ||||||
|  |         doc = doc.reload(10, 'list_field') | ||||||
|  |         self.assertEqual(doc._get_changed_fields(), []) | ||||||
|  |         self.assertEqual(len(doc.list_field), 5) | ||||||
|  |         self.assertEqual(len(doc.dict_field), 3) | ||||||
|  |         self.assertEqual(len(doc.embedded_field.list_field), 4) | ||||||
|  |         self.assertEqual(len(doc.embedded_field.dict_field), 2) | ||||||
|  |  | ||||||
|  |     def test_reload_doesnt_exist(self): | ||||||
|  |         class Foo(Document): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         f = Foo() | ||||||
|  |         try: | ||||||
|  |             f.reload() | ||||||
|  |         except Foo.DoesNotExist: | ||||||
|  |             pass | ||||||
|  |         except Exception as ex: | ||||||
|  |             self.assertFalse("Threw wrong exception") | ||||||
|  |  | ||||||
|  |         f.save() | ||||||
|  |         f.delete() | ||||||
|  |         try: | ||||||
|  |             f.reload() | ||||||
|  |         except Foo.DoesNotExist: | ||||||
|  |             pass | ||||||
|  |         except Exception as ex: | ||||||
|  |             self.assertFalse("Threw wrong exception") | ||||||
|  |  | ||||||
|     def test_dictionary_access(self): |     def test_dictionary_access(self): | ||||||
|         """Ensure that dictionary-style field access works properly. |         """Ensure that dictionary-style field access works properly. | ||||||
|         """ |         """ | ||||||
| @@ -422,7 +462,7 @@ class InstanceTest(unittest.TestCase): | |||||||
|         self.assertEqual(person['name'], 'Another User') |         self.assertEqual(person['name'], 'Another User') | ||||||
|  |  | ||||||
|         # Length = length(assigned fields + id) |         # Length = length(assigned fields + id) | ||||||
|         self.assertEqual(len(person), 3) |         self.assertEqual(len(person), 4) | ||||||
|  |  | ||||||
|         self.assertTrue('age' in person) |         self.assertTrue('age' in person) | ||||||
|         person.age = None |         person.age = None | ||||||
| @@ -490,6 +530,23 @@ class InstanceTest(unittest.TestCase): | |||||||
|         doc = Doc.objects.get() |         doc = Doc.objects.get() | ||||||
|         self.assertEqual(doc, doc.embedded_field[0]._instance) |         self.assertEqual(doc, doc.embedded_field[0]._instance) | ||||||
|  |  | ||||||
|  |     def test_instance_is_set_on_setattr(self): | ||||||
|  |  | ||||||
|  |         class Email(EmbeddedDocument): | ||||||
|  |             email = EmailField() | ||||||
|  |  | ||||||
|  |         class Account(Document): | ||||||
|  |             email = EmbeddedDocumentField(Email) | ||||||
|  |  | ||||||
|  |         Account.drop_collection() | ||||||
|  |         acc = Account() | ||||||
|  |         acc.email = Email(email='test@example.com') | ||||||
|  |         self.assertTrue(hasattr(acc._data["email"], "_instance")) | ||||||
|  |         acc.save() | ||||||
|  |  | ||||||
|  |         acc1 = Account.objects.first() | ||||||
|  |         self.assertTrue(hasattr(acc1._data["email"], "_instance")) | ||||||
|  |  | ||||||
|     def test_document_clean(self): |     def test_document_clean(self): | ||||||
|         class TestDocument(Document): |         class TestDocument(Document): | ||||||
|             status = StringField() |             status = StringField() | ||||||
| @@ -779,6 +836,80 @@ class InstanceTest(unittest.TestCase): | |||||||
|         p1.reload() |         p1.reload() | ||||||
|         self.assertEqual(p1.name, p.parent.name) |         self.assertEqual(p1.name, p.parent.name) | ||||||
|  |  | ||||||
|  |     def test_save_atomicity_condition(self): | ||||||
|  |  | ||||||
|  |         class Widget(Document): | ||||||
|  |             toggle = BooleanField(default=False) | ||||||
|  |             count = IntField(default=0) | ||||||
|  |             save_id = UUIDField() | ||||||
|  |  | ||||||
|  |         def flip(widget): | ||||||
|  |             widget.toggle = not widget.toggle | ||||||
|  |             widget.count += 1 | ||||||
|  |  | ||||||
|  |         def UUID(i): | ||||||
|  |             return uuid.UUID(int=i) | ||||||
|  |  | ||||||
|  |         Widget.drop_collection() | ||||||
|  |  | ||||||
|  |         w1 = Widget(toggle=False, save_id=UUID(1)) | ||||||
|  |  | ||||||
|  |         # ignore save_condition on new record creation | ||||||
|  |         w1.save(save_condition={'save_id':UUID(42)}) | ||||||
|  |         w1.reload() | ||||||
|  |         self.assertFalse(w1.toggle) | ||||||
|  |         self.assertEqual(w1.save_id, UUID(1)) | ||||||
|  |         self.assertEqual(w1.count, 0) | ||||||
|  |  | ||||||
|  |         # mismatch in save_condition prevents save | ||||||
|  |         flip(w1) | ||||||
|  |         self.assertTrue(w1.toggle) | ||||||
|  |         self.assertEqual(w1.count, 1) | ||||||
|  |         w1.save(save_condition={'save_id':UUID(42)}) | ||||||
|  |         w1.reload() | ||||||
|  |         self.assertFalse(w1.toggle) | ||||||
|  |         self.assertEqual(w1.count, 0) | ||||||
|  |  | ||||||
|  |         # matched save_condition allows save | ||||||
|  |         flip(w1) | ||||||
|  |         self.assertTrue(w1.toggle) | ||||||
|  |         self.assertEqual(w1.count, 1) | ||||||
|  |         w1.save(save_condition={'save_id':UUID(1)}) | ||||||
|  |         w1.reload() | ||||||
|  |         self.assertTrue(w1.toggle) | ||||||
|  |         self.assertEqual(w1.count, 1) | ||||||
|  |  | ||||||
|  |         # save_condition can be used to ensure atomic read & updates | ||||||
|  |         # i.e., prevent interleaved reads and writes from separate contexts | ||||||
|  |         w2 = Widget.objects.get() | ||||||
|  |         self.assertEqual(w1, w2) | ||||||
|  |         old_id = w1.save_id | ||||||
|  |  | ||||||
|  |         flip(w1) | ||||||
|  |         w1.save_id = UUID(2) | ||||||
|  |         w1.save(save_condition={'save_id':old_id}) | ||||||
|  |         w1.reload() | ||||||
|  |         self.assertFalse(w1.toggle) | ||||||
|  |         self.assertEqual(w1.count, 2) | ||||||
|  |         flip(w2) | ||||||
|  |         flip(w2) | ||||||
|  |         w2.save(save_condition={'save_id':old_id}) | ||||||
|  |         w2.reload() | ||||||
|  |         self.assertFalse(w2.toggle) | ||||||
|  |         self.assertEqual(w2.count, 2) | ||||||
|  |  | ||||||
|  |         # save_condition uses mongoengine-style operator syntax | ||||||
|  |         flip(w1) | ||||||
|  |         w1.save(save_condition={'count__lt':w1.count}) | ||||||
|  |         w1.reload() | ||||||
|  |         self.assertTrue(w1.toggle) | ||||||
|  |         self.assertEqual(w1.count, 3) | ||||||
|  |         flip(w1) | ||||||
|  |         w1.save(save_condition={'count__gte':w1.count}) | ||||||
|  |         w1.reload() | ||||||
|  |         self.assertTrue(w1.toggle) | ||||||
|  |         self.assertEqual(w1.count, 3) | ||||||
|  |  | ||||||
|     def test_update(self): |     def test_update(self): | ||||||
|         """Ensure that an existing document is updated instead of be |         """Ensure that an existing document is updated instead of be | ||||||
|         overwritten.""" |         overwritten.""" | ||||||
| @@ -943,11 +1074,23 @@ class InstanceTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         self.assertRaises(OperationError, update_no_value_raises) |         self.assertRaises(OperationError, update_no_value_raises) | ||||||
|  |  | ||||||
|         def update_no_op_raises(): |         def update_no_op_should_default_to_set(): | ||||||
|             person = self.Person.objects.first() |             person = self.Person.objects.first() | ||||||
|             person.update(name="Dan") |             person.update(name="Dan") | ||||||
|  |             person.reload() | ||||||
|  |             return person.name | ||||||
|  |  | ||||||
|         self.assertRaises(InvalidQueryError, update_no_op_raises) |         self.assertEqual("Dan", update_no_op_should_default_to_set()) | ||||||
|  |  | ||||||
|  |     def test_update_unique_field(self): | ||||||
|  |         class Doc(Document): | ||||||
|  |             name = StringField(unique=True) | ||||||
|  |  | ||||||
|  |         doc1 = Doc(name="first").save() | ||||||
|  |         doc2 = Doc(name="second").save() | ||||||
|  |  | ||||||
|  |         self.assertRaises(NotUniqueError, lambda: | ||||||
|  |                           doc2.update(set__name=doc1.name)) | ||||||
|  |  | ||||||
|     def test_embedded_update(self): |     def test_embedded_update(self): | ||||||
|         """ |         """ | ||||||
| @@ -2240,6 +2383,8 @@ class InstanceTest(unittest.TestCase): | |||||||
|         log.machine = "Localhost" |         log.machine = "Localhost" | ||||||
|         log.save() |         log.save() | ||||||
|  |  | ||||||
|  |         self.assertTrue(log.id is not None) | ||||||
|  |  | ||||||
|         log.log = "Saving" |         log.log = "Saving" | ||||||
|         log.save() |         log.save() | ||||||
|  |  | ||||||
| @@ -2263,6 +2408,8 @@ class InstanceTest(unittest.TestCase): | |||||||
|         log.machine = "Localhost" |         log.machine = "Localhost" | ||||||
|         log.save() |         log.save() | ||||||
|  |  | ||||||
|  |         self.assertTrue(log.id is not None) | ||||||
|  |  | ||||||
|         log.log = "Saving" |         log.log = "Saving" | ||||||
|         log.save() |         log.save() | ||||||
|  |  | ||||||
| @@ -2370,7 +2517,7 @@ class InstanceTest(unittest.TestCase): | |||||||
|                 for parameter_name, parameter in self.parameters.iteritems(): |                 for parameter_name, parameter in self.parameters.iteritems(): | ||||||
|                     parameter.expand() |                     parameter.expand() | ||||||
|  |  | ||||||
|         class System(Document): |         class NodesSystem(Document): | ||||||
|             name = StringField(required=True) |             name = StringField(required=True) | ||||||
|             nodes = MapField(ReferenceField(Node, dbref=False)) |             nodes = MapField(ReferenceField(Node, dbref=False)) | ||||||
|  |  | ||||||
| @@ -2378,20 +2525,123 @@ class InstanceTest(unittest.TestCase): | |||||||
|                 for node_name, node in self.nodes.iteritems(): |                 for node_name, node in self.nodes.iteritems(): | ||||||
|                     node.expand() |                     node.expand() | ||||||
|                     node.save(*args, **kwargs) |                     node.save(*args, **kwargs) | ||||||
|                 super(System, self).save(*args, **kwargs) |                 super(NodesSystem, self).save(*args, **kwargs) | ||||||
|  |  | ||||||
|         System.drop_collection() |         NodesSystem.drop_collection() | ||||||
|         Node.drop_collection() |         Node.drop_collection() | ||||||
|  |  | ||||||
|         system = System(name="system") |         system = NodesSystem(name="system") | ||||||
|         system.nodes["node"] = Node() |         system.nodes["node"] = Node() | ||||||
|         system.save() |         system.save() | ||||||
|         system.nodes["node"].parameters["param"] = Parameter() |         system.nodes["node"].parameters["param"] = Parameter() | ||||||
|         system.save() |         system.save() | ||||||
|  |  | ||||||
|         system = System.objects.first() |         system = NodesSystem.objects.first() | ||||||
|         self.assertEqual("UNDEFINED", system.nodes["node"].parameters["param"].macros["test"].value) |         self.assertEqual("UNDEFINED", system.nodes["node"].parameters["param"].macros["test"].value) | ||||||
|  |  | ||||||
|  |     def test_embedded_document_equality(self): | ||||||
|  |  | ||||||
|  |         class Test(Document): | ||||||
|  |             field = StringField(required=True) | ||||||
|  |  | ||||||
|  |         class Embedded(EmbeddedDocument): | ||||||
|  |             ref = ReferenceField(Test) | ||||||
|  |  | ||||||
|  |         Test.drop_collection() | ||||||
|  |         test = Test(field='123').save()      # has id | ||||||
|  |  | ||||||
|  |         e = Embedded(ref=test) | ||||||
|  |         f1 = Embedded._from_son(e.to_mongo()) | ||||||
|  |         f2 = Embedded._from_son(e.to_mongo()) | ||||||
|  |  | ||||||
|  |         self.assertEqual(f1, f2) | ||||||
|  |         f1.ref  # Dereferences lazily | ||||||
|  |         self.assertEqual(f1, f2) | ||||||
|  |  | ||||||
|  |     def test_dbref_equality(self): | ||||||
|  |         class Test2(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Test3(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Test(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             test2 = ReferenceField('Test2') | ||||||
|  |             test3 = ReferenceField('Test3') | ||||||
|  |  | ||||||
|  |         Test.drop_collection() | ||||||
|  |         Test2.drop_collection() | ||||||
|  |         Test3.drop_collection() | ||||||
|  |  | ||||||
|  |         t2 = Test2(name='a') | ||||||
|  |         t2.save() | ||||||
|  |  | ||||||
|  |         t3 = Test3(name='x') | ||||||
|  |         t3.id = t2.id | ||||||
|  |         t3.save() | ||||||
|  |  | ||||||
|  |         t = Test(name='b', test2=t2, test3=t3) | ||||||
|  |  | ||||||
|  |         f = Test._from_son(t.to_mongo()) | ||||||
|  |  | ||||||
|  |         dbref2 = f._data['test2'] | ||||||
|  |         obj2 = f.test2 | ||||||
|  |         self.assertTrue(isinstance(dbref2, DBRef)) | ||||||
|  |         self.assertTrue(isinstance(obj2, Test2)) | ||||||
|  |         self.assertTrue(obj2.id == dbref2.id) | ||||||
|  |         self.assertTrue(obj2 == dbref2) | ||||||
|  |         self.assertTrue(dbref2 == obj2) | ||||||
|  |  | ||||||
|  |         dbref3 = f._data['test3'] | ||||||
|  |         obj3 = f.test3 | ||||||
|  |         self.assertTrue(isinstance(dbref3, DBRef)) | ||||||
|  |         self.assertTrue(isinstance(obj3, Test3)) | ||||||
|  |         self.assertTrue(obj3.id == dbref3.id) | ||||||
|  |         self.assertTrue(obj3 == dbref3) | ||||||
|  |         self.assertTrue(dbref3 == obj3) | ||||||
|  |  | ||||||
|  |         self.assertTrue(obj2.id == obj3.id) | ||||||
|  |         self.assertTrue(dbref2.id == dbref3.id) | ||||||
|  |         self.assertFalse(dbref2 == dbref3) | ||||||
|  |         self.assertFalse(dbref3 == dbref2) | ||||||
|  |         self.assertTrue(dbref2 != dbref3) | ||||||
|  |         self.assertTrue(dbref3 != dbref2) | ||||||
|  |  | ||||||
|  |         self.assertFalse(obj2 == dbref3) | ||||||
|  |         self.assertFalse(dbref3 == obj2) | ||||||
|  |         self.assertTrue(obj2 != dbref3) | ||||||
|  |         self.assertTrue(dbref3 != obj2) | ||||||
|  |  | ||||||
|  |         self.assertFalse(obj3 == dbref2) | ||||||
|  |         self.assertFalse(dbref2 == obj3) | ||||||
|  |         self.assertTrue(obj3 != dbref2) | ||||||
|  |         self.assertTrue(dbref2 != obj3) | ||||||
|  |  | ||||||
|  |     def test_default_values(self): | ||||||
|  |         class Person(Document): | ||||||
|  |             created_on = DateTimeField(default=lambda: datetime.utcnow()) | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         p = Person(name='alon') | ||||||
|  |         p.save() | ||||||
|  |         orig_created_on = Person.objects().only('created_on')[0].created_on | ||||||
|  |  | ||||||
|  |         p2 = Person.objects().only('name')[0] | ||||||
|  |         p2.name = 'alon2' | ||||||
|  |         p2.save() | ||||||
|  |         p3 = Person.objects().only('created_on')[0] | ||||||
|  |         self.assertEquals(orig_created_on, p3.created_on) | ||||||
|  |  | ||||||
|  |         class Person(Document): | ||||||
|  |             created_on = DateTimeField(default=lambda: datetime.utcnow()) | ||||||
|  |             name = StringField() | ||||||
|  |             height = IntField(default=189) | ||||||
|  |  | ||||||
|  |         p4 = Person.objects()[0] | ||||||
|  |         p4.save() | ||||||
|  |         self.assertEquals(p4.height, 189) | ||||||
|  |         self.assertEquals(Person.objects(height=189).count(), 1) | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -20,6 +20,28 @@ class TestJson(unittest.TestCase): | |||||||
|     def setUp(self): |     def setUp(self): | ||||||
|         connect(db='mongoenginetest') |         connect(db='mongoenginetest') | ||||||
|  |  | ||||||
|  |     def test_json_names(self): | ||||||
|  |         """ | ||||||
|  |         Going to test reported issue: | ||||||
|  |             https://github.com/MongoEngine/mongoengine/issues/654 | ||||||
|  |         where the reporter asks for the availability to perform | ||||||
|  |         a to_json with the original class names and not the abreviated | ||||||
|  |         mongodb document keys | ||||||
|  |         """ | ||||||
|  |         class Embedded(EmbeddedDocument): | ||||||
|  |             string = StringField(db_field='s') | ||||||
|  |  | ||||||
|  |         class Doc(Document): | ||||||
|  |             string = StringField(db_field='s') | ||||||
|  |             embedded = EmbeddedDocumentField(Embedded, db_field='e') | ||||||
|  |  | ||||||
|  |         doc = Doc( string="Hello", embedded=Embedded(string="Inner Hello")) | ||||||
|  |         doc_json = doc.to_json(sort_keys=True, use_db_field=False,separators=(',', ':')) | ||||||
|  |  | ||||||
|  |         expected_json = """{"embedded":{"string":"Inner Hello"},"string":"Hello"}""" | ||||||
|  |  | ||||||
|  |         self.assertEqual( doc_json, expected_json) | ||||||
|  |  | ||||||
|     def test_json_simple(self): |     def test_json_simple(self): | ||||||
|  |  | ||||||
|         class Embedded(EmbeddedDocument): |         class Embedded(EmbeddedDocument): | ||||||
| @@ -31,6 +53,10 @@ class TestJson(unittest.TestCase): | |||||||
|  |  | ||||||
|         doc = Doc(string="Hi", embedded_field=Embedded(string="Hi")) |         doc = Doc(string="Hi", embedded_field=Embedded(string="Hi")) | ||||||
|  |  | ||||||
|  |         doc_json = doc.to_json(sort_keys=True, separators=(',', ':')) | ||||||
|  |         expected_json = """{"embedded_field":{"string":"Hi"},"string":"Hi"}""" | ||||||
|  |         self.assertEqual(doc_json, expected_json) | ||||||
|  |  | ||||||
|         self.assertEqual(doc, Doc.from_json(doc.to_json())) |         self.assertEqual(doc, Doc.from_json(doc.to_json())) | ||||||
|  |  | ||||||
|     def test_json_complex(self): |     def test_json_complex(self): | ||||||
|   | |||||||
| @@ -141,6 +141,30 @@ class ValidatorErrorTest(unittest.TestCase): | |||||||
|             self.assertEqual(e.to_dict(), { |             self.assertEqual(e.to_dict(), { | ||||||
|                 "e": {'val': 'OK could not be converted to int'}}) |                 "e": {'val': 'OK could not be converted to int'}}) | ||||||
|  |  | ||||||
|  |     def test_embedded_weakref(self): | ||||||
|  |  | ||||||
|  |         class SubDoc(EmbeddedDocument): | ||||||
|  |             val = IntField(required=True) | ||||||
|  |  | ||||||
|  |         class Doc(Document): | ||||||
|  |             e = EmbeddedDocumentField(SubDoc, db_field='eb') | ||||||
|  |  | ||||||
|  |         Doc.drop_collection() | ||||||
|  |  | ||||||
|  |         d1 = Doc() | ||||||
|  |         d2 = Doc() | ||||||
|  |  | ||||||
|  |         s = SubDoc() | ||||||
|  |  | ||||||
|  |         self.assertRaises(ValidationError, lambda: s.validate()) | ||||||
|  |  | ||||||
|  |         d1.e = s | ||||||
|  |         d2.e = s | ||||||
|  |  | ||||||
|  |         del d1 | ||||||
|  |  | ||||||
|  |         self.assertRaises(ValidationError, lambda: d2.validate()) | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -18,6 +18,7 @@ from bson import Binary, DBRef, ObjectId | |||||||
| from mongoengine import * | from mongoengine import * | ||||||
| from mongoengine.connection import get_db | from mongoengine.connection import get_db | ||||||
| from mongoengine.base import _document_registry | from mongoengine.base import _document_registry | ||||||
|  | from mongoengine.base.datastructures import BaseDict | ||||||
| from mongoengine.errors import NotRegistered | from mongoengine.errors import NotRegistered | ||||||
| from mongoengine.python_support import PY3, b, bin_type | from mongoengine.python_support import PY3, b, bin_type | ||||||
|  |  | ||||||
| @@ -47,7 +48,8 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         # Confirm saving now would store values |         # Confirm saving now would store values | ||||||
|         data_to_be_saved = sorted(person.to_mongo().keys()) |         data_to_be_saved = sorted(person.to_mongo().keys()) | ||||||
|         self.assertEqual(data_to_be_saved, ['age', 'created', 'name', 'userid']) |         self.assertEqual( | ||||||
|  |             data_to_be_saved, ['age', 'created', 'name', 'userid']) | ||||||
|  |  | ||||||
|         self.assertTrue(person.validate() is None) |         self.assertTrue(person.validate() is None) | ||||||
|  |  | ||||||
| @@ -63,7 +65,8 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         # Confirm introspection changes nothing |         # Confirm introspection changes nothing | ||||||
|         data_to_be_saved = sorted(person.to_mongo().keys()) |         data_to_be_saved = sorted(person.to_mongo().keys()) | ||||||
|         self.assertEqual(data_to_be_saved, ['age', 'created', 'name', 'userid']) |         self.assertEqual( | ||||||
|  |             data_to_be_saved, ['age', 'created', 'name', 'userid']) | ||||||
|  |  | ||||||
|     def test_default_values_set_to_None(self): |     def test_default_values_set_to_None(self): | ||||||
|         """Ensure that default field values are used when creating a document. |         """Ensure that default field values are used when creating a document. | ||||||
| @@ -384,6 +387,9 @@ class FieldTest(unittest.TestCase): | |||||||
|         person.height = 4.0 |         person.height = 4.0 | ||||||
|         self.assertRaises(ValidationError, person.validate) |         self.assertRaises(ValidationError, person.validate) | ||||||
|  |  | ||||||
|  |         person_2 = Person(height='something invalid') | ||||||
|  |         self.assertRaises(ValidationError, person_2.validate) | ||||||
|  |  | ||||||
|     def test_decimal_validation(self): |     def test_decimal_validation(self): | ||||||
|         """Ensure that invalid values cannot be assigned to decimal fields. |         """Ensure that invalid values cannot be assigned to decimal fields. | ||||||
|         """ |         """ | ||||||
| @@ -405,6 +411,11 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertRaises(ValidationError, person.validate) |         self.assertRaises(ValidationError, person.validate) | ||||||
|         person.height = Decimal('4.0') |         person.height = Decimal('4.0') | ||||||
|         self.assertRaises(ValidationError, person.validate) |         self.assertRaises(ValidationError, person.validate) | ||||||
|  |         person.height = 'something invalid' | ||||||
|  |         self.assertRaises(ValidationError, person.validate) | ||||||
|  |  | ||||||
|  |         person_2 = Person(height='something invalid') | ||||||
|  |         self.assertRaises(ValidationError, person_2.validate) | ||||||
|  |  | ||||||
|         Person.drop_collection() |         Person.drop_collection() | ||||||
|  |  | ||||||
| @@ -579,7 +590,8 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         LogEntry.drop_collection() |         LogEntry.drop_collection() | ||||||
|  |  | ||||||
|         # Post UTC - microseconds are rounded (down) nearest millisecond and dropped |         # Post UTC - microseconds are rounded (down) nearest millisecond and | ||||||
|  |         # dropped | ||||||
|         d1 = datetime.datetime(1970, 01, 01, 00, 00, 01, 999) |         d1 = datetime.datetime(1970, 01, 01, 00, 00, 01, 999) | ||||||
|         d2 = datetime.datetime(1970, 01, 01, 00, 00, 01) |         d2 = datetime.datetime(1970, 01, 01, 00, 00, 01) | ||||||
|         log = LogEntry() |         log = LogEntry() | ||||||
| @@ -680,7 +692,8 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         LogEntry.drop_collection() |         LogEntry.drop_collection() | ||||||
|  |  | ||||||
|         # Post UTC - microseconds are rounded (down) nearest millisecond and dropped - with default datetimefields |         # Post UTC - microseconds are rounded (down) nearest millisecond and | ||||||
|  |         # dropped - with default datetimefields | ||||||
|         d1 = datetime.datetime(1970, 01, 01, 00, 00, 01, 999) |         d1 = datetime.datetime(1970, 01, 01, 00, 00, 01, 999) | ||||||
|         log = LogEntry() |         log = LogEntry() | ||||||
|         log.date = d1 |         log.date = d1 | ||||||
| @@ -688,14 +701,16 @@ class FieldTest(unittest.TestCase): | |||||||
|         log.reload() |         log.reload() | ||||||
|         self.assertEqual(log.date, d1) |         self.assertEqual(log.date, d1) | ||||||
|  |  | ||||||
|         # Post UTC - microseconds are rounded (down) nearest millisecond - with default datetimefields |         # Post UTC - microseconds are rounded (down) nearest millisecond - with | ||||||
|  |         # default datetimefields | ||||||
|         d1 = datetime.datetime(1970, 01, 01, 00, 00, 01, 9999) |         d1 = datetime.datetime(1970, 01, 01, 00, 00, 01, 9999) | ||||||
|         log.date = d1 |         log.date = d1 | ||||||
|         log.save() |         log.save() | ||||||
|         log.reload() |         log.reload() | ||||||
|         self.assertEqual(log.date, d1) |         self.assertEqual(log.date, d1) | ||||||
|  |  | ||||||
|         # Pre UTC dates microseconds below 1000 are dropped - with default datetimefields |         # Pre UTC dates microseconds below 1000 are dropped - with default | ||||||
|  |         # datetimefields | ||||||
|         d1 = datetime.datetime(1969, 12, 31, 23, 59, 59, 999) |         d1 = datetime.datetime(1969, 12, 31, 23, 59, 59, 999) | ||||||
|         log.date = d1 |         log.date = d1 | ||||||
|         log.save() |         log.save() | ||||||
| @@ -921,12 +936,16 @@ class FieldTest(unittest.TestCase): | |||||||
|         post.save() |         post.save() | ||||||
|  |  | ||||||
|         self.assertEqual(BlogPost.objects.count(), 3) |         self.assertEqual(BlogPost.objects.count(), 3) | ||||||
|         self.assertEqual(BlogPost.objects.filter(info__exact='test').count(), 1) |         self.assertEqual( | ||||||
|         self.assertEqual(BlogPost.objects.filter(info__0__test='test').count(), 1) |             BlogPost.objects.filter(info__exact='test').count(), 1) | ||||||
|  |         self.assertEqual( | ||||||
|  |             BlogPost.objects.filter(info__0__test='test').count(), 1) | ||||||
|  |  | ||||||
|         # Confirm handles non strings or non existing keys |         # Confirm handles non strings or non existing keys | ||||||
|         self.assertEqual(BlogPost.objects.filter(info__0__test__exact='5').count(), 0) |         self.assertEqual( | ||||||
|         self.assertEqual(BlogPost.objects.filter(info__100__test__exact='test').count(), 0) |             BlogPost.objects.filter(info__0__test__exact='5').count(), 0) | ||||||
|  |         self.assertEqual( | ||||||
|  |             BlogPost.objects.filter(info__100__test__exact='test').count(), 0) | ||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|     def test_list_field_passed_in_value(self): |     def test_list_field_passed_in_value(self): | ||||||
| @@ -943,7 +962,6 @@ class FieldTest(unittest.TestCase): | |||||||
|         foo.bars.append(bar) |         foo.bars.append(bar) | ||||||
|         self.assertEqual(repr(foo.bars), '[<Bar: Bar object>]') |         self.assertEqual(repr(foo.bars), '[<Bar: Bar object>]') | ||||||
|  |  | ||||||
|  |  | ||||||
|     def test_list_field_strict(self): |     def test_list_field_strict(self): | ||||||
|         """Ensure that list field handles validation if provided a strict field type.""" |         """Ensure that list field handles validation if provided a strict field type.""" | ||||||
|  |  | ||||||
| @@ -1074,20 +1092,28 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertTrue(isinstance(e2.mapping[1], IntegerSetting)) |         self.assertTrue(isinstance(e2.mapping[1], IntegerSetting)) | ||||||
|  |  | ||||||
|         # Test querying |         # Test querying | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__1__value=42).count(), 1) |         self.assertEqual( | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__2__number=1).count(), 1) |             Simple.objects.filter(mapping__1__value=42).count(), 1) | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__2__complex__value=42).count(), 1) |         self.assertEqual( | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__2__list__0__value=42).count(), 1) |             Simple.objects.filter(mapping__2__number=1).count(), 1) | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__2__list__1__value='foo').count(), 1) |         self.assertEqual( | ||||||
|  |             Simple.objects.filter(mapping__2__complex__value=42).count(), 1) | ||||||
|  |         self.assertEqual( | ||||||
|  |             Simple.objects.filter(mapping__2__list__0__value=42).count(), 1) | ||||||
|  |         self.assertEqual( | ||||||
|  |             Simple.objects.filter(mapping__2__list__1__value='foo').count(), 1) | ||||||
|  |  | ||||||
|         # Confirm can update |         # Confirm can update | ||||||
|         Simple.objects().update(set__mapping__1=IntegerSetting(value=10)) |         Simple.objects().update(set__mapping__1=IntegerSetting(value=10)) | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__1__value=10).count(), 1) |         self.assertEqual( | ||||||
|  |             Simple.objects.filter(mapping__1__value=10).count(), 1) | ||||||
|  |  | ||||||
|         Simple.objects().update( |         Simple.objects().update( | ||||||
|             set__mapping__2__list__1=StringSetting(value='Boo')) |             set__mapping__2__list__1=StringSetting(value='Boo')) | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__2__list__1__value='foo').count(), 0) |         self.assertEqual( | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__2__list__1__value='Boo').count(), 1) |             Simple.objects.filter(mapping__2__list__1__value='foo').count(), 0) | ||||||
|  |         self.assertEqual( | ||||||
|  |             Simple.objects.filter(mapping__2__list__1__value='Boo').count(), 1) | ||||||
|  |  | ||||||
|         Simple.drop_collection() |         Simple.drop_collection() | ||||||
|  |  | ||||||
| @@ -1109,9 +1135,15 @@ class FieldTest(unittest.TestCase): | |||||||
|         post.info = {'$title': 'test'} |         post.info = {'$title': 'test'} | ||||||
|         self.assertRaises(ValidationError, post.validate) |         self.assertRaises(ValidationError, post.validate) | ||||||
|  |  | ||||||
|  |         post.info = {'nested': {'$title': 'test'}} | ||||||
|  |         self.assertRaises(ValidationError, post.validate) | ||||||
|  |  | ||||||
|         post.info = {'the.title': 'test'} |         post.info = {'the.title': 'test'} | ||||||
|         self.assertRaises(ValidationError, post.validate) |         self.assertRaises(ValidationError, post.validate) | ||||||
|  |  | ||||||
|  |         post.info = {'nested': {'the.title': 'test'}} | ||||||
|  |         self.assertRaises(ValidationError, post.validate) | ||||||
|  |  | ||||||
|         post.info = {1: 'test'} |         post.info = {1: 'test'} | ||||||
|         self.assertRaises(ValidationError, post.validate) |         self.assertRaises(ValidationError, post.validate) | ||||||
|  |  | ||||||
| @@ -1127,12 +1159,16 @@ class FieldTest(unittest.TestCase): | |||||||
|         post.save() |         post.save() | ||||||
|  |  | ||||||
|         self.assertEqual(BlogPost.objects.count(), 3) |         self.assertEqual(BlogPost.objects.count(), 3) | ||||||
|         self.assertEqual(BlogPost.objects.filter(info__title__exact='test').count(), 1) |         self.assertEqual( | ||||||
|         self.assertEqual(BlogPost.objects.filter(info__details__test__exact='test').count(), 1) |             BlogPost.objects.filter(info__title__exact='test').count(), 1) | ||||||
|  |         self.assertEqual( | ||||||
|  |             BlogPost.objects.filter(info__details__test__exact='test').count(), 1) | ||||||
|  |  | ||||||
|         # Confirm handles non strings or non existing keys |         # Confirm handles non strings or non existing keys | ||||||
|         self.assertEqual(BlogPost.objects.filter(info__details__test__exact=5).count(), 0) |         self.assertEqual( | ||||||
|         self.assertEqual(BlogPost.objects.filter(info__made_up__test__exact='test').count(), 0) |             BlogPost.objects.filter(info__details__test__exact=5).count(), 0) | ||||||
|  |         self.assertEqual( | ||||||
|  |             BlogPost.objects.filter(info__made_up__test__exact='test').count(), 0) | ||||||
|  |  | ||||||
|         post = BlogPost.objects.create(info={'title': 'original'}) |         post = BlogPost.objects.create(info={'title': 'original'}) | ||||||
|         post.info.update({'title': 'updated'}) |         post.info.update({'title': 'updated'}) | ||||||
| @@ -1193,19 +1229,50 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertTrue(isinstance(e2.mapping['someint'], IntegerSetting)) |         self.assertTrue(isinstance(e2.mapping['someint'], IntegerSetting)) | ||||||
|  |  | ||||||
|         # Test querying |         # Test querying | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__someint__value=42).count(), 1) |         self.assertEqual( | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__nested_dict__number=1).count(), 1) |             Simple.objects.filter(mapping__someint__value=42).count(), 1) | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__nested_dict__complex__value=42).count(), 1) |         self.assertEqual( | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__nested_dict__list__0__value=42).count(), 1) |             Simple.objects.filter(mapping__nested_dict__number=1).count(), 1) | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__nested_dict__list__1__value='foo').count(), 1) |         self.assertEqual( | ||||||
|  |             Simple.objects.filter(mapping__nested_dict__complex__value=42).count(), 1) | ||||||
|  |         self.assertEqual( | ||||||
|  |             Simple.objects.filter(mapping__nested_dict__list__0__value=42).count(), 1) | ||||||
|  |         self.assertEqual( | ||||||
|  |             Simple.objects.filter(mapping__nested_dict__list__1__value='foo').count(), 1) | ||||||
|  |  | ||||||
|         # Confirm can update |         # Confirm can update | ||||||
|         Simple.objects().update( |         Simple.objects().update( | ||||||
|             set__mapping={"someint": IntegerSetting(value=10)}) |             set__mapping={"someint": IntegerSetting(value=10)}) | ||||||
|         Simple.objects().update( |         Simple.objects().update( | ||||||
|             set__mapping__nested_dict__list__1=StringSetting(value='Boo')) |             set__mapping__nested_dict__list__1=StringSetting(value='Boo')) | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__nested_dict__list__1__value='foo').count(), 0) |         self.assertEqual( | ||||||
|         self.assertEqual(Simple.objects.filter(mapping__nested_dict__list__1__value='Boo').count(), 1) |             Simple.objects.filter(mapping__nested_dict__list__1__value='foo').count(), 0) | ||||||
|  |         self.assertEqual( | ||||||
|  |             Simple.objects.filter(mapping__nested_dict__list__1__value='Boo').count(), 1) | ||||||
|  |  | ||||||
|  |         Simple.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_atomic_update_dict_field(self): | ||||||
|  |         """Ensure that the entire DictField can be atomically updated.""" | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         class Simple(Document): | ||||||
|  |             mapping = DictField(field=ListField(IntField(required=True))) | ||||||
|  |  | ||||||
|  |         Simple.drop_collection() | ||||||
|  |  | ||||||
|  |         e = Simple() | ||||||
|  |         e.mapping['someints'] = [1, 2] | ||||||
|  |         e.save() | ||||||
|  |         e.update(set__mapping={"ints": [3, 4]}) | ||||||
|  |         e.reload() | ||||||
|  |         self.assertEqual(BaseDict, type(e.mapping)) | ||||||
|  |         self.assertEqual({"ints": [3, 4]}, e.mapping) | ||||||
|  |  | ||||||
|  |         def create_invalid_mapping(): | ||||||
|  |             e.update(set__mapping={"somestrings": ["foo", "bar",]}) | ||||||
|  |  | ||||||
|  |         self.assertRaises(ValueError, create_invalid_mapping) | ||||||
|  |  | ||||||
|         Simple.drop_collection() |         Simple.drop_collection() | ||||||
|  |  | ||||||
| @@ -1463,6 +1530,375 @@ class FieldTest(unittest.TestCase): | |||||||
|         mongoed = p1.to_mongo() |         mongoed = p1.to_mongo() | ||||||
|         self.assertTrue(isinstance(mongoed['parent'], ObjectId)) |         self.assertTrue(isinstance(mongoed['parent'], ObjectId)) | ||||||
|  |  | ||||||
|  |     def test_cached_reference_fields(self): | ||||||
|  |         class Animal(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             tag = StringField() | ||||||
|  |  | ||||||
|  |         class Ocorrence(Document): | ||||||
|  |             person = StringField() | ||||||
|  |             animal = CachedReferenceField( | ||||||
|  |                 Animal, fields=['tag']) | ||||||
|  |  | ||||||
|  |         Animal.drop_collection() | ||||||
|  |         Ocorrence.drop_collection() | ||||||
|  |  | ||||||
|  |         a = Animal(name="Leopard", tag="heavy") | ||||||
|  |         a.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(Animal._cached_reference_fields, [Ocorrence.animal]) | ||||||
|  |         o = Ocorrence(person="teste", animal=a) | ||||||
|  |         o.save() | ||||||
|  |  | ||||||
|  |         p = Ocorrence(person="Wilson") | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(Ocorrence.objects(animal=None).count(), 1) | ||||||
|  |  | ||||||
|  |         self.assertEqual( | ||||||
|  |             a.to_mongo(fields=['tag']), {'tag': 'heavy', "_id": a.pk}) | ||||||
|  |  | ||||||
|  |         self.assertEqual(o.to_mongo()['animal']['tag'], 'heavy') | ||||||
|  |  | ||||||
|  |         # counts | ||||||
|  |         Ocorrence(person="teste 2").save() | ||||||
|  |         Ocorrence(person="teste 3").save() | ||||||
|  |  | ||||||
|  |         count = Ocorrence.objects(animal__tag='heavy').count() | ||||||
|  |         self.assertEqual(count, 1) | ||||||
|  |  | ||||||
|  |         ocorrence = Ocorrence.objects(animal__tag='heavy').first() | ||||||
|  |         self.assertEqual(ocorrence.person, "teste") | ||||||
|  |         self.assertTrue(isinstance(ocorrence.animal, Animal)) | ||||||
|  |  | ||||||
|  |     def test_cached_reference_field_decimal(self): | ||||||
|  |         class PersonAuto(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             salary = DecimalField() | ||||||
|  |  | ||||||
|  |         class SocialTest(Document): | ||||||
|  |             group = StringField() | ||||||
|  |             person = CachedReferenceField( | ||||||
|  |                 PersonAuto, | ||||||
|  |                 fields=('salary',)) | ||||||
|  |  | ||||||
|  |         PersonAuto.drop_collection() | ||||||
|  |         SocialTest.drop_collection() | ||||||
|  |  | ||||||
|  |         p = PersonAuto(name="Alberto", salary=Decimal('7000.00')) | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         s = SocialTest(group="dev", person=p) | ||||||
|  |         s.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual( | ||||||
|  |             SocialTest.objects._collection.find_one({'person.salary': 7000.00}), { | ||||||
|  |                 '_id': s.pk, | ||||||
|  |                 'group': s.group, | ||||||
|  |                 'person': { | ||||||
|  |                     '_id': p.pk, | ||||||
|  |                     'salary': 7000.00 | ||||||
|  |                 } | ||||||
|  |             }) | ||||||
|  |  | ||||||
|  |     def test_cached_reference_field_reference(self): | ||||||
|  |         class Group(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             group = ReferenceField(Group) | ||||||
|  |  | ||||||
|  |         class SocialData(Document): | ||||||
|  |             obs = StringField() | ||||||
|  |             tags = ListField( | ||||||
|  |                 StringField()) | ||||||
|  |             person = CachedReferenceField( | ||||||
|  |                 Person, | ||||||
|  |                 fields=('group',)) | ||||||
|  |  | ||||||
|  |         Group.drop_collection() | ||||||
|  |         Person.drop_collection() | ||||||
|  |         SocialData.drop_collection() | ||||||
|  |  | ||||||
|  |         g1 = Group(name='dev') | ||||||
|  |         g1.save() | ||||||
|  |  | ||||||
|  |         g2 = Group(name="designers") | ||||||
|  |         g2.save() | ||||||
|  |  | ||||||
|  |         p1 = Person(name="Alberto", group=g1) | ||||||
|  |         p1.save() | ||||||
|  |  | ||||||
|  |         p2 = Person(name="Andre", group=g1) | ||||||
|  |         p2.save() | ||||||
|  |  | ||||||
|  |         p3 = Person(name="Afro design", group=g2) | ||||||
|  |         p3.save() | ||||||
|  |  | ||||||
|  |         s1 = SocialData(obs="testing 123", person=p1, tags=['tag1', 'tag2']) | ||||||
|  |         s1.save() | ||||||
|  |  | ||||||
|  |         s2 = SocialData(obs="testing 321", person=p3, tags=['tag3', 'tag4']) | ||||||
|  |         s2.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(SocialData.objects._collection.find_one( | ||||||
|  |             {'tags': 'tag2'}), { | ||||||
|  |                 '_id': s1.pk, | ||||||
|  |                 'obs': 'testing 123', | ||||||
|  |                 'tags': ['tag1', 'tag2'], | ||||||
|  |                 'person': { | ||||||
|  |                     '_id': p1.pk, | ||||||
|  |                     'group': g1.pk | ||||||
|  |                 } | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  |         self.assertEqual(SocialData.objects(person__group=g2).count(), 1) | ||||||
|  |         self.assertEqual(SocialData.objects(person__group=g2).first(), s2) | ||||||
|  |  | ||||||
|  |     def test_cached_reference_field_update_all(self): | ||||||
|  |         class Person(Document): | ||||||
|  |             TYPES = ( | ||||||
|  |                 ('pf', "PF"), | ||||||
|  |                 ('pj', "PJ") | ||||||
|  |             ) | ||||||
|  |             name = StringField() | ||||||
|  |             tp = StringField( | ||||||
|  |                 choices=TYPES | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |             father = CachedReferenceField('self', fields=('tp',)) | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         a1 = Person(name="Wilson Father", tp="pj") | ||||||
|  |         a1.save() | ||||||
|  |  | ||||||
|  |         a2 = Person(name='Wilson Junior', tp='pf', father=a1) | ||||||
|  |         a2.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(dict(a2.to_mongo()), { | ||||||
|  |             "_id": a2.pk, | ||||||
|  |             "name": u"Wilson Junior", | ||||||
|  |             "tp": u"pf", | ||||||
|  |             "father": { | ||||||
|  |                 "_id": a1.pk, | ||||||
|  |                 "tp": u"pj" | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  |         self.assertEqual(Person.objects(father=a1)._query, { | ||||||
|  |             'father._id': a1.pk | ||||||
|  |         }) | ||||||
|  |         self.assertEqual(Person.objects(father=a1).count(), 1) | ||||||
|  |  | ||||||
|  |         Person.objects.update(set__tp="pf") | ||||||
|  |         Person.father.sync_all() | ||||||
|  |  | ||||||
|  |         a2.reload() | ||||||
|  |         self.assertEqual(dict(a2.to_mongo()), { | ||||||
|  |             "_id": a2.pk, | ||||||
|  |             "name": u"Wilson Junior", | ||||||
|  |             "tp": u"pf", | ||||||
|  |             "father": { | ||||||
|  |                 "_id": a1.pk, | ||||||
|  |                 "tp": u"pf" | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  |     def test_cached_reference_fields_on_embedded_documents(self): | ||||||
|  |         def build(): | ||||||
|  |             class Test(Document): | ||||||
|  |                 name = StringField() | ||||||
|  |  | ||||||
|  |             type('WrongEmbeddedDocument', ( | ||||||
|  |                 EmbeddedDocument,), { | ||||||
|  |                     'test': CachedReferenceField(Test) | ||||||
|  |             }) | ||||||
|  |  | ||||||
|  |         self.assertRaises(InvalidDocumentError, build) | ||||||
|  |  | ||||||
|  |     def test_cached_reference_auto_sync(self): | ||||||
|  |         class Person(Document): | ||||||
|  |             TYPES = ( | ||||||
|  |                 ('pf', "PF"), | ||||||
|  |                 ('pj', "PJ") | ||||||
|  |             ) | ||||||
|  |             name = StringField() | ||||||
|  |             tp = StringField( | ||||||
|  |                 choices=TYPES | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |             father = CachedReferenceField('self', fields=('tp',)) | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         a1 = Person(name="Wilson Father", tp="pj") | ||||||
|  |         a1.save() | ||||||
|  |  | ||||||
|  |         a2 = Person(name='Wilson Junior', tp='pf', father=a1) | ||||||
|  |         a2.save() | ||||||
|  |  | ||||||
|  |         a1.tp = 'pf' | ||||||
|  |         a1.save() | ||||||
|  |  | ||||||
|  |         a2.reload() | ||||||
|  |         self.assertEqual(dict(a2.to_mongo()), { | ||||||
|  |             '_id': a2.pk, | ||||||
|  |             'name': 'Wilson Junior', | ||||||
|  |             'tp': 'pf', | ||||||
|  |             'father': { | ||||||
|  |                 '_id': a1.pk, | ||||||
|  |                 'tp': 'pf' | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  |     def test_cached_reference_auto_sync_disabled(self): | ||||||
|  |         class Persone(Document): | ||||||
|  |             TYPES = ( | ||||||
|  |                 ('pf', "PF"), | ||||||
|  |                 ('pj', "PJ") | ||||||
|  |             ) | ||||||
|  |             name = StringField() | ||||||
|  |             tp = StringField( | ||||||
|  |                 choices=TYPES | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |             father = CachedReferenceField( | ||||||
|  |                 'self', fields=('tp',), auto_sync=False) | ||||||
|  |  | ||||||
|  |         Persone.drop_collection() | ||||||
|  |  | ||||||
|  |         a1 = Persone(name="Wilson Father", tp="pj") | ||||||
|  |         a1.save() | ||||||
|  |  | ||||||
|  |         a2 = Persone(name='Wilson Junior', tp='pf', father=a1) | ||||||
|  |         a2.save() | ||||||
|  |  | ||||||
|  |         a1.tp = 'pf' | ||||||
|  |         a1.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(Persone.objects._collection.find_one({'_id': a2.pk}), { | ||||||
|  |             '_id': a2.pk, | ||||||
|  |             'name': 'Wilson Junior', | ||||||
|  |             'tp': 'pf', | ||||||
|  |             'father': { | ||||||
|  |                 '_id': a1.pk, | ||||||
|  |                 'tp': 'pj' | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  |     def test_cached_reference_embedded_fields(self): | ||||||
|  |         class Owner(EmbeddedDocument): | ||||||
|  |             TPS = ( | ||||||
|  |                 ('n', "Normal"), | ||||||
|  |                 ('u', "Urgent") | ||||||
|  |             ) | ||||||
|  |             name = StringField() | ||||||
|  |             tp = StringField( | ||||||
|  |                 verbose_name="Type", | ||||||
|  |                 db_field="t", | ||||||
|  |                 choices=TPS) | ||||||
|  |  | ||||||
|  |         class Animal(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             tag = StringField() | ||||||
|  |  | ||||||
|  |             owner = EmbeddedDocumentField(Owner) | ||||||
|  |  | ||||||
|  |         class Ocorrence(Document): | ||||||
|  |             person = StringField() | ||||||
|  |             animal = CachedReferenceField( | ||||||
|  |                 Animal, fields=['tag', 'owner.tp']) | ||||||
|  |  | ||||||
|  |         Animal.drop_collection() | ||||||
|  |         Ocorrence.drop_collection() | ||||||
|  |  | ||||||
|  |         a = Animal(nam="Leopard", tag="heavy", | ||||||
|  |                    owner=Owner(tp='u', name="Wilson Júnior") | ||||||
|  |                    ) | ||||||
|  |         a.save() | ||||||
|  |  | ||||||
|  |         o = Ocorrence(person="teste", animal=a) | ||||||
|  |         o.save() | ||||||
|  |         self.assertEqual(dict(a.to_mongo(fields=['tag', 'owner.tp'])), { | ||||||
|  |             '_id': a.pk, | ||||||
|  |             'tag': 'heavy', | ||||||
|  |             'owner': { | ||||||
|  |                 't': 'u' | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |         self.assertEqual(o.to_mongo()['animal']['tag'], 'heavy') | ||||||
|  |         self.assertEqual(o.to_mongo()['animal']['owner']['t'], 'u') | ||||||
|  |  | ||||||
|  |         # counts | ||||||
|  |         Ocorrence(person="teste 2").save() | ||||||
|  |         Ocorrence(person="teste 3").save() | ||||||
|  |  | ||||||
|  |         count = Ocorrence.objects( | ||||||
|  |             animal__tag='heavy', animal__owner__tp='u').count() | ||||||
|  |         self.assertEqual(count, 1) | ||||||
|  |  | ||||||
|  |         ocorrence = Ocorrence.objects( | ||||||
|  |             animal__tag='heavy', | ||||||
|  |             animal__owner__tp='u').first() | ||||||
|  |         self.assertEqual(ocorrence.person, "teste") | ||||||
|  |         self.assertTrue(isinstance(ocorrence.animal, Animal)) | ||||||
|  |  | ||||||
|  |     def test_cached_reference_embedded_list_fields(self): | ||||||
|  |         class Owner(EmbeddedDocument): | ||||||
|  |             name = StringField() | ||||||
|  |             tags = ListField(StringField()) | ||||||
|  |  | ||||||
|  |         class Animal(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             tag = StringField() | ||||||
|  |  | ||||||
|  |             owner = EmbeddedDocumentField(Owner) | ||||||
|  |  | ||||||
|  |         class Ocorrence(Document): | ||||||
|  |             person = StringField() | ||||||
|  |             animal = CachedReferenceField( | ||||||
|  |                 Animal, fields=['tag', 'owner.tags']) | ||||||
|  |  | ||||||
|  |         Animal.drop_collection() | ||||||
|  |         Ocorrence.drop_collection() | ||||||
|  |  | ||||||
|  |         a = Animal(nam="Leopard", tag="heavy", | ||||||
|  |                    owner=Owner(tags=['cool', 'funny'], | ||||||
|  |                                name="Wilson Júnior") | ||||||
|  |                    ) | ||||||
|  |         a.save() | ||||||
|  |  | ||||||
|  |         o = Ocorrence(person="teste 2", animal=a) | ||||||
|  |         o.save() | ||||||
|  |         self.assertEqual(dict(a.to_mongo(fields=['tag', 'owner.tags'])), { | ||||||
|  |             '_id': a.pk, | ||||||
|  |             'tag': 'heavy', | ||||||
|  |             'owner': { | ||||||
|  |                 'tags': ['cool', 'funny'] | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  |         self.assertEqual(o.to_mongo()['animal']['tag'], 'heavy') | ||||||
|  |         self.assertEqual(o.to_mongo()['animal']['owner']['tags'], | ||||||
|  |                          ['cool', 'funny']) | ||||||
|  |  | ||||||
|  |         # counts | ||||||
|  |         Ocorrence(person="teste 2").save() | ||||||
|  |         Ocorrence(person="teste 3").save() | ||||||
|  |  | ||||||
|  |         query = Ocorrence.objects( | ||||||
|  |             animal__tag='heavy', animal__owner__tags='cool')._query | ||||||
|  |         self.assertEqual( | ||||||
|  |             query, {'animal.owner.tags': 'cool', 'animal.tag': 'heavy'}) | ||||||
|  |  | ||||||
|  |         ocorrence = Ocorrence.objects( | ||||||
|  |             animal__tag='heavy', | ||||||
|  |             animal__owner__tags='cool').first() | ||||||
|  |         self.assertEqual(ocorrence.person, "teste 2") | ||||||
|  |         self.assertTrue(isinstance(ocorrence.animal, Animal)) | ||||||
|  |  | ||||||
|     def test_objectid_reference_fields(self): |     def test_objectid_reference_fields(self): | ||||||
|  |  | ||||||
|         class Person(Document): |         class Person(Document): | ||||||
| @@ -1822,7 +2258,6 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertEqual(repr(Person.objects(city=None)), |         self.assertEqual(repr(Person.objects(city=None)), | ||||||
|                          "[<Person: Person object>]") |                          "[<Person: Person object>]") | ||||||
|  |  | ||||||
|  |  | ||||||
|     def test_generic_reference_choices(self): |     def test_generic_reference_choices(self): | ||||||
|         """Ensure that a GenericReferenceField can handle choices |         """Ensure that a GenericReferenceField can handle choices | ||||||
|         """ |         """ | ||||||
| @@ -1890,6 +2325,37 @@ class FieldTest(unittest.TestCase): | |||||||
|         Post.drop_collection() |         Post.drop_collection() | ||||||
|         User.drop_collection() |         User.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_generic_reference_list_item_modification(self): | ||||||
|  |         """Ensure that modifications of related documents (through generic reference) don't influence on querying | ||||||
|  |         """ | ||||||
|  |         class Post(Document): | ||||||
|  |             title = StringField() | ||||||
|  |  | ||||||
|  |         class User(Document): | ||||||
|  |             username = StringField() | ||||||
|  |             bookmarks = ListField(GenericReferenceField()) | ||||||
|  |  | ||||||
|  |         Post.drop_collection() | ||||||
|  |         User.drop_collection() | ||||||
|  |  | ||||||
|  |         post_1 = Post(title="Behind the Scenes of the Pavement Reunion") | ||||||
|  |         post_1.save() | ||||||
|  |  | ||||||
|  |         user = User(bookmarks=[post_1]) | ||||||
|  |         user.save() | ||||||
|  |  | ||||||
|  |         post_1.title = "Title was modified" | ||||||
|  |         user.username = "New username" | ||||||
|  |         user.save() | ||||||
|  |  | ||||||
|  |         user = User.objects(bookmarks__all=[post_1]).first() | ||||||
|  |  | ||||||
|  |         self.assertNotEqual(user, None) | ||||||
|  |         self.assertEqual(user.bookmarks[0], post_1) | ||||||
|  |  | ||||||
|  |         Post.drop_collection() | ||||||
|  |         User.drop_collection() | ||||||
|  |  | ||||||
|     def test_binary_fields(self): |     def test_binary_fields(self): | ||||||
|         """Ensure that binary fields can be stored and retrieved. |         """Ensure that binary fields can be stored and retrieved. | ||||||
|         """ |         """ | ||||||
| @@ -1937,7 +2403,8 @@ class FieldTest(unittest.TestCase): | |||||||
|         attachment_required.blob = Binary(b('\xe6\x00\xc4\xff\x07')) |         attachment_required.blob = Binary(b('\xe6\x00\xc4\xff\x07')) | ||||||
|         attachment_required.validate() |         attachment_required.validate() | ||||||
|  |  | ||||||
|         attachment_size_limit = AttachmentSizeLimit(blob=b('\xe6\x00\xc4\xff\x07')) |         attachment_size_limit = AttachmentSizeLimit( | ||||||
|  |             blob=b('\xe6\x00\xc4\xff\x07')) | ||||||
|         self.assertRaises(ValidationError, attachment_size_limit.validate) |         self.assertRaises(ValidationError, attachment_size_limit.validate) | ||||||
|         attachment_size_limit.blob = b('\xe6\x00\xc4\xff') |         attachment_size_limit.blob = b('\xe6\x00\xc4\xff') | ||||||
|         attachment_size_limit.validate() |         attachment_size_limit.validate() | ||||||
| @@ -2134,7 +2601,6 @@ class FieldTest(unittest.TestCase): | |||||||
|         c = self.db['mongoengine.counters'].find_one({'_id': 'person.id'}) |         c = self.db['mongoengine.counters'].find_one({'_id': 'person.id'}) | ||||||
|         self.assertEqual(c['next'], 1000) |         self.assertEqual(c['next'], 1000) | ||||||
|  |  | ||||||
|  |  | ||||||
|     def test_sequence_field_get_next_value(self): |     def test_sequence_field_get_next_value(self): | ||||||
|         class Person(Document): |         class Person(Document): | ||||||
|             id = SequenceField(primary_key=True) |             id = SequenceField(primary_key=True) | ||||||
| @@ -2323,7 +2789,6 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertEqual(1, post.comments[0].id) |         self.assertEqual(1, post.comments[0].id) | ||||||
|         self.assertEqual(2, post.comments[1].id) |         self.assertEqual(2, post.comments[1].id) | ||||||
|  |  | ||||||
|  |  | ||||||
|     def test_generic_embedded_document(self): |     def test_generic_embedded_document(self): | ||||||
|         class Car(EmbeddedDocument): |         class Car(EmbeddedDocument): | ||||||
|             name = StringField() |             name = StringField() | ||||||
| @@ -2454,14 +2919,23 @@ class FieldTest(unittest.TestCase): | |||||||
|         user = User(email="ross@example.com") |         user = User(email="ross@example.com") | ||||||
|         self.assertTrue(user.validate() is None) |         self.assertTrue(user.validate() is None) | ||||||
|  |  | ||||||
|  |         user = User(email="ross@example.co.uk") | ||||||
|  |         self.assertTrue(user.validate() is None) | ||||||
|  |  | ||||||
|         user = User(email=("Kofq@rhom0e4klgauOhpbpNdogawnyIKvQS0wk2mjqrgGQ5S" |         user = User(email=("Kofq@rhom0e4klgauOhpbpNdogawnyIKvQS0wk2mjqrgGQ5S" | ||||||
|                            "ucictfqpdkK9iS1zeFw8sg7s7cwAF7suIfUfeyueLpfosjn3" |                            "ucictfqpdkK9iS1zeFw8sg7s7cwAF7suIfUfeyueLpfosjn3" | ||||||
|                            "aJIazqqWkm7.net")) |                            "aJIazqqWkm7.net")) | ||||||
|         self.assertTrue(user.validate() is None) |         self.assertTrue(user.validate() is None) | ||||||
|  |  | ||||||
|  |         user = User(email="new-tld@example.technology") | ||||||
|  |         self.assertTrue(user.validate() is None) | ||||||
|  |  | ||||||
|         user = User(email='me@localhost') |         user = User(email='me@localhost') | ||||||
|         self.assertRaises(ValidationError, user.validate) |         self.assertRaises(ValidationError, user.validate) | ||||||
|  |  | ||||||
|  |         user = User(email="ross@example.com.") | ||||||
|  |         self.assertRaises(ValidationError, user.validate) | ||||||
|  |  | ||||||
|     def test_email_field_honors_regex(self): |     def test_email_field_honors_regex(self): | ||||||
|         class User(Document): |         class User(Document): | ||||||
|             email = EmailField(regex=r'\w+@example.com') |             email = EmailField(regex=r'\w+@example.com') | ||||||
| @@ -2506,5 +2980,83 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertTrue(tuple(x.items[0]) in tuples) |         self.assertTrue(tuple(x.items[0]) in tuples) | ||||||
|         self.assertTrue(x.items[0] in tuples) |         self.assertTrue(x.items[0] in tuples) | ||||||
|  |  | ||||||
|  |     def test_dynamic_fields_class(self): | ||||||
|  |  | ||||||
|  |         class Doc2(Document): | ||||||
|  |             field_1 = StringField(db_field='f') | ||||||
|  |  | ||||||
|  |         class Doc(Document): | ||||||
|  |             my_id = IntField(required=True, unique=True, primary_key=True) | ||||||
|  |             embed_me = DynamicField(db_field='e') | ||||||
|  |             field_x = StringField(db_field='x') | ||||||
|  |  | ||||||
|  |         Doc.drop_collection() | ||||||
|  |         Doc2.drop_collection() | ||||||
|  |  | ||||||
|  |         doc2 = Doc2(field_1="hello") | ||||||
|  |         doc = Doc(my_id=1, embed_me=doc2, field_x="x") | ||||||
|  |         self.assertRaises(OperationError, doc.save) | ||||||
|  |  | ||||||
|  |         doc2.save() | ||||||
|  |         doc.save() | ||||||
|  |  | ||||||
|  |         doc = Doc.objects.get() | ||||||
|  |         self.assertEqual(doc.embed_me.field_1, "hello") | ||||||
|  |  | ||||||
|  |     def test_dynamic_fields_embedded_class(self): | ||||||
|  |  | ||||||
|  |         class Embed(EmbeddedDocument): | ||||||
|  |             field_1 = StringField(db_field='f') | ||||||
|  |  | ||||||
|  |         class Doc(Document): | ||||||
|  |             my_id = IntField(required=True, unique=True, primary_key=True) | ||||||
|  |             embed_me = DynamicField(db_field='e') | ||||||
|  |             field_x = StringField(db_field='x') | ||||||
|  |  | ||||||
|  |         Doc.drop_collection() | ||||||
|  |  | ||||||
|  |         Doc(my_id=1, embed_me=Embed(field_1="hello"), field_x="x").save() | ||||||
|  |  | ||||||
|  |         doc = Doc.objects.get() | ||||||
|  |         self.assertEqual(doc.embed_me.field_1, "hello") | ||||||
|  |  | ||||||
|  |     def test_invalid_dict_value(self): | ||||||
|  |         class DictFieldTest(Document): | ||||||
|  |             dictionary = DictField(required=True) | ||||||
|  |  | ||||||
|  |         DictFieldTest.drop_collection() | ||||||
|  |  | ||||||
|  |         test = DictFieldTest(dictionary=None) | ||||||
|  |         test.dictionary  # Just access to test getter | ||||||
|  |         self.assertRaises(ValidationError, test.validate) | ||||||
|  |  | ||||||
|  |         test = DictFieldTest(dictionary=False) | ||||||
|  |         test.dictionary  # Just access to test getter | ||||||
|  |         self.assertRaises(ValidationError, test.validate) | ||||||
|  |  | ||||||
|  |     def test_cls_field(self): | ||||||
|  |         class Animal(Document): | ||||||
|  |             meta = {'allow_inheritance': True} | ||||||
|  |  | ||||||
|  |         class Fish(Animal): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         class Mammal(Animal): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         class Dog(Mammal): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         class Human(Mammal): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         Animal.objects.delete() | ||||||
|  |         Dog().save() | ||||||
|  |         Fish().save() | ||||||
|  |         Human().save() | ||||||
|  |         self.assertEquals(Animal.objects(_cls__in=["Animal.Mammal.Dog", "Animal.Fish"]).count(), 2) | ||||||
|  |         self.assertEquals(Animal.objects(_cls__in=["Animal.Fish.Guppy"]).count(), 0) | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -53,11 +53,12 @@ class FileTest(unittest.TestCase): | |||||||
|         content_type = 'text/plain' |         content_type = 'text/plain' | ||||||
|  |  | ||||||
|         putfile = PutFile() |         putfile = PutFile() | ||||||
|         putfile.the_file.put(text, content_type=content_type) |         putfile.the_file.put(text, content_type=content_type, filename="hello") | ||||||
|         putfile.save() |         putfile.save() | ||||||
|  |  | ||||||
|         result = PutFile.objects.first() |         result = PutFile.objects.first() | ||||||
|         self.assertTrue(putfile == result) |         self.assertTrue(putfile == result) | ||||||
|  |         self.assertEqual("%s" % result.the_file, "<GridFSProxy: hello>") | ||||||
|         self.assertEqual(result.the_file.read(), text) |         self.assertEqual(result.the_file.read(), text) | ||||||
|         self.assertEqual(result.the_file.content_type, content_type) |         self.assertEqual(result.the_file.content_type, content_type) | ||||||
|         result.the_file.delete()  # Remove file from GridFS |         result.the_file.delete()  # Remove file from GridFS | ||||||
| @@ -278,7 +279,7 @@ class FileTest(unittest.TestCase): | |||||||
|                 t.image.put(f) |                 t.image.put(f) | ||||||
|                 self.fail("Should have raised an invalidation error") |                 self.fail("Should have raised an invalidation error") | ||||||
|             except ValidationError, e: |             except ValidationError, e: | ||||||
|                 self.assertEquals("%s" % e, "Invalid image: cannot identify image file") |                 self.assertEqual("%s" % e, "Invalid image: cannot identify image file %s" % f) | ||||||
|  |  | ||||||
|         t = TestImage() |         t = TestImage() | ||||||
|         t.image.put(open(TEST_IMAGE_PATH, 'rb')) |         t.image.put(open(TEST_IMAGE_PATH, 'rb')) | ||||||
|   | |||||||
| @@ -19,8 +19,8 @@ class GeoFieldTest(unittest.TestCase): | |||||||
|     def _test_for_expected_error(self, Cls, loc, expected): |     def _test_for_expected_error(self, Cls, loc, expected): | ||||||
|         try: |         try: | ||||||
|             Cls(loc=loc).validate() |             Cls(loc=loc).validate() | ||||||
|             self.fail() |             self.fail('Should not validate the location {0}'.format(loc)) | ||||||
|         except ValidationError, e: |         except ValidationError as e: | ||||||
|             self.assertEqual(expected, e.to_dict()['loc']) |             self.assertEqual(expected, e.to_dict()['loc']) | ||||||
|  |  | ||||||
|     def test_geopoint_validation(self): |     def test_geopoint_validation(self): | ||||||
| @@ -75,6 +75,12 @@ class GeoFieldTest(unittest.TestCase): | |||||||
|             self._test_for_expected_error(Location, coord, expected) |             self._test_for_expected_error(Location, coord, expected) | ||||||
|  |  | ||||||
|         Location(loc=[1, 2]).validate() |         Location(loc=[1, 2]).validate() | ||||||
|  |         Location(loc={ | ||||||
|  |             "type": "Point", | ||||||
|  |             "coordinates": [ | ||||||
|  |               81.4471435546875, | ||||||
|  |               23.61432859499169 | ||||||
|  |             ]}).validate() | ||||||
|  |  | ||||||
|     def test_linestring_validation(self): |     def test_linestring_validation(self): | ||||||
|         class Location(Document): |         class Location(Document): | ||||||
| @@ -149,6 +155,117 @@ class GeoFieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         Location(loc=[[[1, 2], [3, 4], [5, 6], [1, 2]]]).validate() |         Location(loc=[[[1, 2], [3, 4], [5, 6], [1, 2]]]).validate() | ||||||
|  |  | ||||||
|  |     def test_multipoint_validation(self): | ||||||
|  |         class Location(Document): | ||||||
|  |             loc = MultiPointField() | ||||||
|  |  | ||||||
|  |         invalid_coords = {"x": 1, "y": 2} | ||||||
|  |         expected = 'MultiPointField can only accept a valid GeoJson dictionary or lists of (x, y)' | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = {"type": "MadeUp", "coordinates": [[]]} | ||||||
|  |         expected = 'MultiPointField type must be "MultiPoint"' | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = {"type": "MultiPoint", "coordinates": [[1, 2, 3]]} | ||||||
|  |         expected = "Value ([1, 2, 3]) must be a two-dimensional point" | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[]] | ||||||
|  |         expected = "Invalid MultiPoint must contain at least one valid point" | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[[1]], [[1, 2, 3]]] | ||||||
|  |         for coord in invalid_coords: | ||||||
|  |             expected = "Value (%s) must be a two-dimensional point" % repr(coord[0]) | ||||||
|  |             self._test_for_expected_error(Location, coord, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[[{}, {}]], [("a", "b")]] | ||||||
|  |         for coord in invalid_coords: | ||||||
|  |             expected = "Both values (%s) in point must be float or int" % repr(coord[0]) | ||||||
|  |             self._test_for_expected_error(Location, coord, expected) | ||||||
|  |  | ||||||
|  |         Location(loc=[[1, 2]]).validate() | ||||||
|  |         Location(loc={ | ||||||
|  |             "type": "MultiPoint", | ||||||
|  |             "coordinates": [ | ||||||
|  |                 [1, 2], | ||||||
|  |                 [81.4471435546875, 23.61432859499169] | ||||||
|  |             ]}).validate() | ||||||
|  |  | ||||||
|  |     def test_multilinestring_validation(self): | ||||||
|  |         class Location(Document): | ||||||
|  |             loc = MultiLineStringField() | ||||||
|  |  | ||||||
|  |         invalid_coords = {"x": 1, "y": 2} | ||||||
|  |         expected = 'MultiLineStringField can only accept a valid GeoJson dictionary or lists of (x, y)' | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = {"type": "MadeUp", "coordinates": [[]]} | ||||||
|  |         expected = 'MultiLineStringField type must be "MultiLineString"' | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = {"type": "MultiLineString", "coordinates": [[[1, 2, 3]]]} | ||||||
|  |         expected = "Invalid MultiLineString:\nValue ([1, 2, 3]) must be a two-dimensional point" | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [5, "a"] | ||||||
|  |         expected = "Invalid MultiLineString must contain at least one valid linestring" | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[[1]]] | ||||||
|  |         expected = "Invalid MultiLineString:\nValue (%s) must be a two-dimensional point" % repr(invalid_coords[0][0]) | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[[1, 2, 3]]] | ||||||
|  |         expected = "Invalid MultiLineString:\nValue (%s) must be a two-dimensional point" % repr(invalid_coords[0][0]) | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[[[{}, {}]]], [[("a", "b")]]] | ||||||
|  |         for coord in invalid_coords: | ||||||
|  |             expected = "Invalid MultiLineString:\nBoth values (%s) in point must be float or int" % repr(coord[0][0]) | ||||||
|  |             self._test_for_expected_error(Location, coord, expected) | ||||||
|  |  | ||||||
|  |         Location(loc=[[[1, 2], [3, 4], [5, 6], [1,2]]]).validate() | ||||||
|  |  | ||||||
|  |     def test_multipolygon_validation(self): | ||||||
|  |         class Location(Document): | ||||||
|  |             loc = MultiPolygonField() | ||||||
|  |  | ||||||
|  |         invalid_coords = {"x": 1, "y": 2} | ||||||
|  |         expected = 'MultiPolygonField can only accept a valid GeoJson dictionary or lists of (x, y)' | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = {"type": "MadeUp", "coordinates": [[]]} | ||||||
|  |         expected = 'MultiPolygonField type must be "MultiPolygon"' | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = {"type": "MultiPolygon", "coordinates": [[[[1, 2, 3]]]]} | ||||||
|  |         expected = "Invalid MultiPolygon:\nValue ([1, 2, 3]) must be a two-dimensional point" | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[[[5, "a"]]]] | ||||||
|  |         expected = "Invalid MultiPolygon:\nBoth values ([5, 'a']) in point must be float or int" | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[[[]]]] | ||||||
|  |         expected = "Invalid MultiPolygon must contain at least one valid Polygon" | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[[[1, 2, 3]]]] | ||||||
|  |         expected = "Invalid MultiPolygon:\nValue ([1, 2, 3]) must be a two-dimensional point" | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[[[{}, {}]]], [[("a", "b")]]] | ||||||
|  |         expected = "Invalid MultiPolygon:\nBoth values ([{}, {}]) in point must be float or int, Both values (('a', 'b')) in point must be float or int" | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         invalid_coords = [[[[1, 2], [3, 4]]]] | ||||||
|  |         expected = "Invalid MultiPolygon:\nLineStrings must start and end at the same point" | ||||||
|  |         self._test_for_expected_error(Location, invalid_coords, expected) | ||||||
|  |  | ||||||
|  |         Location(loc=[[[[1, 2], [3, 4], [5, 6], [1, 2]]]]).validate() | ||||||
|  |  | ||||||
|     def test_indexes_geopoint(self): |     def test_indexes_geopoint(self): | ||||||
|         """Ensure that indexes are created automatically for GeoPointFields. |         """Ensure that indexes are created automatically for GeoPointFields. | ||||||
|         """ |         """ | ||||||
|   | |||||||
| @@ -3,3 +3,4 @@ from field_list import * | |||||||
| from queryset import * | from queryset import * | ||||||
| from visitor import * | from visitor import * | ||||||
| from geo import * | from geo import * | ||||||
|  | from modify import * | ||||||
| @@ -162,6 +162,10 @@ class OnlyExcludeAllTest(unittest.TestCase): | |||||||
|         self.assertEqual(obj.name, person.name) |         self.assertEqual(obj.name, person.name) | ||||||
|         self.assertEqual(obj.age, person.age) |         self.assertEqual(obj.age, person.age) | ||||||
|  |  | ||||||
|  |         obj = self.Person.objects.only(*('id', 'name',)).get() | ||||||
|  |         self.assertEqual(obj.name, person.name) | ||||||
|  |         self.assertEqual(obj.age, None) | ||||||
|  |  | ||||||
|         # Check polymorphism still works |         # Check polymorphism still works | ||||||
|         class Employee(self.Person): |         class Employee(self.Person): | ||||||
|             salary = IntField(db_field='wage') |             salary = IntField(db_field='wage') | ||||||
| @@ -395,5 +399,28 @@ class OnlyExcludeAllTest(unittest.TestCase): | |||||||
|         numbers = Numbers.objects.fields(embedded__n={"$slice": [-5, 10]}).get() |         numbers = Numbers.objects.fields(embedded__n={"$slice": [-5, 10]}).get() | ||||||
|         self.assertEqual(numbers.embedded.n, [-5, -4, -3, -2, -1]) |         self.assertEqual(numbers.embedded.n, [-5, -4, -3, -2, -1]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def test_exclude_from_subclasses_docs(self): | ||||||
|  |  | ||||||
|  |         class Base(Document): | ||||||
|  |             username = StringField() | ||||||
|  |  | ||||||
|  |             meta = {'allow_inheritance': True} | ||||||
|  |  | ||||||
|  |         class Anon(Base): | ||||||
|  |             anon = BooleanField() | ||||||
|  |  | ||||||
|  |         class User(Base): | ||||||
|  |             password = StringField() | ||||||
|  |             wibble = StringField() | ||||||
|  |  | ||||||
|  |         Base.drop_collection() | ||||||
|  |         User(username="mongodb", password="secret").save() | ||||||
|  |  | ||||||
|  |         user = Base.objects().exclude("password", "wibble").first() | ||||||
|  |         self.assertEqual(user.password, None) | ||||||
|  |  | ||||||
|  |         self.assertRaises(LookUpError, Base.objects.exclude, "made_up") | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -5,6 +5,8 @@ import unittest | |||||||
| from datetime import datetime, timedelta | from datetime import datetime, timedelta | ||||||
| from mongoengine import * | from mongoengine import * | ||||||
|  |  | ||||||
|  | from nose.plugins.skip import SkipTest | ||||||
|  |  | ||||||
| __all__ = ("GeoQueriesTest",) | __all__ = ("GeoQueriesTest",) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -139,6 +141,7 @@ class GeoQueriesTest(unittest.TestCase): | |||||||
|     def test_spherical_geospatial_operators(self): |     def test_spherical_geospatial_operators(self): | ||||||
|         """Ensure that spherical geospatial queries are working |         """Ensure that spherical geospatial queries are working | ||||||
|         """ |         """ | ||||||
|  |         raise SkipTest("https://jira.mongodb.org/browse/SERVER-14039") | ||||||
|         class Point(Document): |         class Point(Document): | ||||||
|             location = GeoPointField() |             location = GeoPointField() | ||||||
|  |  | ||||||
| @@ -414,5 +417,47 @@ class GeoQueriesTest(unittest.TestCase): | |||||||
|         roads = Road.objects.filter(poly__geo_intersects={"$geometry": polygon}).count() |         roads = Road.objects.filter(poly__geo_intersects={"$geometry": polygon}).count() | ||||||
|         self.assertEqual(1, roads) |         self.assertEqual(1, roads) | ||||||
|  |  | ||||||
|  |     def test_2dsphere_point_sets_correctly(self): | ||||||
|  |         class Location(Document): | ||||||
|  |             loc = PointField() | ||||||
|  |  | ||||||
|  |         Location.drop_collection() | ||||||
|  |  | ||||||
|  |         Location(loc=[1,2]).save() | ||||||
|  |         loc = Location.objects.as_pymongo()[0] | ||||||
|  |         self.assertEqual(loc["loc"], {"type": "Point", "coordinates": [1, 2]}) | ||||||
|  |  | ||||||
|  |         Location.objects.update(set__loc=[2,1]) | ||||||
|  |         loc = Location.objects.as_pymongo()[0] | ||||||
|  |         self.assertEqual(loc["loc"], {"type": "Point", "coordinates": [2, 1]}) | ||||||
|  |  | ||||||
|  |     def test_2dsphere_linestring_sets_correctly(self): | ||||||
|  |         class Location(Document): | ||||||
|  |             line = LineStringField() | ||||||
|  |  | ||||||
|  |         Location.drop_collection() | ||||||
|  |  | ||||||
|  |         Location(line=[[1, 2], [2, 2]]).save() | ||||||
|  |         loc = Location.objects.as_pymongo()[0] | ||||||
|  |         self.assertEqual(loc["line"], {"type": "LineString", "coordinates": [[1, 2], [2, 2]]}) | ||||||
|  |  | ||||||
|  |         Location.objects.update(set__line=[[2, 1], [1, 2]]) | ||||||
|  |         loc = Location.objects.as_pymongo()[0] | ||||||
|  |         self.assertEqual(loc["line"], {"type": "LineString", "coordinates": [[2, 1], [1, 2]]}) | ||||||
|  |  | ||||||
|  |     def test_geojson_PolygonField(self): | ||||||
|  |         class Location(Document): | ||||||
|  |             poly = PolygonField() | ||||||
|  |  | ||||||
|  |         Location.drop_collection() | ||||||
|  |  | ||||||
|  |         Location(poly=[[[40, 5], [40, 6], [41, 6], [40, 5]]]).save() | ||||||
|  |         loc = Location.objects.as_pymongo()[0] | ||||||
|  |         self.assertEqual(loc["poly"], {"type": "Polygon", "coordinates": [[[40, 5], [40, 6], [41, 6], [40, 5]]]}) | ||||||
|  |  | ||||||
|  |         Location.objects.update(set__poly=[[[40, 4], [40, 6], [41, 6], [40, 4]]]) | ||||||
|  |         loc = Location.objects.as_pymongo()[0] | ||||||
|  |         self.assertEqual(loc["poly"], {"type": "Polygon", "coordinates": [[[40, 4], [40, 6], [41, 6], [40, 4]]]}) | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
							
								
								
									
										102
									
								
								tests/queryset/modify.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								tests/queryset/modify.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | |||||||
|  | import sys | ||||||
|  | sys.path[0:0] = [""] | ||||||
|  |  | ||||||
|  | import unittest | ||||||
|  |  | ||||||
|  | from mongoengine import connect, Document, IntField | ||||||
|  |  | ||||||
|  | __all__ = ("FindAndModifyTest",) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Doc(Document): | ||||||
|  |     id = IntField(primary_key=True) | ||||||
|  |     value = IntField() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FindAndModifyTest(unittest.TestCase): | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         connect(db="mongoenginetest") | ||||||
|  |         Doc.drop_collection() | ||||||
|  |  | ||||||
|  |     def assertDbEqual(self, docs): | ||||||
|  |         self.assertEqual(list(Doc._collection.find().sort("id")), docs) | ||||||
|  |  | ||||||
|  |     def test_modify(self): | ||||||
|  |         Doc(id=0, value=0).save() | ||||||
|  |         doc = Doc(id=1, value=1).save() | ||||||
|  |  | ||||||
|  |         old_doc = Doc.objects(id=1).modify(set__value=-1) | ||||||
|  |         self.assertEqual(old_doc.to_json(), doc.to_json()) | ||||||
|  |         self.assertDbEqual([{"_id": 0, "value": 0}, {"_id": 1, "value": -1}]) | ||||||
|  |  | ||||||
|  |     def test_modify_with_new(self): | ||||||
|  |         Doc(id=0, value=0).save() | ||||||
|  |         doc = Doc(id=1, value=1).save() | ||||||
|  |  | ||||||
|  |         new_doc = Doc.objects(id=1).modify(set__value=-1, new=True) | ||||||
|  |         doc.value = -1 | ||||||
|  |         self.assertEqual(new_doc.to_json(), doc.to_json()) | ||||||
|  |         self.assertDbEqual([{"_id": 0, "value": 0}, {"_id": 1, "value": -1}]) | ||||||
|  |  | ||||||
|  |     def test_modify_not_existing(self): | ||||||
|  |         Doc(id=0, value=0).save() | ||||||
|  |         self.assertEqual(Doc.objects(id=1).modify(set__value=-1), None) | ||||||
|  |         self.assertDbEqual([{"_id": 0, "value": 0}]) | ||||||
|  |  | ||||||
|  |     def test_modify_with_upsert(self): | ||||||
|  |         Doc(id=0, value=0).save() | ||||||
|  |         old_doc = Doc.objects(id=1).modify(set__value=1, upsert=True) | ||||||
|  |         self.assertEqual(old_doc, None) | ||||||
|  |         self.assertDbEqual([{"_id": 0, "value": 0}, {"_id": 1, "value": 1}]) | ||||||
|  |  | ||||||
|  |     def test_modify_with_upsert_existing(self): | ||||||
|  |         Doc(id=0, value=0).save() | ||||||
|  |         doc = Doc(id=1, value=1).save() | ||||||
|  |  | ||||||
|  |         old_doc = Doc.objects(id=1).modify(set__value=-1, upsert=True) | ||||||
|  |         self.assertEqual(old_doc.to_json(), doc.to_json()) | ||||||
|  |         self.assertDbEqual([{"_id": 0, "value": 0}, {"_id": 1, "value": -1}]) | ||||||
|  |  | ||||||
|  |     def test_modify_with_upsert_with_new(self): | ||||||
|  |         Doc(id=0, value=0).save() | ||||||
|  |         new_doc = Doc.objects(id=1).modify(upsert=True, new=True, set__value=1) | ||||||
|  |         self.assertEqual(new_doc.to_mongo(), {"_id": 1, "value": 1}) | ||||||
|  |         self.assertDbEqual([{"_id": 0, "value": 0}, {"_id": 1, "value": 1}]) | ||||||
|  |  | ||||||
|  |     def test_modify_with_remove(self): | ||||||
|  |         Doc(id=0, value=0).save() | ||||||
|  |         doc = Doc(id=1, value=1).save() | ||||||
|  |  | ||||||
|  |         old_doc = Doc.objects(id=1).modify(remove=True) | ||||||
|  |         self.assertEqual(old_doc.to_json(), doc.to_json()) | ||||||
|  |         self.assertDbEqual([{"_id": 0, "value": 0}]) | ||||||
|  |  | ||||||
|  |     def test_find_and_modify_with_remove_not_existing(self): | ||||||
|  |         Doc(id=0, value=0).save() | ||||||
|  |         self.assertEqual(Doc.objects(id=1).modify(remove=True), None) | ||||||
|  |         self.assertDbEqual([{"_id": 0, "value": 0}]) | ||||||
|  |  | ||||||
|  |     def test_modify_with_order_by(self): | ||||||
|  |         Doc(id=0, value=3).save() | ||||||
|  |         Doc(id=1, value=2).save() | ||||||
|  |         Doc(id=2, value=1).save() | ||||||
|  |         doc = Doc(id=3, value=0).save() | ||||||
|  |  | ||||||
|  |         old_doc = Doc.objects().order_by("-id").modify(set__value=-1) | ||||||
|  |         self.assertEqual(old_doc.to_json(), doc.to_json()) | ||||||
|  |         self.assertDbEqual([ | ||||||
|  |             {"_id": 0, "value": 3}, {"_id": 1, "value": 2}, | ||||||
|  |             {"_id": 2, "value": 1}, {"_id": 3, "value": -1}]) | ||||||
|  |  | ||||||
|  |     def test_modify_with_fields(self): | ||||||
|  |         Doc(id=0, value=0).save() | ||||||
|  |         Doc(id=1, value=1).save() | ||||||
|  |  | ||||||
|  |         old_doc = Doc.objects(id=1).only("id").modify(set__value=-1) | ||||||
|  |         self.assertEqual(old_doc.to_mongo(), {"_id": 1}) | ||||||
|  |         self.assertDbEqual([{"_id": 0, "value": 0}, {"_id": 1, "value": -1}]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     unittest.main() | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -31,6 +31,31 @@ class TransformTest(unittest.TestCase): | |||||||
|         self.assertEqual(transform.query(name__exists=True), |         self.assertEqual(transform.query(name__exists=True), | ||||||
|                          {'name': {'$exists': True}}) |                          {'name': {'$exists': True}}) | ||||||
|  |  | ||||||
|  |     def test_transform_update(self): | ||||||
|  |         class DicDoc(Document): | ||||||
|  |             dictField = DictField() | ||||||
|  |  | ||||||
|  |         class Doc(Document): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         DicDoc.drop_collection() | ||||||
|  |         Doc.drop_collection() | ||||||
|  |  | ||||||
|  |         doc = Doc().save() | ||||||
|  |         dic_doc = DicDoc().save() | ||||||
|  |  | ||||||
|  |         for k, v in (("set", "$set"), ("set_on_insert", "$setOnInsert"), ("push", "$push")): | ||||||
|  |             update = transform.update(DicDoc, **{"%s__dictField__test" % k: doc}) | ||||||
|  |             self.assertTrue(isinstance(update[v]["dictField.test"], dict)) | ||||||
|  |  | ||||||
|  |         # Update special cases | ||||||
|  |         update = transform.update(DicDoc, unset__dictField__test=doc) | ||||||
|  |         self.assertEqual(update["$unset"]["dictField.test"], 1) | ||||||
|  |  | ||||||
|  |         update = transform.update(DicDoc, pull__dictField__test=doc) | ||||||
|  |         self.assertTrue(isinstance(update["$pull"]["dictField"]["test"], dict)) | ||||||
|  |  | ||||||
|  |  | ||||||
|     def test_query_field_name(self): |     def test_query_field_name(self): | ||||||
|         """Ensure that the correct field name is used when querying. |         """Ensure that the correct field name is used when querying. | ||||||
|         """ |         """ | ||||||
| @@ -142,6 +167,35 @@ class TransformTest(unittest.TestCase): | |||||||
|                              {'attachments.views.extracted': 'no'}]} |                              {'attachments.views.extracted': 'no'}]} | ||||||
|         self.assertEqual(expected, raw_query) |         self.assertEqual(expected, raw_query) | ||||||
|  |  | ||||||
|  |     def test_geojson_PointField(self): | ||||||
|  |         class Location(Document): | ||||||
|  |             loc = PointField() | ||||||
|  |  | ||||||
|  |         update = transform.update(Location, set__loc=[1, 2]) | ||||||
|  |         self.assertEqual(update, {'$set': {'loc': {"type": "Point", "coordinates": [1,2]}}}) | ||||||
|  |  | ||||||
|  |         update = transform.update(Location, set__loc={"type": "Point", "coordinates": [1,2]}) | ||||||
|  |         self.assertEqual(update, {'$set': {'loc': {"type": "Point", "coordinates": [1,2]}}}) | ||||||
|  |  | ||||||
|  |     def test_geojson_LineStringField(self): | ||||||
|  |         class Location(Document): | ||||||
|  |             line = LineStringField() | ||||||
|  |  | ||||||
|  |         update = transform.update(Location, set__line=[[1, 2], [2, 2]]) | ||||||
|  |         self.assertEqual(update, {'$set': {'line': {"type": "LineString", "coordinates": [[1, 2], [2, 2]]}}}) | ||||||
|  |  | ||||||
|  |         update = transform.update(Location, set__line={"type": "LineString", "coordinates": [[1, 2], [2, 2]]}) | ||||||
|  |         self.assertEqual(update, {'$set': {'line': {"type": "LineString", "coordinates": [[1, 2], [2, 2]]}}}) | ||||||
|  |  | ||||||
|  |     def test_geojson_PolygonField(self): | ||||||
|  |         class Location(Document): | ||||||
|  |             poly = PolygonField() | ||||||
|  |  | ||||||
|  |         update = transform.update(Location, set__poly=[[[40, 5], [40, 6], [41, 6], [40, 5]]]) | ||||||
|  |         self.assertEqual(update, {'$set': {'poly': {"type": "Polygon", "coordinates": [[[40, 5], [40, 6], [41, 6], [40, 5]]]}}}) | ||||||
|  |  | ||||||
|  |         update = transform.update(Location, set__poly={"type": "Polygon", "coordinates": [[[40, 5], [40, 6], [41, 6], [40, 5]]]}) | ||||||
|  |         self.assertEqual(update, {'$set': {'poly': {"type": "Polygon", "coordinates": [[[40, 5], [40, 6], [41, 6], [40, 5]]]}}}) | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -1,6 +1,11 @@ | |||||||
| import sys | import sys | ||||||
| sys.path[0:0] = [""] | sys.path[0:0] = [""] | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     import unittest2 as unittest | ||||||
|  | except ImportError: | ||||||
|     import unittest |     import unittest | ||||||
|  |  | ||||||
| import datetime | import datetime | ||||||
|  |  | ||||||
| import pymongo | import pymongo | ||||||
| @@ -34,6 +39,17 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         conn = get_connection('testdb') |         conn = get_connection('testdb') | ||||||
|         self.assertTrue(isinstance(conn, pymongo.mongo_client.MongoClient)) |         self.assertTrue(isinstance(conn, pymongo.mongo_client.MongoClient)) | ||||||
|  |  | ||||||
|  |     def test_sharing_connections(self): | ||||||
|  |         """Ensure that connections are shared when the connection settings are exactly the same | ||||||
|  |         """ | ||||||
|  |         connect('mongoenginetest', alias='testdb1') | ||||||
|  |  | ||||||
|  |         expected_connection = get_connection('testdb1') | ||||||
|  |  | ||||||
|  |         connect('mongoenginetest', alias='testdb2') | ||||||
|  |         actual_connection = get_connection('testdb2') | ||||||
|  |         self.assertEqual(expected_connection, actual_connection) | ||||||
|  |  | ||||||
|     def test_connect_uri(self): |     def test_connect_uri(self): | ||||||
|         """Ensure that the connect() method works properly with uri's |         """Ensure that the connect() method works properly with uri's | ||||||
|         """ |         """ | ||||||
| @@ -59,6 +75,32 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         c.admin.system.users.remove({}) |         c.admin.system.users.remove({}) | ||||||
|         c.mongoenginetest.system.users.remove({}) |         c.mongoenginetest.system.users.remove({}) | ||||||
|  |  | ||||||
|  |     def test_connect_uri_without_db(self): | ||||||
|  |         """Ensure that the connect() method works properly with uri's | ||||||
|  |         without database_name | ||||||
|  |         """ | ||||||
|  |         c = connect(db='mongoenginetest', alias='admin') | ||||||
|  |         c.admin.system.users.remove({}) | ||||||
|  |         c.mongoenginetest.system.users.remove({}) | ||||||
|  |  | ||||||
|  |         c.admin.add_user("admin", "password") | ||||||
|  |         c.admin.authenticate("admin", "password") | ||||||
|  |         c.mongoenginetest.add_user("username", "password") | ||||||
|  |  | ||||||
|  |         self.assertRaises(ConnectionError, connect, "testdb_uri_bad", host='mongodb://test:password@localhost') | ||||||
|  |  | ||||||
|  |         connect("mongoenginetest", host='mongodb://localhost/') | ||||||
|  |  | ||||||
|  |         conn = get_connection() | ||||||
|  |         self.assertTrue(isinstance(conn, pymongo.mongo_client.MongoClient)) | ||||||
|  |  | ||||||
|  |         db = get_db() | ||||||
|  |         self.assertTrue(isinstance(db, pymongo.database.Database)) | ||||||
|  |         self.assertEqual(db.name, 'mongoenginetest') | ||||||
|  |  | ||||||
|  |         c.admin.system.users.remove({}) | ||||||
|  |         c.mongoenginetest.system.users.remove({}) | ||||||
|  |  | ||||||
|     def test_register_connection(self): |     def test_register_connection(self): | ||||||
|         """Ensure that connections with different aliases may be registered. |         """Ensure that connections with different aliases may be registered. | ||||||
|         """ |         """ | ||||||
| @@ -72,6 +114,14 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         self.assertTrue(isinstance(db, pymongo.database.Database)) |         self.assertTrue(isinstance(db, pymongo.database.Database)) | ||||||
|         self.assertEqual(db.name, 'mongoenginetest2') |         self.assertEqual(db.name, 'mongoenginetest2') | ||||||
|  |  | ||||||
|  |     def test_register_connection_defaults(self): | ||||||
|  |         """Ensure that defaults are used when the host and port are None. | ||||||
|  |         """ | ||||||
|  |         register_connection('testdb', 'mongoenginetest', host=None, port=None) | ||||||
|  |  | ||||||
|  |         conn = get_connection('testdb') | ||||||
|  |         self.assertTrue(isinstance(conn, pymongo.mongo_client.MongoClient)) | ||||||
|  |  | ||||||
|     def test_connection_kwargs(self): |     def test_connection_kwargs(self): | ||||||
|         """Ensure that connection kwargs get passed to pymongo. |         """Ensure that connection kwargs get passed to pymongo. | ||||||
|         """ |         """ | ||||||
| @@ -97,6 +147,18 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         date_doc = DateDoc.objects.first() |         date_doc = DateDoc.objects.first() | ||||||
|         self.assertEqual(d, date_doc.the_date) |         self.assertEqual(d, date_doc.the_date) | ||||||
|  |  | ||||||
|  |     def test_multiple_connection_settings(self): | ||||||
|  |         connect('mongoenginetest', alias='t1', host="localhost") | ||||||
|  |  | ||||||
|  |         connect('mongoenginetest2', alias='t2', host="127.0.0.1") | ||||||
|  |  | ||||||
|  |         mongo_connections = mongoengine.connection._connections | ||||||
|  |         self.assertEqual(len(mongo_connections.items()), 2) | ||||||
|  |         self.assertTrue('t1' in mongo_connections.keys()) | ||||||
|  |         self.assertTrue('t2' in mongo_connections.keys()) | ||||||
|  |         self.assertEqual(mongo_connections['t1'].host, 'localhost') | ||||||
|  |         self.assertEqual(mongo_connections['t2'].host, '127.0.0.1') | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
							
								
								
									
										107
									
								
								tests/test_datastructures.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								tests/test_datastructures.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,107 @@ | |||||||
|  | import unittest | ||||||
|  | from mongoengine.base.datastructures import StrictDict, SemiStrictDict  | ||||||
|  |  | ||||||
|  | class TestStrictDict(unittest.TestCase): | ||||||
|  |     def strict_dict_class(self, *args, **kwargs): | ||||||
|  |         return StrictDict.create(*args, **kwargs) | ||||||
|  |     def setUp(self): | ||||||
|  |         self.dtype = self.strict_dict_class(("a", "b", "c")) | ||||||
|  |     def test_init(self): | ||||||
|  |         d = self.dtype(a=1, b=1, c=1) | ||||||
|  |         self.assertEqual((d.a, d.b, d.c), (1, 1, 1)) | ||||||
|  |  | ||||||
|  |     def test_init_fails_on_nonexisting_attrs(self): | ||||||
|  |         self.assertRaises(AttributeError, lambda: self.dtype(a=1, b=2, d=3)) | ||||||
|  |          | ||||||
|  |     def test_eq(self): | ||||||
|  |         d = self.dtype(a=1, b=1, c=1) | ||||||
|  |         dd = self.dtype(a=1, b=1, c=1) | ||||||
|  |         e = self.dtype(a=1, b=1, c=3) | ||||||
|  |         f = self.dtype(a=1, b=1) | ||||||
|  |         g = self.strict_dict_class(("a", "b", "c", "d"))(a=1, b=1, c=1, d=1) | ||||||
|  |         h = self.strict_dict_class(("a", "c", "b"))(a=1, b=1, c=1) | ||||||
|  |         i = self.strict_dict_class(("a", "c", "b"))(a=1, b=1, c=2) | ||||||
|  |          | ||||||
|  |         self.assertEqual(d, dd) | ||||||
|  |         self.assertNotEqual(d, e) | ||||||
|  |         self.assertNotEqual(d, f) | ||||||
|  |         self.assertNotEqual(d, g) | ||||||
|  |         self.assertNotEqual(f, d) | ||||||
|  |         self.assertEqual(d, h) | ||||||
|  |         self.assertNotEqual(d, i) | ||||||
|  |  | ||||||
|  |     def test_setattr_getattr(self): | ||||||
|  |         d = self.dtype() | ||||||
|  |         d.a = 1 | ||||||
|  |         self.assertEqual(d.a, 1) | ||||||
|  |         self.assertRaises(AttributeError, lambda: d.b) | ||||||
|  |      | ||||||
|  |     def test_setattr_raises_on_nonexisting_attr(self): | ||||||
|  |         d = self.dtype() | ||||||
|  |         def _f(): | ||||||
|  |             d.x=1 | ||||||
|  |         self.assertRaises(AttributeError, _f) | ||||||
|  |      | ||||||
|  |     def test_setattr_getattr_special(self): | ||||||
|  |         d = self.strict_dict_class(["items"]) | ||||||
|  |         d.items = 1 | ||||||
|  |         self.assertEqual(d.items, 1) | ||||||
|  |      | ||||||
|  |     def test_get(self): | ||||||
|  |         d = self.dtype(a=1) | ||||||
|  |         self.assertEqual(d.get('a'), 1) | ||||||
|  |         self.assertEqual(d.get('b', 'bla'), 'bla') | ||||||
|  |  | ||||||
|  |     def test_items(self): | ||||||
|  |         d = self.dtype(a=1) | ||||||
|  |         self.assertEqual(d.items(), [('a', 1)]) | ||||||
|  |         d = self.dtype(a=1, b=2) | ||||||
|  |         self.assertEqual(d.items(), [('a', 1), ('b', 2)]) | ||||||
|  |  | ||||||
|  |     def test_mappings_protocol(self): | ||||||
|  |         d = self.dtype(a=1, b=2) | ||||||
|  |         assert dict(d) == {'a': 1, 'b': 2} | ||||||
|  |         assert dict(**d) == {'a': 1, 'b': 2} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestSemiSrictDict(TestStrictDict): | ||||||
|  |     def strict_dict_class(self, *args, **kwargs): | ||||||
|  |         return SemiStrictDict.create(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     def test_init_fails_on_nonexisting_attrs(self): | ||||||
|  |         # disable irrelevant test | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     def test_setattr_raises_on_nonexisting_attr(self): | ||||||
|  |         # disable irrelevant test | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     def test_setattr_getattr_nonexisting_attr_succeeds(self): | ||||||
|  |         d = self.dtype() | ||||||
|  |         d.x = 1 | ||||||
|  |         self.assertEqual(d.x, 1) | ||||||
|  |  | ||||||
|  |     def test_init_succeeds_with_nonexisting_attrs(self): | ||||||
|  |         d = self.dtype(a=1, b=1, c=1, x=2) | ||||||
|  |         self.assertEqual((d.a, d.b, d.c, d.x), (1, 1, 1, 2)) | ||||||
|  |     | ||||||
|  |     def test_iter_with_nonexisting_attrs(self): | ||||||
|  |         d = self.dtype(a=1, b=1, c=1, x=2) | ||||||
|  |         self.assertEqual(list(d), ['a', 'b', 'c', 'x']) | ||||||
|  |  | ||||||
|  |     def test_iteritems_with_nonexisting_attrs(self): | ||||||
|  |         d = self.dtype(a=1, b=1, c=1, x=2) | ||||||
|  |         self.assertEqual(list(d.iteritems()), [('a', 1), ('b', 1), ('c', 1), ('x', 2)]) | ||||||
|  |  | ||||||
|  |     def tets_cmp_with_strict_dicts(self): | ||||||
|  |         d = self.dtype(a=1, b=1, c=1) | ||||||
|  |         dd = StrictDict.create(("a", "b", "c"))(a=1, b=1, c=1) | ||||||
|  |         self.assertEqual(d, dd) | ||||||
|  |  | ||||||
|  |     def test_cmp_with_strict_dict_with_nonexisting_attrs(self): | ||||||
|  |         d = self.dtype(a=1, b=1, c=1, x=2) | ||||||
|  |         dd = StrictDict.create(("a", "b", "c", "x"))(a=1, b=1, c=1, x=2) | ||||||
|  |         self.assertEqual(d, dd) | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     unittest.main() | ||||||
| @@ -291,6 +291,30 @@ class FieldTest(unittest.TestCase): | |||||||
|                 self.assertEqual(employee.friends, friends) |                 self.assertEqual(employee.friends, friends) | ||||||
|                 self.assertEqual(q, 2) |                 self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |     def test_list_of_lists_of_references(self): | ||||||
|  |  | ||||||
|  |         class User(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Post(Document): | ||||||
|  |             user_lists = ListField(ListField(ReferenceField(User))) | ||||||
|  |  | ||||||
|  |         class SimpleList(Document): | ||||||
|  |             users = ListField(ReferenceField(User)) | ||||||
|  |  | ||||||
|  |         User.drop_collection() | ||||||
|  |         Post.drop_collection() | ||||||
|  |  | ||||||
|  |         u1 = User.objects.create(name='u1') | ||||||
|  |         u2 = User.objects.create(name='u2') | ||||||
|  |         u3 = User.objects.create(name='u3') | ||||||
|  |  | ||||||
|  |         SimpleList.objects.create(users=[u1, u2, u3]) | ||||||
|  |         self.assertEqual(SimpleList.objects.all()[0].users, [u1, u2, u3]) | ||||||
|  |  | ||||||
|  |         Post.objects.create(user_lists=[[u1, u2], [u3]]) | ||||||
|  |         self.assertEqual(Post.objects.all()[0].user_lists, [[u1, u2], [u3]]) | ||||||
|  |  | ||||||
|     def test_circular_reference(self): |     def test_circular_reference(self): | ||||||
|         """Ensure you can handle circular references |         """Ensure you can handle circular references | ||||||
|         """ |         """ | ||||||
| @@ -1171,6 +1195,54 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         self.assertEqual(2, len([brand for bg in brand_groups for brand in bg.brands])) |         self.assertEqual(2, len([brand for bg in brand_groups for brand in bg.brands])) | ||||||
|  |  | ||||||
|  |     def test_dereferencing_embedded_listfield_referencefield(self): | ||||||
|  |         class Tag(Document): | ||||||
|  |             meta = {'collection': 'tags'} | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Post(EmbeddedDocument): | ||||||
|  |             body = StringField() | ||||||
|  |             tags = ListField(ReferenceField("Tag", dbref=True)) | ||||||
|  |  | ||||||
|  |         class Page(Document): | ||||||
|  |             meta = {'collection': 'pages'} | ||||||
|  |             tags = ListField(ReferenceField("Tag", dbref=True)) | ||||||
|  |             posts = ListField(EmbeddedDocumentField(Post)) | ||||||
|  |  | ||||||
|  |         Tag.drop_collection() | ||||||
|  |         Page.drop_collection() | ||||||
|  |  | ||||||
|  |         tag = Tag(name='test').save() | ||||||
|  |         post = Post(body='test body', tags=[tag]) | ||||||
|  |         Page(tags=[tag], posts=[post]).save() | ||||||
|  |  | ||||||
|  |         page = Page.objects.first() | ||||||
|  |         self.assertEqual(page.tags[0], page.posts[0].tags[0]) | ||||||
|  |  | ||||||
|  |     def test_select_related_follows_embedded_referencefields(self): | ||||||
|  |         class Playlist(Document): | ||||||
|  |             items = ListField(EmbeddedDocumentField("PlaylistItem")) | ||||||
|  |  | ||||||
|  |         class PlaylistItem(EmbeddedDocument): | ||||||
|  |             song = ReferenceField("Song") | ||||||
|  |  | ||||||
|  |         class Song(Document): | ||||||
|  |             title = StringField() | ||||||
|  |  | ||||||
|  |         Playlist.drop_collection() | ||||||
|  |         Song.drop_collection() | ||||||
|  |  | ||||||
|  |         songs = [Song.objects.create(title="song %d" % i) for i in range(3)] | ||||||
|  |         items = [PlaylistItem(song=song) for song in songs] | ||||||
|  |         playlist = Playlist.objects.create(items=items) | ||||||
|  |  | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             playlist = Playlist.objects.first().select_related() | ||||||
|  |             songs = [item.song for item in playlist.items] | ||||||
|  |  | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,11 +2,11 @@ import sys | |||||||
| sys.path[0:0] = [""] | sys.path[0:0] = [""] | ||||||
| import unittest | import unittest | ||||||
| from nose.plugins.skip import SkipTest | from nose.plugins.skip import SkipTest | ||||||
|  |  | ||||||
| from mongoengine import * | from mongoengine import * | ||||||
|  |  | ||||||
|  |  | ||||||
| from mongoengine.django.shortcuts import get_document_or_404 | from mongoengine.django.shortcuts import get_document_or_404 | ||||||
|  |  | ||||||
|  | import django | ||||||
| from django.http import Http404 | from django.http import Http404 | ||||||
| from django.template import Context, Template | from django.template import Context, Template | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| @@ -16,19 +16,27 @@ settings.configure( | |||||||
|     USE_TZ=True, |     USE_TZ=True, | ||||||
|     INSTALLED_APPS=('django.contrib.auth', 'mongoengine.django.mongo_auth'), |     INSTALLED_APPS=('django.contrib.auth', 'mongoengine.django.mongo_auth'), | ||||||
|     AUTH_USER_MODEL=('mongo_auth.MongoUser'), |     AUTH_USER_MODEL=('mongo_auth.MongoUser'), | ||||||
|  |     AUTHENTICATION_BACKENDS = ('mongoengine.django.auth.MongoEngineBackend',) | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | # For Django >= 1.7 | ||||||
|  | if hasattr(django, 'setup'): | ||||||
|  |     django.setup() | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     from django.contrib.auth import authenticate, get_user_model |     from django.contrib.auth import authenticate, get_user_model | ||||||
|     from mongoengine.django.auth import User |     from mongoengine.django.auth import User | ||||||
|     from mongoengine.django.mongo_auth.models import MongoUser, MongoUserManager |     from mongoengine.django.mongo_auth.models import ( | ||||||
|  |         MongoUser, | ||||||
|  |         MongoUserManager, | ||||||
|  |         get_user_document, | ||||||
|  |     ) | ||||||
|     DJ15 = True |     DJ15 = True | ||||||
| except Exception: | except Exception: | ||||||
|     DJ15 = False |     DJ15 = False | ||||||
| from django.contrib.sessions.tests import SessionTestsMixin | from django.contrib.sessions.tests import SessionTestsMixin | ||||||
| from mongoengine.django.sessions import SessionStore, MongoSession | from mongoengine.django.sessions import SessionStore, MongoSession | ||||||
|  | from mongoengine.django.tests import MongoTestCase | ||||||
|  |  | ||||||
| from datetime import tzinfo, timedelta | from datetime import tzinfo, timedelta | ||||||
| ZERO = timedelta(0) | ZERO = timedelta(0) | ||||||
|  |  | ||||||
| @@ -165,6 +173,8 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         class Note(Document): |         class Note(Document): | ||||||
|             text = StringField() |             text = StringField() | ||||||
|  |  | ||||||
|  |         Note.drop_collection() | ||||||
|  |  | ||||||
|         for i in xrange(1, 101): |         for i in xrange(1, 101): | ||||||
|             Note(name="Note: %s" % i).save() |             Note(name="Note: %s" % i).save() | ||||||
|  |  | ||||||
| @@ -258,9 +268,12 @@ class MongoAuthTest(unittest.TestCase): | |||||||
|         User.drop_collection() |         User.drop_collection() | ||||||
|         super(MongoAuthTest, self).setUp() |         super(MongoAuthTest, self).setUp() | ||||||
|  |  | ||||||
|     def test_user_model(self): |     def test_get_user_model(self): | ||||||
|         self.assertEqual(get_user_model(), MongoUser) |         self.assertEqual(get_user_model(), MongoUser) | ||||||
|  |  | ||||||
|  |     def test_get_user_document(self): | ||||||
|  |         self.assertEqual(get_user_document(), User) | ||||||
|  |  | ||||||
|     def test_user_manager(self): |     def test_user_manager(self): | ||||||
|         manager = get_user_model()._default_manager |         manager = get_user_model()._default_manager | ||||||
|         self.assertTrue(isinstance(manager, MongoUserManager)) |         self.assertTrue(isinstance(manager, MongoUserManager)) | ||||||
| @@ -285,5 +298,11 @@ class MongoAuthTest(unittest.TestCase): | |||||||
|         db_user = User.objects.get(username='user') |         db_user = User.objects.get(username='user') | ||||||
|         self.assertEqual(user.id, db_user.id) |         self.assertEqual(user.id, db_user.id) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MongoTestCaseTest(MongoTestCase): | ||||||
|  |     def test_mongo_test_case(self): | ||||||
|  |         self.db.dummy_collection.insert({'collection': 'will be dropped'}) | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -37,7 +37,8 @@ class SignalTests(unittest.TestCase): | |||||||
|  |  | ||||||
|             @classmethod |             @classmethod | ||||||
|             def post_init(cls, sender, document, **kwargs): |             def post_init(cls, sender, document, **kwargs): | ||||||
|                 signal_output.append('post_init signal, %s' % document) |                 signal_output.append('post_init signal, %s, document._created = %s' % (document, document._created)) | ||||||
|  |  | ||||||
|  |  | ||||||
|             @classmethod |             @classmethod | ||||||
|             def pre_save(cls, sender, document, **kwargs): |             def pre_save(cls, sender, document, **kwargs): | ||||||
| @@ -54,7 +55,9 @@ class SignalTests(unittest.TestCase): | |||||||
|  |  | ||||||
|             @classmethod |             @classmethod | ||||||
|             def post_save(cls, sender, document, **kwargs): |             def post_save(cls, sender, document, **kwargs): | ||||||
|  |                 dirty_keys = document._delta()[0].keys() + document._delta()[1].keys() | ||||||
|                 signal_output.append('post_save signal, %s' % document) |                 signal_output.append('post_save signal, %s' % document) | ||||||
|  |                 signal_output.append('post_save dirty keys, %s' % dirty_keys) | ||||||
|                 if 'created' in kwargs: |                 if 'created' in kwargs: | ||||||
|                     if kwargs['created']: |                     if kwargs['created']: | ||||||
|                         signal_output.append('Is created') |                         signal_output.append('Is created') | ||||||
| @@ -191,10 +194,16 @@ class SignalTests(unittest.TestCase): | |||||||
|             a1 = self.Author(name='Bill Shakespeare') |             a1 = self.Author(name='Bill Shakespeare') | ||||||
|             self.Author.objects.insert([a1], load_bulk=False) |             self.Author.objects.insert([a1], load_bulk=False) | ||||||
|  |  | ||||||
|  |         def load_existing_author(): | ||||||
|  |             a  = self.Author(name='Bill Shakespeare') | ||||||
|  |             a.save() | ||||||
|  |             self.get_signal_output(lambda: None) # eliminate signal output | ||||||
|  |             a1 = self.Author.objects(name='Bill Shakespeare')[0] | ||||||
|  |          | ||||||
|         self.assertEqual(self.get_signal_output(create_author), [ |         self.assertEqual(self.get_signal_output(create_author), [ | ||||||
|             "pre_init signal, Author", |             "pre_init signal, Author", | ||||||
|             "{'name': 'Bill Shakespeare'}", |             "{'name': 'Bill Shakespeare'}", | ||||||
|             "post_init signal, Bill Shakespeare", |             "post_init signal, Bill Shakespeare, document._created = True", | ||||||
|         ]) |         ]) | ||||||
|  |  | ||||||
|         a1 = self.Author(name='Bill Shakespeare') |         a1 = self.Author(name='Bill Shakespeare') | ||||||
| @@ -203,6 +212,7 @@ class SignalTests(unittest.TestCase): | |||||||
|             "pre_save_post_validation signal, Bill Shakespeare", |             "pre_save_post_validation signal, Bill Shakespeare", | ||||||
|             "Is created", |             "Is created", | ||||||
|             "post_save signal, Bill Shakespeare", |             "post_save signal, Bill Shakespeare", | ||||||
|  |             "post_save dirty keys, ['name']", | ||||||
|             "Is created" |             "Is created" | ||||||
|         ]) |         ]) | ||||||
|  |  | ||||||
| @@ -213,6 +223,7 @@ class SignalTests(unittest.TestCase): | |||||||
|             "pre_save_post_validation signal, William Shakespeare", |             "pre_save_post_validation signal, William Shakespeare", | ||||||
|             "Is updated", |             "Is updated", | ||||||
|             "post_save signal, William Shakespeare", |             "post_save signal, William Shakespeare", | ||||||
|  |             "post_save dirty keys, ['name']", | ||||||
|             "Is updated" |             "Is updated" | ||||||
|         ]) |         ]) | ||||||
|  |  | ||||||
| @@ -221,12 +232,22 @@ class SignalTests(unittest.TestCase): | |||||||
|             'post_delete signal, William Shakespeare', |             'post_delete signal, William Shakespeare', | ||||||
|         ]) |         ]) | ||||||
|  |  | ||||||
|  |         signal_output = self.get_signal_output(load_existing_author) | ||||||
|  |         # test signal_output lines separately, because of random ObjectID after object load | ||||||
|  |         self.assertEqual(signal_output[0], | ||||||
|  |             "pre_init signal, Author", | ||||||
|  |         ) | ||||||
|  |         self.assertEqual(signal_output[2], | ||||||
|  |             "post_init signal, Bill Shakespeare, document._created = False", | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|         signal_output = self.get_signal_output(bulk_create_author_with_load) |         signal_output = self.get_signal_output(bulk_create_author_with_load) | ||||||
|  |  | ||||||
|         # The output of this signal is not entirely deterministic. The reloaded |         # The output of this signal is not entirely deterministic. The reloaded | ||||||
|         # object will have an object ID. Hence, we only check part of the output |         # object will have an object ID. Hence, we only check part of the output | ||||||
|         self.assertEqual(signal_output[3], |         self.assertEqual(signal_output[3], "pre_bulk_insert signal, [<Author: Bill Shakespeare>]" | ||||||
|             "pre_bulk_insert signal, [<Author: Bill Shakespeare>]") |         ) | ||||||
|         self.assertEqual(signal_output[-2:], |         self.assertEqual(signal_output[-2:], | ||||||
|             ["post_bulk_insert signal, [<Author: Bill Shakespeare>]", |             ["post_bulk_insert signal, [<Author: Bill Shakespeare>]", | ||||||
|              "Is loaded",]) |              "Is loaded",]) | ||||||
| @@ -234,7 +255,7 @@ class SignalTests(unittest.TestCase): | |||||||
|         self.assertEqual(self.get_signal_output(bulk_create_author_without_load), [ |         self.assertEqual(self.get_signal_output(bulk_create_author_without_load), [ | ||||||
|             "pre_init signal, Author", |             "pre_init signal, Author", | ||||||
|             "{'name': 'Bill Shakespeare'}", |             "{'name': 'Bill Shakespeare'}", | ||||||
|             "post_init signal, Bill Shakespeare", |             "post_init signal, Bill Shakespeare, document._created = True", | ||||||
|             "pre_bulk_insert signal, [<Author: Bill Shakespeare>]", |             "pre_bulk_insert signal, [<Author: Bill Shakespeare>]", | ||||||
|             "post_bulk_insert signal, [<Author: Bill Shakespeare>]", |             "post_bulk_insert signal, [<Author: Bill Shakespeare>]", | ||||||
|             "Not loaded", |             "Not loaded", | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user