mirror of
				https://github.com/ae-utbm/sith.git
				synced 2025-10-25 14:13:53 +00:00 
			
		
		
		
	Compare commits
	
		
			1381 Commits
		
	
	
		
			b49f204e20
			...
			windows-up
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 1d03fcf6ea | |||
| a6ba65a494 | |||
| c90fcc838e | |||
|  | bc9cb9b36c | ||
| edafc06c3f | |||
|  | 134f8a7989 | ||
| 771cbdbd77 | |||
| a491baddb9 | |||
| 8d10a5e0ab | |||
| cbe42d3a60 | |||
| 0c4d72e17a | |||
|  | 429df81ec9 | ||
| bb24516474 | |||
|  | 8e339c3d4b | ||
| 25298518bc | |||
|  | 2e26ff2cde | ||
| a8702d4f5e | |||
|  | 7f4cc5fb0f | ||
|  | e7215be00e | ||
|  | 4f35cc00bc | ||
|  | af47587116 | ||
|  | 3c4daeadb0 | ||
|  | 348ab19ac6 | ||
|  | ada74a3e42 | ||
|  | 785ac9bdab | ||
|  | d1e604e7a5 | ||
| 2749a88704 | |||
| eb3db134f8 | |||
| fa7f5d24b0 | |||
| ba76015c71 | |||
| 1887a2790f | |||
| 5d0fc38107 | |||
| 65df55a635 | |||
| a60e1f1fdc | |||
| 0a0f44607e | |||
| 007080ee48 | |||
| a13e3e95b7 | |||
| 169938e1da | |||
| e5fb875968 | |||
| 9bd14f1b4e | |||
| fd2295119d | |||
| eac2709e86 | |||
| 48f6d134bf | |||
| 6d7467e746 | |||
| 0d1629495b | |||
| 63839dc22b | |||
|  | c627944bd1 | ||
|  | f0be4b270b | ||
|  | 728065e771 | ||
|  | 849fac490d | ||
|  | 5752229312 | ||
|  | 6eb860579a | ||
|  | d08d54b4c9 | ||
|  | bb210f8d47 | ||
|  | efca10e252 | ||
|  | b8f851b009 | ||
|  | 1e29ae4171 | ||
|  | 0ae1e850f4 | ||
|  | d380668c0f | ||
|  | 9a72c5eb72 | ||
|  | 407cfbe02b | ||
|  | 6400b2c2c2 | ||
|  | cce7ecbe73 | ||
|  | d200c1e381 | ||
|  | 2f9e5bfee1 | ||
|  | 11702d3d7c | ||
| 43f47e2087 | |||
|  | 4b881903f0 | ||
|  | 761e37ade6 | ||
|  | 10ed2f7404 | ||
| 43768f1691 | |||
| 280d27343d | |||
| 138e1662c7 | |||
| c80fe094a2 | |||
| 139221dd22 | |||
|  | 72c2981d66 | ||
| 6f003ffa53 | |||
| 7f6fd7dc47 | |||
| ccf5118c9d | |||
| 022c19c020 | |||
| 2e5e217842 | |||
| 9c93c004ec | |||
| 472800eff6 | |||
| b8d43a629b | |||
| f6693e12cf | |||
| 38f491cf57 | |||
| 3464d5d860 | |||
|  | 81773dc800 | ||
|  | da400155eb | ||
| 5079938a5b | |||
| b8430adc50 | |||
| eed434aeb2 | |||
| 372470b44b | |||
| 7071553c3b | |||
| eea237b813 | |||
| c37288c285 | |||
| ccf5767a01 | |||
| ffe6fc8c2a | |||
| 5f0b4d2050 | |||
| f9d7dc7d3a | |||
| 8ebea00896 | |||
| a548f4744e | |||
| a383f3e717 | |||
| 60f18669c8 | |||
| a36946529b | |||
|  | eaac0c728f | ||
|  | 9ca95774a3 | ||
|  | fa66851889 | ||
|  | ab81f11199 | ||
|  | bea7741d35 | ||
|  | 81e163812e | ||
|  | 4f233538e0 | ||
| 4ac09ac08b | |||
|  | 6d02970676 | ||
|  | accf1befce | ||
|  | 6953eaa9d0 | ||
|  | 180bae59c8 | ||
|  | 9cafc163e8 | ||
|  | 8f8eef4107 | ||
|  | 7af745087e | ||
|  | aab093200b | ||
|  | 1a9556f811 | ||
|  | 39b36aa509 | ||
|  | 3fc260a12c | ||
|  | 1696a2f579 | ||
|  | baebc0b690 | ||
|  | 9f3a10ca71 | ||
|  | 38ceaf3106 | ||
| 87b619794d | |||
|  | 29c4a36479 | ||
|  | ddeb12f08c | ||
|  | a7b1406e06 | ||
|  | 871ef60cf6 | ||
|  | 7e9071a533 | ||
|  | 8c660e9856 | ||
|  | 6ca641ab7f | ||
|  | 8d6609566f | ||
|  | 17e4c63737 | ||
|  | fad470b670 | ||
|  | c5646b1e59 | ||
|  | 5da27bb266 | ||
|  | be6a077c8e | ||
|  | 8d643fc6b4 | ||
|  | 47876e3971 | ||
|  | c79c251ba7 | ||
|  | 483670e798 | ||
|  | 6c8a6008d5 | ||
|  | e680124d7b | ||
|  | b06a06f50c | ||
|  | 6416de237f | ||
| ad44fd52a4 | |||
| 03c27b10e5 | |||
| fc0ef29738 | |||
| a0eb53a607 | |||
| 66e5ef64fd | |||
|  | f5d5cc18a8 | ||
| 4c65939bbe | |||
| 379527cd58 | |||
| f63fb59cbf | |||
| cde864fdc7 | |||
| e9361697f7 | |||
|  | 830c752971 | ||
| 6bdc1b73ae | |||
|  | 0f003870bb | ||
| 0631c77a1c | |||
| 2cc4308a58 | |||
| 4975475e85 | |||
|  | 466fe58763 | ||
|  | 3b7e338808 | ||
|  | 53b13e7aef | ||
| fa60ecb25a | |||
|  | a975824481 | ||
|  | c51e5eb6cb | ||
|  | f0bc502ec9 | ||
|  | 902cafc5e4 | ||
|  | b2f54aa23e | ||
| 29a5425259 | |||
|  | e2a34c75ea | ||
| de7aa6f6a6 | |||
|  | 9acb421b2e | ||
| 66d2dc74e7 | |||
| 2f613607af | |||
| d4b9c3afb1 | |||
| b81cf49d0a | |||
|  | 1da45fdffc | ||
|  | 10dde3f002 | ||
|  | c2d6af12ab | ||
|  | 6e48f88c06 | ||
|  | 7a91a71565 | ||
|  | c4764110d8 | ||
|  | ff68e65250 | ||
|  | c9d83e5916 | ||
|  | 5dc99dbfcb | ||
|  | 8dbec85c8e | ||
|  | 84d7e40e66 | ||
|  | 0b509f2200 | ||
|  | 9591162cc9 | ||
|  | 007e17fd8b | ||
|  | 95f8e7517c | ||
|  | 9667c79162 | ||
|  | 1c79c25262 | ||
|  | 04b4b34bfe | ||
|  | fc0e689d4e | ||
|  | 83bb4b3b12 | ||
|  | 8dcfc604a0 | ||
|  | d2d639e5f6 | ||
|  | b3eb7693e3 | ||
|  | 10f42b1522 | ||
|  | 76e9f3b1dc | ||
|  | d0ff9bc16c | ||
|  | 5e4ebd16f9 | ||
|  | d2b19424ff | ||
|  | 08286254cd | ||
|  | 4805c39b45 | ||
|  | f845bbf20a | ||
|  | 71c7158124 | ||
|  | c4643ee52c | ||
|  | b46b0882f3 | ||
|  | 1c4efc9431 | ||
|  | 4133e0ccdd | ||
|  | de415e7e75 | ||
|  | 9d17524f45 | ||
|  | 68ad9650af | ||
|  | 8d4d8a3abc | ||
|  | 9617e29ed5 | ||
|  | 75406f7b58 | ||
|  | 70f5ae4f9c | ||
|  | ff307f1d65 | ||
|  | d7ae601c52 | ||
|  | 33b9ff78bb | ||
| 0739ce2fb4 | |||
| 8fc1a754de | |||
| ca8c1c9d92 | |||
| 0485ab1120 | |||
| 8a8851847c | |||
| 7b41051d0d | |||
|  | 3db1f592e2 | ||
|  | 6853ec0b69 | ||
|  | 3b39049c20 | ||
|  | 37d1669a72 | ||
|  | ee9f36d883 | ||
|  | e712f9fdb8 | ||
| 9991f5dc64 | |||
| fce6c3d29c | |||
|  | 346439076e | ||
| 5e8d8b8d5d | |||
|  | db9f86c41e | ||
|  | c7adde62eb | ||
|  | 34559dda08 | ||
|  | 37c4621e9e | ||
|  | dd7ed290f5 | ||
| dc1e1fc897 | |||
| 37abde04d7 | |||
| 40f2f7033e | |||
| aebf909dc6 | |||
| ec7d45fd91 | |||
| 3af5d96bf5 | |||
|  | c7a8a1a91c | ||
|  | 2dd434d987 | ||
| 5e954bae6a | |||
|  | a97dba18c2 | ||
|  | 26770de40e | ||
|  | 583d4ddfb8 | ||
|  | 486047b929 | ||
|  | b65ec6463b | ||
|  | 7cc13ea669 | ||
|  | c2efc969d0 | ||
|  | b091fee035 | ||
|  | 2a0f2454f4 | ||
|  | 97ea1763f1 | ||
|  | b9f51596e9 | ||
| 0610794dbe | |||
| a6b32fcad1 | |||
| e583e78a4e | |||
| 3eb3feea49 | |||
| 935914428b | |||
| ab63ba1c54 | |||
| afdc6b69df | |||
| 8b419dcee6 | |||
| e7181257e3 | |||
| 8e7c09332f | |||
| d9ea5e5538 | |||
| a21460a1b8 | |||
| b6a480ff61 | |||
| 84ee6dd2f5 | |||
| a950585a02 | |||
| 7f8a2c1eaf | |||
| 125157fdf4 | |||
| 517263dd58 | |||
| 301fc73687 | |||
| 45441c351d | |||
| be5ce414ba | |||
| bb3f277ba5 | |||
|  | 23049a8ae2 | ||
| 8bbebfdb13 | |||
|  | 662b4b5c53 | ||
|  | 9675b6372c | ||
|  | 03afd49115 | ||
| 0af3505c2a | |||
| f78b968075 | |||
|  | 7d40316044 | ||
| e3dcad62cc | |||
|  | db6a871854 | ||
| ce4f57bd8f | |||
| 8be8328830 | |||
| 0a0092e189 | |||
| c50f0a2ac5 | |||
| 6b3012d21c | |||
| 729f848c14 | |||
| 56cc4776a6 | |||
|  | b9cbba2309 | ||
| 4165f8d4af | |||
| cac185634d | |||
| 66dceefcf0 | |||
| 677ff51ea5 | |||
| 645b8a543e | |||
| 74a506c48b | |||
| deda2b4055 | |||
|  | 67ebb90ffa | ||
|  | 5d16ba135a | ||
|  | 150d08dc45 | ||
| c1a85486cc | |||
|  | d16a207a83 | ||
| d114b01bcc | |||
| dee54c3b41 | |||
| 670d2fa12e | |||
|  | a68e47ce8c | ||
| 0314aa6733 | |||
|  | 496ad7ce9b | ||
|  | efdd4a6b16 | ||
| 0b31b215f6 | |||
| 7e1734aed5 | |||
|  | 19cd51043a | ||
|  | 5348a451e9 | ||
|  | 83ae21140d | ||
| cdf9519a9f | |||
|  | d77358eaac | ||
| 9609a7615b | |||
| 361a06e5b3 | |||
| 1720307c21 | |||
|  | 15ae24f0bd | ||
|  | 143713fac1 | ||
|  | e4845b580b | ||
| 40c623b202 | |||
| 092ace8432 | |||
| 00cf619c68 | |||
| b6e1c3bc88 | |||
| 3b1d06a71d | |||
| a5d8c96bab | |||
|  | 564d95f701 | ||
| 768e2867b5 | |||
| f07a855e7e | |||
|  | 2fa9daf627 | ||
|  | a1bae7ced3 | ||
|  | 7312580a8d | ||
|  | 1c774aa4a0 | ||
|  | 29b32f6cbf | ||
|  | 465e0f31d9 | ||
|  | 5a8052ae47 | ||
|  | 6a64e05247 | ||
|  | 81a64eed08 | ||
|  | 29b27dc626 | ||
|  | ca25a12be0 | ||
|  | c0a6f5eb30 | ||
|  | 4bc4d266c2 | ||
|  | 8f0ee4df6d | ||
| 579d077b35 | |||
| 32444fac90 | |||
| 849177562d | |||
| 86bbc4cf6e | |||
| 46e58bb49e | |||
| 9199f91151 | |||
| 9247696c1c | |||
| 37f62e15cf | |||
| a98c924b24 | |||
| a71ca60270 | |||
|  | 76cc730d8f | ||
|  | 12bb7e9294 | ||
|  | 1dca0ea003 | ||
|  | b340a6568f | ||
| 6f4e93bb76 | |||
| 93eb09887e | |||
| 09081b03b6 | |||
| ceee393bd8 | |||
|  | b969513d94 | ||
|  | 2111a2c67e | ||
| 7405241b82 | |||
|  | b0884c6b04 | ||
|  | 20bea62542 | ||
| 24925f7726 | |||
| d0c18d4538 | |||
| 37eaa4b912 | |||
| a3cca056ae | |||
| ee965008d1 | |||
| c57d2ece9c | |||
| e5aa7aa866 | |||
|  | cacdf600f4 | ||
|  | 5ee0ee8efb | ||
|  | 08f20796a7 | ||
|  | 58d3a7ee2c | ||
|  | f6be360eab | ||
|  | 543a48b4ab | ||
| 0f657b934d | |||
|  | c4e42212aa | ||
|  | a4fe4996aa | ||
| fbcacb24f8 | |||
|  | 0eaa20e09d | ||
| 271d57051e | |||
|  | 3d6c260e53 | ||
|  | d0f17bd41a | ||
| 655d72a2b1 | |||
|  | 71c96fdf62 | ||
|  | 3f2327dee4 | ||
|  | 06eecfce40 | ||
|  | 67af1485b3 | ||
| a00a85a56a | |||
|  | bb953a6139 | ||
|  | 140dc26dc6 | ||
|  | 3548deebf6 | ||
|  | c67155f02c | ||
|  | c10e1e8cbf | ||
|  | c5f5ad3f75 | ||
|  | 8ec3074488 | ||
|  | 1b1284d3d0 | ||
|  | f71518ed6f | ||
|  | 1800785b80 | ||
|  | 6449724ed5 | ||
|  | 6179c3e7d4 | ||
|  | 3e5d4c5fbb | ||
|  | 3f2b63aaa5 | ||
|  | d29a5cdb44 | ||
|  | bbcc7ffeaa | ||
|  | 93f4dede3e | ||
|  | 683f8235b1 | ||
|  | 43917317b4 | ||
|  | f182de5929 | ||
|  | c6657bffd2 | ||
|  | 3d138d404f | ||
| 9c93162741 | |||
| 6068c6048a | |||
|  | d47461ba40 | ||
|  | 66e88ac6fb | ||
|  | d3cada4c95 | ||
|  | 27443bcd21 | ||
|  | b246e171b7 | ||
|  | ec434bec56 | ||
|  | 7458f622f5 | ||
|  | ab72e01707 | ||
|  | acad74528d | ||
|  | 813bbbb94a | ||
|  | a2a858262a | ||
| 727e5cb199 | |||
| 71602b43bd | |||
|  | bc40b92744 | ||
|  | d545becf24 | ||
|  | 48f605dbe0 | ||
|  | b0d7bbbb79 | ||
|  | f624b7c66d | ||
|  | bf96d8a10c | ||
|  | e8b496cfdc | ||
|  | 79ef151ad3 | ||
|  | 8e48103fd2 | ||
|  | ed4c65600c | ||
|  | ae16a1bd89 | ||
|  | e2b42145e1 | ||
|  | 55ad1f99fd | ||
|  | 5b427bee35 | ||
|  | d1c88a5cef | ||
|  | 99a25d5e9b | ||
|  | d148d6b3a5 | ||
|  | 66189d3ab2 | ||
|  | f1afa3b436 | ||
|  | 6380fb193c | ||
|  | 341ffc9a55 | ||
|  | 6962b39fc9 | ||
|  | d04b4c77c6 | ||
|  | 453b7df0be | ||
|  | 878ee99fe4 | ||
|  | 6918e3044f | ||
|  | cf46c3800f | ||
|  | 7c0c132f40 | ||
|  | e0bf797876 | ||
|  | dd07c374d7 | ||
|  | b3e59b3829 | ||
|  | 352b09d9cd | ||
|  | 93cc6d99f8 | ||
|  | 85a99fc8fa | ||
|  | a4d801bed4 | ||
|  | fbff38c5c3 | ||
|  | 14402f7537 | ||
|  | 88d24f8067 | ||
|  | cc1d700f7d | ||
|  | e82acdabb0 | ||
|  | ea42c98571 | ||
| cc5df9b171 | |||
|  | b4749f297b | ||
|  | e564c6604c | ||
|  | 712615a312 | ||
| d95d4901d2 | |||
|  | 9373654306 | ||
|  | 4a9d9f03a8 | ||
|  | b7261ec629 | ||
|  | 2e1f16fa04 | ||
|  | d295cc5223 | ||
|  | ff088009d9 | ||
|  | 52c19e9962 | ||
|  | 68d0a16d1c | ||
| a422e8d39a | |||
| ef80c1be61 | |||
| 85d9816aaa | |||
| 93b66d980d | |||
| 07d617da91 | |||
| 34aac40e65 | |||
| f54bf2b8af | |||
| e7d04d9817 | |||
| ef1537ac2c | |||
|  | d1f86fe3d9 | ||
| d13b79552b | |||
|  | 4036bfd703 | ||
| 759e360a1d | |||
| 8865529b39 | |||
| cdb73ee49c | |||
| 9188c28ee7 | |||
| 2a6c1f050d | |||
| 2ec1f8cdc0 | |||
|  | 121b388d85 | ||
| 589119c9ee | |||
| b35e1a476e | |||
|  | 8174bce720 | ||
|  | d8a7d62b23 | ||
| a75730d91f | |||
| a2b5f929dd | |||
|  | 7a0fa9f1a0 | ||
|  | 28ff7f24c5 | ||
|  | e6db25357b | ||
|  | 72ea6b6fdd | ||
| bf5f72fd9d | |||
|  | af724a1e0e | ||
| 0eeaf1ce21 | |||
|  | 57a8215c6b | ||
|  | 9163e4dee6 | ||
|  | c56d6e3f6b | ||
| 20e8854467 | |||
|  | 3ef38fabdb | ||
|  | f5cee10761 | ||
|  | d1cbb765c0 | ||
|  | 7ea9a5ca2d | ||
|  | 20c015c312 | ||
|  | ecb48ce663 | ||
|  | 00dc03a235 | ||
|  | d3b203a4a1 | ||
|  | 4506440a62 | ||
|  | da6bd84cdf | ||
|  | 0b9ccf6a57 | ||
|  | a056bd177f | ||
|  | d2ea8f2898 | ||
|  | 5cce4269bb | ||
|  | 0a2ed6dd94 | ||
|  | 417f328206 | ||
|  | cca9732925 | ||
|  | f02864b752 | ||
|  | 62bb15317c | ||
|  | b35751126f | ||
|  | 28d6d8ba96 | ||
|  | 6bdb16e293 | ||
|  | eb45cf6175 | ||
|  | d6d8f56570 | ||
|  | 5322dc1de8 | ||
|  | 51bb1a5c9d | ||
|  | 996dadf6f5 | ||
|  | 29bb0f6712 | ||
|  | f6fbad8403 | ||
|  | e37ce4172e | ||
|  | 1dfd871169 | ||
|  | a637742bb0 | ||
|  | a5e4db99fb | ||
|  | a9f66e2cd9 | ||
|  | 7bc7af8245 | ||
|  | e5dfe1e638 | ||
|  | 284f064cbf | ||
|  | e5c6f00283 | ||
|  | 12d316ebe4 | ||
|  | cbd8932075 | ||
|  | feb6dcbc94 | ||
| 181e74b1d1 | |||
|  | eb04e26b22 | ||
|  | 7b97f0bf47 | ||
|  | 19fdaf4c89 | ||
|  | 7ca9c8dc42 | ||
|  | 946f35c601 | ||
|  | c7b47bdd02 | ||
|  | eef15e05f4 | ||
| f265346a10 | |||
|  | a321bd79ed | ||
|  | 819e2b5f9f | ||
|  | 91344741a5 | ||
|  | 4d2b82235c | ||
| ffa3936878 | |||
|  | 26c70aa071 | ||
| 1bd887567e | |||
| 3304f32ef0 | |||
|  | 0790ae2298 | ||
|  | 39151b61e7 | ||
|  | 3f49d70745 | ||
|  | e5434961de | ||
|  | aab2d3a03f | ||
|  | b022ebb80e | ||
|  | 2737cae4ab | ||
|  | c4e6272535 | ||
|  | aa0c98bf34 | ||
|  | 63b6b262c6 | ||
|  | 424639ea80 | ||
|  | 594776f3a6 | ||
|  | 918e93d211 | ||
|  | b82f98c87f | ||
|  | 6c4251a91f | ||
|  | 2261782920 | ||
|  | 043dcfb283 | ||
|  | 3c76c5e0f1 | ||
|  | d348e6314a | ||
|  | b3fa6f352b | ||
|  | 191b05c305 | ||
|  | 215fdce411 | ||
|  | b25805e0a1 | ||
|  | 13d0d2a300 | ||
| 15f51fb03f | |||
|  | a24b1f5c2a | ||
|  | 04e7f65e8e | ||
| 41b9318028 | |||
|  | 378e8b53f2 | ||
|  | c832e8b1a7 | ||
| fee7ade1a5 | |||
|  | d51dbf8a53 | ||
|  | c03a1b57c5 | ||
| 0c566cfbde | |||
|  | 9295325d21 | ||
|  | cb1aa8bef0 | ||
| b9d19be183 | |||
| 293369f165 | |||
|  | 3046438cb1 | ||
|  | 811e5a5ad1 | ||
| 2c8f18d7fc | |||
| c7f8cdd098 | |||
| 58ff5b934a | |||
|  | 03d15ddded | ||
|  | 002d8f80a6 | ||
|  | 82d3791859 | ||
|  | d9531838f2 | ||
| c7b5c77395 | |||
| 223aa37161 | |||
|  | c1acadbf3d | ||
| 54af894b82 | |||
| e1ac75f394 | |||
|  | 8c69a94488 | ||
|  | 07b625d4aa | ||
|  | a1296dc7af | ||
|  | e5a2236d72 | ||
|  | 588a82426e | ||
|  | 8245ddf2a6 | ||
|  | 775a0c6478 | ||
| bad67a8b65 | |||
| 7e98e184a0 | |||
| 6240eff160 | |||
| a8918ebe86 | |||
|  | b852176958 | ||
|  | c9e398b7ec | ||
|  | e84d5626df | ||
| 0fb61938ce | |||
| d6b27f2f21 | |||
|  | e15bcfae07 | ||
| 72cf5a3d5e | |||
|  | 7de2e00c94 | ||
|  | efe5d75798 | ||
|  | 9f1eedbe1b | ||
|  | 7fe495179f | ||
|  | 7f6c4f6236 | ||
|  | 30948f1701 | ||
|  | 02ec3607b2 | ||
|  | 3c2dcfbfa2 | ||
|  | 8bcf59aaf0 | ||
|  | c6d2ac9100 | ||
|  | 2ac578c3ad | ||
|  | f941435232 | ||
|  | 171a1cb876 | ||
|  | cfc19434d0 | ||
|  | 688871a680 | ||
|  | 44c8558aa3 | ||
|  | 6b923d2310 | ||
|  | 09e0b31bc9 | ||
|  | eb2454eded | ||
|  | 3014d8cead | ||
|  | 70fdc2edf2 | ||
|  | e47f29aa38 | ||
|  | d811896e21 | ||
|  | 5c999b6ef1 | ||
|  | 79a6d9e771 | ||
| e1cf1c786d | |||
|  | 71fe9559b1 | ||
|  | f1fa8d34bf | ||
|  | aa07fa9207 | ||
|  | 47fec973bc | ||
|  | ea8247aa16 | ||
|  | bf18284450 | ||
|  | cd58d5a357 | ||
|  | 75bb3f992c | ||
|  | ae1fcdb8c0 | ||
|  | 507080f75e | ||
|  | 5bcf043d97 | ||
| 99605b98d4 | |||
|  | 6dfd43a8da | ||
| c7135875b8 | |||
|  | e29e1101cd | ||
|  | d97602e60b | ||
|  | a5cbac1f97 | ||
|  | 3143d3d91a | ||
|  | 9bdf3fc4ac | ||
|  | e06bc7dba3 | ||
|  | a8b9f38000 | ||
| ca27b89a8b | |||
| e1bf7caa9a | |||
| e681c17a0f | |||
| 5416d88c97 | |||
|  | 0dca152436 | ||
|  | 9ce7abd31d | ||
|  | f41ff281fb | ||
|  | ee437649f0 | ||
|  | 321cb72ca8 | ||
|  | 4303d51c0a | ||
|  | d16bf12611 | ||
|  | 4231a7972d | ||
|  | c436d39014 | ||
|  | 51a12814f9 | ||
|  | 00ae6e4623 | ||
|  | 4b587e8711 | ||
|  | d2f377b54f | ||
|  | 193c820757 | ||
|  | b9298792ae | ||
|  | aaf30ab965 | ||
|  | 2db66e6154 | ||
|  | 38295e591d | ||
|  | 544b0248b2 | ||
|  | 2bccf633d5 | ||
|  | 4f9d5ae7b1 | ||
|  | 259337dff1 | ||
|  | 84768eb74e | ||
|  | 8852ef990e | ||
|  | 87295ad9b7 | ||
|  | 288764b551 | ||
|  | 5ab5ef681c | ||
|  | c9e70889dd | ||
|  | b30ee0a27a | ||
|  | ef968f3673 | ||
|  | 96dede5077 | ||
|  | 66fcb76cb5 | ||
|  | 63c8e51137 | ||
|  | 12bec5c553 | ||
|  | 08460a6964 | ||
|  | b5a40cfda9 | ||
|  | c78953b036 | ||
|  | 427f7ceaff | ||
|  | 1055385bcc | ||
|  | c1022642a2 | ||
|  | 910a6f8b34 | ||
|  | 06253f029c | ||
|  | fa6527b24f | ||
|  | 0501e6417a | ||
|  | a198f5252d | ||
|  | d83842af27 | ||
|  | f605f7dcc6 | ||
|  | e638bc04ed | ||
|  | 4830c3ea2d | ||
|  | 8e7c025e47 | ||
|  | 1bfe929ab3 | ||
|  | 93cc2c883e | ||
|  | 44290a20a6 | ||
|  | 1f10a284f2 | ||
|  | 28f397574f | ||
|  | 6c1fa6de0b | ||
|  | f0a08afd31 | ||
|  | 982fc09908 | ||
|  | 9e0b5b0b82 | ||
|  | b12e8dc147 | ||
|  | 25c5a3297c | ||
|  | dd3ad42eb5 | ||
|  | 5ea181829e | ||
|  | 0cf203669f | ||
|  | 559bfcac60 | ||
|  | db8a1ed0ab | ||
|  | 16150905a0 | ||
|  | 9a376887ac | ||
|  | 773808fa59 | ||
|  | c1e59a0676 | ||
|  | 05febc60bd | ||
|  | a73fe598ef | ||
|  | b7f20fed6c | ||
|  | 585923c827 | ||
|  | 394e17d599 | ||
|  | 59136850b8 | ||
|  | d726f4b1e8 | ||
|  | 705b9b1e6a | ||
|  | 31e8ad8a3e | ||
|  | 99827e005b | ||
|  | 751c8a8bc6 | ||
|  | 73305c0b28 | ||
|  | 37216cd16b | ||
|  | dae68638cf | ||
|  | 7cadc0bc28 | ||
|  | cce686f3a8 | ||
|  | 4fe46fbcef | ||
|  | fe8b8f46aa | ||
|  | 310f1a2283 | ||
|  | 7079761ffe | ||
|  | f681c981c6 | ||
|  | 5d97146d14 | ||
|  | 7b56bd697d | ||
|  | 14cd268d69 | ||
|  | 754be1c9c9 | ||
|  | da2c155254 | ||
|  | ceb2888f82 | ||
|  | ce3e2bb32b | ||
|  | 26c94c9ec6 | ||
|  | 7b6eed9a47 | ||
|  | 639197f4c8 | ||
|  | 13bae8d2fa | ||
|  | 6b2027550c | ||
|  | 022b365bb2 | ||
|  | d8867fc9ea | ||
|  | 118c58b5fa | ||
|  | faccc1367f | ||
|  | 22b83b0814 | ||
|  | 1d82e2a7d9 | ||
|  | 823bd578f2 | ||
|  | 3e5c36b39e | ||
|  | 8fb0897160 | ||
|  | b8a72c57e1 | ||
|  | 6a0a8e8ab4 | ||
|  | 9188565a86 | ||
|  | 4d7d22c337 | ||
|  | b58116b023 | ||
|  | fe9e5ce861 | ||
|  | e43d53e564 | ||
|  | d4a5039efc | ||
|  | 35506e0175 | ||
|  | 1c27831f92 | ||
|  | cdbf07a835 | ||
|  | b92580943a | ||
|  | 60eff1000f | ||
|  | 96510b270d | ||
|  | 1281104d96 | ||
|  | 3c1724fa81 | ||
|  | 1630af4fbd | ||
|  | e76e2b1537 | ||
|  | 6c276dc596 | ||
|  | d3c115e3f9 | ||
|  | c245ef7149 | ||
|  | 8b09ba2924 | ||
|  | 52eb310f95 | ||
|  | 5bff38fc7b | ||
|  | 2813a59323 | ||
|  | eef33fa263 | ||
|  | 241d3cea53 | ||
|  | 89d6db4208 | ||
|  | e0ad288cf4 | ||
|  | f4d7fae8ca | ||
|  | 95a7493fc1 | ||
|  | 8243dbcbef | ||
|  | c3a4071627 | ||
|  | cef3f22e0d | ||
|  | c206b965ad | ||
|  | e868946fd7 | ||
|  | 254044c36b | ||
|  | c695d6f7a0 | ||
|  | feef855f01 | ||
|  | b3a48ca5af | ||
|  | f3a52d094e | ||
|  | 2901bd919f | ||
|  | 0396a5bf2b | ||
|  | b48ad16f04 | ||
|  | 7cc6250860 | ||
|  | ae2e4b518d | ||
|  | e9b9f3a62b | ||
|  | 3321669726 | ||
|  | 21fc85670e | ||
|  | 18a5ad6541 | ||
|  | 71c5456225 | ||
|  | 50e04164a2 | ||
|  | 3b1d71f317 | ||
|  | 65c2689578 | ||
|  | b45673f04a | ||
|  | cb6e037f5e | ||
|  | 5e6d60bb3a | ||
|  | 64f8d9bad3 | ||
|  | 05b86e1f7a | ||
|  | 700fed860d | ||
|  | 820bf6279b | ||
|  | b97ce81dd2 | ||
|  | f4dfd8f99c | ||
|  | 29139bf360 | ||
|  | 4f9c2724f5 | ||
|  | 7a914f5e94 | ||
|  | 121d04e1d5 | ||
|  | fc6cdba8e2 | ||
|  | 7f39ead159 | ||
|  | 1da82ac2dd | ||
|  | f2dcc39c14 | ||
|  | 705dc56153 | ||
|  | 02047b62d7 | ||
|  | 895d4b33a6 | ||
|  | 142cb3316e | ||
|  | 997fcc9fff | ||
|  | ec65ca11d6 | ||
|  | 0198027544 | ||
|  | 69e0550d4f | ||
|  | 9a1a5635e2 | ||
|  | 863f9ff77e | ||
|  | 4146c4c5cb | ||
|  | b3ad5c5df9 | ||
|  | 9388e2dc88 | ||
|  | 56dec9eaa1 | ||
|  | 596126f4f4 | ||
|  | 8646b2c8f7 | ||
|  | c81bb1fb90 | ||
|  | d17a52a8d6 | ||
|  | 55e0eecc0b | ||
|  | 496adc17ea | ||
|  | ab43d7d2df | ||
|  | 13f0bfe546 | ||
|  | 83a384145b | ||
|  | 8a923761a5 | ||
|  | 6e4a99eba3 | ||
|  | 0470aa185e | ||
|  | 273371db8b | ||
|  | ed3aa0c328 | ||
|  | acfff6b103 | ||
|  | ada4579193 | ||
|  | 3a17c3079e | ||
|  | 26e46de8e1 | ||
|  | 111bcc8e60 | ||
|  | cdaa204ba2 | ||
|  | e85511fcb9 | ||
| 35c120a29f | |||
| 7c4c1bc387 | |||
|  | 6e77edcf67 | ||
|  | effed9c760 | ||
| 0e5c8b53b0 | |||
| 47a332445c | |||
| c904b2d827 | |||
| f56263d6bd | |||
|  | 0c2494cb34 | ||
|  | 9e5743a64c | ||
|  | b5241ec75e | ||
| 4f00224f0d | |||
| 320a896610 | |||
| 08924c5e05 | |||
| 98bfc308a7 | |||
| dee24fbc9c | |||
|  | 2556427c7d | ||
|  | a2b35e5bba | ||
|  | 3e8f1acb96 | ||
|  | 85788977fe | ||
|  | 066ca5bada | ||
|  | 41369f738e | ||
|  | 67377b3cbf | ||
|  | ac3d668655 | ||
|  | c57b15e159 | ||
|  | 66efb8012e | ||
|  | cad0c0dadb | ||
|  | b32c90ed5d | ||
|  | 4d361dc67b | ||
|  | 2b170d91f7 | ||
|  | 9e074d6ca6 | ||
|  | b655b2695b | ||
|  | 366aeed2ba | ||
|  | 454ae5f9e3 | ||
|  | b811114425 | ||
|  | 712e7c8939 | ||
|  | 713cd92141 | ||
|  | 4154b499b1 | ||
|  | 253f204225 | ||
|  | 7241f3eb1d | ||
|  | 2422f60898 | ||
|  | ba6599fa56 | ||
|  | f2666f6fb0 | ||
|  | b33839191d | ||
|  | ee3e375dde | ||
|  | 5b0f7ca21b | ||
|  | f581d91730 | ||
|  | bbf362691b | ||
|  | 15e2c8c7b3 | ||
|  | f838127730 | ||
|  | d4c0bb3b0e | ||
|  | b81aee3f1c | ||
|  | c6caf5dbce | ||
|  | 7acc59f2cd | ||
|  | 757ff7ead7 | ||
|  | bc2fe16b74 | ||
|  | 35363d9ee7 | ||
|  | 52106db6fd | ||
|  | c4b1829e78 | ||
|  | 489a9378c5 | ||
| 28ae109b32 | |||
|  | e7a6a94ff2 | ||
|  | 234556a172 | ||
|  | e4ddceabea | ||
|  | 05dd3ad642 | ||
|  | 6c5db61a97 | ||
|  | a0e4e9e8e3 | ||
|  | c66df77d4a | ||
|  | cfb6b34630 | ||
|  | d8fd0adf47 | ||
|  | 928ae13a8a | ||
|  | c2e0ea70e4 | ||
|  | 782ce24895 | ||
|  | b630742fd4 | ||
|  | b20df930a2 | ||
|  | d60a96fc5c | ||
|  | 05b0a0ab2f | ||
|  | 9eb137e503 | ||
|  | 7d797009bb | ||
|  | 3c1818f229 | ||
|  | d8b69e9b45 | ||
|  | 9177c9d4c2 | ||
|  | 5195352975 | ||
|  | deb8f865df | ||
|  | 5b2c70e4fb | ||
|  | f66db0859e | ||
|  | b6488d1d00 | ||
|  | 6a4ac336ad | ||
|  | 7ac6dcf8a0 | ||
|  | c6a3677cc5 | ||
|  | 707459acd6 | ||
|  | 6390c3320e | ||
|  | b8aabc466c | ||
|  | c66e4232b9 | ||
|  | 336450d43f | ||
|  | 7e66aadd6f | ||
| bf2b796936 | |||
| 85623f48a9 | |||
|  | 4fbee9c3de | ||
|  | bfa3b45547 | ||
|  | 677a9da469 | ||
|  | 1f7752d457 | ||
|  | 89979dbf61 | ||
|  | 8d1abb8f33 | ||
|  | 2df3494c3b | ||
|  | 39bb490257 | ||
|  | 7a7aad0503 | ||
|  | b157a3fa90 | ||
|  | 1b688a8aa5 | ||
|  | e8978cc065 | ||
|  | 7fd68e4825 | ||
|  | 4119eefe37 | ||
|  | aafc2e6e96 | ||
|  | 2cbe6fa11c | ||
|  | eec7bcf296 | ||
|  | 6c45de34a4 | ||
|  | 61a40c47d2 | ||
|  | 007157e2e8 | ||
|  | 49a0ade315 | ||
|  | 782cd9a45a | ||
|  | 6382e631b6 | ||
|  | 12493cffca | ||
|  | a38ab57ddf | ||
|  | 30091ef69c | ||
|  | 1a483bfa2c | ||
|  | 1a091951e8 | ||
|  | bfb66b352a | ||
|  | be26e3df7f | ||
|  | cb3307509d | ||
|  | a3158253a7 | ||
|  | 406380e4f1 | ||
|  | efb70652af | ||
|  | 05256bb99a | ||
|  | 64d0cc2fa8 | ||
|  | f5d7267ba7 | ||
|  | 24c0a21cc1 | ||
|  | 6a352d642b | ||
| 48ae1f7c1c | |||
|  | aaf1adaaa1 | ||
| f34f5fe693 | |||
| f485178422 | |||
|  | 797ca0f926 | ||
| 390a4b0064 | |||
| 94b029dc9c | |||
| 45d5728c3e | |||
|  | 6eabbaf209 | ||
| 03fdd0b947 | |||
| fb8faacddc | |||
| 7ee4557ab5 | |||
|  | 5accdbccbb | ||
|  | 7fb26f9e45 | ||
|  | 26a07f722d | ||
| 9176a03a8a | |||
| 4a1bfc366d | |||
| ebee8c34e1 | |||
| 4ecad1c73b | |||
| d1b3a4d3f6 | |||
| 40832bb3bf | |||
| 4a78157f9a | |||
| bf5fc8750d | |||
| 274a7b7137 | |||
| 8dd2c02d3e | |||
| a73f5cb270 | |||
| 7d40e11144 | |||
| af48553e35 | |||
| ad8bcc7282 | |||
| 22a44415e4 | |||
| 6a153719f9 | |||
| 5c8fa1b9e7 | |||
| d82679e3d7 | |||
| 9cb432a082 | |||
| 869d29d4a4 | |||
| c3d2e64a43 | |||
| e1770ec52c | |||
| 1256744f1b | |||
| 77dddbc581 | |||
| bfa4000365 | |||
| 50c2f8164d | |||
| 5c30de5f22 | |||
| 1c03ce621f | |||
| e634cda318 | |||
| 3501703c15 | |||
| 129f2e53ee | |||
| 209867b3a8 | |||
|  | 59511d255f | ||
| f42daa01c5 | |||
|  | 29ee1b05af | ||
| 42055b9001 | |||
|  | 00c96f5b71 | ||
|  | 5cc7eff94f | ||
| 28077ef0b0 | |||
|  | 143b128891 | ||
|  | 6b06b647bc | ||
|  | 413c613c9f | ||
|  | 1c0d15ba2a | ||
|  | 28bd6b8708 | ||
|  | 419a48ac3a | ||
|  | 6fce27113a | ||
|  | 53a7633700 | ||
|  | 4094394cef | ||
| f533c39e67 | |||
|  | 86bc491df4 | ||
|  | 4759551c16 | ||
|  | b057dbfd60 | ||
|  | bddb88d97f | ||
|  | dbe44a9c1c | ||
|  | eeb791c460 | ||
| 6d0eba6bcf | |||
| 4d04b21f04 | |||
| 2f1b26053b | |||
| 1848945d64 | |||
| 9278419345 | |||
| 566dcc7aee | |||
| a6088c0e4a | |||
|  | 60c9498a56 | ||
| 241650c171 | |||
| 811809895e | |||
| fe9164bfef | |||
|  | ad3f003fbb | ||
|  | 7ecb057b68 | ||
|  | e932abfa74 | ||
|  | 0011f4c7b0 | ||
|  | 13312e9879 | ||
|  | ced90c23db | ||
| 42f5773f51 | |||
| b270c76249 | |||
|  | 34df825718 | ||
|  | aac4e3b99c | ||
|  | 5a55a6c642 | ||
|  | 65c3483c1f | ||
| 660a3161f5 | |||
| 9e6c4b32e3 | |||
| 25225fc451 | |||
| c3f2d0a134 | |||
| cd2d3ee6b4 | |||
| 81fcf411c1 | |||
| d7075eb762 | |||
| cf3f5ea60c | |||
| 59185ab2a8 | |||
| a177fa8232 | |||
| 308cf30a5a | |||
| 99c8d95443 | |||
| 97c316b62e | |||
| 90921fd4cd | |||
| 296cc4144c | |||
| f7548ab8d1 | |||
| 3cb306bc91 | |||
| c20d5855e4 | |||
| 00bd60ef4f | |||
| b8c7fb6f74 | |||
| df531198c9 | |||
| 12b6f0d488 | |||
| 6cc234e8d3 | |||
| 4dadb1dbc0 | |||
| 2616e8b24c | |||
| be855c6c90 | |||
| 7be9077fce | |||
| d48c09a914 | |||
| d5c3dbf864 | |||
| 2a9b89fd2a | |||
| c73f4ca847 | |||
| d63b5335d4 | |||
| a766f7137c | |||
| 5c3c14ab37 | |||
| 775413ac7e | |||
| 1f271c75f0 | |||
| 4df152185e | |||
| c83b30f27b | |||
| db10f7b963 | |||
| ed68c2cb38 | |||
| a6c8dea190 | |||
| 124eaf42cd | |||
| 5489096bf5 | |||
| 3a425c6792 | |||
| 8809753108 | |||
| 4428a2e89c | |||
| 0616597bf2 | |||
| 782a763046 | |||
| 8dcade6890 | |||
| dd49d71cb7 | |||
| b0b52fd714 | |||
| ef40baaa84 | |||
| 7c259bf26b | |||
| 05e5008305 | |||
| 448f5ff40f | |||
| 5482f1174d | |||
| 2da0560ec8 | |||
| 5151fc3792 | |||
| fd5cd56f81 | |||
| 35d9c05abf | |||
| fcb3035b67 | |||
| b7db969f08 | |||
|  | 14303fd46c | ||
| 5e6b17cd19 | |||
|  | 8232ff59a0 | ||
| 411c117f0f | |||
| 298499c749 | |||
| b8ad2d4835 | |||
| d37eb134e2 | |||
| 63ec5d68f4 | |||
| 8330e1eaf2 | |||
| 1f86827e46 | |||
| 321e5e3ff5 | |||
| 3eb8292d15 | |||
| 5a3f90fd28 | |||
| eb975f4de1 | |||
| 405b938e08 | |||
| f899e32fb0 | |||
| 9181e77d55 | |||
| f1b3a174b6 | |||
| eb9821ed36 | |||
| defb7fb3a3 | |||
| 83e225a744 | |||
| f30bea3dc9 | |||
| a69f7b12b1 | |||
| ca042fe75e | |||
|  | dc9111dbcd | ||
| 8a16a66299 | |||
| d7a7613807 | |||
| 3fc8688941 | |||
| d7b351a1aa | |||
|  | 9e0c4e70d4 | ||
| 2232c495be | |||
| 18b1bea664 | |||
| a5d5c41dd6 | |||
| 66d5c71a92 | |||
| 824ea37f44 | |||
| eb29f98c37 | |||
| d903dc58cf | |||
| f09de0ab7d | |||
| d29603c584 | |||
| 3380980c5c | |||
| 38ef13d9b6 | |||
| 6c43b1c43d | |||
| 2b34c46412 | |||
| f9227fa29d | |||
| 96a3eaff1c | |||
| 65cb85a887 | |||
| 640a72c52d | |||
| 9b7b96a310 | |||
| b18746e769 | |||
| a2b431b1ab | |||
| d844bccb04 | |||
| 49f928e754 | |||
| 07fc1014be | |||
| facb6faf75 | |||
| e72338a7d9 | |||
| f37c022538 | |||
| 5229628d48 | |||
| b4b7bf05b4 | |||
| 231415a772 | |||
| f052d307d7 | |||
| f15971cecf | |||
| 99cf59c7a4 | |||
| 0d13014e8a | |||
| fd1f89de1d | |||
| 78b616427f | |||
| c15ea345dd | |||
| 1d319e90f0 | |||
| e6e500e2f9 | |||
| cf1ec1dc86 | |||
| 46a042cde2 | |||
| 52129d7511 | |||
| d03835d737 | |||
| b4b7817baa | |||
| d85152e58c | |||
| f118040432 | |||
| 9f1aff8c07 | |||
| 94bbdf372b | |||
| 240d94bd57 | |||
| 3ee7ff2752 | |||
| 2c5385cf5c | |||
| c8a691044f | |||
| f93eaff876 | |||
| 10faa14bef | |||
| 30ccbdc32d | |||
| 79243aece3 | |||
| a61322b83f | |||
| 3df73f4d1f | |||
| 7165a63e97 | |||
| 2404edd289 | |||
| 3bff09b04c | |||
| 28748af5d3 | |||
| a56a4e2cb8 | |||
| 339497b2c2 | |||
| c05168a2b5 | |||
| 782ee35779 | |||
| 43acee8f1b | |||
| 4a19441a17 | |||
| 11acf5897f | |||
|  | 4be99fe828 | ||
|  | 601193ff3c | ||
| f500dec1f1 | |||
|  | bfb7380715 | ||
| 3e3c576ad7 | |||
| 2aa1314fac | |||
| 3063e4a24f | |||
| 6f8ec4740c | |||
| cbcd84c931 | |||
| e475273cd3 | |||
|  | 851231869b | ||
| 3376f4dfb4 | |||
| 205f93569a | |||
| 4f7a8661ba | |||
| 6e7d351e8e | |||
| 73f1927ce4 | |||
| 56e3f39de1 | |||
| 75a2aefd69 | |||
| 55e822412a | |||
| 171d9a4381 | |||
| 806084e707 | |||
| 04009a6a5b | |||
| 3d0f5c0a15 | |||
| 437af4dd04 | |||
| ba61455017 | |||
| 624f1d653d | |||
| e21821ace5 | |||
| 22c028af11 | |||
| f0560f0d2a | |||
| 502ae09523 | |||
| 2cbef2babc | |||
| e11d45b51e | |||
| 061320a5df | |||
| 2aa465b138 | |||
| d18f0aa829 | |||
| e7b8ddb631 | |||
| 358a625cc4 | |||
| d44fa73b2a | |||
| 5ccb499665 | |||
| c467165bf3 | |||
| 8512f3c5d0 | |||
| 5003e57338 | |||
| b7c2da53fe | |||
| 598cdc0284 | |||
| 692d9a25e3 | |||
| 38f6c27983 | |||
| 1172402166 | |||
| ab344ba02f | |||
| ec33311715 | |||
| 5bf5d0277c | |||
| 31f6ee9ca4 | 
							
								
								
									
										6
									
								
								.envrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.envrc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| if [[ ! -d .venv ]]; then | ||||
|   log_error 'No .venv folder found. Use `uv sync` to create one first.' | ||||
|   exit 2 | ||||
| fi | ||||
|  | ||||
| . .venv/bin/activate | ||||
							
								
								
									
										46
									
								
								.github/actions/setup_project/action.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								.github/actions/setup_project/action.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| name: "Setup project" | ||||
| description: "Setup Python and Poetry" | ||||
| runs: | ||||
|   using: composite | ||||
|   steps: | ||||
|     - name: Install apt packages | ||||
|       uses: awalsh128/cache-apt-pkgs-action@latest | ||||
|       with: | ||||
|         packages: gettext | ||||
|         version: 1.0  # increment to reset cache | ||||
|  | ||||
|     - name: Install uv | ||||
|       uses: astral-sh/setup-uv@v5 | ||||
|       with: | ||||
|         version: "0.5.14" | ||||
|         enable-cache: true | ||||
|         cache-dependency-glob: "uv.lock" | ||||
|  | ||||
|     - name: "Set up Python" | ||||
|       uses: actions/setup-python@v5 | ||||
|       with: | ||||
|         python-version-file: ".python-version" | ||||
|  | ||||
|     - name: Restore cached virtualenv | ||||
|       uses: actions/cache/restore@v4 | ||||
|       with: | ||||
|         key: venv-${{ runner.os }}-${{ hashFiles('.python-version') }}-${{ hashFiles('pyproject.toml') }}-${{ env.CACHE_SUFFIX }} | ||||
|         path: .venv | ||||
|  | ||||
|     - name: Install dependencies | ||||
|       run: uv sync | ||||
|       shell: bash | ||||
|  | ||||
|     - name: Install Xapian | ||||
|       run: uv run ./manage.py install_xapian | ||||
|       shell: bash | ||||
|  | ||||
|     - name: Save cached virtualenv | ||||
|       uses: actions/cache/save@v4 | ||||
|       with: | ||||
|         key: venv-${{ runner.os }}-${{ hashFiles('.python-version') }}-${{ hashFiles('pyproject.toml') }}-${{ env.CACHE_SUFFIX }} | ||||
|         path: .venv | ||||
|  | ||||
|     - name: Compile gettext messages | ||||
|       run: uv run ./manage.py compilemessages | ||||
|       shell: bash | ||||
							
								
								
									
										13
									
								
								.github/auto_assign.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								.github/auto_assign.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| # Set to true to add reviewers to pull requests | ||||
| addReviewers: true | ||||
|  | ||||
| # Set to true to add assignees to pull requests | ||||
| addAssignees: author | ||||
|  | ||||
| # A list of team reviewers to be added to pull requests (GitHub team slug) | ||||
| reviewers: | ||||
|   - ae-utbm/sith-3-developers | ||||
|  | ||||
| # Number of reviewers has no impact on GitHub teams | ||||
| # Set 0 to add all the reviewers (default: 0) | ||||
| numberOfReviewers: 0 | ||||
							
								
								
									
										14
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| # To get started with Dependabot version updates, you'll need to specify which | ||||
| # package ecosystems to update and where the package manifests are located. | ||||
| # Please see the documentation for all configuration options: | ||||
| # https://docs.github.com/github/administering-a-repository/configuration-options-for-dependency-updates | ||||
|  | ||||
| version: 2 | ||||
| updates: | ||||
|   - package-ecosystem: "pip" # See documentation for possible values | ||||
|     directory: "/" # Location of package manifests | ||||
|     schedule: | ||||
|       interval: "weekly" | ||||
|     target-branch: "taiste" | ||||
|     commit-message: | ||||
|       prefix: "[UPDATE] " | ||||
							
								
								
									
										47
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| name: Sith CI | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: [master, taiste] | ||||
|   pull_request: | ||||
|     branches: [master, taiste] | ||||
|   workflow_dispatch: | ||||
|  | ||||
| jobs: | ||||
|   pre-commit: | ||||
|     name: Launch pre-commits checks (ruff) | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
|     - uses: actions/setup-python@v5 | ||||
|       with: | ||||
|         python-version-file: ".python-version" | ||||
|     - uses: pre-commit/action@v3.0.1 | ||||
|       with: | ||||
|         extra_args: --all-files | ||||
|  | ||||
|   tests: | ||||
|     name: Run tests and generate coverage report | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: false  # don't interrupt the other test processes | ||||
|       matrix: | ||||
|         pytest-mark: [slow, not slow] | ||||
|     steps: | ||||
|       - name: Check out repository | ||||
|         uses: actions/checkout@v4 | ||||
|       - uses: ./.github/actions/setup_project | ||||
|         env: | ||||
|           # To avoid race conditions on environment cache | ||||
|           CACHE_SUFFIX: ${{ matrix.pytest-mark }} | ||||
|       - name: Run tests | ||||
|         run: uv run coverage run -m pytest -m "${{ matrix.pytest-mark }}" | ||||
|       - name: Generate coverage report | ||||
|         run: | | ||||
|           uv run coverage report | ||||
|           uv run coverage html | ||||
|       - name: Archive code coverage results | ||||
|         uses: actions/upload-artifact@v3 | ||||
|         with: | ||||
|           name: coverage-report | ||||
|           path: coverage_report | ||||
							
								
								
									
										65
									
								
								.github/workflows/deploy.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								.github/workflows/deploy.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | ||||
| name: Deploy to production | ||||
| concurrency: production | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: [master] | ||||
|   workflow_dispatch: | ||||
|  | ||||
| jobs: | ||||
|   deployment: | ||||
|     runs-on: ubuntu-latest | ||||
|     environment: production | ||||
|     timeout-minutes: 30 | ||||
|    | ||||
|     steps: | ||||
|     - name: SSH Remote Commands | ||||
|       uses: appleboy/ssh-action@v1.1.0 | ||||
|       with: | ||||
|         # Proxy | ||||
|         proxy_host : ${{secrets.PROXY_HOST}} | ||||
|         proxy_port : ${{secrets.PROXY_PORT}} | ||||
|         proxy_username : ${{secrets.PROXY_USER}} | ||||
|         proxy_passphrase: ${{secrets.PROXY_PASSPHRASE}} | ||||
|         proxy_key: ${{secrets.PROXY_KEY}} | ||||
|          | ||||
|         # Serveur web | ||||
|         host: ${{secrets.HOST}} | ||||
|         port : ${{secrets.PORT}} | ||||
|         username : ${{secrets.USER}} | ||||
|         key: ${{secrets.KEY}} | ||||
|  | ||||
|         script_stop: true | ||||
|  | ||||
|         # See https://github.com/ae-utbm/sith/wiki/GitHub-Actions#deployment-action | ||||
|         script: | | ||||
|           cd ${{secrets.SITH_PATH}} | ||||
|  | ||||
|           git fetch | ||||
|           git reset --hard origin/master | ||||
|           uv sync --group prod | ||||
|           npm install | ||||
|           uv run ./manage.py install_xapian | ||||
|           uv run ./manage.py migrate | ||||
|           uv run ./manage.py collectstatic --clear --noinput | ||||
|           uv run ./manage.py compilemessages | ||||
|  | ||||
|           sudo systemctl restart uwsgi | ||||
|    | ||||
|   sentry: | ||||
|     runs-on: ubuntu-latest | ||||
|     environment: production | ||||
|     timeout-minutes: 30 | ||||
|     needs: deployment | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - name: Sentry Release | ||||
|         uses: getsentry/action-release@v1.7.0 | ||||
|         env: | ||||
|           SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }} | ||||
|           SENTRY_ORG: ${{ secrets.SENTRY_ORG }} | ||||
|           SENTRY_PROJECT: ${{ secrets.SENTRY_PROJECT }} | ||||
|           SENTRY_URL: ${{ secrets.SENTRY_URL }} | ||||
|         with: | ||||
|           environment: production | ||||
							
								
								
									
										21
									
								
								.github/workflows/deploy_docs.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								.github/workflows/deploy_docs.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| name: deploy_docs | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
| permissions: | ||||
|   contents: write | ||||
| jobs: | ||||
|   deploy: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: ./.github/actions/setup_project | ||||
|       - run: echo "cache_id=$(date --utc '+%V')" >> $GITHUB_ENV | ||||
|       - uses: actions/cache@v3 | ||||
|         with: | ||||
|           key: mkdocs-material-${{ env.cache_id }} | ||||
|           path: .cache | ||||
|           restore-keys: | | ||||
|             mkdocs-material- | ||||
|       - run: uv run mkdocs gh-deploy --force | ||||
							
								
								
									
										46
									
								
								.github/workflows/taiste.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								.github/workflows/taiste.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| name: Sith taiste | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: [taiste] | ||||
|   workflow_dispatch: | ||||
|  | ||||
| jobs: | ||||
|   deployment: | ||||
|     runs-on: ubuntu-latest | ||||
|     environment: taiste | ||||
|     timeout-minutes: 30 | ||||
|  | ||||
|     steps: | ||||
|     - name: SSH Remote Commands | ||||
|       uses: appleboy/ssh-action@v1.1.0 | ||||
|       with: | ||||
|         # Proxy | ||||
|         proxy_host : ${{secrets.PROXY_HOST}} | ||||
|         proxy_port : ${{secrets.PROXY_PORT}} | ||||
|         proxy_username : ${{secrets.PROXY_USER}} | ||||
|         proxy_passphrase: ${{secrets.PROXY_PASSPHRASE}} | ||||
|         proxy_key: ${{secrets.PROXY_KEY}} | ||||
|  | ||||
|         # Serveur web | ||||
|         host: ${{secrets.HOST}} | ||||
|         port : ${{secrets.PORT}} | ||||
|         username : ${{secrets.USER}} | ||||
|         key: ${{secrets.KEY}} | ||||
|  | ||||
|         script_stop: true | ||||
|  | ||||
|         # See https://github.com/ae-utbm/sith/wiki/GitHub-Actions#deployment-action | ||||
|         script: | | ||||
|           cd ${{secrets.SITH_PATH}} | ||||
|  | ||||
|           git fetch | ||||
|           git reset --hard origin/taiste | ||||
|           uv sync --group prod | ||||
|           npm install | ||||
|           uv run ./manage.py install_xapian | ||||
|           uv run ./manage.py migrate | ||||
|           uv run ./manage.py collectstatic --clear --noinput | ||||
|           uv run ./manage.py compilemessages | ||||
|  | ||||
|           sudo systemctl restart uwsgi | ||||
							
								
								
									
										12
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,15 +1,23 @@ | ||||
| db.sqlite3 | ||||
| *.sqlite3 | ||||
| *.log | ||||
| *.pyc | ||||
| *.mo | ||||
| *__pycache__* | ||||
| .DS_Store | ||||
| pyrightconfig.json | ||||
| dist/ | ||||
| .vscode/ | ||||
| env/ | ||||
| .idea/ | ||||
| .venv/ | ||||
| doc/html | ||||
| data/ | ||||
| galaxy/test_galaxy_state.json | ||||
| /static/ | ||||
| sith/settings_custom.py | ||||
| sith/search_indexes/ | ||||
| .coverage | ||||
| coverage_report/ | ||||
| node_modules/ | ||||
|  | ||||
| # compiled documentation | ||||
| site/ | ||||
|   | ||||
| @@ -1,26 +0,0 @@ | ||||
| stages: | ||||
|   - test | ||||
|  | ||||
| test: | ||||
|   stage: test | ||||
|   script: | ||||
|   - apt-get update | ||||
|   - apt-get install -y gettext python3-xapian | ||||
|   - pushd /usr/lib/python3/dist-packages/xapian && ln -s _xapian* _xapian.so && popd | ||||
|   - export PYTHONPATH="/usr/lib/python3/dist-packages:$PYTHONPATH" | ||||
|   - python -c 'import xapian' # Fail immediately if there is a problem with xapian | ||||
|   - pip install -r requirements.txt | ||||
|   - pip install coverage | ||||
|   - ./manage.py compilemessages | ||||
|   - coverage run ./manage.py test | ||||
|   - coverage html | ||||
|   - coverage report | ||||
|   artifacts: | ||||
|     paths: | ||||
|       - coverage_report/ | ||||
|  | ||||
| black: | ||||
|   stage: test | ||||
|   script: | ||||
|     - pip install black | ||||
|     - black --check . | ||||
							
								
								
									
										19
									
								
								.mailmap
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								.mailmap
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| Code <gregoire.duvauchelle@utbm.fr> | ||||
| Cyl <labetowiez@aol.fr> | ||||
| Juste <maaxleblanc@gmail.com> | ||||
| Krophil <pierre.brunet@krophil.fr> | ||||
| Lo-J <renaudg779@gmail.com> | ||||
| Nabos <gnikwo@hotmail.com> | ||||
| Och <francescowitz68@gmail.com> | ||||
| Partoo <joqaste@gmail.com> | ||||
| Skia <skia@hya.sk> <lordbanana25@mailoo.org> | ||||
| Skia <skia@hya.sk> <skia@libskia.so> | ||||
| Sli <klmp200@klmp200.net> <antoine@bartuccio.fr> | ||||
| Soldat <ryan-68@live.fr> | ||||
| Terre <jbaptiste.lenglet+git@gmail.com> | ||||
| Vial <robin.trioux@utbm.fr> | ||||
| Zar <antoine.charmeau@utbm.fr> <antoine.charmeau@laposte.net> | ||||
| root <root@localhost.localdomain> | ||||
| tleb <tleb@openmailbox.org> <theo.lebrun@live.fr> | ||||
| tleb <tleb@openmailbox.org> <theo.lebrun@utbm.fr> | ||||
| Maréchal <thgirod@hotmail.com> | ||||
							
								
								
									
										26
									
								
								.pre-commit-config.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								.pre-commit-config.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| repos: | ||||
|   - repo: https://github.com/astral-sh/ruff-pre-commit | ||||
|     # Ruff version. | ||||
|     rev: v0.8.3 | ||||
|     hooks: | ||||
|       - id: ruff  # just check the code, and print the errors | ||||
|       - id: ruff  # actually fix the fixable errors, but print nothing | ||||
|         args: ["--fix", "--silent"] | ||||
|       # Run the formatter. | ||||
|       - id: ruff-format | ||||
|   - repo: https://github.com/biomejs/pre-commit | ||||
|     rev: "v0.1.0"  # Use the sha / tag you want to point at | ||||
|     hooks: | ||||
|       - id: biome-check | ||||
|         additional_dependencies: ["@biomejs/biome@1.9.3"] | ||||
|   - repo: https://github.com/rtts/djhtml | ||||
|     rev: 3.0.7 | ||||
|     hooks: | ||||
|       - id: djhtml | ||||
|         name: format templates | ||||
|         entry: djhtml --tabwidth 2 | ||||
|         types: ["jinja"] | ||||
|       - id: djcss | ||||
|         name: format scss files | ||||
|         entry: djcss --tabwidth 2 | ||||
|         types: ["scss"] | ||||
							
								
								
									
										1
									
								
								.python-version
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.python-version
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| 3.12 | ||||
							
								
								
									
										106
									
								
								CONTRIBUTING.md
									
									
									
									
									
								
							
							
						
						
									
										106
									
								
								CONTRIBUTING.md
									
									
									
									
									
								
							| @@ -1,106 +0,0 @@ | ||||
| *Contribuer c'est la vie* | ||||
| ========================= | ||||
|  | ||||
| Hey ! Tu veux devenir un mec bien et en plus devenir bon en python si tu l'es pas déjà ? | ||||
| Il se trouve que le sith AE prévu pour l'été 2016 a besoin de toi ! | ||||
|  | ||||
| Pour faire le sith, on utilise le framework Web [Django](https://docs.djangoproject.com/fr/1.11/intro/)   | ||||
| N'hésite pas à lire les tutos et à nous demander (ae.info@utbm.fr). | ||||
|  | ||||
| Bon, passons aux choses sérieuses, pour bidouiller le sith sans le casser :   | ||||
| Ben en fait, tu peux pas le casser, tu vas juste t'amuser comme un petit fou sur un clone du sith. | ||||
|  | ||||
| C'est pas compliqué, il suffit d'avoir [Git](http://www.git-scm.com/book/fr/v2), python et pip (pour faciliter la gestion des paquets python). | ||||
|  | ||||
| Tout d'abord, tu vas avoir besoin d'un compte Gitlab pour pouvoir te connecter.   | ||||
| Ensuite, tu fais : | ||||
| `git clone https://ae-dev.utbm.fr/ae/Sith.git` | ||||
| Avec cette commande, tu clones le sith AE dans le dossier courant. | ||||
|  | ||||
| ```bash | ||||
| cd Sith | ||||
| virtualenv --system-site-packages --python=python3 env | ||||
| source env/bin/activate | ||||
| pip install -r requirements.txt | ||||
| ./manage runserver | ||||
| ``` | ||||
|  | ||||
| Attention aux dépendances système, à voir dans le README.md | ||||
|  | ||||
| Maintenant, faut passer le sith en mode debug dans le fichier de settings personnalisé. | ||||
|  | ||||
| ```bash | ||||
| echo "DEBUG=True" > sith/settings_custom.py | ||||
| echo 'SITH_URL = "localhost:8000"' >> sith/settings_custom.py | ||||
| ``` | ||||
|  | ||||
| Enfin, il s'agit de créer la base de donnée de test lors de la première utilisation | ||||
|  | ||||
| ```bash | ||||
| ./manage.py setup | ||||
| ``` | ||||
|  | ||||
| Et pour lancer le sith, tu fais `python3 manage.py runserver` | ||||
|  | ||||
| Voilà, c'est le sith AE. Il y a des issues dans le gitlab qui sont à régler. Si tu as un domaine qui t'intéresse, une appli que tu voudrais développer, n'hésites pas et contacte-nous. | ||||
| Va, et que l'AE soit avec toi. | ||||
|  | ||||
| # Black | ||||
|  | ||||
| Pour uniformiser le formattage du code nous utilisons [Black](https://github.com/ambv/black). Cela permet d'avoir le même codestyle et donc le codereview prend moins de temps. Tout étant dans le même format, il est plus facile pour chacun de comprendre le code de chacun ! Cela permet aussi d'éviter des erreurs (y parait 🤷♀️). | ||||
|  | ||||
| Installation de black: | ||||
|  | ||||
| ```bash | ||||
| pip install black | ||||
| ``` | ||||
|  | ||||
| ## Sous VsCode: | ||||
| Attention, pour VsCode, Black doit être installé dans votre virtualenv ! | ||||
| Ajouter ces deux lignes dans les settings de VsCode | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "python.formatting.provider": "black", | ||||
|     "editor.formatOnSave": true | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Sous Sublime Text | ||||
| Il faut installer le plugin [sublack](https://packagecontrol.io/packages/sublack) depuis Package Control. | ||||
|  | ||||
| Il suffit ensuite d'ajouter dans les settings du projet (ou en global) | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "sublack.black_on_save": true | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Si vous utilisez le plugin [anaconda](http://damnwidget.github.io/anaconda/), pensez à modifier les paramètres du linter pep8 pour éviter de recevoir des warnings dans le formatage de black | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "pep8_ignore": [ | ||||
|       "E203", | ||||
|       "E266", | ||||
|       "E501", | ||||
|       "W503" | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Sites et doc cools | ||||
| ------------------ | ||||
|  | ||||
| [Classy Class-Based Views](http://ccbv.co.uk/projects/Django/1.11/) | ||||
|  | ||||
| Helpers: | ||||
|  | ||||
| `./manage.py makemessages --ignore "env/*" -e py,jinja` | ||||
|  | ||||
| `for f in $(find . -name "*.py" ! -path "*migration*" ! -path "./env/*" ! -path "./doc/*"); do cat ./doc/header "$f" > /tmp/temp && mv /tmp/temp "$f"; done` | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
							
								
								
									
										3
									
								
								CONTRIBUTING.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								CONTRIBUTING.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| Pour contribuer au projet, vous pouvez vous référer à la documentation disponible à https://sith-ae.readthedocs.io/. | ||||
|  | ||||
| Et n'oubliez pas, contribuer c'est la vie ! | ||||
							
								
								
									
										21
									
								
								LICENSE.old
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								LICENSE.old
									
									
									
									
									
								
							| @@ -1,21 +0,0 @@ | ||||
| The MIT License (MIT) | ||||
|  | ||||
| Copyright (c) 2016 Skia | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
| in the Software without restriction, including without limitation the rights | ||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| copies of the Software, and to permit persons to whom the Software is | ||||
| furnished to do so, subject to the following conditions: | ||||
|  | ||||
| The above copyright notice and this permission notice shall be included in all | ||||
| copies or substantial portions of the Software. | ||||
|  | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
| SOFTWARE. | ||||
							
								
								
									
										113
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										113
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,104 +1,21 @@ | ||||
| [](https://ae-dev.utbm.fr/ae/Sith/commits/master) | ||||
| [](https://ae-dev.utbm.fr/ae/Sith/commits/master) | ||||
| [](https://github.com/ambv/black) | ||||
| [](https://ae-dev.zulipchat.com) | ||||
| # Sith | ||||
|  | ||||
| ## Sith AE | ||||
| [](#) | ||||
| [](https://www.gnu.org/licenses/gpl-3.0) | ||||
| [](#) | ||||
| [](https://ae-utbm.github.io/sith) | ||||
| [](https://squidfunk.github.io/mkdocs-material/) | ||||
| [](https://biomejs.dev) | ||||
| [](https://discord.gg/xk9wfpsufm) | ||||
|  | ||||
| ### Dependencies: | ||||
| See requirements.txt | ||||
| ### This is the source code of the UTBM's student association available at [https://ae.utbm.fr/](https://ae.utbm.fr/). | ||||
|  | ||||
| You may need to install some dev libraries like `libmysqlclient-dev`, `libssl-dev`, `libjpeg-dev`, `python3-xapian`, or `zlib1g-dev` to install all the | ||||
| requiered dependancies with pip. You may also need `mysql-client`. Don't also forget `python3-dev` if you don't have it | ||||
| already. | ||||
| All documentation is in the `docs` directory and online at [https://ae-utbm.github.io/sith](https://ae-utbm.github.io/sith). This documentation is written in French because it targets a French audience and it's too much work to maintain two versions. The code and code comments are strictly written in English. | ||||
|  | ||||
| You can check all of them with: | ||||
|  | ||||
| ```bash | ||||
| sudo apt install libmysqlclient-dev libssl-dev libjpeg-dev zlib1g-dev python3-dev libffi-dev python3-dev libgraphviz-dev pkg-config python3-xapian gettext | ||||
| ``` | ||||
|  | ||||
| On macos, you will need homebrew | ||||
|  | ||||
| ```bash | ||||
| brew install xapian | ||||
| ``` | ||||
|  | ||||
| If it doesn't work it's because it need [this pull request](https://github.com/Homebrew/homebrew-core/pull/34835) to be validated. | ||||
|  | ||||
| The development is done with sqlite, but it is advised to set a more robust DBMS for production (Postgresql for example) | ||||
|  | ||||
| ### Get started | ||||
|  | ||||
| To start working on the project, just run the following commands: | ||||
|  | ||||
| ```bash | ||||
| git clone https://ae-dev.utbm.fr/ae/Sith.git | ||||
| cd Sith | ||||
| virtualenv --system-site-packages --python=python3 env | ||||
| source env/bin/activate | ||||
| pip install -r requirements.txt | ||||
| ./manage.py setup | ||||
| ``` | ||||
|  | ||||
| To start the simple development server, just run `python3 manage.py runserver` | ||||
|  | ||||
| For more informations, check out the CONTRIBUTING.md file. | ||||
|  | ||||
| ### Logging errors with sentry | ||||
|  | ||||
| To connect the app to sentry.io, you must set the variable SENTRY_DSN in your settings custom. It's composed of the full link given on your sentry project | ||||
|  | ||||
| ### Generating documentation | ||||
|  | ||||
| There is a Doxyfile at the root of the project, meaning that if you have Doxygen, you can run `doxygen Doxyfile` to | ||||
| generate a complete HTML documentation that will be available in the *./doc/html/* folder. | ||||
|  | ||||
| ### Collecting statics for production: | ||||
|  | ||||
| We use scss in the project. In development environment (DEBUG=True), scss is compiled every time the file is needed. For production, it assumes you have already compiled every files and to do so, you need to use the following commands :  | ||||
|  | ||||
| ```bash | ||||
| ./manage.py collectstatic # To collect statics | ||||
| ./manage.py compilestatic # To compile scss in those statics | ||||
| ``` | ||||
|  | ||||
| ### Misc about development | ||||
|  | ||||
| #### Controlling the rights | ||||
|  | ||||
| When you need to protect an object, there are three levels: | ||||
|   * Editing the object properties | ||||
|   * Editing the object various values | ||||
|   * Viewing the object | ||||
|  | ||||
| Now you have many solutions in your model: | ||||
|   * You can define a `is_owned_by(self, user)`, a `can_be_edited_by(self, user)`, and/or a `can_be_viewed_by(self, user)` | ||||
|     method, each returning True is the user passed can edit/view the object, False otherwise.    | ||||
|     This allows you to make complex request when the group solution is not powerful enough.     | ||||
|     It's useful too when you want to define class-wide permissions, e.g. the club members, that are viewable only for | ||||
|     Subscribers. | ||||
|   * You can add an `owner_group` field, as a ForeignKey to Group.  Second is an `edit_groups` field, as a ManyToMany to | ||||
|     Group, and third is a `view_groups`, same as for edit. | ||||
|  | ||||
| Finally, when building a class based view, which is highly advised, you just have to inherit it from CanEditPropMixin, | ||||
| CanEditMixin, or CanViewMixin, which are located in core.views. Your view will then be protected using either the | ||||
| appropriate group fields, or the right method to check user permissions. | ||||
|  | ||||
| #### Counting the number of line of code | ||||
|  | ||||
| ```bash | ||||
| sudo apt install cloc | ||||
| cloc --exclude-dir=doc,env . | ||||
| ``` | ||||
|  | ||||
| #### Updating doc/SYNTAX.md | ||||
|  | ||||
| If you make an update in the Markdown syntax parser, it's good to document | ||||
| update the syntax reference page in `doc/SYNTAX.md`. But updating this file will | ||||
| break the tests if you don't update the corresponding `doc/SYNTAX.html` file at | ||||
| the same time.   | ||||
| To do that, simply run `./manage.py markdown > doc/SYNTAX.html`, | ||||
| and the tests should pass again. | ||||
| #### If you want to contribute, here's how we recommend to read the docs: | ||||
|  | ||||
| * First, it's advised to read the about part of the project to understand the goals and the mindset of the current and previous maintainers and know what to expect to learn. | ||||
| * If in the first part you realize that you need more background about what we use, we provide some links to tutorials and documentation at the end of our documentation. Feel free to use it and complete it with what you found helpful. | ||||
| * Keep in mind that this documentation is thought to be read in order. | ||||
|  | ||||
| > This project is licensed under GNU GPL, see the LICENSE file at the top of the repository for more details. | ||||
|   | ||||
							
								
								
									
										9
									
								
								TODO.md
									
									
									
									
									
								
							
							
						
						
									
										9
									
								
								TODO.md
									
									
									
									
									
								
							| @@ -1,9 +0,0 @@ | ||||
| # TODO | ||||
|  | ||||
| ## Easter eggs | ||||
|  | ||||
|   * 'A' 'L' 'L' 'O': Entendre le Allooo de Madame Coucoune | ||||
|   * idem avec cacafe | ||||
|   * Un meat spin quelque part | ||||
|   * Konami code | ||||
|  | ||||
| @@ -1,23 +1,14 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # Copyright 2023 © AE UTBM | ||||
| # ae@utbm.fr / ae.info@utbm.fr | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # This file is part of the website of the UTBM Student Association (AE UTBM), | ||||
| # https://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # You can find the source code of the website at https://github.com/ae-utbm/sith | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3) | ||||
| # SEE : https://raw.githubusercontent.com/ae-utbm/sith/master/LICENSE | ||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | ||||
| # | ||||
| # | ||||
|   | ||||
| @@ -1,31 +1,30 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # Copyright 2023 © AE UTBM | ||||
| # ae@utbm.fr / ae.info@utbm.fr | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # This file is part of the website of the UTBM Student Association (AE UTBM), | ||||
| # https://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # You can find the source code of the website at https://github.com/ae-utbm/sith | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3) | ||||
| # SEE : https://raw.githubusercontent.com/ae-utbm/sith/master/LICENSE | ||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | ||||
| # | ||||
| # | ||||
|  | ||||
| from django.contrib import admin | ||||
|  | ||||
| from accounting.models import * | ||||
|  | ||||
| from accounting.models import ( | ||||
|     AccountingType, | ||||
|     BankAccount, | ||||
|     ClubAccount, | ||||
|     Company, | ||||
|     GeneralJournal, | ||||
|     Label, | ||||
|     Operation, | ||||
|     SimplifiedAccountingType, | ||||
| ) | ||||
|  | ||||
| admin.site.register(BankAccount) | ||||
| admin.site.register(ClubAccount) | ||||
|   | ||||
							
								
								
									
										23
									
								
								accounting/api.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								accounting/api.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| from typing import Annotated | ||||
|  | ||||
| from annotated_types import MinLen | ||||
| from ninja_extra import ControllerBase, api_controller, paginate, route | ||||
| from ninja_extra.pagination import PageNumberPaginationExtra | ||||
| from ninja_extra.schemas import PaginatedResponseSchema | ||||
|  | ||||
| from accounting.models import ClubAccount, Company | ||||
| from accounting.schemas import ClubAccountSchema, CompanySchema | ||||
| from core.api_permissions import CanAccessLookup | ||||
|  | ||||
|  | ||||
| @api_controller("/lookup", permissions=[CanAccessLookup]) | ||||
| class AccountingController(ControllerBase): | ||||
|     @route.get("/club-account", response=PaginatedResponseSchema[ClubAccountSchema]) | ||||
|     @paginate(PageNumberPaginationExtra, page_size=50) | ||||
|     def search_club_account(self, search: Annotated[str, MinLen(1)]): | ||||
|         return ClubAccount.objects.filter(name__icontains=search).values() | ||||
|  | ||||
|     @route.get("/company", response=PaginatedResponseSchema[CompanySchema]) | ||||
|     @paginate(PageNumberPaginationExtra, page_size=50) | ||||
|     def search_company(self, search: Annotated[str, MinLen(1)]): | ||||
|         return Company.objects.filter(name__icontains=search).values() | ||||
| @@ -1,13 +1,13 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
| import django.core.validators | ||||
| import django.db.models.deletion | ||||
| from django.db import migrations, models | ||||
|  | ||||
| import accounting.models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [] | ||||
|  | ||||
|     operations = [ | ||||
| @@ -243,6 +243,7 @@ class Migration(migrations.Migration): | ||||
|                         verbose_name="accounting type", | ||||
|                         to="accounting.AccountingType", | ||||
|                         blank=True, | ||||
|                         on_delete=django.db.models.deletion.CASCADE, | ||||
|                     ), | ||||
|                 ), | ||||
|             ], | ||||
| @@ -267,6 +268,7 @@ class Migration(migrations.Migration): | ||||
|                         verbose_name="simplified accounting types", | ||||
|                         to="accounting.AccountingType", | ||||
|                         related_name="simplified_types", | ||||
|                         on_delete=django.db.models.deletion.CASCADE, | ||||
|                     ), | ||||
|                 ), | ||||
|             ], | ||||
|   | ||||
| @@ -1,11 +1,10 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import django.db.models.deletion | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [ | ||||
|         ("club", "0001_initial"), | ||||
|         ("accounting", "0001_initial"), | ||||
| @@ -22,6 +21,7 @@ class Migration(migrations.Migration): | ||||
|                 verbose_name="invoice", | ||||
|                 to="core.SithFile", | ||||
|                 blank=True, | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
| @@ -31,12 +31,14 @@ class Migration(migrations.Migration): | ||||
|                 verbose_name="journal", | ||||
|                 to="accounting.GeneralJournal", | ||||
|                 related_name="operations", | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name="operation", | ||||
|             name="linked_operation", | ||||
|             field=models.OneToOneField( | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|                 blank=True, | ||||
|                 to="accounting.Operation", | ||||
|                 null=True, | ||||
| @@ -54,6 +56,7 @@ class Migration(migrations.Migration): | ||||
|                 verbose_name="simple type", | ||||
|                 to="accounting.SimplifiedAccountingType", | ||||
|                 blank=True, | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
| @@ -63,6 +66,7 @@ class Migration(migrations.Migration): | ||||
|                 verbose_name="club account", | ||||
|                 to="accounting.ClubAccount", | ||||
|                 related_name="journals", | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
| @@ -72,23 +76,30 @@ class Migration(migrations.Migration): | ||||
|                 verbose_name="bank account", | ||||
|                 to="accounting.BankAccount", | ||||
|                 related_name="club_accounts", | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name="clubaccount", | ||||
|             name="club", | ||||
|             field=models.ForeignKey( | ||||
|                 verbose_name="club", to="club.Club", related_name="club_account" | ||||
|                 verbose_name="club", | ||||
|                 to="club.Club", | ||||
|                 related_name="club_account", | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name="bankaccount", | ||||
|             name="club", | ||||
|             field=models.ForeignKey( | ||||
|                 verbose_name="club", to="club.Club", related_name="bank_accounts" | ||||
|                 verbose_name="club", | ||||
|                 to="club.Club", | ||||
|                 related_name="bank_accounts", | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AlterUniqueTogether( | ||||
|             name="operation", unique_together=set([("number", "journal")]) | ||||
|             name="operation", unique_together={("number", "journal")} | ||||
|         ), | ||||
|     ] | ||||
|   | ||||
| @@ -1,12 +1,10 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
| import phonenumber_field.modelfields | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("accounting", "0002_auto_20160824_2152")] | ||||
|  | ||||
|     operations = [ | ||||
|   | ||||
| @@ -1,12 +1,10 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
| import django.db.models.deletion | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("accounting", "0003_auto_20160824_2203")] | ||||
|  | ||||
|     operations = [ | ||||
| @@ -29,6 +27,7 @@ class Migration(migrations.Migration): | ||||
|                         related_name="labels", | ||||
|                         verbose_name="club account", | ||||
|                         to="accounting.ClubAccount", | ||||
|                         on_delete=django.db.models.deletion.CASCADE, | ||||
|                     ), | ||||
|                 ), | ||||
|             ], | ||||
| @@ -46,6 +45,6 @@ class Migration(migrations.Migration): | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AlterUniqueTogether( | ||||
|             name="label", unique_together=set([("name", "club_account")]) | ||||
|             name="label", unique_together={("name", "club_account")} | ||||
|         ), | ||||
|     ] | ||||
|   | ||||
| @@ -1,11 +1,9 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("accounting", "0004_auto_20161005_1505")] | ||||
|  | ||||
|     operations = [ | ||||
|   | ||||
| @@ -1,59 +1,60 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # Copyright 2023 © AE UTBM | ||||
| # ae@utbm.fr / ae.info@utbm.fr | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # This file is part of the website of the UTBM Student Association (AE UTBM), | ||||
| # https://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # You can find the source code of the website at https://github.com/ae-utbm/sith | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3) | ||||
| # SEE : https://raw.githubusercontent.com/ae-utbm/sith/master/LICENSE | ||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | ||||
| # | ||||
| # | ||||
|  | ||||
| from django.core.urlresolvers import reverse | ||||
| from django.core.exceptions import ValidationError | ||||
| from django.core import validators | ||||
| from django.db import models | ||||
| from django.conf import settings | ||||
| from django.utils.translation import ugettext_lazy as _ | ||||
| from django.template import defaultfilters | ||||
|  | ||||
| from phonenumber_field.modelfields import PhoneNumberField | ||||
|  | ||||
| from decimal import Decimal | ||||
| from core.models import User, SithFile | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.core import validators | ||||
| from django.core.exceptions import ValidationError | ||||
| from django.db import models | ||||
| from django.template import defaultfilters | ||||
| from django.urls import reverse | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| from phonenumber_field.modelfields import PhoneNumberField | ||||
|  | ||||
| from club.models import Club | ||||
| from core.models import SithFile, User | ||||
|  | ||||
|  | ||||
| class CurrencyField(models.DecimalField): | ||||
|     """ | ||||
|     This is a custom database field used for currency | ||||
|     """ | ||||
|     """Custom database field used for currency.""" | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         kwargs["max_digits"] = 12 | ||||
|         kwargs["decimal_places"] = 2 | ||||
|         super(CurrencyField, self).__init__(*args, **kwargs) | ||||
|         super().__init__(*args, **kwargs) | ||||
|  | ||||
|     def to_python(self, value): | ||||
|         try: | ||||
|             return super(CurrencyField, self).to_python(value).quantize(Decimal("0.01")) | ||||
|             return super().to_python(value).quantize(Decimal("0.01")) | ||||
|         except AttributeError: | ||||
|             return None | ||||
|  | ||||
|  | ||||
| if settings.TESTING: | ||||
|     from model_bakery import baker | ||||
|  | ||||
|     baker.generators.add( | ||||
|         CurrencyField, | ||||
|         lambda: baker.random_gen.gen_decimal(max_digits=8, decimal_places=2), | ||||
|     ) | ||||
| else:  # pragma: no cover | ||||
|     # baker is only used in tests, so we don't need coverage for this part | ||||
|     pass | ||||
|  | ||||
|  | ||||
| # Accounting classes | ||||
|  | ||||
|  | ||||
| @@ -70,31 +71,8 @@ class Company(models.Model): | ||||
|     class Meta: | ||||
|         verbose_name = _("company") | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         return False | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         for club in user.memberships.filter(end_date=None).all(): | ||||
|             if club and club.role == settings.SITH_CLUB_ROLES_ID["Treasurer"]: | ||||
|                 return True | ||||
|         return False | ||||
|  | ||||
|     def can_be_viewed_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be viewed by the given user | ||||
|         """ | ||||
|         for club in user.memberships.filter(end_date=None).all(): | ||||
|             if club and club.role >= settings.SITH_CLUB_ROLES_ID["Treasurer"]: | ||||
|                 return True | ||||
|         return False | ||||
|     def __str__(self): | ||||
|         return self.name | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("accounting:co_edit", kwargs={"co_id": self.id}) | ||||
| @@ -102,89 +80,100 @@ class Company(models.Model): | ||||
|     def get_display_name(self): | ||||
|         return self.name | ||||
|  | ||||
|     def __str__(self): | ||||
|         return self.name | ||||
|     def is_owned_by(self, user): | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         return user.memberships.filter( | ||||
|             end_date=None, club__role=settings.SITH_CLUB_ROLES_ID["Treasurer"] | ||||
|         ).exists() | ||||
|  | ||||
|     def can_be_viewed_by(self, user): | ||||
|         """Check if that object can be viewed by the given user.""" | ||||
|         return user.memberships.filter( | ||||
|             end_date=None, club__role_gte=settings.SITH_CLUB_ROLES_ID["Treasurer"] | ||||
|         ).exists() | ||||
|  | ||||
|  | ||||
| class BankAccount(models.Model): | ||||
|     name = models.CharField(_("name"), max_length=30) | ||||
|     iban = models.CharField(_("iban"), max_length=255, blank=True) | ||||
|     number = models.CharField(_("account number"), max_length=255, blank=True) | ||||
|     club = models.ForeignKey(Club, related_name="bank_accounts", verbose_name=_("club")) | ||||
|     club = models.ForeignKey( | ||||
|         Club, | ||||
|         related_name="bank_accounts", | ||||
|         verbose_name=_("club"), | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("Bank account") | ||||
|         ordering = ["club", "name"] | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         m = self.club.get_membership_for(user) | ||||
|         if m is not None and m.role >= settings.SITH_CLUB_ROLES_ID["Treasurer"]: | ||||
|             return True | ||||
|         return False | ||||
|     def __str__(self): | ||||
|         return self.name | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("accounting:bank_details", kwargs={"b_account_id": self.id}) | ||||
|  | ||||
|     def __str__(self): | ||||
|         return self.name | ||||
|     def is_owned_by(self, user): | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         m = self.club.get_membership_for(user) | ||||
|         return m is not None and m.role >= settings.SITH_CLUB_ROLES_ID["Treasurer"] | ||||
|  | ||||
|  | ||||
| class ClubAccount(models.Model): | ||||
|     name = models.CharField(_("name"), max_length=30) | ||||
|     club = models.ForeignKey(Club, related_name="club_account", verbose_name=_("club")) | ||||
|     club = models.ForeignKey( | ||||
|         Club, | ||||
|         related_name="club_account", | ||||
|         verbose_name=_("club"), | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     bank_account = models.ForeignKey( | ||||
|         BankAccount, related_name="club_accounts", verbose_name=_("bank account") | ||||
|         BankAccount, | ||||
|         related_name="club_accounts", | ||||
|         verbose_name=_("bank account"), | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("Club account") | ||||
|         ordering = ["bank_account", "name"] | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         return False | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         m = self.club.get_membership_for(user) | ||||
|         if m and m.role == settings.SITH_CLUB_ROLES_ID["Treasurer"]: | ||||
|             return True | ||||
|         return False | ||||
|  | ||||
|     def can_be_viewed_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be viewed by the given user | ||||
|         """ | ||||
|         m = self.club.get_membership_for(user) | ||||
|         if m and m.role >= settings.SITH_CLUB_ROLES_ID["Treasurer"]: | ||||
|             return True | ||||
|         return False | ||||
|  | ||||
|     def has_open_journal(self): | ||||
|         for j in self.journals.all(): | ||||
|             if not j.closed: | ||||
|                 return True | ||||
|         return False | ||||
|  | ||||
|     def get_open_journal(self): | ||||
|         return self.journals.filter(closed=False).first() | ||||
|     def __str__(self): | ||||
|         return self.name | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("accounting:club_details", kwargs={"c_account_id": self.id}) | ||||
|  | ||||
|     def __str__(self): | ||||
|         return self.name | ||||
|     def is_owned_by(self, user): | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         m = self.club.get_membership_for(user) | ||||
|         return m and m.role == settings.SITH_CLUB_ROLES_ID["Treasurer"] | ||||
|  | ||||
|     def can_be_viewed_by(self, user): | ||||
|         """Check if that object can be viewed by the given user.""" | ||||
|         m = self.club.get_membership_for(user) | ||||
|         return m and m.role >= settings.SITH_CLUB_ROLES_ID["Treasurer"] | ||||
|  | ||||
|     def has_open_journal(self): | ||||
|         return self.journals.filter(closed=False).exists() | ||||
|  | ||||
|     def get_open_journal(self): | ||||
|         return self.journals.filter(closed=False).first() | ||||
|  | ||||
|     def get_display_name(self): | ||||
|         return _("%(club_account)s on %(bank_account)s") % { | ||||
| @@ -194,16 +183,18 @@ class ClubAccount(models.Model): | ||||
|  | ||||
|  | ||||
| class GeneralJournal(models.Model): | ||||
|     """ | ||||
|     Class storing all the operations for a period of time | ||||
|     """ | ||||
|     """Class storing all the operations for a period of time.""" | ||||
|  | ||||
|     start_date = models.DateField(_("start date")) | ||||
|     end_date = models.DateField(_("end date"), null=True, blank=True, default=None) | ||||
|     name = models.CharField(_("name"), max_length=40) | ||||
|     closed = models.BooleanField(_("is closed"), default=False) | ||||
|     club_account = models.ForeignKey( | ||||
|         ClubAccount, related_name="journals", null=False, verbose_name=_("club account") | ||||
|         ClubAccount, | ||||
|         related_name="journals", | ||||
|         null=False, | ||||
|         verbose_name=_("club account"), | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     amount = CurrencyField(_("amount"), default=0) | ||||
|     effective_amount = CurrencyField(_("effective_amount"), default=0) | ||||
| @@ -212,34 +203,28 @@ class GeneralJournal(models.Model): | ||||
|         verbose_name = _("General journal") | ||||
|         ordering = ["-start_date"] | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         if self.club_account.can_be_edited_by(user): | ||||
|             return True | ||||
|         return False | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         if self.club_account.can_be_edited_by(user): | ||||
|             return True | ||||
|         return False | ||||
|  | ||||
|     def can_be_viewed_by(self, user): | ||||
|         return self.club_account.can_be_viewed_by(user) | ||||
|     def __str__(self): | ||||
|         return self.name | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("accounting:journal_details", kwargs={"j_id": self.id}) | ||||
|  | ||||
|     def __str__(self): | ||||
|         return self.name | ||||
|     def is_owned_by(self, user): | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         return self.club_account.can_be_edited_by(user) | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         return self.club_account.can_be_edited_by(user) | ||||
|  | ||||
|     def can_be_viewed_by(self, user): | ||||
|         return self.club_account.can_be_viewed_by(user) | ||||
|  | ||||
|     def update_amounts(self): | ||||
|         self.amount = 0 | ||||
| @@ -257,13 +242,15 @@ class GeneralJournal(models.Model): | ||||
|  | ||||
|  | ||||
| class Operation(models.Model): | ||||
|     """ | ||||
|     An operation is a line in the journal, a debit or a credit | ||||
|     """ | ||||
|     """An operation is a line in the journal, a debit or a credit.""" | ||||
|  | ||||
|     number = models.IntegerField(_("number")) | ||||
|     journal = models.ForeignKey( | ||||
|         GeneralJournal, related_name="operations", null=False, verbose_name=_("journal") | ||||
|         GeneralJournal, | ||||
|         related_name="operations", | ||||
|         null=False, | ||||
|         verbose_name=_("journal"), | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     amount = CurrencyField(_("amount")) | ||||
|     date = models.DateField(_("date")) | ||||
| @@ -282,6 +269,7 @@ class Operation(models.Model): | ||||
|         verbose_name=_("invoice"), | ||||
|         null=True, | ||||
|         blank=True, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     done = models.BooleanField(_("is done"), default=False) | ||||
|     simpleaccounting_type = models.ForeignKey( | ||||
| @@ -290,6 +278,7 @@ class Operation(models.Model): | ||||
|         verbose_name=_("simple type"), | ||||
|         null=True, | ||||
|         blank=True, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     accounting_type = models.ForeignKey( | ||||
|         "AccountingType", | ||||
| @@ -297,6 +286,7 @@ class Operation(models.Model): | ||||
|         verbose_name=_("accounting type"), | ||||
|         null=True, | ||||
|         blank=True, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     label = models.ForeignKey( | ||||
|         "Label", | ||||
| @@ -328,12 +318,25 @@ class Operation(models.Model): | ||||
|         null=True, | ||||
|         blank=True, | ||||
|         default=None, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         unique_together = ("number", "journal") | ||||
|         ordering = ["-number"] | ||||
|  | ||||
|     def __str__(self): | ||||
|         return f"{self.amount} € | {self.date} | {self.accounting_type} | {self.done}" | ||||
|  | ||||
|     def save(self, *args, **kwargs): | ||||
|         if self.number is None: | ||||
|             self.number = self.journal.operations.count() + 1 | ||||
|         super().save(*args, **kwargs) | ||||
|         self.journal.update_amounts() | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("accounting:journal_details", kwargs={"j_id": self.journal.id}) | ||||
|  | ||||
|     def __getattribute__(self, attr): | ||||
|         if attr == "target": | ||||
|             return self.get_target() | ||||
| @@ -341,7 +344,7 @@ class Operation(models.Model): | ||||
|             return object.__getattribute__(self, attr) | ||||
|  | ||||
|     def clean(self): | ||||
|         super(Operation, self).clean() | ||||
|         super().clean() | ||||
|         if self.date is None: | ||||
|             raise ValidationError(_("The date must be set.")) | ||||
|         elif self.date < self.journal.start_date: | ||||
| @@ -387,55 +390,31 @@ class Operation(models.Model): | ||||
|             tar = Company.objects.filter(id=self.target_id).first() | ||||
|         return tar | ||||
|  | ||||
|     def save(self): | ||||
|         if self.number is None: | ||||
|             self.number = self.journal.operations.count() + 1 | ||||
|         super(Operation, self).save() | ||||
|         self.journal.update_amounts() | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         if self.journal.closed: | ||||
|             return False | ||||
|         m = self.journal.club_account.club.get_membership_for(user) | ||||
|         if m is not None and m.role >= settings.SITH_CLUB_ROLES_ID["Treasurer"]: | ||||
|             return True | ||||
|         return False | ||||
|         return m is not None and m.role >= settings.SITH_CLUB_ROLES_ID["Treasurer"] | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         if self.journal.closed: | ||||
|             return False | ||||
|         m = self.journal.club_account.club.get_membership_for(user) | ||||
|         if m is not None and m.role == settings.SITH_CLUB_ROLES_ID["Treasurer"]: | ||||
|             return True | ||||
|         return False | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("accounting:journal_details", kwargs={"j_id": self.journal.id}) | ||||
|  | ||||
|     def __str__(self): | ||||
|         return "%d € | %s | %s | %s" % ( | ||||
|             self.amount, | ||||
|             self.date, | ||||
|             self.accounting_type, | ||||
|             self.done, | ||||
|         ) | ||||
|         return m is not None and m.role == settings.SITH_CLUB_ROLES_ID["Treasurer"] | ||||
|  | ||||
|  | ||||
| class AccountingType(models.Model): | ||||
|     """ | ||||
|     Class describing the accounting types. | ||||
|     """Accounting types. | ||||
|  | ||||
|     Thoses are numbers used in accounting to classify operations | ||||
|     Those are numbers used in accounting to classify operations | ||||
|     """ | ||||
|  | ||||
|     code = models.CharField( | ||||
| @@ -462,37 +441,43 @@ class AccountingType(models.Model): | ||||
|         verbose_name = _("accounting type") | ||||
|         ordering = ["movement_type", "code"] | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         return False | ||||
|     def __str__(self): | ||||
|         return self.code + " - " + self.get_movement_type_display() + " - " + self.label | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("accounting:type_list") | ||||
|  | ||||
|     def __str__(self): | ||||
|         return self.code + " - " + self.get_movement_type_display() + " - " + self.label | ||||
|     def is_owned_by(self, user): | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) | ||||
|  | ||||
|  | ||||
| class SimplifiedAccountingType(models.Model): | ||||
|     """ | ||||
|     Class describing the simplified accounting types. | ||||
|     """ | ||||
|     """Simplified version of `AccountingType`.""" | ||||
|  | ||||
|     label = models.CharField(_("label"), max_length=128) | ||||
|     accounting_type = models.ForeignKey( | ||||
|         AccountingType, | ||||
|         related_name="simplified_types", | ||||
|         verbose_name=_("simplified accounting types"), | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("simplified type") | ||||
|         ordering = ["accounting_type__movement_type", "accounting_type__code"] | ||||
|  | ||||
|     def __str__(self): | ||||
|         return ( | ||||
|             f"{self.get_movement_type_display()} " | ||||
|             f"- {self.accounting_type.code} - {self.label}" | ||||
|         ) | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("accounting:simple_type_list") | ||||
|  | ||||
|     @property | ||||
|     def movement_type(self): | ||||
|         return self.accounting_type.movement_type | ||||
| @@ -500,25 +485,16 @@ class SimplifiedAccountingType(models.Model): | ||||
|     def get_movement_type_display(self): | ||||
|         return self.accounting_type.get_movement_type_display() | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("accounting:simple_type_list") | ||||
|  | ||||
|     def __str__(self): | ||||
|         return ( | ||||
|             self.get_movement_type_display() | ||||
|             + " - " | ||||
|             + self.accounting_type.code | ||||
|             + " - " | ||||
|             + self.label | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class Label(models.Model): | ||||
|     """Label allow a club to sort its operations""" | ||||
|     """Label allow a club to sort its operations.""" | ||||
|  | ||||
|     name = models.CharField(_("label"), max_length=64) | ||||
|     club_account = models.ForeignKey( | ||||
|         ClubAccount, related_name="labels", verbose_name=_("club account") | ||||
|         ClubAccount, | ||||
|         related_name="labels", | ||||
|         verbose_name=_("club account"), | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
| @@ -533,6 +509,8 @@ class Label(models.Model): | ||||
|         ) | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         return self.club_account.is_owned_by(user) | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
|   | ||||
							
								
								
									
										15
									
								
								accounting/schemas.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								accounting/schemas.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| from ninja import ModelSchema | ||||
|  | ||||
| from accounting.models import ClubAccount, Company | ||||
|  | ||||
|  | ||||
| class ClubAccountSchema(ModelSchema): | ||||
|     class Meta: | ||||
|         model = ClubAccount | ||||
|         fields = ["id", "name"] | ||||
|  | ||||
|  | ||||
| class CompanySchema(ModelSchema): | ||||
|     class Meta: | ||||
|         model = Company | ||||
|         fields = ["id", "name"] | ||||
| @@ -0,0 +1,60 @@ | ||||
| import { AjaxSelect } from "#core:core/components/ajax-select-base"; | ||||
| import { registerComponent } from "#core:utils/web-components"; | ||||
| import type { TomOption } from "tom-select/dist/types/types"; | ||||
| import type { escape_html } from "tom-select/dist/types/utils"; | ||||
| import { | ||||
|   type ClubAccountSchema, | ||||
|   type CompanySchema, | ||||
|   accountingSearchClubAccount, | ||||
|   accountingSearchCompany, | ||||
| } from "#openapi"; | ||||
|  | ||||
| @registerComponent("club-account-ajax-select") | ||||
| export class ClubAccountAjaxSelect extends AjaxSelect { | ||||
|   protected valueField = "id"; | ||||
|   protected labelField = "name"; | ||||
|   protected searchField = ["code", "name"]; | ||||
|  | ||||
|   protected async search(query: string): Promise<TomOption[]> { | ||||
|     const resp = await accountingSearchClubAccount({ query: { search: query } }); | ||||
|     if (resp.data) { | ||||
|       return resp.data.results; | ||||
|     } | ||||
|     return []; | ||||
|   } | ||||
|  | ||||
|   protected renderOption(item: ClubAccountSchema, sanitize: typeof escape_html) { | ||||
|     return `<div class="select-item"> | ||||
|             <span class="select-item-text">${sanitize(item.name)}</span> | ||||
|           </div>`; | ||||
|   } | ||||
|  | ||||
|   protected renderItem(item: ClubAccountSchema, sanitize: typeof escape_html) { | ||||
|     return `<span>${sanitize(item.name)}</span>`; | ||||
|   } | ||||
| } | ||||
|  | ||||
| @registerComponent("company-ajax-select") | ||||
| export class CompanyAjaxSelect extends AjaxSelect { | ||||
|   protected valueField = "id"; | ||||
|   protected labelField = "name"; | ||||
|   protected searchField = ["code", "name"]; | ||||
|  | ||||
|   protected async search(query: string): Promise<TomOption[]> { | ||||
|     const resp = await accountingSearchCompany({ query: { search: query } }); | ||||
|     if (resp.data) { | ||||
|       return resp.data.results; | ||||
|     } | ||||
|     return []; | ||||
|   } | ||||
|  | ||||
|   protected renderOption(item: CompanySchema, sanitize: typeof escape_html) { | ||||
|     return `<div class="select-item"> | ||||
|             <span class="select-item-text">${sanitize(item.name)}</span> | ||||
|           </div>`; | ||||
|   } | ||||
|  | ||||
|   protected renderItem(item: CompanySchema, sanitize: typeof escape_html) { | ||||
|     return `<span>${sanitize(item.name)}</span>`; | ||||
|   } | ||||
| } | ||||
| @@ -1,27 +1,27 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Accounting type list{% endtrans %} | ||||
|   {% trans %}Accounting type list{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="accounting"> | ||||
|         <p> | ||||
|         <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|         {% trans %}Accounting types{% endtrans %} | ||||
|         </p> | ||||
|         <hr> | ||||
|         <p><a href="{{ url('accounting:type_new') }}">{% trans %}New accounting type{% endtrans %}</a></p> | ||||
|         {% if accountingtype_list %} | ||||
|         <h3>{% trans %}Accounting type list{% endtrans %}</h3> | ||||
|         <ul> | ||||
|             {% for a in accountingtype_list  %} | ||||
|             <li><a href="{{ url('accounting:type_edit', type_id=a.id) }}">{{ a }}</a></li> | ||||
|             {% endfor %} | ||||
|         </ul> | ||||
|         {% else %} | ||||
|         {% trans %}There is no types in this website.{% endtrans %} | ||||
|         {% endif %} | ||||
|     </div> | ||||
|   <div id="accounting"> | ||||
|     <p> | ||||
|       <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|       {% trans %}Accounting types{% endtrans %} | ||||
|     </p> | ||||
|     <hr> | ||||
|     <p><a href="{{ url('accounting:type_new') }}">{% trans %}New accounting type{% endtrans %}</a></p> | ||||
|     {% if accountingtype_list %} | ||||
|       <h3>{% trans %}Accounting type list{% endtrans %}</h3> | ||||
|       <ul> | ||||
|         {% for a in accountingtype_list  %} | ||||
|           <li><a href="{{ url('accounting:type_edit', type_id=a.id) }}">{{ a }}</a></li> | ||||
|         {% endfor %} | ||||
|       </ul> | ||||
|     {% else %} | ||||
|       {% trans %}There is no types in this website.{% endtrans %} | ||||
|     {% endif %} | ||||
|   </div> | ||||
| {% endblock %} | ||||
|  | ||||
|   | ||||
| @@ -1,37 +1,37 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Bank account: {% endtrans %}{{ object.name }} | ||||
|   {% trans %}Bank account: {% endtrans %}{{ object.name }} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="accounting"> | ||||
|         <p> | ||||
|         <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|         {{ object.name }} | ||||
|         </p> | ||||
|         <hr> | ||||
|         <h2>{% trans %}Bank account: {% endtrans %}{{ object.name }}</h2> | ||||
|         {% if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) and not object.club_accounts.exists() %} | ||||
|         <a href="{{ url('accounting:bank_delete', b_account_id=object.id) }}">{% trans %}Delete{% endtrans %}</a> | ||||
|         {% endif %} | ||||
|         <h4>{% trans %}Infos{% endtrans %}</h4> | ||||
|         <ul> | ||||
|             <li><strong>{% trans %}IBAN: {% endtrans %}</strong>{{ object.iban }}</li> | ||||
|             <li><strong>{% trans %}Number: {% endtrans %}</strong>{{ object.number }}</li> | ||||
|         </ul> | ||||
|         <p><a href="{{ url('accounting:club_new') }}?parent={{ object.id }}">{% trans %}New club account{% endtrans %}</a></p> | ||||
|         <ul> | ||||
|         {% for c in object.club_accounts.all() %} | ||||
|             <li><a href="{{ url('accounting:club_details', c_account_id=c.id) }}">{{ c }}</a> | ||||
|                 - <a href="{{ url('accounting:club_edit', c_account_id=c.id) }}">{% trans %}Edit{% endtrans %}</a> | ||||
|                 {% if c.journals.count() == 0 %} | ||||
|                 - <a href="{{ url('accounting:club_delete', c_account_id=c.id) }}">{% trans %}Delete{% endtrans %}</a> | ||||
|                 {% endif %} | ||||
|                 </li> | ||||
|         {% endfor %} | ||||
|         </ul> | ||||
|     </div> | ||||
|   <div id="accounting"> | ||||
|     <p> | ||||
|       <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|       {{ object.name }} | ||||
|     </p> | ||||
|     <hr> | ||||
|     <h2>{% trans %}Bank account: {% endtrans %}{{ object.name }}</h2> | ||||
|     {% if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) and not object.club_accounts.exists() %} | ||||
|       <a href="{{ url('accounting:bank_delete', b_account_id=object.id) }}">{% trans %}Delete{% endtrans %}</a> | ||||
|     {% endif %} | ||||
|     <h4>{% trans %}Infos{% endtrans %}</h4> | ||||
|     <ul> | ||||
|       <li><strong>{% trans %}IBAN: {% endtrans %}</strong>{{ object.iban }}</li> | ||||
|       <li><strong>{% trans %}Number: {% endtrans %}</strong>{{ object.number }}</li> | ||||
|     </ul> | ||||
|     <p><a href="{{ url('accounting:club_new') }}?parent={{ object.id }}">{% trans %}New club account{% endtrans %}</a></p> | ||||
|     <ul> | ||||
|       {% for c in object.club_accounts.all() %} | ||||
|         <li><a href="{{ url('accounting:club_details', c_account_id=c.id) }}">{{ c }}</a> | ||||
|           - <a href="{{ url('accounting:club_edit', c_account_id=c.id) }}">{% trans %}Edit{% endtrans %}</a> | ||||
|           {% if c.journals.count() == 0 %} | ||||
|             - <a href="{{ url('accounting:club_delete', c_account_id=c.id) }}">{% trans %}Delete{% endtrans %}</a> | ||||
|           {% endif %} | ||||
|         </li> | ||||
|       {% endfor %} | ||||
|     </ul> | ||||
|   </div> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,32 +1,32 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Bank account list{% endtrans %} | ||||
|   {% trans %}Bank account list{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="accounting"> | ||||
|         <h4> | ||||
|         {% trans %}Accounting{% endtrans %} | ||||
|         </h4> | ||||
|         {% if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) %} | ||||
|         <p><a href="{{ url('accounting:simple_type_list') }}">{% trans %}Manage simplified types{% endtrans %}</a></p> | ||||
|         <p><a href="{{ url('accounting:type_list') }}">{% trans %}Manage accounting types{% endtrans %}</a></p> | ||||
|         <p><a href="{{ url('accounting:bank_new') }}">{% trans %}New bank account{% endtrans %}</a></p> | ||||
|         {% endif %} | ||||
|         {% if bankaccount_list %} | ||||
|         <h3>{% trans %}Bank account list{% endtrans %}</h3> | ||||
|             <ul> | ||||
|                 {% for a in object_list  %} | ||||
|                 <li><a href="{{ url('accounting:bank_details', b_account_id=a.id) }}">{{ a }}</a> | ||||
|                     - <a href="{{ url('accounting:bank_edit', b_account_id=a.id) }}">{% trans %}Edit{% endtrans %}</a> | ||||
|                 </li> | ||||
|                 {% endfor %} | ||||
|             </ul> | ||||
|         {% else %} | ||||
|         {% trans %}There is no accounts in this website.{% endtrans %} | ||||
|         {% endif %} | ||||
|     </div> | ||||
|   <div id="accounting"> | ||||
|     <h4> | ||||
|       {% trans %}Accounting{% endtrans %} | ||||
|     </h4> | ||||
|     {% if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) %} | ||||
|       <p><a href="{{ url('accounting:simple_type_list') }}">{% trans %}Manage simplified types{% endtrans %}</a></p> | ||||
|       <p><a href="{{ url('accounting:type_list') }}">{% trans %}Manage accounting types{% endtrans %}</a></p> | ||||
|       <p><a href="{{ url('accounting:bank_new') }}">{% trans %}New bank account{% endtrans %}</a></p> | ||||
|     {% endif %} | ||||
|     {% if bankaccount_list %} | ||||
|       <h3>{% trans %}Bank account list{% endtrans %}</h3> | ||||
|       <ul> | ||||
|         {% for a in object_list  %} | ||||
|           <li><a href="{{ url('accounting:bank_details', b_account_id=a.id) }}">{{ a }}</a> | ||||
|             - <a href="{{ url('accounting:bank_edit', b_account_id=a.id) }}">{% trans %}Edit{% endtrans %}</a> | ||||
|           </li> | ||||
|         {% endfor %} | ||||
|       </ul> | ||||
|     {% else %} | ||||
|       {% trans %}There is no accounts in this website.{% endtrans %} | ||||
|     {% endif %} | ||||
|   </div> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,68 +1,68 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Club account:{% endtrans %} {{ object.name }} | ||||
|   {% trans %}Club account:{% endtrans %} {{ object.name }} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="accounting"> | ||||
|         <p> | ||||
|         <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|         <a href="{{ url('accounting:bank_details', b_account_id=object.bank_account.id) }}">{{object.bank_account }}</a> > | ||||
|         {{ object }} | ||||
|         </p> | ||||
|         <hr> | ||||
|         <h2>{% trans %}Club account:{% endtrans %} {{ object.name }}</h2> | ||||
|         {% if user.is_root and not object.journals.exists() %} | ||||
|         <a href="{{ url('accounting:club_delete', c_account_id=object.id) }}">{% trans %}Delete{% endtrans %}</a> | ||||
|         {% endif %} | ||||
|         {% if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) %} | ||||
|         <p><a href="{{ url('accounting:label_new') }}?parent={{ object.id }}">{% trans %}New label{% endtrans %}</a></p> | ||||
|         {% endif %} | ||||
|         <p><a href="{{ url('accounting:label_list', clubaccount_id=object.id) }}">{% trans %}Label list{% endtrans %}</a></p> | ||||
|         {% if not object.has_open_journal() %} | ||||
|         <p><a href="{{ url('accounting:journal_new') }}?parent={{ object.id }}">{% trans %}New journal{% endtrans %}</a></p> | ||||
|         {% else %} | ||||
|         <p>{% trans %}You can not create new journal while you still have one opened{% endtrans %}</p> | ||||
|         {% endif %} | ||||
|         <table> | ||||
|             <thead> | ||||
|             <tr> | ||||
|                 <td>{% trans %}Name{% endtrans %}</td> | ||||
|                 <td>{% trans %}Start{% endtrans %}</td> | ||||
|                 <td>{% trans %}End{% endtrans %}</td> | ||||
|                 <td>{% trans %}Amount{% endtrans %}</td> | ||||
|                 <td>{% trans %}Effective amount{% endtrans %}</td> | ||||
|                 <td>{% trans %}Closed{% endtrans %}</td> | ||||
|                 <td>{% trans %}Actions{% endtrans %}</td> | ||||
|             </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|             {% for j in object.journals.all() %} | ||||
|             <tr> | ||||
|                 <td>{{ j.name }}</td> | ||||
|                 <td>{{ j.start_date }}</td> | ||||
|                 {% if j.end_date %} | ||||
|                 <td>{{ j.end_date }}</td> | ||||
|                 {% else %} | ||||
|                 <td> - </td> | ||||
|                 {% endif %} | ||||
|                 <td>{{ j.amount }} €</td> | ||||
|                 <td>{{ j.effective_amount }} €</td> | ||||
|                 {% if j.closed %} | ||||
|                 <td>{% trans %}Yes{% endtrans %}</td> | ||||
|                 {% else %} | ||||
|                 <td>{% trans %}No{% endtrans %}</td> | ||||
|                 {% endif %} | ||||
|                 <td> <a href="{{ url('accounting:journal_details', j_id=j.id) }}">{% trans %}View{% endtrans %}</a> | ||||
|                     <a href="{{ url('accounting:journal_edit', j_id=j.id) }}">{% trans %}Edit{% endtrans %}</a> | ||||
|                     {% if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) and j.operations.count() == 0 %} | ||||
|                         <a href="{{ url('accounting:journal_delete', j_id=j.id) }}">{% trans %}Delete{% endtrans %}</a> | ||||
|                     {% endif %} | ||||
|                     </td> | ||||
|             </tr> | ||||
|             {% endfor %} | ||||
|             </tbody> | ||||
|         </table> | ||||
|     </div> | ||||
|   <div id="accounting"> | ||||
|     <p> | ||||
|       <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|       <a href="{{ url('accounting:bank_details', b_account_id=object.bank_account.id) }}">{{object.bank_account }}</a> > | ||||
|       {{ object }} | ||||
|     </p> | ||||
|     <hr> | ||||
|     <h2>{% trans %}Club account:{% endtrans %} {{ object.name }}</h2> | ||||
|     {% if user.is_root and not object.journals.exists() %} | ||||
|       <a href="{{ url('accounting:club_delete', c_account_id=object.id) }}">{% trans %}Delete{% endtrans %}</a> | ||||
|     {% endif %} | ||||
|     {% if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) %} | ||||
|       <p><a href="{{ url('accounting:label_new') }}?parent={{ object.id }}">{% trans %}New label{% endtrans %}</a></p> | ||||
|     {% endif %} | ||||
|     <p><a href="{{ url('accounting:label_list', clubaccount_id=object.id) }}">{% trans %}Label list{% endtrans %}</a></p> | ||||
|     {% if not object.has_open_journal() %} | ||||
|       <p><a href="{{ url('accounting:journal_new') }}?parent={{ object.id }}">{% trans %}New journal{% endtrans %}</a></p> | ||||
|     {% else %} | ||||
|       <p>{% trans %}You can not create new journal while you still have one opened{% endtrans %}</p> | ||||
|     {% endif %} | ||||
|     <table> | ||||
|       <thead> | ||||
|         <tr> | ||||
|           <td>{% trans %}Name{% endtrans %}</td> | ||||
|           <td>{% trans %}Start{% endtrans %}</td> | ||||
|           <td>{% trans %}End{% endtrans %}</td> | ||||
|           <td>{% trans %}Amount{% endtrans %}</td> | ||||
|           <td>{% trans %}Effective amount{% endtrans %}</td> | ||||
|           <td>{% trans %}Closed{% endtrans %}</td> | ||||
|           <td>{% trans %}Actions{% endtrans %}</td> | ||||
|         </tr> | ||||
|       </thead> | ||||
|       <tbody> | ||||
|         {% for j in object.journals.all() %} | ||||
|           <tr> | ||||
|             <td>{{ j.name }}</td> | ||||
|             <td>{{ j.start_date }}</td> | ||||
|             {% if j.end_date %} | ||||
|               <td>{{ j.end_date }}</td> | ||||
|             {% else %} | ||||
|               <td> - </td> | ||||
|             {% endif %} | ||||
|             <td>{{ j.amount }} €</td> | ||||
|             <td>{{ j.effective_amount }} €</td> | ||||
|             {% if j.closed %} | ||||
|               <td>{% trans %}Yes{% endtrans %}</td> | ||||
|             {% else %} | ||||
|               <td>{% trans %}No{% endtrans %}</td> | ||||
|             {% endif %} | ||||
|             <td> <a href="{{ url('accounting:journal_details', j_id=j.id) }}">{% trans %}View{% endtrans %}</a> | ||||
|               <a href="{{ url('accounting:journal_edit', j_id=j.id) }}">{% trans %}Edit{% endtrans %}</a> | ||||
|               {% if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) and j.operations.count() == 0 %} | ||||
|                 <a href="{{ url('accounting:journal_delete', j_id=j.id) }}">{% trans %}Delete{% endtrans %}</a> | ||||
|               {% endif %} | ||||
|             </td> | ||||
|           </tr> | ||||
|         {% endfor %} | ||||
|       </tbody> | ||||
|     </table> | ||||
|   </div> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -1,29 +1,30 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Company list{% endtrans %} | ||||
|   {% trans %}Company list{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="accounting"> | ||||
|         {% if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) or user.is_root %} | ||||
|         <p><a href="{{ url('accounting:co_new') }}">{% trans %}Create new company{% endtrans %}</a></p> | ||||
|         {% endif %} | ||||
|  | ||||
|         </br> | ||||
|         <table> | ||||
|             <thead> | ||||
|             <tr> | ||||
|                 <td>{% trans %}Companies{% endtrans %}</td> | ||||
|             </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|                 {% for o in object_list %} | ||||
|                     <tr> | ||||
|                         <td><a href="{{ url('accounting:co_edit', co_id=o.id) }}">{{ o.get_display_name() }}</a></td> | ||||
|                     </tr> | ||||
|                 {% endfor %} | ||||
|             </tbody> | ||||
|         </table> | ||||
|     </div> | ||||
|   <div id="accounting"> | ||||
|     {% if user.is_root | ||||
|     or user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) | ||||
|     %} | ||||
|     <p><a href="{{ url('accounting:co_new') }}">{% trans %}Create new company{% endtrans %}</a></p> | ||||
| {% endif %} | ||||
| <br/> | ||||
| <table> | ||||
|   <thead> | ||||
|     <tr> | ||||
|       <td>{% trans %}Companies{% endtrans %}</td> | ||||
|     </tr> | ||||
|   </thead> | ||||
|   <tbody> | ||||
|     {% for o in object_list %} | ||||
|       <tr> | ||||
|         <td><a href="{{ url('accounting:co_edit', co_id=o.id) }}">{{ o.get_display_name() }}</a></td> | ||||
|       </tr> | ||||
|     {% endfor %} | ||||
|   </tbody> | ||||
| </table> | ||||
| </div> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -1,100 +1,103 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}General journal:{% endtrans %} {{ object.name }} | ||||
|   {% trans %}General journal:{% endtrans %} {{ object.name }} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="accounting"> | ||||
|         <p> | ||||
|         <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|         <a href="{{ url('accounting:bank_details', b_account_id=object.club_account.bank_account.id) }}">{{object.club_account.bank_account }}</a> > | ||||
|         <a href="{{ url('accounting:club_details', c_account_id=object.club_account.id) }}">{{ object.club_account }}</a> > | ||||
|         {{ object.name }} | ||||
|         </p> | ||||
|         <hr> | ||||
|         <h2>{% trans %}General journal:{% endtrans %} {{ object.name }}</h2> | ||||
|         <p><a href="{{ url('accounting:label_new') }}?parent={{ object.club_account.id }}">{% trans %}New label{% endtrans %}</a></p> | ||||
|         <p><a href="{{ url('accounting:label_list', clubaccount_id=object.club_account.id) }}">{% trans %}Label list{% endtrans %}</a></p> | ||||
|         <p><a href="{{ url('accounting:co_list') }}">{% trans %}Company list{% endtrans %}</a></p> | ||||
|         <p><strong>{% trans %}Amount: {% endtrans %}</strong>{{ object.amount }} € - | ||||
|         <strong>{% trans %}Effective amount: {% endtrans %}</strong>{{ object.effective_amount }} €</p> | ||||
|         {% if object.closed %} | ||||
|         <p>{% trans %}Journal is closed, you can not create operation{% endtrans %}</p> | ||||
|         {% else %} | ||||
|         <p><a href="{{ url('accounting:op_new', j_id=object.id) }}">{% trans %}New operation{% endtrans %}</a></p> | ||||
|         </br> | ||||
|         {% endif %} | ||||
|         <div class="journal-table"> | ||||
|         <table> | ||||
|             <thead> | ||||
|   <div id="accounting"> | ||||
|     <p> | ||||
|       <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|       <a href="{{ url('accounting:bank_details', b_account_id=object.club_account.bank_account.id) }}">{{object.club_account.bank_account }}</a> > | ||||
|       <a href="{{ url('accounting:club_details', c_account_id=object.club_account.id) }}">{{ object.club_account }}</a> > | ||||
|       {{ object.name }} | ||||
|     </p> | ||||
|     <hr> | ||||
|     <h2>{% trans %}General journal:{% endtrans %} {{ object.name }}</h2> | ||||
|     <p><a href="{{ url('accounting:label_new') }}?parent={{ object.club_account.id }}">{% trans %}New label{% endtrans %}</a></p> | ||||
|     <p><a href="{{ url('accounting:label_list', clubaccount_id=object.club_account.id) }}">{% trans %}Label list{% endtrans %}</a></p> | ||||
|     <p><a href="{{ url('accounting:co_list') }}">{% trans %}Company list{% endtrans %}</a></p> | ||||
|     <p><strong>{% trans %}Amount: {% endtrans %}</strong>{{ object.amount }} € - | ||||
|       <strong>{% trans %}Effective amount: {% endtrans %}</strong>{{ object.effective_amount }} €</p> | ||||
|     {% if object.closed %} | ||||
|       <p>{% trans %}Journal is closed, you can not create operation{% endtrans %}</p> | ||||
|     {% else %} | ||||
|       <p><a href="{{ url('accounting:op_new', j_id=object.id) }}">{% trans %}New operation{% endtrans %}</a></p> | ||||
|       </br> | ||||
|     {% endif %} | ||||
|     <div class="journal-table"> | ||||
|       <table> | ||||
|         <thead> | ||||
|           <tr> | ||||
|             <td>{% trans %}Nb{% endtrans %}</td> | ||||
|             <td>{% trans %}Date{% endtrans %}</td> | ||||
|             <td>{% trans %}Label{% endtrans %}</td> | ||||
|             <td>{% trans %}Amount{% endtrans %}</td> | ||||
|             <td>{% trans %}Payment mode{% endtrans %}</td> | ||||
|             <td>{% trans %}Target{% endtrans %}</td> | ||||
|             <td>{% trans %}Code{% endtrans %}</td> | ||||
|             <td>{% trans %}Nature{% endtrans %}</td> | ||||
|             <td>{% trans %}Done{% endtrans %}</td> | ||||
|             <td>{% trans %}Comment{% endtrans %}</td> | ||||
|             <td>{% trans %}File{% endtrans %}</td> | ||||
|             <td>{% trans %}Actions{% endtrans %}</td> | ||||
|             <td>{% trans %}PDF{% endtrans %}</td> | ||||
|           </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
|           {% for o in object.operations.all() %} | ||||
|             <tr> | ||||
|                 <td>{% trans %}Nb{% endtrans %}</td> | ||||
|                 <td>{% trans %}Date{% endtrans %}</td> | ||||
|                 <td>{% trans %}Label{% endtrans %}</td> | ||||
|                 <td>{% trans %}Amount{% endtrans %}</td> | ||||
|                 <td>{% trans %}Payment mode{% endtrans %}</td> | ||||
|                 <td>{% trans %}Target{% endtrans %}</td> | ||||
|                 <td>{% trans %}Code{% endtrans %}</td> | ||||
|                 <td>{% trans %}Nature{% endtrans %}</td> | ||||
|                 <td>{% trans %}Done{% endtrans %}</td> | ||||
|                 <td>{% trans %}Comment{% endtrans %}</td> | ||||
|                 <td>{% trans %}File{% endtrans %}</td> | ||||
|                 <td>{% trans %}Actions{% endtrans %}</td> | ||||
|                 <td>{% trans %}PDF{% endtrans %}</td> | ||||
|             </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|             {% for o in object.operations.all() %} | ||||
|             <tr> | ||||
|                 <td>{{ o.number }}</td> | ||||
|                 <td>{{ o.date }}</td> | ||||
|                 <td>{{ o.label or "" }}</td> | ||||
|                 {% if o.accounting_type.movement_type == "DEBIT" %} | ||||
|                     <td class="neg-amount"> {{ o.amount }} €</td> | ||||
|                 {% else %} | ||||
|                     <td class="pos-amount"> {{ o.amount }} €</td> | ||||
|                 {% endif %} | ||||
|                 <td>{{ o.get_mode_display() }}</td> | ||||
|                 {% if o.target_type == "OTHER" %} | ||||
|               <td>{{ o.number }}</td> | ||||
|               <td>{{ o.date }}</td> | ||||
|               <td>{{ o.label or "" }}</td> | ||||
|               {% if o.accounting_type.movement_type == "DEBIT" %} | ||||
|                 <td class="neg-amount"> {{ o.amount }} €</td> | ||||
|               {% else %} | ||||
|                 <td class="pos-amount"> {{ o.amount }} €</td> | ||||
|               {% endif %} | ||||
|               <td>{{ o.get_mode_display() }}</td> | ||||
|               {% if o.target_type == "OTHER" %} | ||||
|                 <td>{{ o.target_label }}</td> | ||||
|                 {% else %} | ||||
|               {% else %} | ||||
|                 <td><a href="{{ o.target.get_absolute_url() }}">{{ o.target.get_display_name() }}</a></td> | ||||
|                 {% endif %} | ||||
|                 <td>{{ o.accounting_type.code }}</td> | ||||
|                 <td>{{ o.accounting_type.label }}</td> | ||||
|                 {% if o.done %} | ||||
|               {% endif %} | ||||
|               <td>{{ o.accounting_type.code }}</td> | ||||
|               <td>{{ o.accounting_type.label }}</td> | ||||
|               {% if o.done %} | ||||
|                 <td>{% trans %}Yes{% endtrans %}</td> | ||||
|                 {% else %} | ||||
|               {% else %} | ||||
|                 <td>{% trans %}No{% endtrans %}</td> | ||||
|                 {% endif %} | ||||
|                 <td>{{ o.remark }} | ||||
|               {% endif %} | ||||
|               <td>{{ o.remark }} | ||||
|                 {% if not o.linked_operation and o.target_type == "ACCOUNT" and not o.target.has_open_journal() %} | ||||
|                     <p><strong> | ||||
|     {% trans %}Warning: this operation has no linked operation because the targeted club account has no opened journal.{% endtrans %} | ||||
|                 </strong></p> | ||||
|                 <p><strong> | ||||
|     {% trans url=o.target.get_absolute_url() %}Open a journal in <a href="{{ url }}">this club account</a>, then save this operation again to make the linked operation.{% endtrans %} | ||||
|             </strong></p> | ||||
|                   <p><strong> | ||||
|                     {% trans %}Warning: this operation has no linked operation because the targeted club account has no opened journal.{% endtrans %} | ||||
|                   </strong></p> | ||||
|                   <p><strong> | ||||
|                     {% trans url=o.target.get_absolute_url() %}Open a journal in <a href="{{ url }}">this club account</a>, then save this operation again to make the linked operation.{% endtrans %} | ||||
|                   </strong></p> | ||||
|                 {% endif %} | ||||
|                 </td> | ||||
|                 {% if o.invoice %} | ||||
|               </td> | ||||
|               {% if o.invoice %} | ||||
|                 <td><a href="{{ url('core:download', file_id=o.invoice.id) }}">{{ o.invoice.name }}</a></td> | ||||
|                 {% else %} | ||||
|               {% else %} | ||||
|                 <td>-</td> | ||||
|               {% endif %} | ||||
|               <td> | ||||
|                 {% | ||||
|                 if o.journal.club_account.bank_account.name not in ["AE TI", "TI"] | ||||
|                 or user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) | ||||
|                 %} | ||||
|                 {% if not o.journal.closed %} | ||||
|                   <a href="{{ url('accounting:op_edit', op_id=o.id) }}">{% trans %}Edit{% endtrans %}</a> | ||||
|                 {% endif %} | ||||
|                 <td> | ||||
|                     {% if o.journal.club_account.bank_account.name != "AE TI" and o.journal.club_account.bank_account.name != "TI" or user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) %} | ||||
|                     {% if not o.journal.closed %} | ||||
|                     <a href="{{ url('accounting:op_edit', op_id=o.id) }}">{% trans %}Edit{% endtrans %}</a> | ||||
|                     {% endif %} | ||||
|                     {% endif %} | ||||
|                 </td> | ||||
|                 <td><a href="{{ url('accounting:op_pdf', op_id=o.id) }}">{% trans %}Generate{% endtrans %}</a></td> | ||||
|             </tr> | ||||
|             {% endfor %} | ||||
|             </tbody> | ||||
|         </table> | ||||
|         </div> | ||||
|     </div> | ||||
|           {% endif %} | ||||
|         </td> | ||||
|         <td><a href="{{ url('accounting:op_pdf', op_id=o.id) }}">{% trans %}Generate{% endtrans %}</a></td> | ||||
|       </tr> | ||||
| {% endfor %} | ||||
| </tbody> | ||||
| </table> | ||||
| </div> | ||||
| </div> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -1,33 +1,33 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}General journal:{% endtrans %} {{ object.name }} | ||||
|   {% trans %}General journal:{% endtrans %} {{ object.name }} | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
| {% block content %} | ||||
| <div id="accounting"> | ||||
|   <div id="accounting"> | ||||
|     <h3>{% trans %}Accounting statement: {% endtrans %} {{ object.name }}</h3> | ||||
|  | ||||
|     <table> | ||||
|         <thead> | ||||
|             <tr> | ||||
|                 <td>{% trans %}Operation type{% endtrans %}</td> | ||||
|                 <td>{% trans %}Sum{% endtrans %}</td> | ||||
|             </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
|             {% for k,v in statement.items() %} | ||||
|             <tr> | ||||
|                 <td>{{ k }}</td> | ||||
|                 <td>{{ v }}</td> | ||||
|             </tr> | ||||
|             {% endfor %} | ||||
|         </tbody> | ||||
|       <thead> | ||||
|         <tr> | ||||
|           <td>{% trans %}Operation type{% endtrans %}</td> | ||||
|           <td>{% trans %}Sum{% endtrans %}</td> | ||||
|         </tr> | ||||
|       </thead> | ||||
|       <tbody> | ||||
|         {% for k,v in statement.items() %} | ||||
|           <tr> | ||||
|             <td>{{ k }}</td> | ||||
|             <td>{{ "%.2f" % v }}</td> | ||||
|           </tr> | ||||
|         {% endfor %} | ||||
|       </tbody> | ||||
|  | ||||
|     </table> | ||||
|  | ||||
|     <p><strong>{% trans %}Amount: {% endtrans %}</strong>{{ object.amount }} €</p> | ||||
|     <p><strong>{% trans %}Effective amount: {% endtrans %}</strong>{{ object.effective_amount }} €</p> | ||||
| </div> | ||||
|     <p><strong>{% trans %}Amount: {% endtrans %}</strong>{{ "%.2f" % object.amount }} €</p> | ||||
|     <p><strong>{% trans %}Effective amount: {% endtrans %}</strong>{{ "%.2f" %object.effective_amount }} €</p> | ||||
|   </div> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -1,57 +1,57 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}General journal:{% endtrans %} {{ object.name }} | ||||
|   {% trans %}General journal:{% endtrans %} {{ object.name }} | ||||
| {% endblock %} | ||||
|  | ||||
| {% macro display_tables(dict) %} | ||||
| <div id="accounting"> | ||||
|   <div id="accounting"> | ||||
|     <h6>{% trans %}Credit{% endtrans %}</h6> | ||||
|     <table> | ||||
|         <thead> | ||||
|             <tr> | ||||
|                 <td>{% trans %}Nature of operation{% endtrans %}</td> | ||||
|                 <td>{% trans %}Sum{% endtrans %}</td> | ||||
|             </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
|             {% for k,v in dict['CREDIT'].items() %} | ||||
|             <tr> | ||||
|                 <td>{{ k }}</td> | ||||
|                 <td>{{ v }}</td> | ||||
|             </tr> | ||||
|             {% endfor %} | ||||
|         </tbody> | ||||
|       <thead> | ||||
|         <tr> | ||||
|           <td>{% trans %}Nature of operation{% endtrans %}</td> | ||||
|           <td>{% trans %}Sum{% endtrans %}</td> | ||||
|         </tr> | ||||
|       </thead> | ||||
|       <tbody> | ||||
|         {% for k,v in dict['CREDIT'].items() %} | ||||
|           <tr> | ||||
|             <td>{{ k }}</td> | ||||
|             <td>{{ "%.2f" % v }}</td> | ||||
|           </tr> | ||||
|         {% endfor %} | ||||
|       </tbody> | ||||
|     </table> | ||||
|     {% trans %}Total: {% endtrans %}{{ dict['CREDIT_sum'] }} | ||||
|     {% trans %}Total: {% endtrans %}{{ "%.2f" % dict['CREDIT_sum'] }} | ||||
|  | ||||
|     <h6>{% trans %}Debit{% endtrans %}</h6> | ||||
|     <table> | ||||
|         <thead> | ||||
|             <tr> | ||||
|                 <td>{% trans %}Nature of operation{% endtrans %}</td> | ||||
|                 <td>{% trans %}Sum{% endtrans %}</td> | ||||
|             </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
|             {% for k,v in dict['DEBIT'].items() %} | ||||
|             <tr> | ||||
|                 <td>{{ k }}</td> | ||||
|                 <td>{{ v }}</td> | ||||
|             </tr> | ||||
|             {% endfor %} | ||||
|         </tbody> | ||||
|       <thead> | ||||
|         <tr> | ||||
|           <td>{% trans %}Nature of operation{% endtrans %}</td> | ||||
|           <td>{% trans %}Sum{% endtrans %}</td> | ||||
|         </tr> | ||||
|       </thead> | ||||
|       <tbody> | ||||
|         {% for k,v in dict['DEBIT'].items() %} | ||||
|           <tr> | ||||
|             <td>{{ k }}</td> | ||||
|             <td>{{ "%.2f" % v }}</td> | ||||
|           </tr> | ||||
|         {% endfor %} | ||||
|       </tbody> | ||||
|     </table> | ||||
|     {% trans %}Total: {% endtrans %}{{ dict['DEBIT_sum'] }} | ||||
|     {% endmacro %} | ||||
|     {% trans %}Total: {% endtrans %}{{ "%.2f" % dict['DEBIT_sum'] }} | ||||
| {% endmacro %} | ||||
|  | ||||
|     {% block content %} | ||||
|     <h3>{% trans %}Statement by nature: {% endtrans %} {{ object.name }}</h3> | ||||
| {% block content %} | ||||
|   <h3>{% trans %}Statement by nature: {% endtrans %} {{ object.name }}</h3> | ||||
|  | ||||
|     {% for k,v in statement.items() %} | ||||
|         <h4 style="background: lightblue; padding: 4px;">{{ k }} : {{ v['CREDIT_sum'] - v['DEBIT_sum'] }}</h4> | ||||
|   {% for k,v in statement.items() %} | ||||
|     <h4 style="background: lightblue; padding: 4px;">{{ k }} : {{ "%.2f" % (v['CREDIT_sum'] - v['DEBIT_sum']) }}</h4> | ||||
|     {{ display_tables(v) }} | ||||
|     <hr> | ||||
|     {% endfor %} | ||||
| </div> | ||||
|   {% endfor %} | ||||
|   </div> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -1,68 +1,68 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}General journal:{% endtrans %} {{ object.name }} | ||||
|   {% trans %}General journal:{% endtrans %} {{ object.name }} | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
| {% block content %} | ||||
| <div id="accounting"> | ||||
|   <div id="accounting"> | ||||
|     <h3>{% trans %}Statement by person: {% endtrans %} {{ object.name }}</h3> | ||||
|  | ||||
|     <h4>{% trans %}Credit{% endtrans %}</h4> | ||||
|  | ||||
|     <table> | ||||
|         <thead> | ||||
|             <tr> | ||||
|                 <td>{% trans %}Target of the operation{% endtrans %}</td> | ||||
|                 <td>{% trans %}Sum{% endtrans %}</td> | ||||
|             </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
|             {% for key in credit_statement.keys() %} | ||||
|             <tr> | ||||
|                 {% if key.target_type == "OTHER" %} | ||||
|                 <td>{{ o.target_label }}</td> | ||||
|                 {% elif key %} | ||||
|                 <td><a href="{{ key.get_absolute_url() }}">{{ key.get_display_name() }}</a></td> | ||||
|                 {% else %} | ||||
|                 <td></td> | ||||
|                 {% endif %} | ||||
|                 <td>{{ credit_statement[key] }}</td> | ||||
|             </tr> | ||||
|             {% endfor %} | ||||
|         </tbody> | ||||
|       <thead> | ||||
|         <tr> | ||||
|           <td>{% trans %}Target of the operation{% endtrans %}</td> | ||||
|           <td>{% trans %}Sum{% endtrans %}</td> | ||||
|         </tr> | ||||
|       </thead> | ||||
|       <tbody> | ||||
|         {% for key in credit_statement.keys() %} | ||||
|           <tr> | ||||
|             {% if key.target_type == "OTHER" %} | ||||
|               <td>{{ o.target_label }}</td> | ||||
|             {% elif key %} | ||||
|               <td><a href="{{ key.get_absolute_url() }}">{{ key.get_display_name() }}</a></td> | ||||
|             {% else %} | ||||
|               <td></td> | ||||
|             {% endif %} | ||||
|             <td>{{ "%.2f" % credit_statement[key] }}</td> | ||||
|           </tr> | ||||
|         {% endfor %} | ||||
|       </tbody> | ||||
|  | ||||
|     </table> | ||||
|  | ||||
|     <p>Total : {{ total_credit }}</p> | ||||
|     <p>Total : {{ "%.2f" % total_credit }}</p> | ||||
|  | ||||
|     <h4>{% trans %}Debit{% endtrans %}</h4> | ||||
|  | ||||
|     <table> | ||||
|         <thead> | ||||
|             <tr> | ||||
|                 <td>{% trans %}Target of the operation{% endtrans %}</td> | ||||
|                 <td>{% trans %}Sum{% endtrans %}</td> | ||||
|             </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
|             {% for key in debit_statement.keys() %} | ||||
|             <tr> | ||||
|                 {% if key.target_type == "OTHER" %} | ||||
|                 <td>{{ o.target_label }}</td> | ||||
|                 {% elif key %} | ||||
|                 <td><a href="{{ key.get_absolute_url() }}">{{ key.get_display_name() }}</a></td> | ||||
|                 {% else %} | ||||
|                 <td></td> | ||||
|                 {% endif %} | ||||
|                 <td>{{ debit_statement[key] }}</td> | ||||
|             </tr> | ||||
|             {% endfor %} | ||||
|         </tbody> | ||||
|       <thead> | ||||
|         <tr> | ||||
|           <td>{% trans %}Target of the operation{% endtrans %}</td> | ||||
|           <td>{% trans %}Sum{% endtrans %}</td> | ||||
|         </tr> | ||||
|       </thead> | ||||
|       <tbody> | ||||
|         {% for key in debit_statement.keys() %} | ||||
|           <tr> | ||||
|             {% if key.target_type == "OTHER" %} | ||||
|               <td>{{ o.target_label }}</td> | ||||
|             {% elif key %} | ||||
|               <td><a href="{{ key.get_absolute_url() }}">{{ key.get_display_name() }}</a></td> | ||||
|             {% else %} | ||||
|               <td></td> | ||||
|             {% endif %} | ||||
|             <td>{{ "%.2f" % debit_statement[key] }}</td> | ||||
|           </tr> | ||||
|         {% endfor %} | ||||
|       </tbody> | ||||
|  | ||||
|     </table> | ||||
|  | ||||
|     <p>Total : {{ total_debit }}</p> | ||||
| </div> | ||||
|     <p>Total : {{ "%.2f" % total_debit }}</p> | ||||
|   </div> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -1,36 +1,36 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Label list{% endtrans %} | ||||
|   {% trans %}Label list{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="accounting"> | ||||
|         <p> | ||||
|         <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|         <a href="{{ url('accounting:bank_details', b_account_id=object.bank_account.id) }}">{{object.bank_account }}</a> > | ||||
|         <a href="{{ url('accounting:club_details', c_account_id=object.id) }}">{{ object }}</a> | ||||
|         </p> | ||||
|         <hr> | ||||
|         <p><a href="{{ url('accounting:club_details', c_account_id=object.id) }}">{% trans %}Back to club account{% endtrans %}</a></p> | ||||
|         {% if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) %} | ||||
|         <p><a href="{{ url('accounting:label_new') }}?parent={{ object.id }}">{% trans %}New label{% endtrans %}</a></p> | ||||
|         {% endif %} | ||||
|         {% if object.labels.all() %} | ||||
|         <h3>{% trans %}Label list{% endtrans %}</h3> | ||||
|         <ul> | ||||
|             {% for l in object.labels.all()  %} | ||||
|             <li><a href="{{ url('accounting:label_edit', label_id=l.id) }}">{{ l }}</a> | ||||
|             {% if user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) %} | ||||
|              - | ||||
|                 <a href="{{ url('accounting:label_delete', label_id=l.id) }}">{% trans %}Delete{% endtrans %}</a> | ||||
|   <div id="accounting"> | ||||
|     <p> | ||||
|       <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|       <a href="{{ url('accounting:bank_details', b_account_id=object.bank_account.id) }}">{{object.bank_account }}</a> > | ||||
|       <a href="{{ url('accounting:club_details', c_account_id=object.id) }}">{{ object }}</a> | ||||
|     </p> | ||||
|     <hr> | ||||
|     <p><a href="{{ url('accounting:club_details', c_account_id=object.id) }}">{% trans %}Back to club account{% endtrans %}</a></p> | ||||
|     {% if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) %} | ||||
|       <p><a href="{{ url('accounting:label_new') }}?parent={{ object.id }}">{% trans %}New label{% endtrans %}</a></p> | ||||
|     {% endif %} | ||||
|     {% if object.labels.all() %} | ||||
|       <h3>{% trans %}Label list{% endtrans %}</h3> | ||||
|       <ul> | ||||
|         {% for l in object.labels.all()  %} | ||||
|           <li><a href="{{ url('accounting:label_edit', label_id=l.id) }}">{{ l }}</a> | ||||
|             {% if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID) %} | ||||
|               - | ||||
|               <a href="{{ url('accounting:label_delete', label_id=l.id) }}">{% trans %}Delete{% endtrans %}</a> | ||||
|             {% endif %} | ||||
|             </li> | ||||
|             {% endfor %} | ||||
|         </ul> | ||||
|         {% else %} | ||||
|         {% trans %}There is no label in this club account.{% endtrans %} | ||||
|         {% endif %} | ||||
|     </div> | ||||
|           </li> | ||||
|         {% endfor %} | ||||
|       </ul> | ||||
|     {% else %} | ||||
|       {% trans %}There is no label in this club account.{% endtrans %} | ||||
|     {% endif %} | ||||
|   </div> | ||||
| {% endblock %} | ||||
|  | ||||
|   | ||||
| @@ -1,123 +1,123 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Edit operation{% endtrans %} | ||||
|   {% trans %}Edit operation{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
| <div id="accounting"> | ||||
|   <div id="accounting"> | ||||
|     <p> | ||||
|     <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|     <a href="{{ url('accounting:bank_details', b_account_id=object.club_account.bank_account.id) }}">{{object.club_account.bank_account }}</a> > | ||||
|     <a href="{{ url('accounting:club_details', c_account_id=object.club_account.id) }}">{{ object.club_account }}</a> > | ||||
|     <a href="{{ url('accounting:journal_details', j_id=object.id) }}">{{ object.name }}</a> > | ||||
|     {% trans %}Edit operation{% endtrans %} | ||||
|       <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|       <a href="{{ url('accounting:bank_details', b_account_id=object.club_account.bank_account.id) }}">{{object.club_account.bank_account }}</a> > | ||||
|       <a href="{{ url('accounting:club_details', c_account_id=object.club_account.id) }}">{{ object.club_account }}</a> > | ||||
|       <a href="{{ url('accounting:journal_details', j_id=object.id) }}">{{ object.name }}</a> > | ||||
|       {% trans %}Edit operation{% endtrans %} | ||||
|     </p> | ||||
|     <hr> | ||||
|     <h2>{% trans %}Edit operation{% endtrans %}</h2> | ||||
|     <form action="" method="post"> | ||||
|         {% csrf_token %} | ||||
|         {{ form.non_field_errors() }} | ||||
|         {{ form.journal }} | ||||
|         {{ form.target_id }} | ||||
|         <p>{{ form.amount.errors }}<label for="{{ form.amount.name }}">{{ form.amount.label }}</label> {{ form.amount }}</p> | ||||
|         <p>{{ form.remark.errors }}<label for="{{ form.remark.name }}">{{ form.remark.label }}</label> {{ form.remark }}</p> | ||||
|         <br /> | ||||
|         <strong>{% trans %}Warning: if you select <em>Account</em>, the opposite operation will be created in the target account. If you don't want that, select <em>Club</em> instead of <em>Account</em>.{% endtrans %}</strong> | ||||
|         <p>{{ form.target_type.errors }}<label for="{{ form.target_type.name }}">{{ form.target_type.label }}</label> {{ form.target_type }}</p> | ||||
|             {{ form.user }} | ||||
|             {{ form.club }} | ||||
|             {{ form.club_account }} | ||||
|             {{ form.company }} | ||||
|             {{ form.target_label }} | ||||
|             <span id="id_need_link_full"><label>{{ form.need_link.label }}</label> {{ form.need_link }}</span> | ||||
|         <p>{{ form.date.errors }}<label for="{{ form.date.name }}">{{ form.date.label }}</label> {{ form.date }}</p> | ||||
|         <p>{{ form.mode.errors }}<label for="{{ form.mode.name }}">{{ form.mode.label }}</label> {{ form.mode }}</p> | ||||
|         <p>{{ form.cheque_number.errors }}<label for="{{ form.cheque_number.name }}">{{ form.cheque_number.label }}</label> {{ | ||||
|       {% csrf_token %} | ||||
|       {{ form.non_field_errors() }} | ||||
|       {{ form.journal }} | ||||
|       {{ form.target_id }} | ||||
|       <p>{{ form.amount.errors }}<label for="{{ form.amount.name }}">{{ form.amount.label }}</label> {{ form.amount }}</p> | ||||
|       <p>{{ form.remark.errors }}<label for="{{ form.remark.name }}">{{ form.remark.label }}</label> {{ form.remark }}</p> | ||||
|       <br /> | ||||
|       <strong>{% trans %}Warning: if you select <em>Account</em>, the opposite operation will be created in the target account. If you don't want that, select <em>Club</em> instead of <em>Account</em>.{% endtrans %}</strong> | ||||
|       <p>{{ form.target_type.errors }}<label for="{{ form.target_type.name }}">{{ form.target_type.label }}</label> {{ form.target_type }}</p> | ||||
|       {{ form.user }} | ||||
|       {{ form.club }} | ||||
|       {{ form.club_account }} | ||||
|       {{ form.company }} | ||||
|       {{ form.target_label }} | ||||
|       <span id="id_need_link_full"><label>{{ form.need_link.label }}</label> {{ form.need_link }}</span> | ||||
|       <p>{{ form.date.errors }}<label for="{{ form.date.name }}">{{ form.date.label }}</label> {{ form.date }}</p> | ||||
|       <p>{{ form.mode.errors }}<label for="{{ form.mode.name }}">{{ form.mode.label }}</label> {{ form.mode }}</p> | ||||
|       <p>{{ form.cheque_number.errors }}<label for="{{ form.cheque_number.name }}">{{ form.cheque_number.label }}</label> {{ | ||||
|         form.cheque_number }}</p> | ||||
|         <p>{{ form.invoice.errors }}<label for="{{ form.invoice.name }}">{{ form.invoice.label }}</label> {{ form.invoice }}</p> | ||||
|         <p>{{ form.simpleaccounting_type.errors }}<label for="{{ form.simpleaccounting_type.name }}">{{ | ||||
|             form.simpleaccounting_type.label }}</label> {{ form.simpleaccounting_type }}</p> | ||||
|         <p>{{ form.accounting_type.errors }}<label for="{{ form.accounting_type.name }}">{{ form.accounting_type.label }}</label> {{ | ||||
|       <p>{{ form.invoice.errors }}<label for="{{ form.invoice.name }}">{{ form.invoice.label }}</label> {{ form.invoice }}</p> | ||||
|       <p>{{ form.simpleaccounting_type.errors }}<label for="{{ form.simpleaccounting_type.name }}">{{ | ||||
|         form.simpleaccounting_type.label }}</label> {{ form.simpleaccounting_type }}</p> | ||||
|       <p>{{ form.accounting_type.errors }}<label for="{{ form.accounting_type.name }}">{{ form.accounting_type.label }}</label> {{ | ||||
|         form.accounting_type }}</p> | ||||
|         <p>{{ form.label.errors }}<label for="{{ form.label.name }}">{{ form.label.label }}</label> {{ form.label }}</p> | ||||
|         <p>{{ form.done.errors }}<label for="{{ form.done.name }}">{{ form.done.label }}</label> {{ form.done }}</p> | ||||
|         {% if form.instance.linked_operation %} | ||||
|       <p>{{ form.label.errors }}<label for="{{ form.label.name }}">{{ form.label.label }}</label> {{ form.label }}</p> | ||||
|       <p>{{ form.done.errors }}<label for="{{ form.done.name }}">{{ form.done.label }}</label> {{ form.done }}</p> | ||||
|       {% if form.instance.linked_operation %} | ||||
|         {% set obj = form.instance.linked_operation %} | ||||
|         <p><strong>{% trans %}Linked operation:{% endtrans %}</strong><br> | ||||
|         <a href="{{ url('accounting:bank_details', b_account_id=obj.journal.club_account.bank_account.id) }}"> | ||||
|                 {{obj.journal.club_account.bank_account }}</a> > | ||||
|         <a href="{{ url('accounting:club_details', c_account_id=obj.journal.club_account.id) }}">{{ obj.journal.club_account }}</a> > | ||||
|         <a href="{{ url('accounting:journal_details', j_id=obj.journal.id) }}">{{ obj.journal }}</a> > | ||||
|         n°{{ obj.number }} | ||||
|           <a href="{{ url('accounting:bank_details', b_account_id=obj.journal.club_account.bank_account.id) }}"> | ||||
|             {{obj.journal.club_account.bank_account }}</a> > | ||||
|           <a href="{{ url('accounting:club_details', c_account_id=obj.journal.club_account.id) }}">{{ obj.journal.club_account }}</a> > | ||||
|           <a href="{{ url('accounting:journal_details', j_id=obj.journal.id) }}">{{ obj.journal }}</a> > | ||||
|           n°{{ obj.number }} | ||||
|         </p> | ||||
|         {% endif %} | ||||
|         <p><input type="submit" value="{% trans %}Save{% endtrans %}" /></p> | ||||
|       {% endif %} | ||||
|       <p><input type="submit" value="{% trans %}Save{% endtrans %}" /></p> | ||||
|     </form> | ||||
|     {% endblock %} | ||||
| {% endblock %} | ||||
|  | ||||
|     {% block script %} | ||||
|         {{ super() }} | ||||
|     <script> | ||||
| {% block script %} | ||||
|   {{ super() }} | ||||
|   <script> | ||||
|     $( function() { | ||||
|             var target_type = $('#id_target_type'); | ||||
|             var user = $('#id_user_wrapper'); | ||||
|             var club = $('#id_club_wrapper'); | ||||
|             var club_account = $('#id_club_account_wrapper'); | ||||
|             var company = $('#id_company_wrapper'); | ||||
|             var other = $('#id_target_label'); | ||||
|             var need_link = $('#id_need_link_full'); | ||||
|             function update_targets () { | ||||
|                 if (target_type.val() == "USER") { | ||||
|                     console.log(user); | ||||
|                     user.show(); | ||||
|                     club.hide(); | ||||
|                     club_account.hide(); | ||||
|                     company.hide(); | ||||
|                     other.hide(); | ||||
|                     need_link.hide(); | ||||
|                 } else if (target_type.val() == "ACCOUNT") { | ||||
|                     club_account.show(); | ||||
|                     need_link.show(); | ||||
|                     user.hide(); | ||||
|                     club.hide(); | ||||
|                     company.hide(); | ||||
|                     other.hide(); | ||||
|                 } else if (target_type.val() == "CLUB") { | ||||
|                     club.show(); | ||||
|                     user.hide(); | ||||
|                     club_account.hide(); | ||||
|                     company.hide(); | ||||
|                     other.hide(); | ||||
|                     need_link.hide(); | ||||
|                 } else if (target_type.val() == "COMPANY") { | ||||
|                     company.show(); | ||||
|                     user.hide(); | ||||
|                     club_account.hide(); | ||||
|                     club.hide(); | ||||
|                     other.hide(); | ||||
|                     need_link.hide(); | ||||
|                 } else if (target_type.val() == "OTHER") { | ||||
|                     other.show(); | ||||
|                     user.hide(); | ||||
|                     club.hide(); | ||||
|                     club_account.hide(); | ||||
|                     company.hide(); | ||||
|                     need_link.hide(); | ||||
|                 } else { | ||||
|                     company.hide(); | ||||
|                     user.hide(); | ||||
|                     club_account.hide(); | ||||
|                     club.hide(); | ||||
|                     other.hide(); | ||||
|                     need_link.hide(); | ||||
|                 } | ||||
|             } | ||||
|             update_targets(); | ||||
|             target_type.change(update_targets); | ||||
|             } ); | ||||
|     </script> | ||||
| </div> | ||||
|       var target_type = $('#id_target_type'); | ||||
|       var user = $('user-ajax-select'); | ||||
|       var club = $('club-ajax-select'); | ||||
|       var club_account = $('club-account-ajax-select'); | ||||
|       var company = $('company-ajax-select'); | ||||
|       var other = $('#id_target_label'); | ||||
|       var need_link = $('#id_need_link_full'); | ||||
|       function update_targets () { | ||||
|         if (target_type.val() == "USER") { | ||||
|           console.log(user); | ||||
|           user.show(); | ||||
|           club.hide(); | ||||
|           club_account.hide(); | ||||
|           company.hide(); | ||||
|           other.hide(); | ||||
|           need_link.hide(); | ||||
|         } else if (target_type.val() == "ACCOUNT") { | ||||
|           club_account.show(); | ||||
|           need_link.show(); | ||||
|           user.hide(); | ||||
|           club.hide(); | ||||
|           company.hide(); | ||||
|           other.hide(); | ||||
|         } else if (target_type.val() == "CLUB") { | ||||
|           club.show(); | ||||
|           user.hide(); | ||||
|           club_account.hide(); | ||||
|           company.hide(); | ||||
|           other.hide(); | ||||
|           need_link.hide(); | ||||
|         } else if (target_type.val() == "COMPANY") { | ||||
|           company.show(); | ||||
|           user.hide(); | ||||
|           club_account.hide(); | ||||
|           club.hide(); | ||||
|           other.hide(); | ||||
|           need_link.hide(); | ||||
|         } else if (target_type.val() == "OTHER") { | ||||
|           other.show(); | ||||
|           user.hide(); | ||||
|           club.hide(); | ||||
|           club_account.hide(); | ||||
|           company.hide(); | ||||
|           need_link.hide(); | ||||
|         } else { | ||||
|           company.hide(); | ||||
|           user.hide(); | ||||
|           club_account.hide(); | ||||
|           club.hide(); | ||||
|           other.hide(); | ||||
|           need_link.hide(); | ||||
|         } | ||||
|       } | ||||
|       update_targets(); | ||||
|       target_type.change(update_targets); | ||||
|     } ); | ||||
|   </script> | ||||
|   </div> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,16 +1,16 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Refound account{% endtrans %} | ||||
|   {% trans %}Refound account{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
| 	<div id="accounting"> | ||||
| 	    <h3>{% trans %}Refound account{% endtrans %}</h3> | ||||
| 	    <form action="" method="post"> | ||||
| 	        {% csrf_token %} | ||||
| 	        {{ form.as_p() }} | ||||
| 	        <p><input type="submit" value="{% trans %}Refound{% endtrans %}" /></p> | ||||
| 	    </form> | ||||
| 	</div> | ||||
|   <div id="accounting"> | ||||
|     <h3>{% trans %}Refound account{% endtrans %}</h3> | ||||
|     <form action="" method="post"> | ||||
|       {% csrf_token %} | ||||
|       {{ form.as_p() }} | ||||
|       <p><input type="submit" value="{% trans %}Refound{% endtrans %}" /></p> | ||||
|     </form> | ||||
|   </div> | ||||
| {% endblock %} | ||||
| @@ -1,27 +1,27 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Simplified type list{% endtrans %} | ||||
|   {% trans %}Simplified type list{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="accounting"> | ||||
|         <p> | ||||
|         <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|         {% trans %}Simplified types{% endtrans %} | ||||
|         </p> | ||||
|         <hr> | ||||
|         <p><a href="{{ url('accounting:simple_type_new') }}">{% trans %}New simplified type{% endtrans %}</a></p> | ||||
|         {% if simplifiedaccountingtype_list %} | ||||
|         <h3>{% trans %}Simplified type list{% endtrans %}</h3> | ||||
|         <ul> | ||||
|             {% for a in simplifiedaccountingtype_list  %} | ||||
|             <li><a href="{{ url('accounting:simple_type_edit', type_id=a.id) }}">{{ a }}</a></li> | ||||
|             {% endfor %} | ||||
|         </ul> | ||||
|         {% else %} | ||||
|         {% trans %}There is no types in this website.{% endtrans %} | ||||
|         {% endif %} | ||||
|     </div> | ||||
|   <div id="accounting"> | ||||
|     <p> | ||||
|       <a href="{{ url('accounting:bank_list') }}">{% trans %}Accounting{% endtrans %}</a> > | ||||
|       {% trans %}Simplified types{% endtrans %} | ||||
|     </p> | ||||
|     <hr> | ||||
|     <p><a href="{{ url('accounting:simple_type_new') }}">{% trans %}New simplified type{% endtrans %}</a></p> | ||||
|     {% if simplifiedaccountingtype_list %} | ||||
|       <h3>{% trans %}Simplified type list{% endtrans %}</h3> | ||||
|       <ul> | ||||
|         {% for a in simplifiedaccountingtype_list  %} | ||||
|           <li><a href="{{ url('accounting:simple_type_edit', type_id=a.id) }}">{{ a }}</a></li> | ||||
|         {% endfor %} | ||||
|       </ul> | ||||
|     {% else %} | ||||
|       {% trans %}There is no types in this website.{% endtrans %} | ||||
|     {% endif %} | ||||
|   </div> | ||||
| {% endblock %} | ||||
|  | ||||
|   | ||||
| @@ -1,124 +1,109 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # Copyright 2023 © AE UTBM | ||||
| # ae@utbm.fr / ae.info@utbm.fr | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # This file is part of the website of the UTBM Student Association (AE UTBM), | ||||
| # https://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # You can find the source code of the website at https://github.com/ae-utbm/sith | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3) | ||||
| # SEE : https://raw.githubusercontent.com/ae-utbm/sith/master/LICENSE | ||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | ||||
| # | ||||
| # | ||||
|  | ||||
| from datetime import date, timedelta | ||||
|  | ||||
| from django.test import TestCase | ||||
| from django.core.urlresolvers import reverse | ||||
| from django.core.management import call_command | ||||
| from datetime import date | ||||
| from django.urls import reverse | ||||
|  | ||||
| from core.models import User | ||||
| from accounting.models import ( | ||||
|     GeneralJournal, | ||||
|     Operation, | ||||
|     Label, | ||||
|     AccountingType, | ||||
|     GeneralJournal, | ||||
|     Label, | ||||
|     Operation, | ||||
|     SimplifiedAccountingType, | ||||
| ) | ||||
| from core.models import User | ||||
|  | ||||
|  | ||||
| class RefoundAccountTest(TestCase): | ||||
|     def setUp(self): | ||||
|         call_command("populate") | ||||
|         self.skia = User.objects.filter(username="skia").first() | ||||
| class TestRefoundAccount(TestCase): | ||||
|     @classmethod | ||||
|     def setUpTestData(cls): | ||||
|         cls.skia = User.objects.get(username="skia") | ||||
|         # reffil skia's account | ||||
|         self.skia.customer.amount = 800 | ||||
|         self.skia.customer.save() | ||||
|         cls.skia.customer.amount = 800 | ||||
|         cls.skia.customer.save() | ||||
|         cls.refound_account_url = reverse("accounting:refound_account") | ||||
|  | ||||
|     def test_permission_denied(self): | ||||
|         self.client.login(username="guy", password="plop") | ||||
|         self.client.force_login(User.objects.get(username="guy")) | ||||
|         response_post = self.client.post( | ||||
|             reverse("accounting:refound_account"), {"user": self.skia.id} | ||||
|             self.refound_account_url, {"user": self.skia.id} | ||||
|         ) | ||||
|         response_get = self.client.get(reverse("accounting:refound_account")) | ||||
|         self.assertTrue(response_get.status_code == 403) | ||||
|         self.assertTrue(response_post.status_code == 403) | ||||
|         response_get = self.client.get(self.refound_account_url) | ||||
|         assert response_get.status_code == 403 | ||||
|         assert response_post.status_code == 403 | ||||
|  | ||||
|     def test_root_granteed(self): | ||||
|         self.client.login(username="root", password="plop") | ||||
|         response_post = self.client.post( | ||||
|             reverse("accounting:refound_account"), {"user": self.skia.id} | ||||
|         ) | ||||
|         self.skia = User.objects.filter(username="skia").first() | ||||
|         response_get = self.client.get(reverse("accounting:refound_account")) | ||||
|         self.assertFalse(response_get.status_code == 403) | ||||
|         self.assertTrue('<form action="" method="post">' in str(response_get.content)) | ||||
|         self.assertFalse(response_post.status_code == 403) | ||||
|         self.assertTrue(self.skia.customer.amount == 0) | ||||
|         self.client.force_login(User.objects.get(username="root")) | ||||
|         response = self.client.post(self.refound_account_url, {"user": self.skia.id}) | ||||
|         self.assertRedirects(response, self.refound_account_url) | ||||
|         self.skia.refresh_from_db() | ||||
|         response = self.client.get(self.refound_account_url) | ||||
|         assert response.status_code == 200 | ||||
|         assert '<form action="" method="post">' in str(response.content) | ||||
|         assert self.skia.customer.amount == 0 | ||||
|  | ||||
|     def test_comptable_granteed(self): | ||||
|         self.client.login(username="comptable", password="plop") | ||||
|         response_post = self.client.post( | ||||
|             reverse("accounting:refound_account"), {"user": self.skia.id} | ||||
|         ) | ||||
|         self.skia = User.objects.filter(username="skia").first() | ||||
|         response_get = self.client.get(reverse("accounting:refound_account")) | ||||
|         self.assertFalse(response_get.status_code == 403) | ||||
|         self.assertTrue('<form action="" method="post">' in str(response_get.content)) | ||||
|         self.assertFalse(response_post.status_code == 403) | ||||
|         self.assertTrue(self.skia.customer.amount == 0) | ||||
|         self.client.force_login(User.objects.get(username="comptable")) | ||||
|         response = self.client.post(self.refound_account_url, {"user": self.skia.id}) | ||||
|         self.assertRedirects(response, self.refound_account_url) | ||||
|         self.skia.refresh_from_db() | ||||
|         response = self.client.get(self.refound_account_url) | ||||
|         assert response.status_code == 200 | ||||
|         assert '<form action="" method="post">' in str(response.content) | ||||
|         assert self.skia.customer.amount == 0 | ||||
|  | ||||
|  | ||||
| class JournalTest(TestCase): | ||||
|     def setUp(self): | ||||
|         call_command("populate") | ||||
|         self.journal = GeneralJournal.objects.filter(id=1).first() | ||||
| class TestJournal(TestCase): | ||||
|     @classmethod | ||||
|     def setUpTestData(cls): | ||||
|         cls.journal = GeneralJournal.objects.get(id=1) | ||||
|  | ||||
|     def test_permission_granted(self): | ||||
|         self.client.login(username="comptable", password="plop") | ||||
|         self.client.force_login(User.objects.get(username="comptable")) | ||||
|         response_get = self.client.get( | ||||
|             reverse("accounting:journal_details", args=[self.journal.id]) | ||||
|         ) | ||||
|  | ||||
|         self.assertTrue(response_get.status_code == 200) | ||||
|         self.assertTrue( | ||||
|             "<td>M\\xc3\\xa9thode de paiement</td>" in str(response_get.content) | ||||
|         ) | ||||
|         assert response_get.status_code == 200 | ||||
|         assert "<td>M\\xc3\\xa9thode de paiement</td>" in str(response_get.content) | ||||
|  | ||||
|     def test_permission_not_granted(self): | ||||
|         self.client.login(username="skia", password="plop") | ||||
|         self.client.force_login(User.objects.get(username="skia")) | ||||
|         response_get = self.client.get( | ||||
|             reverse("accounting:journal_details", args=[self.journal.id]) | ||||
|         ) | ||||
|  | ||||
|         self.assertTrue(response_get.status_code == 403) | ||||
|         self.assertFalse( | ||||
|             "<td>M\xc3\xa9thode de paiement</td>" in str(response_get.content) | ||||
|         ) | ||||
|         assert response_get.status_code == 403 | ||||
|         assert "<td>M\xc3\xa9thode de paiement</td>" not in str(response_get.content) | ||||
|  | ||||
|  | ||||
| class OperationTest(TestCase): | ||||
| class TestOperation(TestCase): | ||||
|     def setUp(self): | ||||
|         call_command("populate") | ||||
|         self.tomorrow_formatted = (date.today() + timedelta(days=1)).strftime( | ||||
|             "%d/%m/%Y" | ||||
|         ) | ||||
|         self.journal = GeneralJournal.objects.filter(id=1).first() | ||||
|         self.skia = User.objects.filter(username="skia").first() | ||||
|         at = AccountingType( | ||||
|             code="443", label="Ce code n'existe pas", movement_type="CREDIT" | ||||
|         ) | ||||
|         at.save() | ||||
|         l = Label(club_account=self.journal.club_account, name="bob") | ||||
|         l.save() | ||||
|         self.client.login(username="comptable", password="plop") | ||||
|         label = Label.objects.create(club_account=self.journal.club_account, name="bob") | ||||
|         self.client.force_login(User.objects.get(username="comptable")) | ||||
|         self.op1 = Operation( | ||||
|             journal=self.journal, | ||||
|             date=date.today(), | ||||
| @@ -126,7 +111,7 @@ class OperationTest(TestCase): | ||||
|             remark="Test bilan", | ||||
|             mode="CASH", | ||||
|             done=True, | ||||
|             label=l, | ||||
|             label=label, | ||||
|             accounting_type=at, | ||||
|             target_type="USER", | ||||
|             target_id=self.skia.id, | ||||
| @@ -139,7 +124,7 @@ class OperationTest(TestCase): | ||||
|             remark="Test bilan", | ||||
|             mode="CASH", | ||||
|             done=True, | ||||
|             label=l, | ||||
|             label=label, | ||||
|             accounting_type=at, | ||||
|             target_type="USER", | ||||
|             target_id=self.skia.id, | ||||
| @@ -147,8 +132,7 @@ class OperationTest(TestCase): | ||||
|         self.op2.save() | ||||
|  | ||||
|     def test_new_operation(self): | ||||
|         self.client.login(username="comptable", password="plop") | ||||
|         at = AccountingType.objects.filter(code="604").first() | ||||
|         at = AccountingType.objects.get(code="604") | ||||
|         response = self.client.post( | ||||
|             reverse("accounting:op_new", args=[self.journal.id]), | ||||
|             { | ||||
| @@ -158,7 +142,7 @@ class OperationTest(TestCase): | ||||
|                 "target_type": "OTHER", | ||||
|                 "target_id": "", | ||||
|                 "target_label": "Le fantome de la nuit", | ||||
|                 "date": "04/12/2020", | ||||
|                 "date": self.tomorrow_formatted, | ||||
|                 "mode": "CASH", | ||||
|                 "cheque_number": "", | ||||
|                 "invoice": "", | ||||
| @@ -180,8 +164,7 @@ class OperationTest(TestCase): | ||||
|         self.assertTrue("<td>Le fantome de la nuit</td>" in str(response_get.content)) | ||||
|  | ||||
|     def test_bad_new_operation(self): | ||||
|         self.client.login(username="comptable", password="plop") | ||||
|         AccountingType.objects.filter(code="604").first() | ||||
|         AccountingType.objects.get(code="604") | ||||
|         response = self.client.post( | ||||
|             reverse("accounting:op_new", args=[self.journal.id]), | ||||
|             { | ||||
| @@ -191,7 +174,7 @@ class OperationTest(TestCase): | ||||
|                 "target_type": "OTHER", | ||||
|                 "target_id": "", | ||||
|                 "target_label": "Le fantome de la nuit", | ||||
|                 "date": "04/12/2020", | ||||
|                 "date": self.tomorrow_formatted, | ||||
|                 "mode": "CASH", | ||||
|                 "cheque_number": "", | ||||
|                 "invoice": "", | ||||
| @@ -207,7 +190,7 @@ class OperationTest(TestCase): | ||||
|         ) | ||||
|  | ||||
|     def test_new_operation_not_authorized(self): | ||||
|         self.client.login(username="skia", password="plop") | ||||
|         self.client.force_login(self.skia) | ||||
|         at = AccountingType.objects.filter(code="604").first() | ||||
|         response = self.client.post( | ||||
|             reverse("accounting:op_new", args=[self.journal.id]), | ||||
| @@ -218,7 +201,7 @@ class OperationTest(TestCase): | ||||
|                 "target_type": "OTHER", | ||||
|                 "target_id": "", | ||||
|                 "target_label": "Le fantome du jour", | ||||
|                 "date": "04/12/2020", | ||||
|                 "date": self.tomorrow_formatted, | ||||
|                 "mode": "CASH", | ||||
|                 "cheque_number": "", | ||||
|                 "invoice": "", | ||||
| @@ -233,8 +216,7 @@ class OperationTest(TestCase): | ||||
|             self.journal.operations.filter(target_label="Le fantome du jour").exists() | ||||
|         ) | ||||
|  | ||||
|     def test__operation_simple_accounting(self): | ||||
|         self.client.login(username="comptable", password="plop") | ||||
|     def test_operation_simple_accounting(self): | ||||
|         sat = SimplifiedAccountingType.objects.all().first() | ||||
|         response = self.client.post( | ||||
|             reverse("accounting:op_new", args=[self.journal.id]), | ||||
| @@ -245,7 +227,7 @@ class OperationTest(TestCase): | ||||
|                 "target_type": "OTHER", | ||||
|                 "target_id": "", | ||||
|                 "target_label": "Le fantome de l'aurore", | ||||
|                 "date": "04/12/2020", | ||||
|                 "date": self.tomorrow_formatted, | ||||
|                 "mode": "CASH", | ||||
|                 "cheque_number": "", | ||||
|                 "invoice": "", | ||||
| @@ -255,15 +237,14 @@ class OperationTest(TestCase): | ||||
|                 "done": False, | ||||
|             }, | ||||
|         ) | ||||
|         self.assertFalse(response.status_code == 403) | ||||
|         self.assertTrue(self.journal.operations.filter(amount=23).exists()) | ||||
|         assert response.status_code != 403 | ||||
|         assert self.journal.operations.filter(amount=23).exists() | ||||
|         response_get = self.client.get( | ||||
|             reverse("accounting:journal_details", args=[self.journal.id]) | ||||
|         ) | ||||
|         self.assertTrue( | ||||
|             "<td>Le fantome de l'aurore</td>" in str(response_get.content) | ||||
|         ) | ||||
|         self.assertTrue( | ||||
|         assert "<td>Le fantome de l'aurore</td>" in str(response_get.content) | ||||
|  | ||||
|         assert ( | ||||
|             self.journal.operations.filter(amount=23) | ||||
|             .values("accounting_type") | ||||
|             .first()["accounting_type"] | ||||
| @@ -271,31 +252,41 @@ class OperationTest(TestCase): | ||||
|         ) | ||||
|  | ||||
|     def test_nature_statement(self): | ||||
|         self.client.login(username="comptable", password="plop") | ||||
|         response_get = self.client.get( | ||||
|         response = self.client.get( | ||||
|             reverse("accounting:journal_nature_statement", args=[self.journal.id]) | ||||
|         ) | ||||
|         self.assertTrue( | ||||
|             "bob (Troll Pench\\xc3\\xa9) : 3.00" in str(response_get.content) | ||||
|         ) | ||||
|         self.assertContains(response, "bob (Troll Penché) : 3.00", status_code=200) | ||||
|  | ||||
|     def test_person_statement(self): | ||||
|         self.client.login(username="comptable", password="plop") | ||||
|         response_get = self.client.get( | ||||
|         response = self.client.get( | ||||
|             reverse("accounting:journal_person_statement", args=[self.journal.id]) | ||||
|         ) | ||||
|         self.assertTrue( | ||||
|             "<td>3.00</td>" in str(response_get.content) | ||||
|             and '<td><a href="/user/1/">S' Kia</a></td>' | ||||
|             in str(response_get.content) | ||||
|         self.assertContains(response, "Total : 5575.72", status_code=200) | ||||
|         self.assertContains(response, "Total : 71.42") | ||||
|         content = response.content.decode() | ||||
|         self.assertInHTML( | ||||
|             """<td><a href="/user/1/">S' Kia</a></td><td>3.00</td>""", content | ||||
|         ) | ||||
|         self.assertInHTML( | ||||
|             """<td><a href="/user/1/">S' Kia</a></td><td>823.00</td>""", content | ||||
|         ) | ||||
|  | ||||
|     def test_accounting_statement(self): | ||||
|         self.client.login(username="comptable", password="plop") | ||||
|         response_get = self.client.get( | ||||
|         response = self.client.get( | ||||
|             reverse("accounting:journal_accounting_statement", args=[self.journal.id]) | ||||
|         ) | ||||
|         self.assertTrue( | ||||
|             "<td>443 - Cr\\xc3\\xa9dit - Ce code n'existe pas</td>" | ||||
|             in str(response_get.content) | ||||
|         assert response.status_code == 200 | ||||
|         self.assertInHTML( | ||||
|             """ | ||||
|             <tr> | ||||
|                 <td>443 - Crédit - Ce code n'existe pas</td> | ||||
|                 <td>3.00</td> | ||||
|             </tr>""", | ||||
|             response.content.decode(), | ||||
|         ) | ||||
|         self.assertContains( | ||||
|             response, | ||||
|             """ | ||||
|     <p><strong>Montant : </strong>-5504.30 €</p> | ||||
|     <p><strong>Montant effectif: </strong>-5504.30 €</p>""", | ||||
|         ) | ||||
|   | ||||
| @@ -1,152 +1,173 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # Copyright 2023 © AE UTBM | ||||
| # ae@utbm.fr / ae.info@utbm.fr | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # This file is part of the website of the UTBM Student Association (AE UTBM), | ||||
| # https://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # You can find the source code of the website at https://github.com/ae-utbm/sith | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3) | ||||
| # SEE : https://raw.githubusercontent.com/ae-utbm/sith/master/LICENSE | ||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | ||||
| # | ||||
| # | ||||
|  | ||||
| from django.conf.urls import url | ||||
| from django.urls import path | ||||
|  | ||||
| from accounting.views import * | ||||
| from accounting.views import ( | ||||
|     AccountingTypeCreateView, | ||||
|     AccountingTypeEditView, | ||||
|     AccountingTypeListView, | ||||
|     BankAccountCreateView, | ||||
|     BankAccountDeleteView, | ||||
|     BankAccountDetailView, | ||||
|     BankAccountEditView, | ||||
|     BankAccountListView, | ||||
|     ClubAccountCreateView, | ||||
|     ClubAccountDeleteView, | ||||
|     ClubAccountDetailView, | ||||
|     ClubAccountEditView, | ||||
|     CompanyCreateView, | ||||
|     CompanyEditView, | ||||
|     CompanyListView, | ||||
|     JournalAccountingStatementView, | ||||
|     JournalCreateView, | ||||
|     JournalDeleteView, | ||||
|     JournalDetailView, | ||||
|     JournalEditView, | ||||
|     JournalNatureStatementView, | ||||
|     JournalPersonStatementView, | ||||
|     LabelCreateView, | ||||
|     LabelDeleteView, | ||||
|     LabelEditView, | ||||
|     LabelListView, | ||||
|     OperationCreateView, | ||||
|     OperationEditView, | ||||
|     OperationPDFView, | ||||
|     RefoundAccountView, | ||||
|     SimplifiedAccountingTypeCreateView, | ||||
|     SimplifiedAccountingTypeEditView, | ||||
|     SimplifiedAccountingTypeListView, | ||||
| ) | ||||
|  | ||||
| urlpatterns = [ | ||||
|     # Accounting types | ||||
|     url( | ||||
|         r"^simple_type$", | ||||
|     path( | ||||
|         "simple_type/", | ||||
|         SimplifiedAccountingTypeListView.as_view(), | ||||
|         name="simple_type_list", | ||||
|     ), | ||||
|     url( | ||||
|         r"^simple_type/create$", | ||||
|     path( | ||||
|         "simple_type/create/", | ||||
|         SimplifiedAccountingTypeCreateView.as_view(), | ||||
|         name="simple_type_new", | ||||
|     ), | ||||
|     url( | ||||
|         r"^simple_type/(?P<type_id>[0-9]+)/edit$", | ||||
|     path( | ||||
|         "simple_type/<int:type_id>/edit/", | ||||
|         SimplifiedAccountingTypeEditView.as_view(), | ||||
|         name="simple_type_edit", | ||||
|     ), | ||||
|     # Accounting types | ||||
|     url(r"^type$", AccountingTypeListView.as_view(), name="type_list"), | ||||
|     url(r"^type/create$", AccountingTypeCreateView.as_view(), name="type_new"), | ||||
|     url( | ||||
|         r"^type/(?P<type_id>[0-9]+)/edit$", | ||||
|     path("type/", AccountingTypeListView.as_view(), name="type_list"), | ||||
|     path("type/create/", AccountingTypeCreateView.as_view(), name="type_new"), | ||||
|     path( | ||||
|         "type/<int:type_id>/edit/", | ||||
|         AccountingTypeEditView.as_view(), | ||||
|         name="type_edit", | ||||
|     ), | ||||
|     # Bank accounts | ||||
|     url(r"^$", BankAccountListView.as_view(), name="bank_list"), | ||||
|     url(r"^bank/create$", BankAccountCreateView.as_view(), name="bank_new"), | ||||
|     url( | ||||
|         r"^bank/(?P<b_account_id>[0-9]+)$", | ||||
|     path("", BankAccountListView.as_view(), name="bank_list"), | ||||
|     path("bank/create", BankAccountCreateView.as_view(), name="bank_new"), | ||||
|     path( | ||||
|         "bank/<int:b_account_id>/", | ||||
|         BankAccountDetailView.as_view(), | ||||
|         name="bank_details", | ||||
|     ), | ||||
|     url( | ||||
|         r"^bank/(?P<b_account_id>[0-9]+)/edit$", | ||||
|     path( | ||||
|         "bank/<int:b_account_id>/edit/", | ||||
|         BankAccountEditView.as_view(), | ||||
|         name="bank_edit", | ||||
|     ), | ||||
|     url( | ||||
|         r"^bank/(?P<b_account_id>[0-9]+)/delete$", | ||||
|     path( | ||||
|         "bank/<int:b_account_id>/delete/", | ||||
|         BankAccountDeleteView.as_view(), | ||||
|         name="bank_delete", | ||||
|     ), | ||||
|     # Club accounts | ||||
|     url(r"^club/create$", ClubAccountCreateView.as_view(), name="club_new"), | ||||
|     url( | ||||
|         r"^club/(?P<c_account_id>[0-9]+)$", | ||||
|     path("club/create/", ClubAccountCreateView.as_view(), name="club_new"), | ||||
|     path( | ||||
|         "club/<int:c_account_id>/", | ||||
|         ClubAccountDetailView.as_view(), | ||||
|         name="club_details", | ||||
|     ), | ||||
|     url( | ||||
|         r"^club/(?P<c_account_id>[0-9]+)/edit$", | ||||
|     path( | ||||
|         "club/<int:c_account_id>/edit/", | ||||
|         ClubAccountEditView.as_view(), | ||||
|         name="club_edit", | ||||
|     ), | ||||
|     url( | ||||
|         r"^club/(?P<c_account_id>[0-9]+)/delete$", | ||||
|     path( | ||||
|         "club/<int:c_account_id>/delete/", | ||||
|         ClubAccountDeleteView.as_view(), | ||||
|         name="club_delete", | ||||
|     ), | ||||
|     # Journals | ||||
|     url(r"^journal/create$", JournalCreateView.as_view(), name="journal_new"), | ||||
|     url( | ||||
|         r"^journal/(?P<j_id>[0-9]+)$", | ||||
|     path("journal/create/", JournalCreateView.as_view(), name="journal_new"), | ||||
|     path( | ||||
|         "journal/<int:j_id>/", | ||||
|         JournalDetailView.as_view(), | ||||
|         name="journal_details", | ||||
|     ), | ||||
|     url( | ||||
|         r"^journal/(?P<j_id>[0-9]+)/edit$", | ||||
|     path( | ||||
|         "journal/<int:j_id>/edit/", | ||||
|         JournalEditView.as_view(), | ||||
|         name="journal_edit", | ||||
|     ), | ||||
|     url( | ||||
|         r"^journal/(?P<j_id>[0-9]+)/delete$", | ||||
|     path( | ||||
|         "journal/<int:j_id>/delete/", | ||||
|         JournalDeleteView.as_view(), | ||||
|         name="journal_delete", | ||||
|     ), | ||||
|     url( | ||||
|         r"^journal/(?P<j_id>[0-9]+)/statement/nature$", | ||||
|     path( | ||||
|         "journal/<int:j_id>/statement/nature/", | ||||
|         JournalNatureStatementView.as_view(), | ||||
|         name="journal_nature_statement", | ||||
|     ), | ||||
|     url( | ||||
|         r"^journal/(?P<j_id>[0-9]+)/statement/person$", | ||||
|     path( | ||||
|         "journal/<int:j_id>/statement/person/", | ||||
|         JournalPersonStatementView.as_view(), | ||||
|         name="journal_person_statement", | ||||
|     ), | ||||
|     url( | ||||
|         r"^journal/(?P<j_id>[0-9]+)/statement/accounting$", | ||||
|     path( | ||||
|         "journal/<int:j_id>/statement/accounting/", | ||||
|         JournalAccountingStatementView.as_view(), | ||||
|         name="journal_accounting_statement", | ||||
|     ), | ||||
|     # Operations | ||||
|     url( | ||||
|         r"^operation/create/(?P<j_id>[0-9]+)$", | ||||
|     path( | ||||
|         "operation/create/<int:j_id>/", | ||||
|         OperationCreateView.as_view(), | ||||
|         name="op_new", | ||||
|     ), | ||||
|     url(r"^operation/(?P<op_id>[0-9]+)$", OperationEditView.as_view(), name="op_edit"), | ||||
|     url( | ||||
|         r"^operation/(?P<op_id>[0-9]+)/pdf$", OperationPDFView.as_view(), name="op_pdf" | ||||
|     ), | ||||
|     path("operation/<int:op_id>/", OperationEditView.as_view(), name="op_edit"), | ||||
|     path("operation/<int:op_id>/pdf/", OperationPDFView.as_view(), name="op_pdf"), | ||||
|     # Companies | ||||
|     url(r"^company/list$", CompanyListView.as_view(), name="co_list"), | ||||
|     url(r"^company/create$", CompanyCreateView.as_view(), name="co_new"), | ||||
|     url(r"^company/(?P<co_id>[0-9]+)$", CompanyEditView.as_view(), name="co_edit"), | ||||
|     path("company/list/", CompanyListView.as_view(), name="co_list"), | ||||
|     path("company/create/", CompanyCreateView.as_view(), name="co_new"), | ||||
|     path("company/<int:co_id>/", CompanyEditView.as_view(), name="co_edit"), | ||||
|     # Labels | ||||
|     url(r"^label/new$", LabelCreateView.as_view(), name="label_new"), | ||||
|     url( | ||||
|         r"^label/(?P<clubaccount_id>[0-9]+)$", | ||||
|     path("label/new/", LabelCreateView.as_view(), name="label_new"), | ||||
|     path( | ||||
|         "label/<int:clubaccount_id>/", | ||||
|         LabelListView.as_view(), | ||||
|         name="label_list", | ||||
|     ), | ||||
|     url( | ||||
|         r"^label/(?P<label_id>[0-9]+)/edit$", LabelEditView.as_view(), name="label_edit" | ||||
|     ), | ||||
|     url( | ||||
|         r"^label/(?P<label_id>[0-9]+)/delete$", | ||||
|     path("label/<int:label_id>/edit/", LabelEditView.as_view(), name="label_edit"), | ||||
|     path( | ||||
|         "label/<int:label_id>/delete/", | ||||
|         LabelDeleteView.as_view(), | ||||
|         name="label_delete", | ||||
|     ), | ||||
|     # User account | ||||
|     url(r"^refound/account$", RefoundAccountView.as_view(), name="refound_account"), | ||||
|     path("refound/account/", RefoundAccountView.as_view(), name="refound_account"), | ||||
| ] | ||||
|   | ||||
| @@ -1,70 +1,66 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # Copyright 2023 © AE UTBM | ||||
| # ae@utbm.fr / ae.info@utbm.fr | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # This file is part of the website of the UTBM Student Association (AE UTBM), | ||||
| # https://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # You can find the source code of the website at https://github.com/ae-utbm/sith | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3) | ||||
| # SEE : https://raw.githubusercontent.com/ae-utbm/sith/master/LICENSE | ||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | ||||
| # | ||||
| # | ||||
|  | ||||
| from django.views.generic import ListView, DetailView | ||||
| from django.views.generic.edit import UpdateView, CreateView, DeleteView, FormView | ||||
| from django.core.urlresolvers import reverse_lazy, reverse | ||||
| from django.utils.translation import ugettext_lazy as _ | ||||
| from django.forms.models import modelform_factory | ||||
| from django.core.exceptions import PermissionDenied, ValidationError | ||||
| from django.forms import HiddenInput | ||||
| from django.db import transaction | ||||
| from django.db.models import Sum | ||||
| from django.conf import settings | ||||
| from django import forms | ||||
| from django.http import HttpResponse | ||||
| import collections | ||||
|  | ||||
| from ajax_select.fields import AutoCompleteSelectField | ||||
| from django import forms | ||||
| from django.conf import settings | ||||
| from django.core.exceptions import PermissionDenied, ValidationError | ||||
| from django.db import transaction | ||||
| from django.db.models import Sum | ||||
| from django.forms import HiddenInput | ||||
| from django.forms.models import modelform_factory | ||||
| from django.http import HttpResponse | ||||
| from django.urls import reverse, reverse_lazy | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| from django.views.generic import DetailView, ListView | ||||
| from django.views.generic.edit import CreateView, DeleteView, FormView, UpdateView | ||||
|  | ||||
| from core.views import ( | ||||
|     CanViewMixin, | ||||
|     CanEditMixin, | ||||
|     CanEditPropMixin, | ||||
|     CanCreateMixin, | ||||
|     TabedViewMixin, | ||||
| ) | ||||
| from core.views.forms import SelectFile, SelectDate | ||||
| from accounting.models import ( | ||||
|     AccountingType, | ||||
|     BankAccount, | ||||
|     ClubAccount, | ||||
|     GeneralJournal, | ||||
|     Operation, | ||||
|     AccountingType, | ||||
|     Company, | ||||
|     SimplifiedAccountingType, | ||||
|     GeneralJournal, | ||||
|     Label, | ||||
|     Operation, | ||||
|     SimplifiedAccountingType, | ||||
| ) | ||||
| from counter.models import Counter, Selling, Product | ||||
| from accounting.widgets.select import ( | ||||
|     AutoCompleteSelectClubAccount, | ||||
|     AutoCompleteSelectCompany, | ||||
| ) | ||||
| from club.models import Club | ||||
| from club.widgets.select import AutoCompleteSelectClub | ||||
| from core.models import User | ||||
| from core.views import ( | ||||
|     CanCreateMixin, | ||||
|     CanEditMixin, | ||||
|     CanEditPropMixin, | ||||
|     CanViewMixin, | ||||
|     TabedViewMixin, | ||||
| ) | ||||
| from core.views.forms import SelectDate, SelectFile | ||||
| from core.views.widgets.select import AutoCompleteSelectUser | ||||
| from counter.models import Counter, Product, Selling | ||||
|  | ||||
| # Main accounting view | ||||
|  | ||||
|  | ||||
| class BankAccountListView(CanViewMixin, ListView): | ||||
|     """ | ||||
|     A list view for the admins | ||||
|     """ | ||||
|     """A list view for the admins.""" | ||||
|  | ||||
|     model = BankAccount | ||||
|     template_name = "accounting/bank_account_list.jinja" | ||||
| @@ -75,18 +71,14 @@ class BankAccountListView(CanViewMixin, ListView): | ||||
|  | ||||
|  | ||||
| class SimplifiedAccountingTypeListView(CanViewMixin, ListView): | ||||
|     """ | ||||
|     A list view for the admins | ||||
|     """ | ||||
|     """A list view for the admins.""" | ||||
|  | ||||
|     model = SimplifiedAccountingType | ||||
|     template_name = "accounting/simplifiedaccountingtype_list.jinja" | ||||
|  | ||||
|  | ||||
| class SimplifiedAccountingTypeEditView(CanViewMixin, UpdateView): | ||||
|     """ | ||||
|     An edit view for the admins | ||||
|     """ | ||||
|     """An edit view for the admins.""" | ||||
|  | ||||
|     model = SimplifiedAccountingType | ||||
|     pk_url_kwarg = "type_id" | ||||
| @@ -95,9 +87,7 @@ class SimplifiedAccountingTypeEditView(CanViewMixin, UpdateView): | ||||
|  | ||||
|  | ||||
| class SimplifiedAccountingTypeCreateView(CanCreateMixin, CreateView): | ||||
|     """ | ||||
|     Create an accounting type (for the admins) | ||||
|     """ | ||||
|     """Create an accounting type (for the admins).""" | ||||
|  | ||||
|     model = SimplifiedAccountingType | ||||
|     fields = ["label", "accounting_type"] | ||||
| @@ -108,18 +98,14 @@ class SimplifiedAccountingTypeCreateView(CanCreateMixin, CreateView): | ||||
|  | ||||
|  | ||||
| class AccountingTypeListView(CanViewMixin, ListView): | ||||
|     """ | ||||
|     A list view for the admins | ||||
|     """ | ||||
|     """A list view for the admins.""" | ||||
|  | ||||
|     model = AccountingType | ||||
|     template_name = "accounting/accountingtype_list.jinja" | ||||
|  | ||||
|  | ||||
| class AccountingTypeEditView(CanViewMixin, UpdateView): | ||||
|     """ | ||||
|     An edit view for the admins | ||||
|     """ | ||||
|     """An edit view for the admins.""" | ||||
|  | ||||
|     model = AccountingType | ||||
|     pk_url_kwarg = "type_id" | ||||
| @@ -128,9 +114,7 @@ class AccountingTypeEditView(CanViewMixin, UpdateView): | ||||
|  | ||||
|  | ||||
| class AccountingTypeCreateView(CanCreateMixin, CreateView): | ||||
|     """ | ||||
|     Create an accounting type (for the admins) | ||||
|     """ | ||||
|     """Create an accounting type (for the admins).""" | ||||
|  | ||||
|     model = AccountingType | ||||
|     fields = ["code", "label", "movement_type"] | ||||
| @@ -141,9 +125,7 @@ class AccountingTypeCreateView(CanCreateMixin, CreateView): | ||||
|  | ||||
|  | ||||
| class BankAccountEditView(CanViewMixin, UpdateView): | ||||
|     """ | ||||
|     An edit view for the admins | ||||
|     """ | ||||
|     """An edit view for the admins.""" | ||||
|  | ||||
|     model = BankAccount | ||||
|     pk_url_kwarg = "b_account_id" | ||||
| @@ -152,9 +134,7 @@ class BankAccountEditView(CanViewMixin, UpdateView): | ||||
|  | ||||
|  | ||||
| class BankAccountDetailView(CanViewMixin, DetailView): | ||||
|     """ | ||||
|     A detail view, listing every club account | ||||
|     """ | ||||
|     """A detail view, listing every club account.""" | ||||
|  | ||||
|     model = BankAccount | ||||
|     pk_url_kwarg = "b_account_id" | ||||
| @@ -162,9 +142,7 @@ class BankAccountDetailView(CanViewMixin, DetailView): | ||||
|  | ||||
|  | ||||
| class BankAccountCreateView(CanCreateMixin, CreateView): | ||||
|     """ | ||||
|     Create a bank account (for the admins) | ||||
|     """ | ||||
|     """Create a bank account (for the admins).""" | ||||
|  | ||||
|     model = BankAccount | ||||
|     fields = ["name", "club", "iban", "number"] | ||||
| @@ -174,9 +152,7 @@ class BankAccountCreateView(CanCreateMixin, CreateView): | ||||
| class BankAccountDeleteView( | ||||
|     CanEditPropMixin, DeleteView | ||||
| ):  # TODO change Delete to Close | ||||
|     """ | ||||
|     Delete a bank account (for the admins) | ||||
|     """ | ||||
|     """Delete a bank account (for the admins).""" | ||||
|  | ||||
|     model = BankAccount | ||||
|     pk_url_kwarg = "b_account_id" | ||||
| @@ -188,9 +164,7 @@ class BankAccountDeleteView( | ||||
|  | ||||
|  | ||||
| class ClubAccountEditView(CanViewMixin, UpdateView): | ||||
|     """ | ||||
|     An edit view for the admins | ||||
|     """ | ||||
|     """An edit view for the admins.""" | ||||
|  | ||||
|     model = ClubAccount | ||||
|     pk_url_kwarg = "c_account_id" | ||||
| @@ -199,9 +173,7 @@ class ClubAccountEditView(CanViewMixin, UpdateView): | ||||
|  | ||||
|  | ||||
| class ClubAccountDetailView(CanViewMixin, DetailView): | ||||
|     """ | ||||
|     A detail view, listing every journal | ||||
|     """ | ||||
|     """A detail view, listing every journal.""" | ||||
|  | ||||
|     model = ClubAccount | ||||
|     pk_url_kwarg = "c_account_id" | ||||
| @@ -209,17 +181,15 @@ class ClubAccountDetailView(CanViewMixin, DetailView): | ||||
|  | ||||
|  | ||||
| class ClubAccountCreateView(CanCreateMixin, CreateView): | ||||
|     """ | ||||
|     Create a club account (for the admins) | ||||
|     """ | ||||
|     """Create a club account (for the admins).""" | ||||
|  | ||||
|     model = ClubAccount | ||||
|     fields = ["name", "club", "bank_account"] | ||||
|     template_name = "core/create.jinja" | ||||
|  | ||||
|     def get_initial(self): | ||||
|         ret = super(ClubAccountCreateView, self).get_initial() | ||||
|         if "parent" in self.request.GET.keys(): | ||||
|         ret = super().get_initial() | ||||
|         if "parent" in self.request.GET: | ||||
|             obj = BankAccount.objects.filter(id=int(self.request.GET["parent"])).first() | ||||
|             if obj is not None: | ||||
|                 ret["bank_account"] = obj.id | ||||
| @@ -229,9 +199,7 @@ class ClubAccountCreateView(CanCreateMixin, CreateView): | ||||
| class ClubAccountDeleteView( | ||||
|     CanEditPropMixin, DeleteView | ||||
| ):  # TODO change Delete to Close | ||||
|     """ | ||||
|     Delete a club account (for the admins) | ||||
|     """ | ||||
|     """Delete a club account (for the admins).""" | ||||
|  | ||||
|     model = ClubAccount | ||||
|     pk_url_kwarg = "c_account_id" | ||||
| @@ -247,17 +215,14 @@ class JournalTabsMixin(TabedViewMixin): | ||||
|         return _("Journal") | ||||
|  | ||||
|     def get_list_of_tabs(self): | ||||
|         tab_list = [] | ||||
|         tab_list.append( | ||||
|         return [ | ||||
|             { | ||||
|                 "url": reverse( | ||||
|                     "accounting:journal_details", kwargs={"j_id": self.object.id} | ||||
|                 ), | ||||
|                 "slug": "journal", | ||||
|                 "name": _("Journal"), | ||||
|             } | ||||
|         ) | ||||
|         tab_list.append( | ||||
|             }, | ||||
|             { | ||||
|                 "url": reverse( | ||||
|                     "accounting:journal_nature_statement", | ||||
| @@ -265,9 +230,7 @@ class JournalTabsMixin(TabedViewMixin): | ||||
|                 ), | ||||
|                 "slug": "nature_statement", | ||||
|                 "name": _("Statement by nature"), | ||||
|             } | ||||
|         ) | ||||
|         tab_list.append( | ||||
|             }, | ||||
|             { | ||||
|                 "url": reverse( | ||||
|                     "accounting:journal_person_statement", | ||||
| @@ -275,9 +238,7 @@ class JournalTabsMixin(TabedViewMixin): | ||||
|                 ), | ||||
|                 "slug": "person_statement", | ||||
|                 "name": _("Statement by person"), | ||||
|             } | ||||
|         ) | ||||
|         tab_list.append( | ||||
|             }, | ||||
|             { | ||||
|                 "url": reverse( | ||||
|                     "accounting:journal_accounting_statement", | ||||
| @@ -285,15 +246,12 @@ class JournalTabsMixin(TabedViewMixin): | ||||
|                 ), | ||||
|                 "slug": "accounting_statement", | ||||
|                 "name": _("Accounting statement"), | ||||
|             } | ||||
|         ) | ||||
|         return tab_list | ||||
|             }, | ||||
|         ] | ||||
|  | ||||
|  | ||||
| class JournalCreateView(CanCreateMixin, CreateView): | ||||
|     """ | ||||
|     Create a general journal | ||||
|     """ | ||||
|     """Create a general journal.""" | ||||
|  | ||||
|     model = GeneralJournal | ||||
|     form_class = modelform_factory( | ||||
| @@ -304,8 +262,8 @@ class JournalCreateView(CanCreateMixin, CreateView): | ||||
|     template_name = "core/create.jinja" | ||||
|  | ||||
|     def get_initial(self): | ||||
|         ret = super(JournalCreateView, self).get_initial() | ||||
|         if "parent" in self.request.GET.keys(): | ||||
|         ret = super().get_initial() | ||||
|         if "parent" in self.request.GET: | ||||
|             obj = ClubAccount.objects.filter(id=int(self.request.GET["parent"])).first() | ||||
|             if obj is not None: | ||||
|                 ret["club_account"] = obj.id | ||||
| @@ -313,9 +271,7 @@ class JournalCreateView(CanCreateMixin, CreateView): | ||||
|  | ||||
|  | ||||
| class JournalDetailView(JournalTabsMixin, CanViewMixin, DetailView): | ||||
|     """ | ||||
|     A detail view, listing every operation | ||||
|     """ | ||||
|     """A detail view, listing every operation.""" | ||||
|  | ||||
|     model = GeneralJournal | ||||
|     pk_url_kwarg = "j_id" | ||||
| @@ -324,9 +280,7 @@ class JournalDetailView(JournalTabsMixin, CanViewMixin, DetailView): | ||||
|  | ||||
|  | ||||
| class JournalEditView(CanEditMixin, UpdateView): | ||||
|     """ | ||||
|     Update a general journal | ||||
|     """ | ||||
|     """Update a general journal.""" | ||||
|  | ||||
|     model = GeneralJournal | ||||
|     pk_url_kwarg = "j_id" | ||||
| @@ -335,9 +289,7 @@ class JournalEditView(CanEditMixin, UpdateView): | ||||
|  | ||||
|  | ||||
| class JournalDeleteView(CanEditPropMixin, DeleteView): | ||||
|     """ | ||||
|     Delete a club account (for the admins) | ||||
|     """ | ||||
|     """Delete a club account (for the admins).""" | ||||
|  | ||||
|     model = GeneralJournal | ||||
|     pk_url_kwarg = "j_id" | ||||
| @@ -347,7 +299,7 @@ class JournalDeleteView(CanEditPropMixin, DeleteView): | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         self.object = self.get_object() | ||||
|         if self.object.operations.count() == 0: | ||||
|             return super(JournalDeleteView, self).dispatch(request, *args, **kwargs) | ||||
|             return super().dispatch(request, *args, **kwargs) | ||||
|         else: | ||||
|             raise PermissionDenied | ||||
|  | ||||
| @@ -381,12 +333,30 @@ class OperationForm(forms.ModelForm): | ||||
|             "invoice": SelectFile, | ||||
|         } | ||||
|  | ||||
|     user = AutoCompleteSelectField("users", help_text=None, required=False) | ||||
|     club_account = AutoCompleteSelectField( | ||||
|         "club_accounts", help_text=None, required=False | ||||
|     user = forms.ModelChoiceField( | ||||
|         help_text=None, | ||||
|         required=False, | ||||
|         widget=AutoCompleteSelectUser, | ||||
|         queryset=User.objects.all(), | ||||
|     ) | ||||
|     club_account = forms.ModelChoiceField( | ||||
|         help_text=None, | ||||
|         required=False, | ||||
|         widget=AutoCompleteSelectClubAccount, | ||||
|         queryset=ClubAccount.objects.all(), | ||||
|     ) | ||||
|     club = forms.ModelChoiceField( | ||||
|         help_text=None, | ||||
|         required=False, | ||||
|         widget=AutoCompleteSelectClub, | ||||
|         queryset=Club.objects.all(), | ||||
|     ) | ||||
|     company = forms.ModelChoiceField( | ||||
|         help_text=None, | ||||
|         required=False, | ||||
|         widget=AutoCompleteSelectCompany, | ||||
|         queryset=Company.objects.all(), | ||||
|     ) | ||||
|     club = AutoCompleteSelectField("clubs", help_text=None, required=False) | ||||
|     company = AutoCompleteSelectField("companies", help_text=None, required=False) | ||||
|     need_link = forms.BooleanField( | ||||
|         label=_("Link this operation to the target account"), | ||||
|         required=False, | ||||
| @@ -395,7 +365,7 @@ class OperationForm(forms.ModelForm): | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         club_account = kwargs.pop("club_account", None) | ||||
|         super(OperationForm, self).__init__(*args, **kwargs) | ||||
|         super().__init__(*args, **kwargs) | ||||
|         if club_account: | ||||
|             self.fields["label"].queryset = club_account.labels.order_by("name").all() | ||||
|         if self.instance.target_type == "USER": | ||||
| @@ -408,8 +378,8 @@ class OperationForm(forms.ModelForm): | ||||
|             self.fields["company"].initial = self.instance.target_id | ||||
|  | ||||
|     def clean(self): | ||||
|         self.cleaned_data = super(OperationForm, self).clean() | ||||
|         if "target_type" in self.cleaned_data.keys(): | ||||
|         self.cleaned_data = super().clean() | ||||
|         if "target_type" in self.cleaned_data: | ||||
|             if ( | ||||
|                 self.cleaned_data.get("user") is None | ||||
|                 and self.cleaned_data.get("club") is None | ||||
| @@ -438,7 +408,7 @@ class OperationForm(forms.ModelForm): | ||||
|         return self.cleaned_data | ||||
|  | ||||
|     def save(self): | ||||
|         ret = super(OperationForm, self).save() | ||||
|         ret = super().save() | ||||
|         if ( | ||||
|             self.instance.target_type == "ACCOUNT" | ||||
|             and not self.instance.linked_operation | ||||
| @@ -476,9 +446,7 @@ class OperationForm(forms.ModelForm): | ||||
|  | ||||
|  | ||||
| class OperationCreateView(CanCreateMixin, CreateView): | ||||
|     """ | ||||
|     Create an operation | ||||
|     """ | ||||
|     """Create an operation.""" | ||||
|  | ||||
|     model = Operation | ||||
|     form_class = OperationForm | ||||
| @@ -490,23 +458,21 @@ class OperationCreateView(CanCreateMixin, CreateView): | ||||
|         return self.form_class(club_account=ca, **self.get_form_kwargs()) | ||||
|  | ||||
|     def get_initial(self): | ||||
|         ret = super(OperationCreateView, self).get_initial() | ||||
|         ret = super().get_initial() | ||||
|         if self.journal is not None: | ||||
|             ret["journal"] = self.journal.id | ||||
|         return ret | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         """ Add journal to the context """ | ||||
|         kwargs = super(OperationCreateView, self).get_context_data(**kwargs) | ||||
|         """Add journal to the context.""" | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         if self.journal: | ||||
|             kwargs["object"] = self.journal | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class OperationEditView(CanEditMixin, UpdateView): | ||||
|     """ | ||||
|     An edit view, working as detail for the moment | ||||
|     """ | ||||
|     """An edit view, working as detail for the moment.""" | ||||
|  | ||||
|     model = Operation | ||||
|     pk_url_kwarg = "op_id" | ||||
| @@ -514,29 +480,27 @@ class OperationEditView(CanEditMixin, UpdateView): | ||||
|     template_name = "accounting/operation_edit.jinja" | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         """ Add journal to the context """ | ||||
|         kwargs = super(OperationEditView, self).get_context_data(**kwargs) | ||||
|         """Add journal to the context.""" | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["object"] = self.object.journal | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class OperationPDFView(CanViewMixin, DetailView): | ||||
|     """ | ||||
|     Display the PDF of a given operation | ||||
|     """ | ||||
|     """Display the PDF of a given operation.""" | ||||
|  | ||||
|     model = Operation | ||||
|     pk_url_kwarg = "op_id" | ||||
|  | ||||
|     def get(self, request, *args, **kwargs): | ||||
|         from reportlab.pdfgen import canvas | ||||
|         from reportlab.lib.units import cm | ||||
|         from reportlab.platypus import Table, TableStyle | ||||
|         from reportlab.lib import colors | ||||
|         from reportlab.lib.pagesizes import letter | ||||
|         from reportlab.lib.units import cm | ||||
|         from reportlab.lib.utils import ImageReader | ||||
|         from reportlab.pdfbase.ttfonts import TTFont | ||||
|         from reportlab.pdfbase import pdfmetrics | ||||
|         from reportlab.pdfbase.ttfonts import TTFont | ||||
|         from reportlab.pdfgen import canvas | ||||
|         from reportlab.platypus import Table, TableStyle | ||||
|  | ||||
|         pdfmetrics.registerFont(TTFont("DejaVu", "DejaVuSerif.ttf")) | ||||
|  | ||||
| @@ -607,7 +571,7 @@ class OperationPDFView(CanViewMixin, DetailView): | ||||
|         payment_mode = "" | ||||
|         for m in settings.SITH_ACCOUNTING_PAYMENT_METHOD: | ||||
|             if m[0] == mode: | ||||
|                 payment_mode += "[\u00D7]" | ||||
|                 payment_mode += "[\u00d7]" | ||||
|             else: | ||||
|                 payment_mode += "[  ]" | ||||
|             payment_mode += " %s\n" % (m[1]) | ||||
| @@ -675,9 +639,7 @@ class OperationPDFView(CanViewMixin, DetailView): | ||||
|  | ||||
|  | ||||
| class JournalNatureStatementView(JournalTabsMixin, CanViewMixin, DetailView): | ||||
|     """ | ||||
|     Display a statement sorted by labels | ||||
|     """ | ||||
|     """Display a statement sorted by labels.""" | ||||
|  | ||||
|     model = GeneralJournal | ||||
|     pk_url_kwarg = "j_id" | ||||
| @@ -688,19 +650,17 @@ class JournalNatureStatementView(JournalTabsMixin, CanViewMixin, DetailView): | ||||
|         ret = collections.OrderedDict() | ||||
|         statement = collections.OrderedDict() | ||||
|         total_sum = 0 | ||||
|         for sat in [None] + list( | ||||
|             SimplifiedAccountingType.objects.order_by("label").all() | ||||
|         ): | ||||
|             sum = queryset.filter( | ||||
|         for sat in [ | ||||
|             None, | ||||
|             *list(SimplifiedAccountingType.objects.order_by("label")), | ||||
|         ]: | ||||
|             amount = queryset.filter( | ||||
|                 accounting_type__movement_type=movement_type, simpleaccounting_type=sat | ||||
|             ).aggregate(amount_sum=Sum("amount"))["amount_sum"] | ||||
|             if sat: | ||||
|                 sat = sat.label | ||||
|             else: | ||||
|                 sat = "" | ||||
|             if sum: | ||||
|                 total_sum += sum | ||||
|                 statement[sat] = sum | ||||
|             label = sat.label if sat is not None else "" | ||||
|             if amount: | ||||
|                 total_sum += amount | ||||
|                 statement[label] = amount | ||||
|         ret[movement_type] = statement | ||||
|         ret[movement_type + "_sum"] = total_sum | ||||
|         return ret | ||||
| @@ -723,28 +683,23 @@ class JournalNatureStatementView(JournalTabsMixin, CanViewMixin, DetailView): | ||||
|             self.statement(self.object.operations.filter(label=None).all(), "DEBIT") | ||||
|         ) | ||||
|         statement[_("No label operations")] = no_label_statement | ||||
|         for l in labels: | ||||
|         for label in labels: | ||||
|             l_stmt = collections.OrderedDict() | ||||
|             l_stmt.update( | ||||
|                 self.statement(self.object.operations.filter(label=l).all(), "CREDIT") | ||||
|             ) | ||||
|             l_stmt.update( | ||||
|                 self.statement(self.object.operations.filter(label=l).all(), "DEBIT") | ||||
|             ) | ||||
|             statement[l] = l_stmt | ||||
|             journals = self.object.operations.filter(label=label).all() | ||||
|             l_stmt.update(self.statement(journals, "CREDIT")) | ||||
|             l_stmt.update(self.statement(journals, "DEBIT")) | ||||
|             statement[label] = l_stmt | ||||
|         return statement | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         """ Add infos to the context """ | ||||
|         kwargs = super(JournalNatureStatementView, self).get_context_data(**kwargs) | ||||
|         """Add infos to the context.""" | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["statement"] = self.big_statement() | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class JournalPersonStatementView(JournalTabsMixin, CanViewMixin, DetailView): | ||||
|     """ | ||||
|     Calculate a dictionary with operation target and sum of operations | ||||
|     """ | ||||
|     """Calculate a dictionary with operation target and sum of operations.""" | ||||
|  | ||||
|     model = GeneralJournal | ||||
|     pk_url_kwarg = "j_id" | ||||
| @@ -774,8 +729,8 @@ class JournalPersonStatementView(JournalTabsMixin, CanViewMixin, DetailView): | ||||
|         return sum(self.statement(movement_type).values()) | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         """ Add journal to the context """ | ||||
|         kwargs = super(JournalPersonStatementView, self).get_context_data(**kwargs) | ||||
|         """Add journal to the context.""" | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["credit_statement"] = self.statement("CREDIT") | ||||
|         kwargs["debit_statement"] = self.statement("DEBIT") | ||||
|         kwargs["total_credit"] = self.total("CREDIT") | ||||
| @@ -784,9 +739,7 @@ class JournalPersonStatementView(JournalTabsMixin, CanViewMixin, DetailView): | ||||
|  | ||||
|  | ||||
| class JournalAccountingStatementView(JournalTabsMixin, CanViewMixin, DetailView): | ||||
|     """ | ||||
|     Calculate a dictionary with operation type and sum of operations | ||||
|     """ | ||||
|     """Calculate a dictionary with operation type and sum of operations.""" | ||||
|  | ||||
|     model = GeneralJournal | ||||
|     pk_url_kwarg = "j_id" | ||||
| @@ -804,8 +757,8 @@ class JournalAccountingStatementView(JournalTabsMixin, CanViewMixin, DetailView) | ||||
|         return statement | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         """ Add journal to the context """ | ||||
|         kwargs = super(JournalAccountingStatementView, self).get_context_data(**kwargs) | ||||
|         """Add journal to the context.""" | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["statement"] = self.statement() | ||||
|         return kwargs | ||||
|  | ||||
| @@ -819,9 +772,7 @@ class CompanyListView(CanViewMixin, ListView): | ||||
|  | ||||
|  | ||||
| class CompanyCreateView(CanCreateMixin, CreateView): | ||||
|     """ | ||||
|     Create a company | ||||
|     """ | ||||
|     """Create a company.""" | ||||
|  | ||||
|     model = Company | ||||
|     fields = ["name"] | ||||
| @@ -830,9 +781,7 @@ class CompanyCreateView(CanCreateMixin, CreateView): | ||||
|  | ||||
|  | ||||
| class CompanyEditView(CanCreateMixin, UpdateView): | ||||
|     """ | ||||
|     Edit a company | ||||
|     """ | ||||
|     """Edit a company.""" | ||||
|  | ||||
|     model = Company | ||||
|     pk_url_kwarg = "co_id" | ||||
| @@ -860,8 +809,8 @@ class LabelCreateView( | ||||
|     template_name = "core/create.jinja" | ||||
|  | ||||
|     def get_initial(self): | ||||
|         ret = super(LabelCreateView, self).get_initial() | ||||
|         if "parent" in self.request.GET.keys(): | ||||
|         ret = super().get_initial() | ||||
|         if "parent" in self.request.GET: | ||||
|             obj = ClubAccount.objects.filter(id=int(self.request.GET["parent"])).first() | ||||
|             if obj is not None: | ||||
|                 ret["club_account"] = obj.id | ||||
| @@ -885,39 +834,41 @@ class LabelDeleteView(CanEditMixin, DeleteView): | ||||
|  | ||||
|  | ||||
| class CloseCustomerAccountForm(forms.Form): | ||||
|     user = AutoCompleteSelectField( | ||||
|         "users", label=_("Refound this account"), help_text=None, required=True | ||||
|     user = forms.ModelChoiceField( | ||||
|         label=_("Refound this account"), | ||||
|         help_text=None, | ||||
|         required=True, | ||||
|         widget=AutoCompleteSelectUser, | ||||
|         queryset=User.objects.all(), | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class RefoundAccountView(FormView): | ||||
|     """ | ||||
|     Create a selling with the same amount than the current user money | ||||
|     """ | ||||
|     """Create a selling with the same amount than the current user money.""" | ||||
|  | ||||
|     template_name = "accounting/refound_account.jinja" | ||||
|     form_class = CloseCustomerAccountForm | ||||
|  | ||||
|     def permission(self, user): | ||||
|         if user.is_root or user.is_in_group(settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|         if user.is_root or user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID): | ||||
|             return True | ||||
|         else: | ||||
|             raise PermissionDenied | ||||
|  | ||||
|     def dispatch(self, request, *arg, **kwargs): | ||||
|         res = super(RefoundAccountView, self).dispatch(request, *arg, **kwargs) | ||||
|         res = super().dispatch(request, *arg, **kwargs) | ||||
|         if self.permission(request.user): | ||||
|             return res | ||||
|  | ||||
|     def post(self, request, *arg, **kwargs): | ||||
|         self.operator = request.user | ||||
|         if self.permission(request.user): | ||||
|             return super(RefoundAccountView, self).post(self, request, *arg, **kwargs) | ||||
|             return super().post(self, request, *arg, **kwargs) | ||||
|  | ||||
|     def form_valid(self, form): | ||||
|         self.customer = form.cleaned_data["user"] | ||||
|         self.create_selling() | ||||
|         return super(RefoundAccountView, self).form_valid(form) | ||||
|         return super().form_valid(form) | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse("accounting:refound_account") | ||||
|   | ||||
							
								
								
									
										39
									
								
								accounting/widgets/select.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								accounting/widgets/select.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| from pydantic import TypeAdapter | ||||
|  | ||||
| from accounting.models import ClubAccount, Company | ||||
| from accounting.schemas import ClubAccountSchema, CompanySchema | ||||
| from core.views.widgets.select import AutoCompleteSelect, AutoCompleteSelectMultiple | ||||
|  | ||||
| _js = ["bundled/accounting/components/ajax-select-index.ts"] | ||||
|  | ||||
|  | ||||
| class AutoCompleteSelectClubAccount(AutoCompleteSelect): | ||||
|     component_name = "club-account-ajax-select" | ||||
|     model = ClubAccount | ||||
|     adapter = TypeAdapter(list[ClubAccountSchema]) | ||||
|  | ||||
|     js = _js | ||||
|  | ||||
|  | ||||
| class AutoCompleteSelectMultipleClubAccount(AutoCompleteSelectMultiple): | ||||
|     component_name = "club-account-ajax-select" | ||||
|     model = ClubAccount | ||||
|     adapter = TypeAdapter(list[ClubAccountSchema]) | ||||
|  | ||||
|     js = _js | ||||
|  | ||||
|  | ||||
| class AutoCompleteSelectCompany(AutoCompleteSelect): | ||||
|     component_name = "company-ajax-select" | ||||
|     model = Company | ||||
|     adapter = TypeAdapter(list[CompanySchema]) | ||||
|  | ||||
|     js = _js | ||||
|  | ||||
|  | ||||
| class AutoCompleteSelectMultipleCompany(AutoCompleteSelectMultiple): | ||||
|     component_name = "company-ajax-select" | ||||
|     model = Company | ||||
|     adapter = TypeAdapter(list[CompanySchema]) | ||||
|  | ||||
|     js = _js | ||||
							
								
								
									
										10
									
								
								antispam/admin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								antispam/admin.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| from django.contrib import admin | ||||
|  | ||||
| from antispam.models import ToxicDomain | ||||
|  | ||||
|  | ||||
| @admin.register(ToxicDomain) | ||||
| class ToxicDomainAdmin(admin.ModelAdmin): | ||||
|     list_display = ("domain", "is_externally_managed", "created") | ||||
|     search_fields = ("domain", "is_externally_managed", "created") | ||||
|     list_filter = ("is_externally_managed",) | ||||
							
								
								
									
										7
									
								
								antispam/apps.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								antispam/apps.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| from django.apps import AppConfig | ||||
|  | ||||
|  | ||||
| class AntispamConfig(AppConfig): | ||||
|     default_auto_field = "django.db.models.BigAutoField" | ||||
|     verbose_name = "antispam" | ||||
|     name = "antispam" | ||||
							
								
								
									
										18
									
								
								antispam/forms.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								antispam/forms.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| import re | ||||
|  | ||||
| from django import forms | ||||
| from django.core.validators import EmailValidator | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
| from antispam.models import ToxicDomain | ||||
|  | ||||
|  | ||||
| class AntiSpamEmailField(forms.EmailField): | ||||
|     """An email field that email addresses with a known toxic domain.""" | ||||
|  | ||||
|     def run_validators(self, value: str): | ||||
|         super().run_validators(value) | ||||
|         # Domain part should exist since email validation is guaranteed to run first | ||||
|         domain = re.search(EmailValidator.domain_regex, value) | ||||
|         if ToxicDomain.objects.filter(domain=domain[0]).exists(): | ||||
|             raise forms.ValidationError(_("Email domain is not allowed.")) | ||||
							
								
								
									
										0
									
								
								antispam/management/commands/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								antispam/management/commands/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										69
									
								
								antispam/management/commands/update_spam_database.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								antispam/management/commands/update_spam_database.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| import requests | ||||
| from django.conf import settings | ||||
| from django.core.management import BaseCommand | ||||
| from django.db.models import Max | ||||
| from django.utils import timezone | ||||
|  | ||||
| from antispam.models import ToxicDomain | ||||
|  | ||||
|  | ||||
| class Command(BaseCommand): | ||||
|     """Update blocked ips/mails database""" | ||||
|  | ||||
|     help = "Update blocked ips/mails database" | ||||
|  | ||||
|     def add_arguments(self, parser): | ||||
|         parser.add_argument( | ||||
|             "--force", action="store_true", help="Force re-creation even if up to date" | ||||
|         ) | ||||
|  | ||||
|     def _should_update(self, *, force: bool = False) -> bool: | ||||
|         if force: | ||||
|             return True | ||||
|         oldest = ToxicDomain.objects.filter(is_externally_managed=True).aggregate( | ||||
|             res=Max("created") | ||||
|         )["res"] | ||||
|         return not (oldest and timezone.now() < (oldest + timezone.timedelta(days=1))) | ||||
|  | ||||
|     def _download_domains(self, providers: list[str]) -> set[str]: | ||||
|         domains = set() | ||||
|         for provider in providers: | ||||
|             res = requests.get(provider) | ||||
|             if not res.ok: | ||||
|                 self.stderr.write( | ||||
|                     f"Source {provider} responded with code {res.status_code}" | ||||
|                 ) | ||||
|                 continue | ||||
|             domains |= set(res.content.decode().splitlines()) | ||||
|         return domains | ||||
|  | ||||
|     def _update_domains(self, domains: set[str]): | ||||
|         # Cleanup database | ||||
|         ToxicDomain.objects.filter(is_externally_managed=True).delete() | ||||
|  | ||||
|         # Create database | ||||
|         ToxicDomain.objects.bulk_create( | ||||
|             [ | ||||
|                 ToxicDomain(domain=domain, is_externally_managed=True) | ||||
|                 for domain in domains | ||||
|             ], | ||||
|             ignore_conflicts=True, | ||||
|         ) | ||||
|         self.stdout.write("Domain database updated") | ||||
|  | ||||
|     def handle(self, *args, **options): | ||||
|         if not self._should_update(force=options["force"]): | ||||
|             self.stdout.write("Domain database is up to date") | ||||
|             return | ||||
|         self.stdout.write("Updating domain database") | ||||
|  | ||||
|         domains = self._download_domains(settings.TOXIC_DOMAINS_PROVIDERS) | ||||
|  | ||||
|         if not domains: | ||||
|             self.stderr.write( | ||||
|                 "No domains could be fetched from settings.TOXIC_DOMAINS_PROVIDERS. " | ||||
|                 "Please, have a look at your settings." | ||||
|             ) | ||||
|             return | ||||
|  | ||||
|         self._update_domains(domains) | ||||
							
								
								
									
										35
									
								
								antispam/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								antispam/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| # Generated by Django 4.2.14 on 2024-08-03 23:05 | ||||
|  | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|     initial = True | ||||
|  | ||||
|     dependencies = [] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.CreateModel( | ||||
|             name="ToxicDomain", | ||||
|             fields=[ | ||||
|                 ( | ||||
|                     "domain", | ||||
|                     models.URLField( | ||||
|                         max_length=253, | ||||
|                         primary_key=True, | ||||
|                         serialize=False, | ||||
|                         verbose_name="domain", | ||||
|                     ), | ||||
|                 ), | ||||
|                 ("created", models.DateTimeField(auto_now_add=True)), | ||||
|                 ( | ||||
|                     "is_externally_managed", | ||||
|                     models.BooleanField( | ||||
|                         default=False, | ||||
|                         help_text="True if kept up-to-date using external toxic domain providers, else False", | ||||
|                         verbose_name="is externally managed", | ||||
|                     ), | ||||
|                 ), | ||||
|             ], | ||||
|         ), | ||||
|     ] | ||||
							
								
								
									
										0
									
								
								antispam/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								antispam/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										19
									
								
								antispam/models.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								antispam/models.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| from django.db import models | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
|  | ||||
| class ToxicDomain(models.Model): | ||||
|     """Domain marked as spam in public databases""" | ||||
|  | ||||
|     domain = models.URLField(_("domain"), max_length=253, primary_key=True) | ||||
|     created = models.DateTimeField(auto_now_add=True) | ||||
|     is_externally_managed = models.BooleanField( | ||||
|         _("is externally managed"), | ||||
|         default=False, | ||||
|         help_text=_( | ||||
|             "True if kept up-to-date using external toxic domain providers, else False" | ||||
|         ), | ||||
|     ) | ||||
|  | ||||
|     def __str__(self) -> str: | ||||
|         return self.domain | ||||
| @@ -1,23 +0,0 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # | ||||
| # | ||||
							
								
								
									
										56
									
								
								api/urls.py
									
									
									
									
									
								
							
							
						
						
									
										56
									
								
								api/urls.py
									
									
									
									
									
								
							| @@ -1,56 +0,0 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # | ||||
| # | ||||
|  | ||||
| from django.conf.urls import url, include | ||||
|  | ||||
| from api.views import * | ||||
| from rest_framework import routers | ||||
|  | ||||
| # Router config | ||||
| router = routers.DefaultRouter() | ||||
| router.register(r"counter", CounterViewSet, base_name="api_counter") | ||||
| router.register(r"user", UserViewSet, base_name="api_user") | ||||
| router.register(r"club", ClubViewSet, base_name="api_club") | ||||
| router.register(r"group", GroupViewSet, base_name="api_group") | ||||
|  | ||||
| # Launderette | ||||
| router.register( | ||||
|     r"launderette/place", LaunderettePlaceViewSet, base_name="api_launderette_place" | ||||
| ) | ||||
| router.register( | ||||
|     r"launderette/machine", | ||||
|     LaunderetteMachineViewSet, | ||||
|     base_name="api_launderette_machine", | ||||
| ) | ||||
| router.register( | ||||
|     r"launderette/token", LaunderetteTokenViewSet, base_name="api_launderette_token" | ||||
| ) | ||||
|  | ||||
| urlpatterns = [ | ||||
|     # API | ||||
|     url(r"^", include(router.urls)), | ||||
|     url(r"^login/", include("rest_framework.urls", namespace="rest_framework")), | ||||
|     url(r"^markdown$", RenderMarkdown, name="api_markdown"), | ||||
|     url(r"^mailings$", FetchMailingLists, name="mailings_fetch"), | ||||
| ] | ||||
| @@ -1,79 +0,0 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # | ||||
| # | ||||
|  | ||||
| from rest_framework.response import Response | ||||
| from rest_framework import viewsets | ||||
| from django.core.exceptions import PermissionDenied | ||||
| from rest_framework.decorators import detail_route | ||||
| from django.db.models.query import QuerySet | ||||
|  | ||||
| from core.views import can_view, can_edit | ||||
|  | ||||
|  | ||||
| def check_if(obj, user, test): | ||||
|     """ | ||||
|         Detect if it's a single object or a queryset | ||||
|         aply a given test on individual object and return global permission | ||||
|     """ | ||||
|     if isinstance(obj, QuerySet): | ||||
|         for o in obj: | ||||
|             if test(o, user) is False: | ||||
|                 return False | ||||
|         return True | ||||
|     else: | ||||
|         return test(obj, user) | ||||
|  | ||||
|  | ||||
| class ManageModelMixin: | ||||
|     @detail_route() | ||||
|     def id(self, request, pk=None): | ||||
|         """ | ||||
|             Get by id (api/v1/router/{pk}/id/) | ||||
|         """ | ||||
|         self.queryset = get_object_or_404(self.queryset.filter(id=pk)) | ||||
|         serializer = self.get_serializer(self.queryset) | ||||
|         return Response(serializer.data) | ||||
|  | ||||
|  | ||||
| class RightModelViewSet(ManageModelMixin, viewsets.ModelViewSet): | ||||
|     def dispatch(self, request, *arg, **kwargs): | ||||
|         res = super(RightModelViewSet, self).dispatch(request, *arg, **kwargs) | ||||
|         obj = self.queryset | ||||
|         user = self.request.user | ||||
|         try: | ||||
|             if request.method == "GET" and check_if(obj, user, can_view): | ||||
|                 return res | ||||
|             if request.method != "GET" and check_if(obj, user, can_edit): | ||||
|                 return res | ||||
|         except: | ||||
|             pass  # To prevent bug with Anonymous user | ||||
|         raise PermissionDenied | ||||
|  | ||||
|  | ||||
| from .api import * | ||||
| from .counter import * | ||||
| from .user import * | ||||
| from .club import * | ||||
| from .group import * | ||||
| from .launderette import * | ||||
| @@ -1,43 +0,0 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # | ||||
| # | ||||
|  | ||||
| from rest_framework.response import Response | ||||
| from rest_framework.decorators import api_view, renderer_classes | ||||
| from rest_framework.renderers import StaticHTMLRenderer | ||||
| from rest_framework.views import APIView | ||||
|  | ||||
| from core.templatetags.renderer import markdown | ||||
|  | ||||
|  | ||||
| @api_view(["POST"]) | ||||
| @renderer_classes((StaticHTMLRenderer,)) | ||||
| def RenderMarkdown(request): | ||||
|     """ | ||||
|         Render Markdown | ||||
|     """ | ||||
|     try: | ||||
|         data = markdown(request.POST["text"]) | ||||
|     except: | ||||
|         data = "Error" | ||||
|     return Response(data) | ||||
| @@ -1,64 +0,0 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # | ||||
| # | ||||
|  | ||||
| from rest_framework.response import Response | ||||
| from rest_framework import serializers | ||||
| from rest_framework.decorators import api_view, renderer_classes | ||||
| from rest_framework.renderers import StaticHTMLRenderer | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.core.exceptions import PermissionDenied | ||||
|  | ||||
| from club.models import Club, Mailing | ||||
|  | ||||
| from api.views import RightModelViewSet | ||||
|  | ||||
|  | ||||
| class ClubSerializer(serializers.ModelSerializer): | ||||
|     class Meta: | ||||
|         model = Club | ||||
|         fields = ("id", "name", "unix_name", "address", "members") | ||||
|  | ||||
|  | ||||
| class ClubViewSet(RightModelViewSet): | ||||
|     """ | ||||
|         Manage Clubs (api/v1/club/) | ||||
|     """ | ||||
|  | ||||
|     serializer_class = ClubSerializer | ||||
|     queryset = Club.objects.all() | ||||
|  | ||||
|  | ||||
| @api_view(["GET"]) | ||||
| @renderer_classes((StaticHTMLRenderer,)) | ||||
| def FetchMailingLists(request): | ||||
|     key = request.GET.get("key", "") | ||||
|     if key != settings.SITH_MAILING_FETCH_KEY: | ||||
|         raise PermissionDenied | ||||
|     data = "" | ||||
|     for mailing in Mailing.objects.filter( | ||||
|         is_moderated=True, club__is_active=True | ||||
|     ).all(): | ||||
|         data += mailing.fetch_format() + "\n" | ||||
|     return Response(data) | ||||
| @@ -1,61 +0,0 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # | ||||
| # | ||||
|  | ||||
| from rest_framework import serializers | ||||
| from rest_framework.response import Response | ||||
| from rest_framework.decorators import list_route | ||||
|  | ||||
| from counter.models import Counter | ||||
|  | ||||
| from api.views import RightModelViewSet | ||||
|  | ||||
|  | ||||
| class CounterSerializer(serializers.ModelSerializer): | ||||
|  | ||||
|     is_open = serializers.BooleanField(read_only=True) | ||||
|     barman_list = serializers.ListField( | ||||
|         child=serializers.IntegerField(), read_only=True | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         model = Counter | ||||
|         fields = ("id", "name", "type", "club", "products", "is_open", "barman_list") | ||||
|  | ||||
|  | ||||
| class CounterViewSet(RightModelViewSet): | ||||
|     """ | ||||
|         Manage Counters (api/v1/counter/) | ||||
|     """ | ||||
|  | ||||
|     serializer_class = CounterSerializer | ||||
|     queryset = Counter.objects.all() | ||||
|  | ||||
|     @list_route() | ||||
|     def bar(self, request): | ||||
|         """ | ||||
|             Return all bars (api/v1/counter/bar/) | ||||
|         """ | ||||
|         self.queryset = self.queryset.filter(type="BAR") | ||||
|         serializer = self.get_serializer(self.queryset, many=True) | ||||
|         return Response(serializer.data) | ||||
| @@ -1,137 +0,0 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # | ||||
| # | ||||
|  | ||||
| from rest_framework import serializers | ||||
| from rest_framework.response import Response | ||||
| from rest_framework.decorators import list_route | ||||
|  | ||||
| from launderette.models import Launderette, Machine, Token | ||||
|  | ||||
| from api.views import RightModelViewSet | ||||
|  | ||||
|  | ||||
| class LaunderettePlaceSerializer(serializers.ModelSerializer): | ||||
|  | ||||
|     machine_list = serializers.ListField( | ||||
|         child=serializers.IntegerField(), read_only=True | ||||
|     ) | ||||
|     token_list = serializers.ListField(child=serializers.IntegerField(), read_only=True) | ||||
|  | ||||
|     class Meta: | ||||
|         model = Launderette | ||||
|         fields = ( | ||||
|             "id", | ||||
|             "name", | ||||
|             "counter", | ||||
|             "machine_list", | ||||
|             "token_list", | ||||
|             "get_absolute_url", | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class LaunderetteMachineSerializer(serializers.ModelSerializer): | ||||
|     class Meta: | ||||
|         model = Machine | ||||
|         fields = ("id", "name", "type", "is_working", "launderette") | ||||
|  | ||||
|  | ||||
| class LaunderetteTokenSerializer(serializers.ModelSerializer): | ||||
|     class Meta: | ||||
|         model = Token | ||||
|         fields = ( | ||||
|             "id", | ||||
|             "name", | ||||
|             "type", | ||||
|             "launderette", | ||||
|             "borrow_date", | ||||
|             "user", | ||||
|             "is_avaliable", | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class LaunderettePlaceViewSet(RightModelViewSet): | ||||
|     """ | ||||
|         Manage Launderette (api/v1/launderette/place/) | ||||
|     """ | ||||
|  | ||||
|     serializer_class = LaunderettePlaceSerializer | ||||
|     queryset = Launderette.objects.all() | ||||
|  | ||||
|  | ||||
| class LaunderetteMachineViewSet(RightModelViewSet): | ||||
|     """ | ||||
|         Manage Washing Machines (api/v1/launderette/machine/) | ||||
|     """ | ||||
|  | ||||
|     serializer_class = LaunderetteMachineSerializer | ||||
|     queryset = Machine.objects.all() | ||||
|  | ||||
|  | ||||
| class LaunderetteTokenViewSet(RightModelViewSet): | ||||
|     """ | ||||
|         Manage Launderette's tokens (api/v1/launderette/token/) | ||||
|     """ | ||||
|  | ||||
|     serializer_class = LaunderetteTokenSerializer | ||||
|     queryset = Token.objects.all() | ||||
|  | ||||
|     @list_route() | ||||
|     def washing(self, request): | ||||
|         """ | ||||
|             Return all washing tokens (api/v1/launderette/token/washing) | ||||
|         """ | ||||
|         self.queryset = self.queryset.filter(type="WASHING") | ||||
|         serializer = self.get_serializer(self.queryset, many=True) | ||||
|         return Response(serializer.data) | ||||
|  | ||||
|     @list_route() | ||||
|     def drying(self, request): | ||||
|         """ | ||||
|             Return all drying tokens (api/v1/launderette/token/drying) | ||||
|         """ | ||||
|         self.queryset = self.queryset.filter(type="DRYING") | ||||
|         serializer = self.get_serializer(self.queryset, many=True) | ||||
|         return Response(serializer.data) | ||||
|  | ||||
|     @list_route() | ||||
|     def avaliable(self, request): | ||||
|         """ | ||||
|             Return all avaliable tokens (api/v1/launderette/token/avaliable) | ||||
|         """ | ||||
|         self.queryset = self.queryset.filter( | ||||
|             borrow_date__isnull=True, user__isnull=True | ||||
|         ) | ||||
|         serializer = self.get_serializer(self.queryset, many=True) | ||||
|         return Response(serializer.data) | ||||
|  | ||||
|     @list_route() | ||||
|     def unavaliable(self, request): | ||||
|         """ | ||||
|             Return all unavaliable tokens (api/v1/launderette/token/unavaliable) | ||||
|         """ | ||||
|         self.queryset = self.queryset.filter( | ||||
|             borrow_date__isnull=False, user__isnull=False | ||||
|         ) | ||||
|         serializer = self.get_serializer(self.queryset, many=True) | ||||
|         return Response(serializer.data) | ||||
| @@ -1,68 +0,0 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # | ||||
| # | ||||
|  | ||||
| import datetime | ||||
|  | ||||
| from rest_framework import serializers | ||||
| from rest_framework.response import Response | ||||
| from rest_framework.decorators import list_route | ||||
|  | ||||
| from core.models import User | ||||
|  | ||||
| from api.views import RightModelViewSet | ||||
|  | ||||
|  | ||||
| class UserSerializer(serializers.ModelSerializer): | ||||
|     class Meta: | ||||
|         model = User | ||||
|         fields = ( | ||||
|             "id", | ||||
|             "first_name", | ||||
|             "last_name", | ||||
|             "email", | ||||
|             "date_of_birth", | ||||
|             "nick_name", | ||||
|             "is_active", | ||||
|             "date_joined", | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class UserViewSet(RightModelViewSet): | ||||
|     """ | ||||
|         Manage Users (api/v1/user/) | ||||
|         Only show active users | ||||
|     """ | ||||
|  | ||||
|     serializer_class = UserSerializer | ||||
|     queryset = User.objects.filter(is_active=True) | ||||
|  | ||||
|     @list_route() | ||||
|     def birthday(self, request): | ||||
|         """ | ||||
|             Return all users born today (api/v1/user/birstdays) | ||||
|         """ | ||||
|         date = datetime.datetime.today() | ||||
|         self.queryset = self.queryset.filter(date_of_birth=date) | ||||
|         serializer = self.get_serializer(self.queryset, many=True) | ||||
|         return Response(serializer.data) | ||||
							
								
								
									
										29
									
								
								biome.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								biome.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| { | ||||
|   "$schema": "./node_modules/@biomejs/biome/configuration_schema.json", | ||||
|   "vcs": { | ||||
|     "enabled": true, | ||||
|     "clientKind": "git", | ||||
|     "useIgnoreFile": true | ||||
|   }, | ||||
|   "files": { | ||||
|     "ignoreUnknown": false, | ||||
|     "ignore": ["*.min.*", "staticfiles/generated"] | ||||
|   }, | ||||
|   "formatter": { | ||||
|     "enabled": true, | ||||
|     "indentStyle": "space", | ||||
|     "lineWidth": 88 | ||||
|   }, | ||||
|   "organizeImports": { | ||||
|     "enabled": true | ||||
|   }, | ||||
|   "linter": { | ||||
|     "enabled": true, | ||||
|     "rules": { | ||||
|       "all": true | ||||
|     } | ||||
|   }, | ||||
|   "javascript": { | ||||
|     "globals": ["Alpine", "$", "jQuery", "gettext", "interpolate"] | ||||
|   } | ||||
| } | ||||
| @@ -1,23 +1,14 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # Copyright 2023 © AE UTBM | ||||
| # ae@utbm.fr / ae.info@utbm.fr | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # This file is part of the website of the UTBM Student Association (AE UTBM), | ||||
| # https://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # You can find the source code of the website at https://github.com/ae-utbm/sith | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3) | ||||
| # SEE : https://raw.githubusercontent.com/ae-utbm/sith/master/LICENSE | ||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | ||||
| # | ||||
| # | ||||
|   | ||||
| @@ -1,31 +1,42 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| # Copyright 2023 © AE UTBM | ||||
| # ae@utbm.fr / ae.info@utbm.fr | ||||
| # | ||||
| # Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM, | ||||
| # http://ae.utbm.fr. | ||||
| # This file is part of the website of the UTBM Student Association (AE UTBM), | ||||
| # https://ae.utbm.fr. | ||||
| # | ||||
| # This program is free software; you can redistribute it and/or modify it under | ||||
| # the terms of the GNU General Public License a published by the Free Software | ||||
| # Foundation; either version 3 of the License, or (at your option) any later | ||||
| # version. | ||||
| # You can find the source code of the website at https://github.com/ae-utbm/sith | ||||
| # | ||||
| # This program is distributed in the hope that it will be useful, but WITHOUT | ||||
| # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
| # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | ||||
| # details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License along with | ||||
| # this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3) | ||||
| # SEE : https://raw.githubusercontent.com/ae-utbm/sith/master/LICENSE | ||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | ||||
| # | ||||
| # | ||||
|  | ||||
| from django.contrib import admin | ||||
|  | ||||
| from club.models import Club, Membership | ||||
|  | ||||
|  | ||||
| admin.site.register(Club) | ||||
| admin.site.register(Membership) | ||||
| @admin.register(Club) | ||||
| class ClubAdmin(admin.ModelAdmin): | ||||
|     list_display = ("name", "unix_name", "parent", "is_active") | ||||
|     search_fields = ("name", "unix_name") | ||||
|     autocomplete_fields = ( | ||||
|         "parent", | ||||
|         "board_group", | ||||
|         "members_group", | ||||
|         "home", | ||||
|         "page", | ||||
|     ) | ||||
|  | ||||
|  | ||||
| @admin.register(Membership) | ||||
| class MembershipAdmin(admin.ModelAdmin): | ||||
|     list_display = ("user", "club", "role", "start_date", "end_date") | ||||
|     search_fields = ( | ||||
|         "user__username", | ||||
|         "user__first_name", | ||||
|         "user__last_name", | ||||
|         "club__name", | ||||
|     ) | ||||
|     autocomplete_fields = ("user",) | ||||
|   | ||||
							
								
								
									
										22
									
								
								club/api.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								club/api.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| from typing import Annotated | ||||
|  | ||||
| from annotated_types import MinLen | ||||
| from ninja_extra import ControllerBase, api_controller, paginate, route | ||||
| from ninja_extra.pagination import PageNumberPaginationExtra | ||||
| from ninja_extra.schemas import PaginatedResponseSchema | ||||
|  | ||||
| from club.models import Club | ||||
| from club.schemas import ClubSchema | ||||
| from core.api_permissions import CanAccessLookup | ||||
|  | ||||
|  | ||||
| @api_controller("/club") | ||||
| class ClubController(ControllerBase): | ||||
|     @route.get( | ||||
|         "/search", | ||||
|         response=PaginatedResponseSchema[ClubSchema], | ||||
|         permissions=[CanAccessLookup], | ||||
|     ) | ||||
|     @paginate(PageNumberPaginationExtra, page_size=50) | ||||
|     def search_club(self, search: Annotated[str, MinLen(1)]): | ||||
|         return Club.objects.filter(name__icontains=search).values() | ||||
							
								
								
									
										109
									
								
								club/forms.py
									
									
									
									
									
								
							
							
						
						
									
										109
									
								
								club/forms.py
									
									
									
									
									
								
							| @@ -1,4 +1,3 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| @@ -23,16 +22,14 @@ | ||||
| # | ||||
| # | ||||
|  | ||||
| from django.conf import settings | ||||
| from django import forms | ||||
| from django.utils.translation import ugettext_lazy as _ | ||||
|  | ||||
| from ajax_select.fields import AutoCompleteSelectField, AutoCompleteSelectMultipleField | ||||
|  | ||||
| from club.models import Mailing, MailingSubscription, Club, Membership | ||||
| from django.conf import settings | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
| from club.models import Club, Mailing, MailingSubscription, Membership | ||||
| from core.models import User | ||||
| from core.views.forms import SelectDate, SelectDateTime | ||||
| from core.views.widgets.select import AutoCompleteSelectMultipleUser | ||||
| from counter.models import Counter | ||||
|  | ||||
|  | ||||
| @@ -42,31 +39,30 @@ class ClubEditForm(forms.ModelForm): | ||||
|         fields = ["address", "logo", "short_description"] | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         super(ClubEditForm, self).__init__(*args, **kwargs) | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.fields["short_description"].widget = forms.Textarea() | ||||
|  | ||||
|  | ||||
| class MailingForm(forms.Form): | ||||
|     """ | ||||
|     Form handling mailing lists right | ||||
|     """ | ||||
|     """Form handling mailing lists right.""" | ||||
|  | ||||
|     ACTION_NEW_MAILING = 1 | ||||
|     ACTION_NEW_SUBSCRIPTION = 2 | ||||
|     ACTION_REMOVE_SUBSCRIPTION = 3 | ||||
|  | ||||
|     subscription_users = AutoCompleteSelectMultipleField( | ||||
|         "users", | ||||
|     subscription_users = forms.ModelMultipleChoiceField( | ||||
|         label=_("Users to add"), | ||||
|         help_text=_("Search users to add (one or more)."), | ||||
|         required=False, | ||||
|         widget=AutoCompleteSelectMultipleUser, | ||||
|         queryset=User.objects.all(), | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, club_id, user_id, mailings, *args, **kwargs): | ||||
|         super(MailingForm, self).__init__(*args, **kwargs) | ||||
|         super().__init__(*args, **kwargs) | ||||
|  | ||||
|         self.fields["action"] = forms.TypedChoiceField( | ||||
|             ( | ||||
|             choices=( | ||||
|                 (self.ACTION_NEW_MAILING, _("New Mailing")), | ||||
|                 (self.ACTION_NEW_SUBSCRIPTION, _("Subscribe")), | ||||
|                 (self.ACTION_REMOVE_SUBSCRIPTION, _("Remove")), | ||||
| @@ -108,24 +104,15 @@ class MailingForm(forms.Form): | ||||
|         ) | ||||
|  | ||||
|     def check_required(self, cleaned_data, field): | ||||
|         """ | ||||
|         If the given field doesn't exist or has no value, add a required error on it | ||||
|         """ | ||||
|         """If the given field doesn't exist or has no value, add a required error on it.""" | ||||
|         if not cleaned_data.get(field, None): | ||||
|             self.add_error(field, _("This field is required")) | ||||
|  | ||||
|     def clean_subscription_users(self): | ||||
|         """ | ||||
|         Convert given users into real users and check their validity | ||||
|         """ | ||||
|         cleaned_data = super(MailingForm, self).clean() | ||||
|         """Convert given users into real users and check their validity.""" | ||||
|         cleaned_data = super().clean() | ||||
|         users = [] | ||||
|         for user in cleaned_data["subscription_users"]: | ||||
|             user = User.objects.filter(id=user).first() | ||||
|             if not user: | ||||
|                 raise forms.ValidationError( | ||||
|                     _("One of the selected users doesn't exist"), code="invalid" | ||||
|                 ) | ||||
|             if not user.email: | ||||
|                 raise forms.ValidationError( | ||||
|                     _("One of the selected users doesn't have an email address"), | ||||
| @@ -135,9 +122,9 @@ class MailingForm(forms.Form): | ||||
|         return users | ||||
|  | ||||
|     def clean(self): | ||||
|         cleaned_data = super(MailingForm, self).clean() | ||||
|         cleaned_data = super().clean() | ||||
|  | ||||
|         if not "action" in cleaned_data: | ||||
|         if "action" not in cleaned_data: | ||||
|             # If there is no action provided, we can stop here | ||||
|             raise forms.ValidationError(_("An action is required"), code="invalid") | ||||
|  | ||||
| @@ -157,37 +144,44 @@ class MailingForm(forms.Form): | ||||
|         return cleaned_data | ||||
|  | ||||
|  | ||||
| class SellingsFormBase(forms.Form): | ||||
| class SellingsForm(forms.Form): | ||||
|     begin_date = forms.DateTimeField( | ||||
|         ["%Y-%m-%d %H:%M:%S"], | ||||
|         label=_("Begin date"), | ||||
|         required=False, | ||||
|         widget=SelectDateTime, | ||||
|         label=_("Begin date"), widget=SelectDateTime, required=False | ||||
|     ) | ||||
|     end_date = forms.DateTimeField( | ||||
|         ["%Y-%m-%d %H:%M:%S"], | ||||
|         label=_("End date"), | ||||
|         required=False, | ||||
|         widget=SelectDateTime, | ||||
|         label=_("End date"), widget=SelectDateTime, required=False | ||||
|     ) | ||||
|     counter = forms.ModelChoiceField( | ||||
|  | ||||
|     counters = forms.ModelMultipleChoiceField( | ||||
|         Counter.objects.order_by("name").all(), label=_("Counter"), required=False | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, club, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.fields["products"] = forms.ModelMultipleChoiceField( | ||||
|             club.products.order_by("name").filter(archived=False).all(), | ||||
|             label=_("Products"), | ||||
|             required=False, | ||||
|         ) | ||||
|         self.fields["archived_products"] = forms.ModelMultipleChoiceField( | ||||
|             club.products.order_by("name").filter(archived=True).all(), | ||||
|             label=_("Archived products"), | ||||
|             required=False, | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class ClubMemberForm(forms.Form): | ||||
|     """ | ||||
|     Form handling the members of a club | ||||
|     """ | ||||
|     """Form handling the members of a club.""" | ||||
|  | ||||
|     error_css_class = "error" | ||||
|     required_css_class = "required" | ||||
|  | ||||
|     users = AutoCompleteSelectMultipleField( | ||||
|         "users", | ||||
|     users = forms.ModelMultipleChoiceField( | ||||
|         label=_("Users to add"), | ||||
|         help_text=_("Search users to add (one or more)."), | ||||
|         required=False, | ||||
|         widget=AutoCompleteSelectMultipleUser, | ||||
|         queryset=User.objects.all(), | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
| @@ -199,7 +193,7 @@ class ClubMemberForm(forms.Form): | ||||
|                 self.club.members.filter(end_date=None).order_by("-role").all() | ||||
|             ) | ||||
|         self.request_user_membership = self.club.get_membership_for(self.request_user) | ||||
|         super(ClubMemberForm, self).__init__(*args, **kwargs) | ||||
|         super().__init__(*args, **kwargs) | ||||
|  | ||||
|         # Using a ModelForm binds too much the form with the model and we don't want that | ||||
|         # We want the view to process the model creation since they are multiple users | ||||
| @@ -224,9 +218,7 @@ class ClubMemberForm(forms.Form): | ||||
|                 id__in=[ | ||||
|                     ms.user.id | ||||
|                     for ms in self.club_members | ||||
|                     if ms.can_be_edited_by( | ||||
|                         self.request_user, self.request_user_membership | ||||
|                     ) | ||||
|                     if ms.can_be_edited_by(self.request_user) | ||||
|                 ] | ||||
|             ).all(), | ||||
|             label=_("Mark as old"), | ||||
| @@ -237,18 +229,13 @@ class ClubMemberForm(forms.Form): | ||||
|             self.fields.pop("start_date") | ||||
|  | ||||
|     def clean_users(self): | ||||
|         """Check that the user is not trying to add an user already in the club. | ||||
|  | ||||
|         Also check that the user is valid and has a valid subscription. | ||||
|         """ | ||||
|             Check that the user is not trying to add an user already in the club | ||||
|             Also check that the user is valid and has a valid subscription | ||||
|         """ | ||||
|         cleaned_data = super(ClubMemberForm, self).clean() | ||||
|         cleaned_data = super().clean() | ||||
|         users = [] | ||||
|         for user_id in cleaned_data["users"]: | ||||
|             user = User.objects.filter(id=user_id).first() | ||||
|             if not user: | ||||
|                 raise forms.ValidationError( | ||||
|                     _("One of the selected users doesn't exist"), code="invalid" | ||||
|                 ) | ||||
|         for user in cleaned_data["users"]: | ||||
|             if not user.is_subscribed: | ||||
|                 raise forms.ValidationError( | ||||
|                     _("User must be subscriber to take part to a club"), code="invalid" | ||||
| @@ -261,10 +248,8 @@ class ClubMemberForm(forms.Form): | ||||
|         return users | ||||
|  | ||||
|     def clean(self): | ||||
|         """ | ||||
|             Check user rights for adding an user | ||||
|         """ | ||||
|         cleaned_data = super(ClubMemberForm, self).clean() | ||||
|         """Check user rights for adding an user.""" | ||||
|         cleaned_data = super().clean() | ||||
|  | ||||
|         if "start_date" in cleaned_data and not cleaned_data["start_date"]: | ||||
|             # Drop start_date if allowed to edition but not specified | ||||
|   | ||||
| @@ -1,12 +1,11 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
| import django.core.validators | ||||
| import django.db.models.deletion | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [] | ||||
|  | ||||
|     operations = [ | ||||
| @@ -90,7 +89,10 @@ class Migration(migrations.Migration): | ||||
|                 ( | ||||
|                     "club", | ||||
|                     models.ForeignKey( | ||||
|                         verbose_name="club", to="club.Club", related_name="members" | ||||
|                         verbose_name="club", | ||||
|                         to="club.Club", | ||||
|                         related_name="members", | ||||
|                         on_delete=django.db.models.deletion.CASCADE, | ||||
|                     ), | ||||
|                 ), | ||||
|             ], | ||||
|   | ||||
| @@ -1,12 +1,11 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
| import django.db.models.deletion | ||||
| from django.conf import settings | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [ | ||||
|         migrations.swappable_dependency(settings.AUTH_USER_MODEL), | ||||
|         ("club", "0001_initial"), | ||||
| @@ -18,6 +17,7 @@ class Migration(migrations.Migration): | ||||
|             model_name="membership", | ||||
|             name="user", | ||||
|             field=models.ForeignKey( | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|                 verbose_name="user", | ||||
|                 to=settings.AUTH_USER_MODEL, | ||||
|                 related_name="membership", | ||||
| @@ -34,6 +34,7 @@ class Migration(migrations.Migration): | ||||
|             model_name="club", | ||||
|             name="home", | ||||
|             field=models.OneToOneField( | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|                 blank=True, | ||||
|                 null=True, | ||||
|                 related_name="home_of_club", | ||||
| @@ -45,14 +46,21 @@ class Migration(migrations.Migration): | ||||
|             model_name="club", | ||||
|             name="owner_group", | ||||
|             field=models.ForeignKey( | ||||
|                 default=1, to="core.Group", related_name="owned_club" | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|                 default=1, | ||||
|                 to="core.Group", | ||||
|                 related_name="owned_club", | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name="club", | ||||
|             name="parent", | ||||
|             field=models.ForeignKey( | ||||
|                 null=True, to="club.Club", related_name="children", blank=True | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|                 null=True, | ||||
|                 to="club.Club", | ||||
|                 related_name="children", | ||||
|                 blank=True, | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|   | ||||
| @@ -1,11 +1,9 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("club", "0002_auto_20160824_2152")] | ||||
|  | ||||
|     operations = [ | ||||
|   | ||||
| @@ -1,12 +1,11 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
| import django.db.models.deletion | ||||
| from django.conf import settings | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("club", "0003_auto_20160902_2042")] | ||||
|  | ||||
|     operations = [ | ||||
| @@ -14,6 +13,7 @@ class Migration(migrations.Migration): | ||||
|             model_name="membership", | ||||
|             name="user", | ||||
|             field=models.ForeignKey( | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|                 verbose_name="user", | ||||
|                 related_name="memberships", | ||||
|                 to=settings.AUTH_USER_MODEL, | ||||
|   | ||||
| @@ -1,12 +1,10 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
| import django.db.models.deletion | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("club", "0004_auto_20160915_1057")] | ||||
|  | ||||
|     operations = [ | ||||
|   | ||||
| @@ -1,12 +1,10 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
| import django.utils.timezone | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("club", "0005_auto_20161120_1149")] | ||||
|  | ||||
|     operations = [ | ||||
|   | ||||
| @@ -1,11 +1,9 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
| from django.db import migrations | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("club", "0006_auto_20161229_0040")] | ||||
|  | ||||
|     operations = [ | ||||
|   | ||||
| @@ -1,11 +1,9 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("club", "0007_auto_20170324_0917")] | ||||
|  | ||||
|     operations = [ | ||||
|   | ||||
| @@ -1,14 +1,14 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
| from django.conf import settings | ||||
| import re | ||||
|  | ||||
| import django.core.validators | ||||
| import django.db.models.deletion | ||||
| from django.conf import settings | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [ | ||||
|         migrations.swappable_dependency(settings.AUTH_USER_MODEL), | ||||
|         ("club", "0008_auto_20170515_2214"), | ||||
| @@ -51,12 +51,16 @@ class Migration(migrations.Migration): | ||||
|                 ( | ||||
|                     "club", | ||||
|                     models.ForeignKey( | ||||
|                         verbose_name="Club", related_name="mailings", to="club.Club" | ||||
|                         on_delete=django.db.models.deletion.CASCADE, | ||||
|                         verbose_name="Club", | ||||
|                         related_name="mailings", | ||||
|                         to="club.Club", | ||||
|                     ), | ||||
|                 ), | ||||
|                 ( | ||||
|                     "moderator", | ||||
|                     models.ForeignKey( | ||||
|                         on_delete=django.db.models.deletion.CASCADE, | ||||
|                         null=True, | ||||
|                         verbose_name="moderator", | ||||
|                         related_name="moderated_mailings", | ||||
| @@ -84,6 +88,7 @@ class Migration(migrations.Migration): | ||||
|                 ( | ||||
|                     "mailing", | ||||
|                     models.ForeignKey( | ||||
|                         on_delete=django.db.models.deletion.CASCADE, | ||||
|                         verbose_name="Mailing", | ||||
|                         related_name="subscriptions", | ||||
|                         to="club.Mailing", | ||||
| @@ -92,6 +97,7 @@ class Migration(migrations.Migration): | ||||
|                 ( | ||||
|                     "user", | ||||
|                     models.ForeignKey( | ||||
|                         on_delete=django.db.models.deletion.CASCADE, | ||||
|                         null=True, | ||||
|                         verbose_name="User", | ||||
|                         related_name="mailing_subscriptions", | ||||
| @@ -103,6 +109,6 @@ class Migration(migrations.Migration): | ||||
|         ), | ||||
|         migrations.AlterUniqueTogether( | ||||
|             name="mailingsubscription", | ||||
|             unique_together=set([("user", "email", "mailing")]), | ||||
|             unique_together={("user", "email", "mailing")}, | ||||
|         ), | ||||
|     ] | ||||
|   | ||||
| @@ -1,24 +1,10 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import django.db.models.deletion | ||||
| from django.db import migrations, models | ||||
|  | ||||
| from club.models import Club | ||||
| from core.operations import PsqlRunOnly | ||||
|  | ||||
|  | ||||
| def generate_club_pages(apps, schema_editor): | ||||
|     def recursive_generate_club_page(club): | ||||
|         club.make_page() | ||||
|         for child in Club.objects.filter(parent=club).all(): | ||||
|             recursive_generate_club_page(child) | ||||
|  | ||||
|     for club in Club.objects.filter(parent=None).all(): | ||||
|         recursive_generate_club_page(club) | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("core", "0024_auto_20170906_1317"), ("club", "0010_club_logo")] | ||||
|  | ||||
|     operations = [ | ||||
| @@ -31,7 +17,11 @@ class Migration(migrations.Migration): | ||||
|             model_name="club", | ||||
|             name="page", | ||||
|             field=models.OneToOneField( | ||||
|                 related_name="club", blank=True, null=True, to="core.Page" | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|                 related_name="club", | ||||
|                 blank=True, | ||||
|                 null=True, | ||||
|                 to="core.Page", | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
| @@ -45,11 +35,4 @@ class Migration(migrations.Migration): | ||||
|                 null=True, | ||||
|             ), | ||||
|         ), | ||||
|         PsqlRunOnly( | ||||
|             "SET CONSTRAINTS ALL IMMEDIATE", reverse_sql=migrations.RunSQL.noop | ||||
|         ), | ||||
|         migrations.RunPython(generate_club_pages), | ||||
|         PsqlRunOnly( | ||||
|             migrations.RunSQL.noop, reverse_sql="SET CONSTRAINTS ALL IMMEDIATE" | ||||
|         ), | ||||
|     ] | ||||
|   | ||||
| @@ -1,11 +1,9 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("club", "0009_auto_20170822_2232")] | ||||
|  | ||||
|     operations = [ | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import django.db.models.deletion | ||||
| from django.db import migrations, models | ||||
|  | ||||
| import club.models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [("club", "0010_auto_20170912_2028")] | ||||
|  | ||||
|     operations = [ | ||||
| @@ -14,7 +14,8 @@ class Migration(migrations.Migration): | ||||
|             model_name="club", | ||||
|             name="owner_group", | ||||
|             field=models.ForeignKey( | ||||
|                 default=club.models.Club.get_default_owner_group, | ||||
|                 on_delete=django.db.models.deletion.CASCADE, | ||||
|                 default=club.models.get_default_owner_group, | ||||
|                 related_name="owned_club", | ||||
|                 to="core.Group", | ||||
|             ), | ||||
|   | ||||
							
								
								
									
										106
									
								
								club/migrations/0012_club_board_group_club_members_group.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										106
									
								
								club/migrations/0012_club_board_group_club_members_group.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,106 @@ | ||||
| # Generated by Django 4.2.16 on 2024-11-20 17:08 | ||||
|  | ||||
| import django.db.models.deletion | ||||
| import django.db.models.functions.datetime | ||||
| from django.conf import settings | ||||
| from django.db import migrations, models | ||||
| from django.db.migrations.state import StateApps | ||||
| from django.db.models import Q | ||||
| from django.utils.timezone import localdate | ||||
|  | ||||
|  | ||||
| def migrate_meta_groups(apps: StateApps, schema_editor): | ||||
|     """Attach the existing meta groups to the clubs. | ||||
|  | ||||
|     Until now, the meta groups were not attached to the clubs, | ||||
|     nor to the users. | ||||
|     This creates actual foreign relationships between the clubs | ||||
|     and theirs groups and the users and theirs groups. | ||||
|  | ||||
|     Warnings: | ||||
|         When the meta groups associated with the clubs aren't found, | ||||
|         they are created. | ||||
|         Thus the migration shouldn't fail, and all the clubs will | ||||
|         have their groups. | ||||
|         However, there will probably be some groups that have | ||||
|         not been found but exist nonetheless, | ||||
|         so there will be duplicates and dangling groups. | ||||
|         There must be a manual cleanup after this migration. | ||||
|     """ | ||||
|     Group = apps.get_model("core", "Group") | ||||
|     Club = apps.get_model("club", "Club") | ||||
|  | ||||
|     meta_groups = Group.objects.filter(is_meta=True) | ||||
|     clubs = list(Club.objects.all()) | ||||
|     for club in clubs: | ||||
|         club.board_group = meta_groups.get_or_create( | ||||
|             name=club.unix_name + settings.SITH_BOARD_SUFFIX, | ||||
|             defaults={"is_meta": True}, | ||||
|         )[0] | ||||
|         club.members_group = meta_groups.get_or_create( | ||||
|             name=club.unix_name + settings.SITH_MEMBER_SUFFIX, | ||||
|             defaults={"is_meta": True}, | ||||
|         )[0] | ||||
|         club.save() | ||||
|         club.refresh_from_db() | ||||
|         memberships = club.members.filter( | ||||
|             Q(end_date=None) | Q(end_date__gt=localdate()) | ||||
|         ).select_related("user") | ||||
|         club.members_group.users.set([m.user for m in memberships]) | ||||
|         club.board_group.users.set( | ||||
|             [ | ||||
|                 m.user | ||||
|                 for m in memberships.filter(role__gt=settings.SITH_MAXIMUM_FREE_ROLE) | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|  | ||||
| # steps of the migration : | ||||
| # - Create a nullable field for the board group and the member group | ||||
| # - Edit those new fields to make them point to currently existing meta groups | ||||
| # - When this data migration is done, make the fields non-nullable | ||||
| class Migration(migrations.Migration): | ||||
|     dependencies = [ | ||||
|         ("core", "0040_alter_user_options_user_user_permissions_and_more"), | ||||
|         ("club", "0011_auto_20180426_2013"), | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.RemoveField( | ||||
|             model_name="club", | ||||
|             name="edit_groups", | ||||
|         ), | ||||
|         migrations.RemoveField( | ||||
|             model_name="club", | ||||
|             name="owner_group", | ||||
|         ), | ||||
|         migrations.RemoveField( | ||||
|             model_name="club", | ||||
|             name="view_groups", | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name="club", | ||||
|             name="board_group", | ||||
|             field=models.OneToOneField( | ||||
|                 blank=True, | ||||
|                 null=True, | ||||
|                 on_delete=django.db.models.deletion.PROTECT, | ||||
|                 related_name="club_board", | ||||
|                 to="core.group", | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name="club", | ||||
|             name="members_group", | ||||
|             field=models.OneToOneField( | ||||
|                 blank=True, | ||||
|                 null=True, | ||||
|                 on_delete=django.db.models.deletion.PROTECT, | ||||
|                 related_name="club", | ||||
|                 to="core.group", | ||||
|             ), | ||||
|         ), | ||||
|         migrations.RunPython( | ||||
|             migrate_meta_groups, reverse_code=migrations.RunPython.noop, elidable=True | ||||
|         ), | ||||
|     ] | ||||
| @@ -0,0 +1,36 @@ | ||||
| # Generated by Django 4.2.17 on 2025-01-04 16:46 | ||||
|  | ||||
| import django.db.models.deletion | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|     dependencies = [("club", "0012_club_board_group_club_members_group")] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.AlterField( | ||||
|             model_name="club", | ||||
|             name="board_group", | ||||
|             field=models.OneToOneField( | ||||
|                 on_delete=django.db.models.deletion.PROTECT, | ||||
|                 related_name="club_board", | ||||
|                 to="core.group", | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AlterField( | ||||
|             model_name="club", | ||||
|             name="members_group", | ||||
|             field=models.OneToOneField( | ||||
|                 on_delete=django.db.models.deletion.PROTECT, | ||||
|                 related_name="club", | ||||
|                 to="core.group", | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AddConstraint( | ||||
|             model_name="membership", | ||||
|             constraint=models.CheckConstraint( | ||||
|                 check=models.Q(("end_date__gte", models.F("start_date"))), | ||||
|                 name="end_after_start", | ||||
|             ), | ||||
|         ), | ||||
|     ] | ||||
							
								
								
									
										571
									
								
								club/models.py
									
									
									
									
									
								
							
							
						
						
									
										571
									
								
								club/models.py
									
									
									
									
									
								
							| @@ -1,4 +1,3 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| @@ -22,31 +21,41 @@ | ||||
| # Place - Suite 330, Boston, MA 02111-1307, USA. | ||||
| # | ||||
| # | ||||
| from __future__ import annotations | ||||
|  | ||||
| from typing import Iterable, Self | ||||
|  | ||||
| from django.db import models | ||||
| from django.core import validators | ||||
| from django.conf import settings | ||||
| from django.utils.translation import ugettext_lazy as _ | ||||
| from django.core.exceptions import ValidationError, ObjectDoesNotExist | ||||
| from django.db import transaction | ||||
| from django.core.urlresolvers import reverse | ||||
| from django.utils import timezone | ||||
| from django.core import validators | ||||
| from django.core.cache import cache | ||||
| from django.core.exceptions import ObjectDoesNotExist, ValidationError | ||||
| from django.core.validators import RegexValidator, validate_email | ||||
| from django.db import models, transaction | ||||
| from django.db.models import Exists, F, OuterRef, Q | ||||
| from django.urls import reverse | ||||
| from django.utils import timezone | ||||
| from django.utils.functional import cached_property | ||||
| from django.utils.timezone import localdate | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
| from core.models import User, MetaGroup, Group, SithFile, RealGroup, Notification, Page | ||||
| from core.models import Group, Notification, Page, SithFile, User | ||||
|  | ||||
| # Create your models here. | ||||
|  | ||||
|  | ||||
| # This function prevents generating migration upon settings change | ||||
| def get_default_owner_group(): | ||||
|     return settings.SITH_GROUP_ROOT_ID | ||||
|  | ||||
|  | ||||
| class Club(models.Model): | ||||
|     """ | ||||
|     The Club class, made as a tree to allow nice tidy organization | ||||
|     """ | ||||
|     """The Club class, made as a tree to allow nice tidy organization.""" | ||||
|  | ||||
|     id = models.AutoField(primary_key=True, db_index=True) | ||||
|     name = models.CharField(_("name"), max_length=64) | ||||
|     parent = models.ForeignKey("Club", related_name="children", null=True, blank=True) | ||||
|     parent = models.ForeignKey( | ||||
|         "Club", related_name="children", null=True, blank=True, on_delete=models.CASCADE | ||||
|     ) | ||||
|     unix_name = models.CharField( | ||||
|         _("unix name"), | ||||
|         max_length=30, | ||||
| @@ -70,19 +79,6 @@ class Club(models.Model): | ||||
|         _("short description"), max_length=1000, default="", blank=True, null=True | ||||
|     ) | ||||
|     address = models.CharField(_("address"), max_length=254) | ||||
|     # This function prevents generating migration upon settings change | ||||
|     def get_default_owner_group(): | ||||
|         return settings.SITH_GROUP_ROOT_ID | ||||
|  | ||||
|     owner_group = models.ForeignKey( | ||||
|         Group, related_name="owned_club", default=get_default_owner_group | ||||
|     ) | ||||
|     edit_groups = models.ManyToManyField( | ||||
|         Group, related_name="editable_club", blank=True | ||||
|     ) | ||||
|     view_groups = models.ManyToManyField( | ||||
|         Group, related_name="viewable_club", blank=True | ||||
|     ) | ||||
|     home = models.OneToOneField( | ||||
|         SithFile, | ||||
|         related_name="home_of_club", | ||||
| @@ -91,19 +87,60 @@ class Club(models.Model): | ||||
|         blank=True, | ||||
|         on_delete=models.SET_NULL, | ||||
|     ) | ||||
|     page = models.OneToOneField(Page, related_name="club", blank=True, null=True) | ||||
|     page = models.OneToOneField( | ||||
|         Page, related_name="club", blank=True, null=True, on_delete=models.CASCADE | ||||
|     ) | ||||
|     members_group = models.OneToOneField( | ||||
|         Group, related_name="club", on_delete=models.PROTECT | ||||
|     ) | ||||
|     board_group = models.OneToOneField( | ||||
|         Group, related_name="club_board", on_delete=models.PROTECT | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         ordering = ["name", "unix_name"] | ||||
|  | ||||
|     def __str__(self): | ||||
|         return self.name | ||||
|  | ||||
|     @transaction.atomic() | ||||
|     def save(self, *args, **kwargs): | ||||
|         creation = self._state.adding | ||||
|         if not creation: | ||||
|             db_club = Club.objects.get(id=self.id) | ||||
|             if self.unix_name != db_club.unix_name: | ||||
|                 self.home.name = self.unix_name | ||||
|                 self.home.save() | ||||
|             if self.name != db_club.name: | ||||
|                 self.board_group.name = f"{self.name} - Bureau" | ||||
|                 self.board_group.save() | ||||
|                 self.members_group.name = f"{self.name} - Membres" | ||||
|                 self.members_group.save() | ||||
|         if creation: | ||||
|             self.board_group = Group.objects.create( | ||||
|                 name=f"{self.name} - Bureau", is_manually_manageable=False | ||||
|             ) | ||||
|             self.members_group = Group.objects.create( | ||||
|                 name=f"{self.name} - Membres", is_manually_manageable=False | ||||
|             ) | ||||
|         super().save(*args, **kwargs) | ||||
|         if creation: | ||||
|             self.make_home() | ||||
|         self.make_page() | ||||
|         cache.set(f"sith_club_{self.unix_name}", self) | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("club:club_view", kwargs={"club_id": self.id}) | ||||
|  | ||||
|     @cached_property | ||||
|     def president(self): | ||||
|     def president(self) -> Membership | None: | ||||
|         """Fetch the membership of the current president of this club.""" | ||||
|         return self.members.filter( | ||||
|             role=settings.SITH_CLUB_ROLES_ID["President"], end_date=None | ||||
|         ).first() | ||||
|  | ||||
|     def check_loop(self): | ||||
|         """Raise a validation error when a loop is found within the parent list""" | ||||
|         """Raise a validation error when a loop is found within the parent list.""" | ||||
|         objs = [] | ||||
|         cur = self | ||||
|         while cur.parent is not None: | ||||
| @@ -115,26 +152,18 @@ class Club(models.Model): | ||||
|     def clean(self): | ||||
|         self.check_loop() | ||||
|  | ||||
|     def _change_unixname(self, new_name): | ||||
|         c = Club.objects.filter(unix_name=new_name).first() | ||||
|         if c is None: | ||||
|             if self.home: | ||||
|                 self.home.name = new_name | ||||
|                 self.home.save() | ||||
|         else: | ||||
|             raise ValidationError(_("A club with that unix_name already exists")) | ||||
|     def make_home(self) -> None: | ||||
|         if self.home: | ||||
|             return | ||||
|         home_root = SithFile.objects.filter(parent=None, name="clubs").first() | ||||
|         root = User.objects.filter(username="root").first() | ||||
|         if home_root and root: | ||||
|             home = SithFile(parent=home_root, name=self.unix_name, owner=root) | ||||
|             home.save() | ||||
|             self.home = home | ||||
|             self.save() | ||||
|  | ||||
|     def make_home(self): | ||||
|         if not self.home: | ||||
|             home_root = SithFile.objects.filter(parent=None, name="clubs").first() | ||||
|             root = User.objects.filter(username="root").first() | ||||
|             if home_root and root: | ||||
|                 home = SithFile(parent=home_root, name=self.unix_name, owner=root) | ||||
|                 home.save() | ||||
|                 self.home = home | ||||
|                 self.save() | ||||
|  | ||||
|     def make_page(self): | ||||
|     def make_page(self) -> None: | ||||
|         root = User.objects.filter(username="root").first() | ||||
|         if not self.page: | ||||
|             club_root = Page.objects.filter(name=settings.SITH_CLUB_ROOT_PAGE).first() | ||||
| @@ -164,88 +193,137 @@ class Club(models.Model): | ||||
|             self.page.parent = self.parent.page | ||||
|             self.page.save(force_lock=True) | ||||
|  | ||||
|     def save(self, *args, **kwargs): | ||||
|         with transaction.atomic(): | ||||
|             creation = False | ||||
|             old = Club.objects.filter(id=self.id).first() | ||||
|             if not old: | ||||
|                 creation = True | ||||
|             else: | ||||
|                 if old.unix_name != self.unix_name: | ||||
|                     self._change_unixname(self.unix_name) | ||||
|             super(Club, self).save(*args, **kwargs) | ||||
|             if creation: | ||||
|                 board = MetaGroup(name=self.unix_name + settings.SITH_BOARD_SUFFIX) | ||||
|                 board.save() | ||||
|                 member = MetaGroup(name=self.unix_name + settings.SITH_MEMBER_SUFFIX) | ||||
|                 member.save() | ||||
|                 subscribers = Group.objects.filter( | ||||
|                     name=settings.SITH_MAIN_MEMBERS_GROUP | ||||
|                 ).first() | ||||
|                 self.make_home() | ||||
|                 self.home.edit_groups = [board] | ||||
|                 self.home.view_groups = [member, subscribers] | ||||
|                 self.home.save() | ||||
|             self.make_page() | ||||
|     def delete(self, *args, **kwargs) -> tuple[int, dict[str, int]]: | ||||
|         # Invalidate the cache of this club and of its memberships | ||||
|         for membership in self.members.ongoing().select_related("user"): | ||||
|             cache.delete(f"membership_{self.id}_{membership.user.id}") | ||||
|         cache.delete(f"sith_club_{self.unix_name}") | ||||
|         self.board_group.delete() | ||||
|         self.members_group.delete() | ||||
|         return super().delete(*args, **kwargs) | ||||
|  | ||||
|     def __str__(self): | ||||
|     def get_display_name(self) -> str: | ||||
|         return self.name | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("club:club_view", kwargs={"club_id": self.id}) | ||||
|     def is_owned_by(self, user: User) -> bool: | ||||
|         """Method to see if that object can be super edited by the given user.""" | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         return user.is_root or user.is_board_member | ||||
|  | ||||
|     def get_display_name(self): | ||||
|         return self.name | ||||
|     def get_full_logo_url(self) -> str: | ||||
|         return f"https://{settings.SITH_URL}{self.logo.url}" | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be super edited by the given user | ||||
|         """ | ||||
|         return user.is_in_group(settings.SITH_MAIN_BOARD_GROUP) | ||||
|  | ||||
|     def get_full_logo_url(self): | ||||
|         return "https://%s%s" % (settings.SITH_URL, self.logo.url) | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|     def can_be_edited_by(self, user: User) -> bool: | ||||
|         """Method to see if that object can be edited by the given user.""" | ||||
|         return self.has_rights_in_club(user) | ||||
|  | ||||
|     def can_be_viewed_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be seen by the given user | ||||
|         """ | ||||
|         sub = User.objects.filter(pk=user.pk).first() | ||||
|         if sub is None: | ||||
|             return False | ||||
|         return sub.was_subscribed | ||||
|     def can_be_viewed_by(self, user: User) -> bool: | ||||
|         """Method to see if that object can be seen by the given user.""" | ||||
|         return user.was_subscribed | ||||
|  | ||||
|     _memberships = {} | ||||
|     def get_membership_for(self, user: User) -> Membership | None: | ||||
|         """Return the current membership the given user. | ||||
|  | ||||
|     def get_membership_for(self, user): | ||||
|         Note: | ||||
|             The result is cached. | ||||
|         """ | ||||
|         Returns the current membership the given user | ||||
|         """ | ||||
|         try: | ||||
|             return Club._memberships[self.id][user.id] | ||||
|         except: | ||||
|             m = self.members.filter(user=user.id).filter(end_date=None).first() | ||||
|             try: | ||||
|                 Club._memberships[self.id][user.id] = m | ||||
|             except: | ||||
|                 Club._memberships[self.id] = {} | ||||
|                 Club._memberships[self.id][user.id] = m | ||||
|             return m | ||||
|         if user.is_anonymous: | ||||
|             return None | ||||
|         membership = cache.get(f"membership_{self.id}_{user.id}") | ||||
|         if membership == "not_member": | ||||
|             return None | ||||
|         if membership is None: | ||||
|             membership = self.members.filter(user=user, end_date=None).first() | ||||
|             if membership is None: | ||||
|                 cache.set(f"membership_{self.id}_{user.id}", "not_member") | ||||
|             else: | ||||
|                 cache.set(f"membership_{self.id}_{user.id}", membership) | ||||
|         return membership | ||||
|  | ||||
|     def has_rights_in_club(self, user): | ||||
|         m = self.get_membership_for(user) | ||||
|         return m is not None and m.role > settings.SITH_MAXIMUM_FREE_ROLE | ||||
|     def has_rights_in_club(self, user: User) -> bool: | ||||
|         return user.is_in_group(pk=self.board_group_id) | ||||
|  | ||||
|  | ||||
| class MembershipQuerySet(models.QuerySet): | ||||
|     def ongoing(self) -> Self: | ||||
|         """Filter all memberships which are not finished yet.""" | ||||
|         return self.filter(Q(end_date=None) | Q(end_date__gt=localdate())) | ||||
|  | ||||
|     def board(self) -> Self: | ||||
|         """Filter all memberships where the user is/was in the board. | ||||
|  | ||||
|         Be aware that users who were in the board in the past | ||||
|         are included, even if there are no more members. | ||||
|  | ||||
|         If you want to get the users who are currently in the board, | ||||
|         mind combining this with the :meth:`ongoing` queryset method | ||||
|         """ | ||||
|         return self.filter(role__gt=settings.SITH_MAXIMUM_FREE_ROLE) | ||||
|  | ||||
|     def update(self, **kwargs) -> int: | ||||
|         """Refresh the cache and edit group ownership. | ||||
|  | ||||
|         Update the cache, when necessary, remove | ||||
|         users from club groups they are no more in | ||||
|         and add them in the club groups they should be in. | ||||
|  | ||||
|         Be aware that this adds three db queries : | ||||
|         one to retrieve the updated memberships, | ||||
|         one to perform group removal and one to perform | ||||
|         group attribution. | ||||
|         """ | ||||
|         nb_rows = super().update(**kwargs) | ||||
|         if nb_rows == 0: | ||||
|             # if no row was affected, no need to refresh the cache | ||||
|             return 0 | ||||
|  | ||||
|         cache_memberships = {} | ||||
|         memberships = set(self.select_related("club")) | ||||
|         # delete all User-Group relations and recreate the necessary ones | ||||
|         # It's more concise to write and more reliable | ||||
|         Membership._remove_club_groups(memberships) | ||||
|         Membership._add_club_groups(memberships) | ||||
|         for member in memberships: | ||||
|             cache_key = f"membership_{member.club_id}_{member.user_id}" | ||||
|             if member.end_date is None: | ||||
|                 cache_memberships[cache_key] = member | ||||
|             else: | ||||
|                 cache_memberships[cache_key] = "not_member" | ||||
|         cache.set_many(cache_memberships) | ||||
|         return nb_rows | ||||
|  | ||||
|     def delete(self) -> tuple[int, dict[str, int]]: | ||||
|         """Work just like the default Django's delete() method, | ||||
|         but add a cache invalidation for the elements of the queryset | ||||
|         before the deletion, | ||||
|         and a removal of the user from the club groups. | ||||
|  | ||||
|         Be aware that this adds some db queries : | ||||
|  | ||||
|         - 1 to retrieve the deleted elements in order to perform | ||||
|           post-delete operations. | ||||
|           As we can't know if a delete will affect rows or not, | ||||
|           this query will always happen | ||||
|         - 1 query to remove the users from the club groups. | ||||
|           If the delete operation affected no row, | ||||
|           this query won't happen. | ||||
|         """ | ||||
|         memberships = set(self.all()) | ||||
|         nb_rows, rows_counts = super().delete() | ||||
|         if nb_rows > 0: | ||||
|             Membership._remove_club_groups(memberships) | ||||
|             cache.set_many( | ||||
|                 { | ||||
|                     f"membership_{m.club_id}_{m.user_id}": "not_member" | ||||
|                     for m in memberships | ||||
|                 } | ||||
|             ) | ||||
|         return nb_rows, rows_counts | ||||
|  | ||||
|  | ||||
| class Membership(models.Model): | ||||
|     """ | ||||
|     The Membership class makes the connection between User and Clubs | ||||
|     """The Membership class makes the connection between User and Clubs. | ||||
|  | ||||
|     Both Users and Clubs can have many Membership objects: | ||||
|        - a user can be a member of many clubs at a time | ||||
| @@ -261,9 +339,15 @@ class Membership(models.Model): | ||||
|         related_name="memberships", | ||||
|         null=False, | ||||
|         blank=False, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     club = models.ForeignKey( | ||||
|         Club, verbose_name=_("club"), related_name="members", null=False, blank=False | ||||
|         Club, | ||||
|         verbose_name=_("club"), | ||||
|         related_name="members", | ||||
|         null=False, | ||||
|         blank=False, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     start_date = models.DateField(_("start date"), default=timezone.now) | ||||
|     end_date = models.DateField(_("end date"), null=True, blank=True) | ||||
| @@ -276,48 +360,153 @@ class Membership(models.Model): | ||||
|         _("description"), max_length=128, null=False, blank=True | ||||
|     ) | ||||
|  | ||||
|     objects = MembershipQuerySet.as_manager() | ||||
|  | ||||
|     class Meta: | ||||
|         constraints = [ | ||||
|             models.CheckConstraint( | ||||
|                 check=Q(end_date__gte=F("start_date")), name="end_after_start" | ||||
|             ), | ||||
|         ] | ||||
|  | ||||
|     def __str__(self): | ||||
|         return ( | ||||
|             self.club.name | ||||
|             + " - " | ||||
|             + self.user.username | ||||
|             + " - " | ||||
|             + str(settings.SITH_CLUB_ROLES[self.role]) | ||||
|             + str(" - " + str(_("past member")) if self.end_date is not None else "") | ||||
|             f"{self.club.name} - {self.user.username} " | ||||
|             f"- {settings.SITH_CLUB_ROLES[self.role]} " | ||||
|             f"- {str(_('past member')) if self.end_date is not None else ''}" | ||||
|         ) | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         """ | ||||
|         Method to see if that object can be super edited by the given user | ||||
|         """ | ||||
|         return user.is_in_group(settings.SITH_MAIN_BOARD_GROUP) | ||||
|  | ||||
|     def can_be_edited_by(self, user, membership=None): | ||||
|         """ | ||||
|         Method to see if that object can be edited by the given user | ||||
|         """ | ||||
|         if user.memberships: | ||||
|             if membership:  # This is for optimisation purpose | ||||
|                 ms = membership | ||||
|             else: | ||||
|                 ms = user.memberships.filter(club=self.club, end_date=None).first() | ||||
|             return (ms and ms.role >= self.role) or user.is_in_group( | ||||
|                 settings.SITH_MAIN_BOARD_GROUP | ||||
|             ) | ||||
|         return user.is_in_group(settings.SITH_MAIN_BOARD_GROUP) | ||||
|     def save(self, *args, **kwargs): | ||||
|         super().save(*args, **kwargs) | ||||
|         # a save may either be an update or a creation | ||||
|         # and may result in either an ongoing or an ended membership. | ||||
|         # It could also be a retrogradation from the board to being a simple member. | ||||
|         # To avoid problems, the user is removed from the club groups beforehand ; | ||||
|         # he will be added back if necessary | ||||
|         self._remove_club_groups([self]) | ||||
|         if self.end_date is None: | ||||
|             self._add_club_groups([self]) | ||||
|             cache.set(f"membership_{self.club_id}_{self.user_id}", self) | ||||
|         else: | ||||
|             cache.set(f"membership_{self.club_id}_{self.user_id}", "not_member") | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("club:club_members", kwargs={"club_id": self.club.id}) | ||||
|         return reverse("club:club_members", kwargs={"club_id": self.club_id}) | ||||
|  | ||||
|     def is_owned_by(self, user: User) -> bool: | ||||
|         """Method to see if that object can be super edited by the given user.""" | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         return user.is_root or user.is_board_member | ||||
|  | ||||
|     def can_be_edited_by(self, user: User) -> bool: | ||||
|         """Check if that object can be edited by the given user.""" | ||||
|         if user.is_root or user.is_board_member: | ||||
|             return True | ||||
|         membership = self.club.get_membership_for(user) | ||||
|         return membership is not None and membership.role >= self.role | ||||
|  | ||||
|     def delete(self, *args, **kwargs): | ||||
|         self._remove_club_groups([self]) | ||||
|         super().delete(*args, **kwargs) | ||||
|         cache.delete(f"membership_{self.club_id}_{self.user_id}") | ||||
|  | ||||
|     @staticmethod | ||||
|     def _remove_club_groups( | ||||
|         memberships: Iterable[Membership], | ||||
|     ) -> tuple[int, dict[str, int]]: | ||||
|         """Remove users of those memberships from the club groups. | ||||
|  | ||||
|         For example, if a user is in the Troll club board, | ||||
|         he is in the board group and the members group of the Troll. | ||||
|         After calling this function, he will be in neither. | ||||
|  | ||||
|         Returns: | ||||
|             The result of the deletion queryset. | ||||
|  | ||||
|         Warnings: | ||||
|             If this function isn't used in combination | ||||
|             with an actual deletion of the memberships, | ||||
|             it will result in an inconsistent state, | ||||
|             where users will be in the clubs, without | ||||
|             having the associated rights. | ||||
|         """ | ||||
|         clubs = {m.club_id for m in memberships} | ||||
|         users = {m.user_id for m in memberships} | ||||
|         groups = Group.objects.filter(Q(club__in=clubs) | Q(club_board__in=clubs)) | ||||
|         return User.groups.through.objects.filter( | ||||
|             Q(group__in=groups) & Q(user__in=users) | ||||
|         ).delete() | ||||
|  | ||||
|     @staticmethod | ||||
|     def _add_club_groups( | ||||
|         memberships: Iterable[Membership], | ||||
|     ) -> list[User.groups.through]: | ||||
|         """Add users of those memberships to the club groups. | ||||
|  | ||||
|         For example, if a user just joined the Troll club board, | ||||
|         he will be added in both the members group and the board group | ||||
|         of the club. | ||||
|  | ||||
|         Returns: | ||||
|             The created User-Group relations. | ||||
|  | ||||
|         Warnings: | ||||
|             If this function isn't used in combination | ||||
|             with an actual update/creation of the memberships, | ||||
|             it will result in an inconsistent state, | ||||
|             where users will have the rights associated to the | ||||
|             club, without actually being part of it. | ||||
|         """ | ||||
|         # only active membership (i.e. `end_date=None`) | ||||
|         # grant the attribution of club groups. | ||||
|         memberships = [m for m in memberships if m.end_date is None] | ||||
|         if not memberships: | ||||
|             return [] | ||||
|  | ||||
|         if sum(1 for m in memberships if not hasattr(m, "club")) > 1: | ||||
|             # if more than one membership hasn't its `club` attribute set | ||||
|             # it's less expensive to reload the whole query with | ||||
|             # a select_related than perform a distinct query | ||||
|             # to fetch each club. | ||||
|             ids = {m.id for m in memberships} | ||||
|             memberships = list( | ||||
|                 Membership.objects.filter(id__in=ids).select_related("club") | ||||
|             ) | ||||
|         club_groups = [] | ||||
|         for membership in memberships: | ||||
|             club_groups.append( | ||||
|                 User.groups.through( | ||||
|                     user_id=membership.user_id, | ||||
|                     group_id=membership.club.members_group_id, | ||||
|                 ) | ||||
|             ) | ||||
|             if membership.role > settings.SITH_MAXIMUM_FREE_ROLE: | ||||
|                 club_groups.append( | ||||
|                     User.groups.through( | ||||
|                         user_id=membership.user_id, | ||||
|                         group_id=membership.club.board_group_id, | ||||
|                     ) | ||||
|                 ) | ||||
|         return User.groups.through.objects.bulk_create( | ||||
|             club_groups, ignore_conflicts=True | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class Mailing(models.Model): | ||||
|     """ | ||||
|     This class correspond to a mailing list | ||||
|     Remember that mailing lists should be validated by UTBM | ||||
|     """A Mailing list for a club. | ||||
|  | ||||
|     Warning: | ||||
|         Remember that mailing lists should be validated by UTBM. | ||||
|     """ | ||||
|  | ||||
|     club = models.ForeignKey( | ||||
|         Club, verbose_name=_("Club"), related_name="mailings", null=False, blank=False | ||||
|         Club, | ||||
|         verbose_name=_("Club"), | ||||
|         related_name="mailings", | ||||
|         null=False, | ||||
|         blank=False, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     email = models.CharField( | ||||
|         _("Email address"), | ||||
| @@ -334,9 +523,32 @@ class Mailing(models.Model): | ||||
|     ) | ||||
|     is_moderated = models.BooleanField(_("is moderated"), default=False) | ||||
|     moderator = models.ForeignKey( | ||||
|         User, related_name="moderated_mailings", verbose_name=_("moderator"), null=True | ||||
|         User, | ||||
|         related_name="moderated_mailings", | ||||
|         verbose_name=_("moderator"), | ||||
|         null=True, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|  | ||||
|     def __str__(self): | ||||
|         return "%s - %s" % (self.club, self.email_full) | ||||
|  | ||||
|     def save(self, *args, **kwargs): | ||||
|         if not self.is_moderated: | ||||
|             unread_notif_subquery = Notification.objects.filter( | ||||
|                 user=OuterRef("pk"), type="MAILING_MODERATION", viewed=False | ||||
|             ) | ||||
|             for user in User.objects.filter( | ||||
|                 ~Exists(unread_notif_subquery), | ||||
|                 groups__id__in=[settings.SITH_GROUP_COM_ADMIN_ID], | ||||
|             ): | ||||
|                 Notification( | ||||
|                     user=user, | ||||
|                     url=reverse("com:mailing_admin"), | ||||
|                     type="MAILING_MODERATION", | ||||
|                 ).save(*args, **kwargs) | ||||
|         super().save(*args, **kwargs) | ||||
|  | ||||
|     def clean(self): | ||||
|         if Mailing.objects.filter(email=self.email).exists(): | ||||
|             raise ValidationError(_("This mailing list already exists.")) | ||||
| @@ -344,21 +556,19 @@ class Mailing(models.Model): | ||||
|             self.is_moderated = True | ||||
|         else: | ||||
|             self.moderator = None | ||||
|         super(Mailing, self).clean() | ||||
|         super().clean() | ||||
|  | ||||
|     @property | ||||
|     def email_full(self): | ||||
|         return self.email + "@" + settings.SITH_MAILING_DOMAIN | ||||
|  | ||||
|     def can_moderate(self, user): | ||||
|         return user.is_root or user.is_in_group(settings.SITH_GROUP_COM_ADMIN_ID) | ||||
|         return user.is_root or user.is_com_admin | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         return ( | ||||
|             user.is_in_group(self) | ||||
|             or user.is_root | ||||
|             or user.is_in_group(settings.SITH_GROUP_COM_ADMIN_ID) | ||||
|         ) | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         return user.is_root or user.is_com_admin | ||||
|  | ||||
|     def can_view(self, user): | ||||
|         return self.club.has_rights_in_club(user) | ||||
| @@ -366,42 +576,17 @@ class Mailing(models.Model): | ||||
|     def can_be_edited_by(self, user): | ||||
|         return self.club.has_rights_in_club(user) | ||||
|  | ||||
|     def delete(self): | ||||
|         for sub in self.subscriptions.all(): | ||||
|             sub.delete() | ||||
|         super(Mailing, self).delete() | ||||
|     def delete(self, *args, **kwargs): | ||||
|         self.subscriptions.all().delete() | ||||
|         super().delete() | ||||
|  | ||||
|     def fetch_format(self): | ||||
|         resp = self.email + ": " | ||||
|         for sub in self.subscriptions.all(): | ||||
|             resp += sub.fetch_format() | ||||
|         return resp | ||||
|  | ||||
|     def save(self): | ||||
|         if not self.is_moderated: | ||||
|             for user in ( | ||||
|                 RealGroup.objects.filter(id=settings.SITH_GROUP_COM_ADMIN_ID) | ||||
|                 .first() | ||||
|                 .users.all() | ||||
|             ): | ||||
|                 if not user.notifications.filter( | ||||
|                     type="MAILING_MODERATION", viewed=False | ||||
|                 ).exists(): | ||||
|                     Notification( | ||||
|                         user=user, | ||||
|                         url=reverse("com:mailing_admin"), | ||||
|                         type="MAILING_MODERATION", | ||||
|                     ).save() | ||||
|         super(Mailing, self).save() | ||||
|  | ||||
|     def __str__(self): | ||||
|         return "%s - %s" % (self.club, self.email_full) | ||||
|         destination = "".join(s.fetch_format() for s in self.subscriptions.all()) | ||||
|         return f"{self.email}: {destination}" | ||||
|  | ||||
|  | ||||
| class MailingSubscription(models.Model): | ||||
|     """ | ||||
|     This class makes the link between user and mailing list | ||||
|     """ | ||||
|     """Link between user and mailing list.""" | ||||
|  | ||||
|     mailing = models.ForeignKey( | ||||
|         Mailing, | ||||
| @@ -409,6 +594,7 @@ class MailingSubscription(models.Model): | ||||
|         related_name="subscriptions", | ||||
|         null=False, | ||||
|         blank=False, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     user = models.ForeignKey( | ||||
|         User, | ||||
| @@ -416,12 +602,16 @@ class MailingSubscription(models.Model): | ||||
|         related_name="mailing_subscriptions", | ||||
|         null=True, | ||||
|         blank=True, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     email = models.EmailField(_("Email address"), blank=False, null=False) | ||||
|  | ||||
|     class Meta: | ||||
|         unique_together = (("user", "email", "mailing"),) | ||||
|  | ||||
|     def __str__(self): | ||||
|         return "(%s) - %s : %s" % (self.mailing, self.get_username, self.email) | ||||
|  | ||||
|     def clean(self): | ||||
|         if not self.user and not self.email: | ||||
|             raise ValidationError(_("At least user or email is required")) | ||||
| @@ -436,13 +626,15 @@ class MailingSubscription(models.Model): | ||||
|                     ) | ||||
|         except ObjectDoesNotExist: | ||||
|             pass | ||||
|         super(MailingSubscription, self).clean() | ||||
|         super().clean() | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         return ( | ||||
|             self.mailing.club.has_rights_in_club(user) | ||||
|             or user.is_root | ||||
|             or self.user.is_in_group(settings.SITH_GROUP_COM_ADMIN_ID) | ||||
|             or self.user.is_com_admin | ||||
|         ) | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
| @@ -462,6 +654,3 @@ class MailingSubscription(models.Model): | ||||
|  | ||||
|     def fetch_format(self): | ||||
|         return self.get_email + " " | ||||
|  | ||||
|     def __str__(self): | ||||
|         return "(%s) - %s : %s" % (self.mailing, self.get_username, self.email) | ||||
|   | ||||
							
								
								
									
										9
									
								
								club/schemas.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								club/schemas.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| from ninja import ModelSchema | ||||
|  | ||||
| from club.models import Club | ||||
|  | ||||
|  | ||||
| class ClubSchema(ModelSchema): | ||||
|     class Meta: | ||||
|         model = Club | ||||
|         fields = ["id", "name"] | ||||
							
								
								
									
										30
									
								
								club/static/bundled/club/components/ajax-select-index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								club/static/bundled/club/components/ajax-select-index.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| import { AjaxSelect } from "#core:core/components/ajax-select-base"; | ||||
| import { registerComponent } from "#core:utils/web-components"; | ||||
| import type { TomOption } from "tom-select/dist/types/types"; | ||||
| import type { escape_html } from "tom-select/dist/types/utils"; | ||||
| import { type ClubSchema, clubSearchClub } from "#openapi"; | ||||
|  | ||||
| @registerComponent("club-ajax-select") | ||||
| export class ClubAjaxSelect extends AjaxSelect { | ||||
|   protected valueField = "id"; | ||||
|   protected labelField = "name"; | ||||
|   protected searchField = ["code", "name"]; | ||||
|  | ||||
|   protected async search(query: string): Promise<TomOption[]> { | ||||
|     const resp = await clubSearchClub({ query: { search: query } }); | ||||
|     if (resp.data) { | ||||
|       return resp.data.results; | ||||
|     } | ||||
|     return []; | ||||
|   } | ||||
|  | ||||
|   protected renderOption(item: ClubSchema, sanitize: typeof escape_html) { | ||||
|     return `<div class="select-item"> | ||||
|             <span class="select-item-text">${sanitize(item.name)}</span> | ||||
|           </div>`; | ||||
|   } | ||||
|  | ||||
|   protected renderItem(item: ClubSchema, sanitize: typeof escape_html) { | ||||
|     return `<span>${sanitize(item.name)}</span>`; | ||||
|   } | ||||
| } | ||||
| @@ -2,16 +2,16 @@ | ||||
| {% from 'core/macros.jinja' import user_profile_link %} | ||||
|  | ||||
| {% block content %} | ||||
| 	<div id="club_detail"> | ||||
| 		{% if club.logo %} | ||||
| 			<div class="club_logo"><img src="{{ club.logo.url }}" alt="{{ club.unix_name }}"></div> | ||||
| 		{% endif %} | ||||
| 		{% if page_revision %} | ||||
| 			{{ page_revision|markdown }} | ||||
|  		{% else %} | ||||
|  		<h3>{% trans %}Club{% endtrans %}</h3> | ||||
| 	    {% endif %} | ||||
|     </div> | ||||
|   <div id="club_detail"> | ||||
|     {% if club.logo %} | ||||
|       <div class="club_logo"><img src="{{ club.logo.url }}" alt="{{ club.unix_name }}"></div> | ||||
|     {% endif %} | ||||
|     {% if page_revision %} | ||||
|       {{ page_revision|markdown }} | ||||
|     {% else %} | ||||
|       <h3>{% trans %}Club{% endtrans %}</h3> | ||||
|     {% endif %} | ||||
|   </div> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,48 +1,48 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Club list{% endtrans %} | ||||
|   {% trans %}Club list{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% macro display_club(club) -%} | ||||
|  | ||||
|         {% if club.is_active or user.is_root %} | ||||
|          | ||||
|         <li><a href="{{ url('club:club_view', club_id=club.id) }}">{{ club.name }}</a> | ||||
|          | ||||
|         {% if not club.is_active %} | ||||
|             ({% trans %}inactive{% endtrans %}) | ||||
|         {% endif %} | ||||
|   {% if club.is_active or user.is_root %} | ||||
|  | ||||
|         {% if club.president %} - <a href="{{ url('core:user_profile', user_id=club.president.user.id) }}">{{ club.president.user }}</a>{% endif %} | ||||
|         {% if club.short_description %}<p>{{ club.short_description|markdown }}</p>{% endif %} | ||||
|              | ||||
|         {% endif %} | ||||
|     <li><a href="{{ url('club:club_view', club_id=club.id) }}">{{ club.name }}</a> | ||||
|  | ||||
|         {%- if club.children.all()|length != 0 %} | ||||
|         <ul> | ||||
|             {%- for c in club.children.order_by('name') %} | ||||
|                 {{ display_club(c) }} | ||||
|             {%- endfor %} | ||||
|         </ul> | ||||
|         {%- endif -%} | ||||
|     </li> | ||||
|       {% if not club.is_active %} | ||||
|         ({% trans %}inactive{% endtrans %}) | ||||
|       {% endif %} | ||||
|  | ||||
|       {% if club.president %} - <a href="{{ url('core:user_profile', user_id=club.president.user.id) }}">{{ club.president.user }}</a>{% endif %} | ||||
|       {% if club.short_description %}<p>{{ club.short_description|markdown }}</p>{% endif %} | ||||
|  | ||||
|   {% endif %} | ||||
|  | ||||
|   {%- if club.children.all()|length != 0 %} | ||||
|     <ul> | ||||
|       {%- for c in club.children.order_by('name') %} | ||||
|         {{ display_club(c) }} | ||||
|       {%- endfor %} | ||||
|     </ul> | ||||
|   {%- endif -%} | ||||
|   </li> | ||||
| {%- endmacro %} | ||||
|  | ||||
| {% block content %} | ||||
|     {% if user.is_root %} | ||||
|   {% if user.is_root %} | ||||
|     <p><a href="{{ url('club:club_new') }}">{% trans %}New club{% endtrans %}</a></p> | ||||
|     {% endif %} | ||||
|     {% if club_list %} | ||||
|   {% endif %} | ||||
|   {% if club_list %} | ||||
|     <h3>{% trans %}Club list{% endtrans %}</h3> | ||||
|     <ul> | ||||
|         {%- for c in club_list.all().order_by('name') if c.parent is none %} | ||||
|       {%- for c in club_list.all().order_by('name') if c.parent is none %} | ||||
|         {{ display_club(c) }} | ||||
|         {%- endfor %} | ||||
|       {%- endfor %} | ||||
|     </ul> | ||||
|     {% else %} | ||||
|   {% else %} | ||||
|     {% trans %}There is no club in this website.{% endtrans %} | ||||
|     {% endif %} | ||||
|   {% endif %} | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -2,79 +2,81 @@ | ||||
| {% from 'core/macros.jinja' import user_profile_link, select_all_checkbox %} | ||||
|  | ||||
| {% block content %} | ||||
|     <h2>{% trans %}Club members{% endtrans %}</h2> | ||||
|     {% if members %} | ||||
|   <h2>{% trans %}Club members{% endtrans %}</h2> | ||||
|   {% if members %} | ||||
|     <form action="{{ url('club:club_members', club_id=club.id) }}" id="users_old" method="post"> | ||||
|         {% csrf_token %} | ||||
|         {% set users_old = dict(form.users_old | groupby("choice_label")) %} | ||||
|         {% if users_old %} | ||||
|             {{ select_all_checkbox("users_old") }} | ||||
|             <p></p> | ||||
|         {% endif %} | ||||
|         <table> | ||||
|             <thead> | ||||
|                 <td>{% trans %}User{% endtrans %}</td> | ||||
|                 <td>{% trans %}Role{% endtrans %}</td> | ||||
|                 <td>{% trans %}Description{% endtrans %}</td> | ||||
|                 <td>{% trans %}Since{% endtrans %}</td> | ||||
|                 {% if users_old %} | ||||
|                     <td>{% trans %}Mark as old{% endtrans %}</td> | ||||
|                 {% endif %} | ||||
|             </thead> | ||||
|             <tbody> | ||||
|             {% for m in members %} | ||||
|                 <tr> | ||||
|                     <td>{{ user_profile_link(m.user) }}</td> | ||||
|                     <td>{{ settings.SITH_CLUB_ROLES[m.role] }}</td> | ||||
|                     <td>{{ m.description }}</td> | ||||
|                     <td>{{ m.start_date }}</td> | ||||
|                     {% if users_old %} | ||||
|                         <td> | ||||
|                         {% set user_old = users_old[m.user.get_display_name()] %} | ||||
|                         {% if user_old %} | ||||
|                             {{ user_old[0].tag() }} | ||||
|                         {% endif %} | ||||
|                         </td> | ||||
|                     {% endif %} | ||||
|                 </tr> | ||||
|             {% endfor %} | ||||
|             </tbody> | ||||
|         </table> | ||||
|         {{ form.users_old.errors }} | ||||
|         {% if users_old %} | ||||
|             <p></p> | ||||
|             <input type="submit" name="submit" value="{% trans %}Mark as old{% endtrans %}"> | ||||
|         {% endif %} | ||||
|       {% csrf_token %} | ||||
|       {% set users_old = dict(form.users_old | groupby("choice_label")) %} | ||||
|       {% if users_old %} | ||||
|         {{ select_all_checkbox("users_old") }} | ||||
|         <p></p> | ||||
|       {% endif %} | ||||
|       <table> | ||||
|         <thead> | ||||
|           <tr> | ||||
|             <td>{% trans %}User{% endtrans %}</td> | ||||
|             <td>{% trans %}Role{% endtrans %}</td> | ||||
|             <td>{% trans %}Description{% endtrans %}</td> | ||||
|             <td>{% trans %}Since{% endtrans %}</td> | ||||
|             {% if users_old %} | ||||
|               <td>{% trans %}Mark as old{% endtrans %}</td> | ||||
|             {% endif %} | ||||
|           </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
|           {% for m in members %} | ||||
|             <tr> | ||||
|               <td>{{ user_profile_link(m.user) }}</td> | ||||
|               <td>{{ settings.SITH_CLUB_ROLES[m.role] }}</td> | ||||
|               <td>{{ m.description }}</td> | ||||
|               <td>{{ m.start_date }}</td> | ||||
|               {% if users_old %} | ||||
|                 <td> | ||||
|                   {% set user_old = users_old[m.user.get_display_name()] %} | ||||
|                   {% if user_old %} | ||||
|                     {{ user_old[0].tag() }} | ||||
|                   {% endif %} | ||||
|                 </td> | ||||
|               {% endif %} | ||||
|             </tr> | ||||
|           {% endfor %} | ||||
|         </tbody> | ||||
|       </table> | ||||
|       {{ form.users_old.errors }} | ||||
|       {% if users_old %} | ||||
|         <p></p> | ||||
|         <input type="submit" name="submit" value="{% trans %}Mark as old{% endtrans %}"> | ||||
|       {% endif %} | ||||
|     </form> | ||||
|     {% else %} | ||||
|   {% else %} | ||||
|     <p>{% trans %}There are no members in this club.{% endtrans %}</p> | ||||
|   {% endif %} | ||||
|   <form action="{{ url('club:club_members', club_id=club.id) }}" id="add_users" method="post"> | ||||
|     {% csrf_token %} | ||||
|     {{ form.non_field_errors() }} | ||||
|     <p> | ||||
|       {{ form.users.errors }} | ||||
|       <label for="{{ form.users.id_for_label }}">{{ form.users.label }} :</label> | ||||
|       {{ form.users }} | ||||
|       <span class="helptext">{{ form.users.help_text }}</span> | ||||
|     </p> | ||||
|     <p> | ||||
|       {{ form.role.errors }} | ||||
|       <label for="{{ form.role.id_for_label }}">{{ form.role.label }} :</label> | ||||
|       {{ form.role }} | ||||
|     </p> | ||||
|     {% if form.start_date %} | ||||
|       <p> | ||||
|         {{ form.start_date.errors }} | ||||
|         <label for="{{ form.start_date.id_for_label }}">{{ form.start_date.label }} :</label> | ||||
|         {{ form.start_date }} | ||||
|       </p> | ||||
|     {% endif %} | ||||
|     <form action="{{ url('club:club_members', club_id=club.id) }}" id="add_users" method="post"> | ||||
|         {% csrf_token %} | ||||
|         {{ form.non_field_errors() }} | ||||
|         <p> | ||||
|             {{ form.users.errors }} | ||||
|             <label for="{{ form.users.id_for_label }}">{{ form.users.label }} :</label> | ||||
|             {{ form.users }} | ||||
|             <span class="helptext">{{ form.users.help_text }}</span> | ||||
|         </p> | ||||
|         <p> | ||||
|             {{ form.role.errors }} | ||||
|             <label for="{{ form.role.id_for_label }}">{{ form.role.label }} :</label> | ||||
|             {{ form.role }} | ||||
|         </p> | ||||
|         {% if form.start_date %} | ||||
|         <p> | ||||
|             {{ form.start_date.errors }} | ||||
|             <label for="{{ form.start_date.id_for_label }}">{{ form.start_date.label }} :</label> | ||||
|             {{ form.start_date }} | ||||
|         </p> | ||||
|         {% endif %} | ||||
|         <p> | ||||
|             {{ form.description.errors }} | ||||
|             <label for="{{ form.description.id_for_label }}">{{ form.description.label }} :</label> | ||||
|             {{ form.description }} | ||||
|         </p> | ||||
|         <p><input type="submit" value="{% trans %}Add{% endtrans %}" /></p> | ||||
|     </form> | ||||
|     <p> | ||||
|       {{ form.description.errors }} | ||||
|       <label for="{{ form.description.id_for_label }}">{{ form.description.label }} :</label> | ||||
|       {{ form.description }} | ||||
|     </p> | ||||
|     <p><input type="submit" value="{% trans %}Add{% endtrans %}" /></p> | ||||
|   </form> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -2,27 +2,27 @@ | ||||
| {% from 'core/macros.jinja' import user_profile_link %} | ||||
|  | ||||
| {% block content %} | ||||
|     <h2>{% trans %}Club old members{% endtrans %}</h2> | ||||
|     <table> | ||||
|         <thead> | ||||
|             <td>{% trans %}User{% endtrans %}</td> | ||||
|             <td>{% trans %}Role{% endtrans %}</td> | ||||
|             <td>{% trans %}Description{% endtrans %}</td> | ||||
|             <td>{% trans %}From{% endtrans %}</td> | ||||
|             <td>{% trans %}To{% endtrans %}</td> | ||||
|         </thead> | ||||
|         <tbody> | ||||
|         {% for m in club.members.exclude(end_date=None).order_by('-role', 'description', '-end_date').all() %} | ||||
|             <tr> | ||||
|                 <td>{{ user_profile_link(m.user) }}</td> | ||||
|                 <td>{{ settings.SITH_CLUB_ROLES[m.role] }}</td> | ||||
|                 <td>{{ m.description }}</td> | ||||
|                 <td>{{ m.start_date }}</td> | ||||
|                 <td>{{ m.end_date }}</td> | ||||
|             </tr> | ||||
|         {% endfor %} | ||||
|         </tbody> | ||||
|     </table> | ||||
|   <h2>{% trans %}Club old members{% endtrans %}</h2> | ||||
|   <table> | ||||
|     <thead> | ||||
|       <td>{% trans %}User{% endtrans %}</td> | ||||
|       <td>{% trans %}Role{% endtrans %}</td> | ||||
|       <td>{% trans %}Description{% endtrans %}</td> | ||||
|       <td>{% trans %}From{% endtrans %}</td> | ||||
|       <td>{% trans %}To{% endtrans %}</td> | ||||
|     </thead> | ||||
|     <tbody> | ||||
|       {% for m in club.members.exclude(end_date=None).order_by('-role', 'description', '-end_date').all() %} | ||||
|         <tr> | ||||
|           <td>{{ user_profile_link(m.user) }}</td> | ||||
|           <td>{{ settings.SITH_CLUB_ROLES[m.role] }}</td> | ||||
|           <td>{{ m.description }}</td> | ||||
|           <td>{{ m.start_date }}</td> | ||||
|           <td>{{ m.end_date }}</td> | ||||
|         </tr> | ||||
|       {% endfor %} | ||||
|     </tbody> | ||||
|   </table> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,58 +1,94 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
| {% from 'core/macros.jinja' import user_profile_link %} | ||||
|  | ||||
| {# This page uses a custom macro instead of the core `paginate_jinja` and `paginate_alpine` | ||||
| because it works with a somewhat dynamic form, | ||||
| but was written before Alpine was introduced in the project. | ||||
| TODO : rewrite the pagination used in this template an Alpine one | ||||
| #} | ||||
| {% macro paginate(page_obj, paginator, js_action) %} | ||||
|   {% set js = js_action|default('') %} | ||||
|   {% if page_obj.has_previous() or page_obj.has_next() %} | ||||
|     {% if page_obj.has_previous() %} | ||||
|       <a {% if js %} type="submit" onclick="{{ js }}" {% endif %} href="?page={{ page_obj.previous_page_number() }}">{% trans %}Previous{% endtrans %}</a> | ||||
|     {% else %} | ||||
|       <span class="disabled">{% trans %}Previous{% endtrans %}</span> | ||||
|     {% endif %} | ||||
|     {% for i in paginator.page_range %} | ||||
|       {% if page_obj.number == i %} | ||||
|         <span class="active">{{ i }} <span class="sr-only">({% trans %}current{% endtrans %})</span></span> | ||||
|       {% else %} | ||||
|         <a {% if js %} type="submit" onclick="{{ js }}" {% endif %} href="?page={{ i }}">{{ i }}</a> | ||||
|       {% endif %} | ||||
|     {% endfor %} | ||||
|     {% if page_obj.has_next() %} | ||||
|       <a {% if js %} type="submit" onclick="{{ js }}" {% endif %} href="?page={{ page_obj.next_page_number() }}">{% trans %}Next{% endtrans %}</a> | ||||
|     {% else %} | ||||
|       <span class="disabled">{% trans %}Next{% endtrans %}</span> | ||||
|     {% endif %} | ||||
|   {% endif %} | ||||
| {% endmacro %} | ||||
|  | ||||
| {% block content %} | ||||
| <h3>{% trans %}Sellings{% endtrans %}</h3> | ||||
| <form action="" method="get"> | ||||
|   <h3>{% trans %}Sales{% endtrans %}</h3> | ||||
|   <form id="form" action="?page=1" method="post"> | ||||
|     {% csrf_token %} | ||||
|     {{ form }} | ||||
|     <p><input type="submit" value="{% trans %}Show{% endtrans %}" /></p> | ||||
|     <p><input type="submit" value="{% trans %}Download as cvs{% endtrans %}" formaction="{{ url('club:sellings_csv', club_id=object.id) }}"/></p> | ||||
| </form> | ||||
| <p> | ||||
| {% trans %}Quantity: {% endtrans %}{{ total_quantity }} {% trans %}units{% endtrans %}<br/> | ||||
| {% trans %}Total: {% endtrans %}{{ total }} €<br/> | ||||
| {% trans %}Benefit: {% endtrans %}{{ benefit }} € | ||||
| </p> | ||||
| <table> | ||||
|   </form> | ||||
|   <p> | ||||
|     {% trans %}Quantity: {% endtrans %}{{ total_quantity }} {% trans %}units{% endtrans %}<br/> | ||||
|     {% trans %}Total: {% endtrans %}{{ total }} €<br/> | ||||
|     {% trans %}Benefit: {% endtrans %}{{ benefit }} € | ||||
|   </p> | ||||
|   <table> | ||||
|     <thead> | ||||
|         <tr> | ||||
|             <td>{% trans %}Date{% endtrans %}</td> | ||||
|             <td>{% trans %}Counter{% endtrans %}</td> | ||||
|             <td>{% trans %}Barman{% endtrans %}</td> | ||||
|             <td>{% trans %}Customer{% endtrans %}</td> | ||||
|             <td>{% trans %}Label{% endtrans %}</td> | ||||
|             <td>{% trans %}Quantity{% endtrans %}</td> | ||||
|             <td>{% trans %}Total{% endtrans %}</td> | ||||
|             <td>{% trans %}Payment method{% endtrans %}</td> | ||||
|         </tr> | ||||
|       <tr> | ||||
|         <td>{% trans %}Date{% endtrans %}</td> | ||||
|         <td>{% trans %}Counter{% endtrans %}</td> | ||||
|         <td>{% trans %}Barman{% endtrans %}</td> | ||||
|         <td>{% trans %}Customer{% endtrans %}</td> | ||||
|         <td>{% trans %}Label{% endtrans %}</td> | ||||
|         <td>{% trans %}Quantity{% endtrans %}</td> | ||||
|         <td>{% trans %}Total{% endtrans %}</td> | ||||
|         <td>{% trans %}Payment method{% endtrans %}</td> | ||||
|       </tr> | ||||
|     </thead> | ||||
|     <tbody> | ||||
|     {% for s in result %} | ||||
|       {% for s in paginated_result %} | ||||
|         <tr> | ||||
|             <td>{{ s.date|localtime|date(DATETIME_FORMAT) }} {{ s.date|localtime|time(DATETIME_FORMAT) }}</td> | ||||
|             <td>{{ s.counter }}</td> | ||||
|             {% if s.seller %} | ||||
|           <td>{{ s.date|localtime|date(DATETIME_FORMAT) }} {{ s.date|localtime|time(DATETIME_FORMAT) }}</td> | ||||
|           <td>{{ s.counter }}</td> | ||||
|           {% if s.seller %} | ||||
|             <td><a href="{{ s.seller.get_absolute_url() }}">{{ s.seller.get_display_name() }}</a></td> | ||||
|             {% else %} | ||||
|           {% else %} | ||||
|             <td></td> | ||||
|             {% endif %} | ||||
|             {% if s.customer %} | ||||
|           {% endif %} | ||||
|           {% if s.customer %} | ||||
|             <td><a href="{{ s.customer.user.get_absolute_url() }}">{{ s.customer.user.get_display_name() }}</a></td> | ||||
|             {% else %} | ||||
|           {% else %} | ||||
|             <td></td> | ||||
|             {% endif %} | ||||
|             <td>{{ s.label }}</td> | ||||
|             <td>{{ s.quantity }}</td> | ||||
|             <td>{{ s.quantity * s.unit_price }} €</td> | ||||
|             <td>{{ s.get_payment_method_display() }}</td> | ||||
|             {% if s.is_owned_by(user) %} | ||||
|                 <td><a href="{{ url('counter:selling_delete', selling_id=s.id) }}">{% trans %}Delete{% endtrans %}</a></td> | ||||
|             {% endif %} | ||||
|           {% endif %} | ||||
|           <td>{{ s.label }}</td> | ||||
|           <td>{{ s.quantity }}</td> | ||||
|           <td>{{ s.quantity * s.unit_price }} €</td> | ||||
|           <td>{{ s.get_payment_method_display() }}</td> | ||||
|           {% if s.is_owned_by(user) %} | ||||
|             <td><a href="{{ url('counter:selling_delete', selling_id=s.id) }}">{% trans %}Delete{% endtrans %}</a></td> | ||||
|           {% endif %} | ||||
|         </tr> | ||||
|     {% endfor %} | ||||
|       {% endfor %} | ||||
|     </tbody> | ||||
| </table> | ||||
|   </table> | ||||
|   <script type="text/javascript"> | ||||
|     function formPagination(link){ | ||||
|       $("form").attr("action", link.href); | ||||
|       link.href = "javascript:void(0)"; // block link action | ||||
|       $("form").submit(); | ||||
|     } | ||||
|   </script> | ||||
|   {{ paginate(paginated_result, paginator, "formPagination(this)") }} | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,46 +1,46 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block content %} | ||||
| <h3>{% trans %}Club tools{% endtrans %}</h3> | ||||
| <div> | ||||
|   <h3>{% trans %}Club tools{% endtrans %}</h3> | ||||
|   <div> | ||||
|     <h4>{% trans %}Communication:{% endtrans %}</h4> | ||||
|     <ul> | ||||
|         <li> <a href="{{ url('com:news_new') }}?club={{ object.id }}">{% trans %}Create a news{% endtrans %}</a></li> | ||||
|         <li> <a href="{{ url('com:weekmail_article') }}?club={{ object.id }}">{% trans %}Post in the Weekmail{% endtrans %}</a></li> | ||||
|         {% if object.trombi %} | ||||
|       <li> <a href="{{ url('com:news_new') }}?club={{ object.id }}">{% trans %}Create a news{% endtrans %}</a></li> | ||||
|       <li> <a href="{{ url('com:weekmail_article') }}?club={{ object.id }}">{% trans %}Post in the Weekmail{% endtrans %}</a></li> | ||||
|       {% if object.trombi %} | ||||
|         <li> <a href="{{ url('trombi:detail', trombi_id=object.trombi.id) }}">{% trans %}Edit Trombi{% endtrans %}</a></li> | ||||
|         {% else %} | ||||
|       {% else %} | ||||
|         <li> <a href="{{ url('trombi:create', club_id=object.id) }}">{% trans %}New Trombi{% endtrans %}</a></li> | ||||
|         <li> <a href="{{ url('club:poster_list', club_id=object.id) }}">{% trans %}Posters{% endtrans %}</a></li> | ||||
|         {% endif %} | ||||
|       {% endif %} | ||||
|     </ul> | ||||
|     <h4>{% trans %}Counters:{% endtrans %}</h4> | ||||
|     <ul> | ||||
|     {% if object.unix_name == settings.SITH_LAUNDERETTE_MANAGER['unix_name'] %} | ||||
|     {% for l in Launderette.objects.all() %} | ||||
|         <li><a href="{{ url('launderette:main_click', launderette_id=l.id) }}">{{ l }}</a></li> | ||||
|     {% endfor %} | ||||
|     {% elif object.counters.filter(type="OFFICE")|count > 0 %} | ||||
|       {% if object.unix_name == settings.SITH_LAUNDERETTE_MANAGER['unix_name'] %} | ||||
|         {% for l in Launderette.objects.all() %} | ||||
|           <li><a href="{{ url('launderette:main_click', launderette_id=l.id) }}">{{ l }}</a></li> | ||||
|         {% endfor %} | ||||
|       {% elif object.counters.filter(type="OFFICE")|count > 0 %} | ||||
|         {% for c in object.counters.filter(type="OFFICE") %} | ||||
|         <li>{{ c }}: | ||||
|           <li>{{ c }}: | ||||
|             <a href="{{ url('counter:details', counter_id=c.id) }}">View</a> | ||||
|             <a href="{{ url('counter:admin', counter_id=c.id) }}">Edit</a> | ||||
|         </li> | ||||
|           </li> | ||||
|         {% endfor %} | ||||
|     {% endif %} | ||||
|       {% endif %} | ||||
|     </ul> | ||||
|     {% if object.club_account.exists() %} | ||||
|         <h4>{% trans %}Accouting: {% endtrans %}</h4> | ||||
|         <ul> | ||||
|       <h4>{% trans %}Accounting: {% endtrans %}</h4> | ||||
|       <ul> | ||||
|         {% for ca in object.club_account.all() %} | ||||
|             <li><a href="{{ url('accounting:club_details', c_account_id=ca.id) }}">{{ ca.get_display_name() }}</a></li> | ||||
|           <li><a href="{{ url('accounting:club_details', c_account_id=ca.id) }}">{{ ca.get_display_name() }}</a></li> | ||||
|         {% endfor %} | ||||
|         </ul> | ||||
|       </ul> | ||||
|     {% endif %} | ||||
|     {% if object.unix_name == settings.SITH_LAUNDERETTE_MANAGER['unix_name'] %} | ||||
|     <li><a href="{{ url('launderette:launderette_list') }}">{% trans %}Manage launderettes{% endtrans %}</a></li> | ||||
|       <li><a href="{{ url('launderette:launderette_list') }}">{% trans %}Manage launderettes{% endtrans %}</a></li> | ||||
|     {% endif %} | ||||
| </div> | ||||
|   </div> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -2,107 +2,107 @@ | ||||
| {% from 'core/macros.jinja' import select_all_checkbox %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Mailing lists{% endtrans %} | ||||
|   {% trans %}Mailing lists{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|  | ||||
|     <b>{% trans %}Remember : mailing lists need to be moderated, if your new created list is not shown wait until moderation takes action{% endtrans %}</b> | ||||
|   <b>{% trans %}Remember : mailing lists need to be moderated, if your new created list is not shown wait until moderation takes action{% endtrans %}</b> | ||||
|  | ||||
|     {% if mailings_not_moderated %} | ||||
|         <p>{% trans %}Mailing lists waiting for moderation{% endtrans %}</p> | ||||
|         <ul> | ||||
|             {% for mailing in mailings_not_moderated %} | ||||
|                 <li>{{ mailing.email_full }}<a href="{{ url('club:mailing_delete', mailing_id=mailing.id) }}"> - {% trans %}Delete{% endtrans %}</a></li> | ||||
|             {% endfor %} | ||||
|         </ul> | ||||
|     {% endif %} | ||||
|   {% if mailings_not_moderated %} | ||||
|     <p>{% trans %}Mailing lists waiting for moderation{% endtrans %}</p> | ||||
|     <ul> | ||||
|       {% for mailing in mailings_not_moderated %} | ||||
|         <li>{{ mailing.email_full }}<a href="{{ url('club:mailing_delete', mailing_id=mailing.id) }}"> - {% trans %}Delete{% endtrans %}</a></li> | ||||
|       {% endfor %} | ||||
|     </ul> | ||||
|   {% endif %} | ||||
|  | ||||
|     {% if mailings_moderated %} | ||||
|   {% if mailings_moderated %} | ||||
|     {% for mailing in mailings_moderated %} | ||||
|     <h2>{% trans %}Mailing{% endtrans %} {{ mailing.email_full }} | ||||
|       <h2>{% trans %}Mailing{% endtrans %} {{ mailing.email_full }} | ||||
|         {%- if user.is_owner(mailing) -%} | ||||
|         <a href="{{ url('club:mailing_delete', mailing_id=mailing.id) }}"> - {% trans %}Delete{% endtrans %}</a> | ||||
|           <a href="{{ url('club:mailing_delete', mailing_id=mailing.id) }}"> - {% trans %}Delete{% endtrans %}</a> | ||||
|         {%- endif -%} | ||||
|     </h2> | ||||
|     <form method="GET" action="{{ url('club:mailing_generate', mailing_id=mailing.id) }}" style="display:inline-block;"> | ||||
|       </h2> | ||||
|       <form method="GET" action="{{ url('club:mailing_generate', mailing_id=mailing.id) }}" style="display:inline-block;"> | ||||
|         <input type="submit" name="generateMalingList" value="{% trans %}Generate mailing list{% endtrans %}"> | ||||
|     </form> | ||||
|     {% set form_mailing_removal = form["removal_" + mailing.id|string] %} | ||||
|     {% if form_mailing_removal.field.choices %} | ||||
|     {% set ms = dict(mailing.subscriptions.all() | groupby('id')) %} | ||||
|     <form action="{{ url('club:mailing', club_id=club.id) }}" id="{{ form_mailing_removal.auto_id }}" method="post" enctype="multipart/form-data"> | ||||
|         <p style="margin-bottom: 1em;">{{ select_all_checkbox(form_mailing_removal.auto_id) }}</p> | ||||
|         {% csrf_token %} | ||||
|         <input hidden type="number" name="{{ form.action.name }}" value="{{ form_actions.REMOVE_SUBSCRIPTION }}" /> | ||||
|         <table> | ||||
|       </form> | ||||
|       {% set form_mailing_removal = form["removal_" + mailing.id|string] %} | ||||
|       {% if form_mailing_removal.field.choices %} | ||||
|         {% set ms = dict(mailing.subscriptions.all() | groupby('id')) %} | ||||
|         <form action="{{ url('club:mailing', club_id=club.id) }}" id="{{ form_mailing_removal.auto_id }}" method="post" enctype="multipart/form-data"> | ||||
|           <p style="margin-bottom: 1em;">{{ select_all_checkbox(form_mailing_removal.auto_id) }}</p> | ||||
|           {% csrf_token %} | ||||
|           <input hidden type="number" name="{{ form.action.name }}" value="{{ form_actions.REMOVE_SUBSCRIPTION }}" /> | ||||
|           <table> | ||||
|             <thead> | ||||
|                 <tr> | ||||
|                     <td>{% trans %}User{% endtrans %}</td> | ||||
|                     <td>{% trans %}Email{% endtrans %}</td> | ||||
|                     <td>{% trans %}Delete{% endtrans %}</td> | ||||
|                 </tr> | ||||
|               <tr> | ||||
|                 <td>{% trans %}User{% endtrans %}</td> | ||||
|                 <td>{% trans %}Email{% endtrans %}</td> | ||||
|                 <td>{% trans %}Delete{% endtrans %}</td> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|                 {% for widget in form_mailing_removal.subwidgets %} | ||||
|                 {% set user = ms[widget.data.value][0] %} | ||||
|               {% for widget in form_mailing_removal.subwidgets %} | ||||
|                 {% set user = ms[widget.data.value.value][0] %} | ||||
|                 <tr> | ||||
|                     <td>{{ user.get_username }}</td> | ||||
|                     <td>{{ user.get_email }}</td> | ||||
|                     <td>{{ widget.tag() }}</td> | ||||
|                   <td>{{ user.get_username }}</td> | ||||
|                   <td>{{ user.get_email }}</td> | ||||
|                   <td>{{ widget.tag() }}</td> | ||||
|                 </tr> | ||||
|                 {% endfor %} | ||||
|               {% endfor %} | ||||
|             </tbody> | ||||
|         </table> | ||||
|         {{ form_mailing_removal.errors }} | ||||
|         <p><input type="submit" value="{% trans %}Remove from mailing list{% endtrans %}" /></p> | ||||
|     </form> | ||||
|           </table> | ||||
|           {{ form_mailing_removal.errors }} | ||||
|           <p><input type="submit" value="{% trans %}Remove from mailing list{% endtrans %}" /></p> | ||||
|         </form> | ||||
|  | ||||
|     {% else %} | ||||
|       {% else %} | ||||
|         <p><b>{% trans %}There is no subscriber for this mailing list{% endtrans %}</b></p> | ||||
|     {% endif %} | ||||
|       {% endif %} | ||||
|     {% endfor %} | ||||
|  | ||||
|     {% else %} | ||||
|   {% else %} | ||||
|     <p>{% trans %}No mailing list existing for this club{% endtrans %}</p> | ||||
|     {% endif %} | ||||
|   {% endif %} | ||||
|  | ||||
|     <p>{{ form.non_field_errors() }}</p> | ||||
|     {% if mailings_moderated %} | ||||
|         <h2>{% trans %}New member{% endtrans %}</h2> | ||||
|         <form action="{{ url('club:mailing', club_id=club.id) }}" method="post" enctype="multipart/form-data"> | ||||
|             {% csrf_token %} | ||||
|             <p> | ||||
|                 {{ form.subscription_mailing.errors }} | ||||
|                 <label for="{{ form.subscription_mailing.id_for_label }}">{{ form.subscription_mailing.label }}</label> | ||||
|                 {{ form.subscription_mailing }} | ||||
|             </p> | ||||
|             <p> | ||||
|                 {{ form.subscription_users.errors }} | ||||
|                 <label for="{{ form.subscription_users.id_for_label }}">{{ form.subscription_users.label }}</label> | ||||
|                 {{ form.subscription_users }} | ||||
|                 <span class="helptext">{{ form.subscription_users.help_text }}</span> | ||||
|             </p> | ||||
|             <p> | ||||
|                 {{ form.subscription_email.errors }} | ||||
|                 <label for="{{ form.subscription_email.id_for_label }}">{{ form.subscription_email.label }}</label> | ||||
|                 {{ form.subscription_email }} | ||||
|             </p> | ||||
|             <input hidden type="number" name="{{ form.action.name }}" value="{{ form_actions.NEW_SUBSCRIPTION }}" /> | ||||
|             <p><input type="submit" value="{% trans %}Add to mailing list{% endtrans %}" /></p> | ||||
|         </form> | ||||
|     {% endif %} | ||||
|  | ||||
|     <h2>{% trans %}New mailing{% endtrans %}</h2> | ||||
|   <p>{{ form.non_field_errors() }}</p> | ||||
|   {% if mailings_moderated %} | ||||
|     <h2>{% trans %}New member{% endtrans %}</h2> | ||||
|     <form action="{{ url('club:mailing', club_id=club.id) }}" method="post" enctype="multipart/form-data"> | ||||
|         {% csrf_token %} | ||||
|         <p> | ||||
|             {{ form.mailing_email.errors }} | ||||
|             <label for="{{ form.mailing_email.id_for_label }}">{{ form.mailing_email.label }}</label> | ||||
|             {{ form.mailing_email }} | ||||
|         </p> | ||||
|         <input hidden type="number" name="{{ form.action.name }}" value="{{ form_actions.NEW_MALING }}" /> | ||||
|         <p><input type="submit" value="{% trans %}Create mailing list{% endtrans %}" /></p> | ||||
|       {% csrf_token %} | ||||
|       <p> | ||||
|         {{ form.subscription_mailing.errors }} | ||||
|         <label for="{{ form.subscription_mailing.id_for_label }}">{{ form.subscription_mailing.label }}</label> | ||||
|         {{ form.subscription_mailing }} | ||||
|       </p> | ||||
|       <p> | ||||
|         {{ form.subscription_users.errors }} | ||||
|         <label for="{{ form.subscription_users.id_for_label }}">{{ form.subscription_users.label }}</label> | ||||
|         {{ form.subscription_users }} | ||||
|         <span class="helptext">{{ form.subscription_users.help_text }}</span> | ||||
|       </p> | ||||
|       <p> | ||||
|         {{ form.subscription_email.errors }} | ||||
|         <label for="{{ form.subscription_email.id_for_label }}">{{ form.subscription_email.label }}</label> | ||||
|         {{ form.subscription_email }} | ||||
|       </p> | ||||
|       <input hidden type="number" name="{{ form.action.name }}" value="{{ form_actions.NEW_SUBSCRIPTION }}" /> | ||||
|       <p><input type="submit" value="{% trans %}Add to mailing list{% endtrans %}" /></p> | ||||
|     </form> | ||||
|   {% endif %} | ||||
|  | ||||
|   <h2>{% trans %}New mailing{% endtrans %}</h2> | ||||
|   <form action="{{ url('club:mailing', club_id=club.id) }}" method="post" enctype="multipart/form-data"> | ||||
|     {% csrf_token %} | ||||
|     <p> | ||||
|       {{ form.mailing_email.errors }} | ||||
|       <label for="{{ form.mailing_email.id_for_label }}">{{ form.mailing_email.label }}</label> | ||||
|       {{ form.mailing_email }} | ||||
|     </p> | ||||
|     <input hidden type="number" name="{{ form.action.name }}" value="{{ form_actions.NEW_MALING }}" /> | ||||
|     <p><input type="submit" value="{% trans %}Create mailing list{% endtrans %}" /></p> | ||||
|   </form> | ||||
|  | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -2,11 +2,11 @@ | ||||
| {% from 'core/macros_pages.jinja' import page_history %} | ||||
|  | ||||
| {% block content %} | ||||
| 	{% if club.page %} | ||||
| 	{{ page_history(club.page) }} | ||||
| 	{% else %} | ||||
| 	    {% trans %}No page existing for this club{% endtrans %} | ||||
| 	{% endif %} | ||||
|   {% if club.page %} | ||||
|     {{ page_history(club.page) }} | ||||
|   {% else %} | ||||
|     {% trans %}No page existing for this club{% endtrans %} | ||||
|   {% endif %} | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
| {% from 'core/macros_pages.jinja' import page_edit_form %} | ||||
|  | ||||
| {% block content %} | ||||
| {{ page_edit_form(page, form, url('club:club_edit_page', club_id=page.club.id), csrf_token) }} | ||||
|   {{ page_edit_form(page, form, url('club:club_edit_page', club_id=page.club.id), csrf_token) }} | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,48 +1,48 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block title %} | ||||
| {% trans %}Club stats{% endtrans %} | ||||
|   {% trans %}Club stats{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     {% if club_list %} | ||||
|   {% if club_list %} | ||||
|     <h3>{% trans %}Club stats{% endtrans %}</h3> | ||||
|     <form action="" method="GET"> | ||||
|         {% csrf_token %} | ||||
|         <p> | ||||
|       {% csrf_token %} | ||||
|       <p> | ||||
|         <select name="branch"> | ||||
|             {% for b in settings.SITH_PROFILE_DEPARTMENTS %} | ||||
|                 <option value="{{ b[0] }}">{{ b[0] }}</option> | ||||
|             {% endfor %} | ||||
|           {% for b in settings.SITH_PROFILE_DEPARTMENTS %} | ||||
|             <option value="{{ b[0] }}">{{ b[0] }}</option> | ||||
|           {% endfor %} | ||||
|         </select> | ||||
|         </p> | ||||
|         <p><input type="submit" value="{% trans %}Show{% endtrans %}" /></p> | ||||
|       </p> | ||||
|       <p><input type="submit" value="{% trans %}Show{% endtrans %}" /></p> | ||||
|     </form> | ||||
|     <table> | ||||
|         <thead> | ||||
|             <tr> | ||||
|                 <td>Club</td> | ||||
|                 <td>Member number</td> | ||||
|                 <td>Old member number</td> | ||||
|             </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
|       <thead> | ||||
|         <tr> | ||||
|           <td>Club</td> | ||||
|           <td>Member number</td> | ||||
|           <td>Old member number</td> | ||||
|         </tr> | ||||
|       </thead> | ||||
|       <tbody> | ||||
|         {% for c in club_list.order_by('id') %} | ||||
|             {% set members = c.members.all() %} | ||||
|             {% if request.GET['branch'] %} | ||||
|           {% set members = c.members.all() %} | ||||
|           {% if request.GET['branch'] %} | ||||
|             {% set members = members.filter(user__department=request.GET['branch']) %} | ||||
|             {% endif %} | ||||
|             <tr> | ||||
|                 <td>{{ c.get_display_name() }}</td> | ||||
|                 <td>{{ members.filter(end_date=None, role__gt=settings.SITH_MAXIMUM_FREE_ROLE).count() }}</td> | ||||
|                 <td>{{ members.exclude(end_date=None, role__gt=settings.SITH_MAXIMUM_FREE_ROLE).count() }}</td> | ||||
|             </tr> | ||||
|           {% endif %} | ||||
|           <tr> | ||||
|             <td>{{ c.get_display_name() }}</td> | ||||
|             <td>{{ members.filter(end_date=None, role__gt=settings.SITH_MAXIMUM_FREE_ROLE).count() }}</td> | ||||
|             <td>{{ members.exclude(end_date=None, role__gt=settings.SITH_MAXIMUM_FREE_ROLE).count() }}</td> | ||||
|           </tr> | ||||
|         {% endfor %} | ||||
|         </tbody> | ||||
|       </tbody> | ||||
|     </table> | ||||
|     {% else %} | ||||
|   {% else %} | ||||
|     {% trans %}There is no club in this website.{% endtrans %} | ||||
|     {% endif %} | ||||
|   {% endif %} | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										1197
									
								
								club/tests.py
									
									
									
									
									
								
							
							
						
						
									
										1197
									
								
								club/tests.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										117
									
								
								club/urls.py
									
									
									
									
									
								
							
							
						
						
									
										117
									
								
								club/urls.py
									
									
									
									
									
								
							| @@ -1,4 +1,3 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| @@ -23,81 +22,95 @@ | ||||
| # | ||||
| # | ||||
|  | ||||
| from django.conf.urls import url | ||||
| from django.urls import path | ||||
|  | ||||
| from club.views import * | ||||
| from club.views import ( | ||||
|     ClubCreateView, | ||||
|     ClubEditPropView, | ||||
|     ClubEditView, | ||||
|     ClubListView, | ||||
|     ClubMailingView, | ||||
|     ClubMembersView, | ||||
|     ClubOldMembersView, | ||||
|     ClubPageEditView, | ||||
|     ClubPageHistView, | ||||
|     ClubRevView, | ||||
|     ClubSellingCSVView, | ||||
|     ClubSellingView, | ||||
|     ClubStatView, | ||||
|     ClubToolsView, | ||||
|     ClubView, | ||||
|     MailingAutoGenerationView, | ||||
|     MailingDeleteView, | ||||
|     MailingSubscriptionDeleteView, | ||||
|     MembershipDeleteView, | ||||
|     MembershipSetOldView, | ||||
|     PosterCreateView, | ||||
|     PosterDeleteView, | ||||
|     PosterEditView, | ||||
|     PosterListView, | ||||
| ) | ||||
|  | ||||
| urlpatterns = [ | ||||
|     url(r"^$", ClubListView.as_view(), name="club_list"), | ||||
|     url(r"^new$", ClubCreateView.as_view(), name="club_new"), | ||||
|     url(r"^stats$", ClubStatView.as_view(), name="club_stats"), | ||||
|     url(r"^(?P<club_id>[0-9]+)/$", ClubView.as_view(), name="club_view"), | ||||
|     url( | ||||
|         r"^(?P<club_id>[0-9]+)/rev/(?P<rev_id>[0-9]+)/$", | ||||
|         ClubRevView.as_view(), | ||||
|         name="club_view_rev", | ||||
|     path("", ClubListView.as_view(), name="club_list"), | ||||
|     path("new/", ClubCreateView.as_view(), name="club_new"), | ||||
|     path("stats/", ClubStatView.as_view(), name="club_stats"), | ||||
|     path("<int:club_id>/", ClubView.as_view(), name="club_view"), | ||||
|     path( | ||||
|         "<int:club_id>/rev/<int:rev_id>/", ClubRevView.as_view(), name="club_view_rev" | ||||
|     ), | ||||
|     url(r"^(?P<club_id>[0-9]+)/hist$", ClubPageHistView.as_view(), name="club_hist"), | ||||
|     url(r"^(?P<club_id>[0-9]+)/edit$", ClubEditView.as_view(), name="club_edit"), | ||||
|     url( | ||||
|         r"^(?P<club_id>[0-9]+)/edit/page$", | ||||
|         ClubPageEditView.as_view(), | ||||
|         name="club_edit_page", | ||||
|     ), | ||||
|     url( | ||||
|         r"^(?P<club_id>[0-9]+)/members$", ClubMembersView.as_view(), name="club_members" | ||||
|     ), | ||||
|     url( | ||||
|         r"^(?P<club_id>[0-9]+)/elderlies$", | ||||
|     path("<int:club_id>/hist/", ClubPageHistView.as_view(), name="club_hist"), | ||||
|     path("<int:club_id>/edit/", ClubEditView.as_view(), name="club_edit"), | ||||
|     path("<int:club_id>/edit/page/", ClubPageEditView.as_view(), name="club_edit_page"), | ||||
|     path("<int:club_id>/members/", ClubMembersView.as_view(), name="club_members"), | ||||
|     path( | ||||
|         "<int:club_id>/elderlies/", | ||||
|         ClubOldMembersView.as_view(), | ||||
|         name="club_old_members", | ||||
|     ), | ||||
|     url( | ||||
|         r"^(?P<club_id>[0-9]+)/sellings$", | ||||
|         ClubSellingView.as_view(), | ||||
|         name="club_sellings", | ||||
|     path("<int:club_id>/sellings/", ClubSellingView.as_view(), name="club_sellings"), | ||||
|     path( | ||||
|         "<int:club_id>/sellings/csv/", ClubSellingCSVView.as_view(), name="sellings_csv" | ||||
|     ), | ||||
|     url( | ||||
|         r"^(?P<club_id>[0-9]+)/sellings/csv$", | ||||
|         ClubSellingCSVView.as_view(), | ||||
|         name="sellings_csv", | ||||
|     ), | ||||
|     url(r"^(?P<club_id>[0-9]+)/prop$", ClubEditPropView.as_view(), name="club_prop"), | ||||
|     url(r"^(?P<club_id>[0-9]+)/tools$", ClubToolsView.as_view(), name="tools"), | ||||
|     url(r"^(?P<club_id>[0-9]+)/mailing$", ClubMailingView.as_view(), name="mailing"), | ||||
|     url( | ||||
|         r"^(?P<mailing_id>[0-9]+)/mailing/generate$", | ||||
|     path("<int:club_id>/prop/", ClubEditPropView.as_view(), name="club_prop"), | ||||
|     path("<int:club_id>/tools/", ClubToolsView.as_view(), name="tools"), | ||||
|     path("<int:club_id>/mailing/", ClubMailingView.as_view(), name="mailing"), | ||||
|     path( | ||||
|         "<int:mailing_id>/mailing/generate/", | ||||
|         MailingAutoGenerationView.as_view(), | ||||
|         name="mailing_generate", | ||||
|     ), | ||||
|     url( | ||||
|         r"^(?P<mailing_id>[0-9]+)/mailing/delete$", | ||||
|     path( | ||||
|         "<int:mailing_id>/mailing/delete/", | ||||
|         MailingDeleteView.as_view(), | ||||
|         name="mailing_delete", | ||||
|     ), | ||||
|     url( | ||||
|         r"^(?P<mailing_subscription_id>[0-9]+)/mailing/delete/subscription$", | ||||
|     path( | ||||
|         "<int:mailing_subscription_id>/mailing/delete/subscription/", | ||||
|         MailingSubscriptionDeleteView.as_view(), | ||||
|         name="mailing_subscription_delete", | ||||
|     ), | ||||
|     url( | ||||
|         r"^membership/(?P<membership_id>[0-9]+)/set_old$", | ||||
|     path( | ||||
|         "membership/<int:membership_id>/set_old/", | ||||
|         MembershipSetOldView.as_view(), | ||||
|         name="membership_set_old", | ||||
|     ), | ||||
|     url(r"^(?P<club_id>[0-9]+)/poster$", PosterListView.as_view(), name="poster_list"), | ||||
|     url( | ||||
|         r"^(?P<club_id>[0-9]+)/poster/create$", | ||||
|         PosterCreateView.as_view(), | ||||
|         name="poster_create", | ||||
|     path( | ||||
|         "membership/<int:membership_id>/delete/", | ||||
|         MembershipDeleteView.as_view(), | ||||
|         name="membership_delete", | ||||
|     ), | ||||
|     url( | ||||
|         r"^(?P<club_id>[0-9]+)/poster/(?P<poster_id>[0-9]+)/edit$", | ||||
|     path("<int:club_id>/poster/", PosterListView.as_view(), name="poster_list"), | ||||
|     path( | ||||
|         "<int:club_id>/poster/create/", PosterCreateView.as_view(), name="poster_create" | ||||
|     ), | ||||
|     path( | ||||
|         "<int:club_id>/poster/<int:poster_id>/edit/", | ||||
|         PosterEditView.as_view(), | ||||
|         name="poster_edit", | ||||
|     ), | ||||
|     url( | ||||
|         r"^(?P<club_id>[0-9]+)/poster/(?P<poster_id>[0-9]+)/delete$", | ||||
|     path( | ||||
|         "<int:club_id>/poster/<int:poster_id>/delete/", | ||||
|         PosterDeleteView.as_view(), | ||||
|         name="poster_delete", | ||||
|     ), | ||||
|   | ||||
							
								
								
									
										371
									
								
								club/views.py
									
									
									
									
									
								
							
							
						
						
									
										371
									
								
								club/views.py
									
									
									
									
									
								
							| @@ -1,4 +1,3 @@ | ||||
| # -*- coding:utf-8 -* | ||||
| # | ||||
| # Copyright 2016,2017 | ||||
| # - Skia <skia@libskia.so> | ||||
| @@ -23,43 +22,45 @@ | ||||
| # | ||||
| # | ||||
|  | ||||
| import csv | ||||
|  | ||||
| from django.conf import settings | ||||
| from django import forms | ||||
| from django.views.generic import ListView, DetailView, TemplateView, View | ||||
| from django.views.generic.edit import DeleteView | ||||
| from django.views.generic.detail import SingleObjectMixin | ||||
| from django.views.generic.edit import UpdateView, CreateView | ||||
| from django.http import HttpResponseRedirect, HttpResponse, Http404 | ||||
| from django.core.urlresolvers import reverse, reverse_lazy | ||||
| from django.utils import timezone | ||||
| from django.utils.translation import ugettext_lazy as _ | ||||
| from django.utils.translation import ugettext as _t | ||||
| from django.core.exceptions import PermissionDenied, ValidationError, NON_FIELD_ERRORS | ||||
| from django.core.exceptions import NON_FIELD_ERRORS, PermissionDenied, ValidationError | ||||
| from django.core.paginator import InvalidPage, Paginator | ||||
| from django.db.models import Sum | ||||
| from django.http import ( | ||||
|     Http404, | ||||
|     HttpResponseRedirect, | ||||
|     StreamingHttpResponse, | ||||
| ) | ||||
| from django.shortcuts import get_object_or_404, redirect | ||||
| from django.urls import reverse, reverse_lazy | ||||
| from django.utils import timezone | ||||
| from django.utils.translation import gettext as _t | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| from django.views.generic import DetailView, ListView, TemplateView, View | ||||
| from django.views.generic.edit import CreateView, DeleteView, UpdateView | ||||
|  | ||||
| from core.views import ( | ||||
|     CanCreateMixin, | ||||
|     CanViewMixin, | ||||
|     CanEditMixin, | ||||
|     CanEditPropMixin, | ||||
|     TabedViewMixin, | ||||
|     PageEditViewBase, | ||||
|     DetailFormView, | ||||
| from club.forms import ClubEditForm, ClubMemberForm, MailingForm, SellingsForm | ||||
| from club.models import Club, Mailing, MailingSubscription, Membership | ||||
| from com.views import ( | ||||
|     PosterCreateBaseView, | ||||
|     PosterDeleteBaseView, | ||||
|     PosterEditBaseView, | ||||
|     PosterListBaseView, | ||||
| ) | ||||
| from core.models import PageRev | ||||
|  | ||||
| from counter.models import Selling | ||||
|  | ||||
| from com.views import ( | ||||
|     PosterListBaseView, | ||||
|     PosterCreateBaseView, | ||||
|     PosterEditBaseView, | ||||
|     PosterDeleteBaseView, | ||||
| from core.views import ( | ||||
|     CanCreateMixin, | ||||
|     CanEditMixin, | ||||
|     CanEditPropMixin, | ||||
|     CanViewMixin, | ||||
|     DetailFormView, | ||||
|     PageEditViewBase, | ||||
|     TabedViewMixin, | ||||
|     UserIsRootMixin, | ||||
| ) | ||||
|  | ||||
| from club.models import Club, Membership, Mailing, MailingSubscription | ||||
| from club.forms import MailingForm, ClubEditForm, ClubMemberForm, SellingsFormBase | ||||
| from counter.models import Selling | ||||
|  | ||||
|  | ||||
| class ClubTabsMixin(TabedViewMixin): | ||||
| @@ -70,14 +71,13 @@ class ClubTabsMixin(TabedViewMixin): | ||||
|         return self.object.get_display_name() | ||||
|  | ||||
|     def get_list_of_tabs(self): | ||||
|         tab_list = [] | ||||
|         tab_list.append( | ||||
|         tab_list = [ | ||||
|             { | ||||
|                 "url": reverse("club:club_view", kwargs={"club_id": self.object.id}), | ||||
|                 "slug": "infos", | ||||
|                 "name": _("Infos"), | ||||
|             } | ||||
|         ) | ||||
|         ] | ||||
|         if self.request.user.can_view(self.object): | ||||
|             tab_list.append( | ||||
|                 { | ||||
| @@ -174,18 +174,14 @@ class ClubTabsMixin(TabedViewMixin): | ||||
|  | ||||
|  | ||||
| class ClubListView(ListView): | ||||
|     """ | ||||
|     List the Clubs | ||||
|     """ | ||||
|     """List the Clubs.""" | ||||
|  | ||||
|     model = Club | ||||
|     template_name = "club/club_list.jinja" | ||||
|  | ||||
|  | ||||
| class ClubView(ClubTabsMixin, DetailView): | ||||
|     """ | ||||
|     Front page of a Club | ||||
|     """ | ||||
|     """Front page of a Club.""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
| @@ -193,24 +189,22 @@ class ClubView(ClubTabsMixin, DetailView): | ||||
|     current_tab = "infos" | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super(ClubView, self).get_context_data(**kwargs) | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         if self.object.page and self.object.page.revisions.exists(): | ||||
|             kwargs["page_revision"] = self.object.page.revisions.last().content | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class ClubRevView(ClubView): | ||||
|     """ | ||||
|     Display a specific page revision | ||||
|     """ | ||||
|     """Display a specific page revision.""" | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         obj = self.get_object() | ||||
|         self.revision = get_object_or_404(PageRev, pk=kwargs["rev_id"], page__club=obj) | ||||
|         return super(ClubRevView, self).dispatch(request, *args, **kwargs) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super(ClubRevView, self).get_context_data(**kwargs) | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["page_revision"] = self.revision.content | ||||
|         return kwargs | ||||
|  | ||||
| @@ -223,7 +217,7 @@ class ClubPageEditView(ClubTabsMixin, PageEditViewBase): | ||||
|         self.club = get_object_or_404(Club, pk=kwargs["club_id"]) | ||||
|         if not self.club.page: | ||||
|             raise Http404 | ||||
|         return super(ClubPageEditView, self).dispatch(request, *args, **kwargs) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get_object(self): | ||||
|         self.page = self.club.page | ||||
| @@ -234,9 +228,7 @@ class ClubPageEditView(ClubTabsMixin, PageEditViewBase): | ||||
|  | ||||
|  | ||||
| class ClubPageHistView(ClubTabsMixin, CanViewMixin, DetailView): | ||||
|     """ | ||||
|     Modification hostory of the page | ||||
|     """ | ||||
|     """Modification hostory of the page.""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
| @@ -245,9 +237,7 @@ class ClubPageHistView(ClubTabsMixin, CanViewMixin, DetailView): | ||||
|  | ||||
|  | ||||
| class ClubToolsView(ClubTabsMixin, CanEditMixin, DetailView): | ||||
|     """ | ||||
|     Tools page of a Club | ||||
|     """ | ||||
|     """Tools page of a Club.""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
| @@ -256,9 +246,7 @@ class ClubToolsView(ClubTabsMixin, CanEditMixin, DetailView): | ||||
|  | ||||
|  | ||||
| class ClubMembersView(ClubTabsMixin, CanViewMixin, DetailFormView): | ||||
|     """ | ||||
|     View of a club's members | ||||
|     """ | ||||
|     """View of a club's members.""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
| @@ -267,22 +255,20 @@ class ClubMembersView(ClubTabsMixin, CanViewMixin, DetailFormView): | ||||
|     current_tab = "members" | ||||
|  | ||||
|     def get_form_kwargs(self): | ||||
|         kwargs = super(ClubMembersView, self).get_form_kwargs() | ||||
|         kwargs = super().get_form_kwargs() | ||||
|         kwargs["request_user"] = self.request.user | ||||
|         kwargs["club"] = self.get_object() | ||||
|         kwargs["club_members"] = self.members | ||||
|         return kwargs | ||||
|  | ||||
|     def get_context_data(self, *args, **kwargs): | ||||
|         kwargs = super(ClubMembersView, self).get_context_data(*args, **kwargs) | ||||
|         kwargs = super().get_context_data(*args, **kwargs) | ||||
|         kwargs["members"] = self.members | ||||
|         return kwargs | ||||
|  | ||||
|     def form_valid(self, form): | ||||
|         """ | ||||
|             Check user rights | ||||
|         """ | ||||
|         resp = super(ClubMembersView, self).form_valid(form) | ||||
|         """Check user rights.""" | ||||
|         resp = super().form_valid(form) | ||||
|  | ||||
|         data = form.clean() | ||||
|         users = data.pop("users", []) | ||||
| @@ -296,10 +282,8 @@ class ClubMembersView(ClubTabsMixin, CanViewMixin, DetailFormView): | ||||
|         return resp | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         self.members = ( | ||||
|             self.get_object().members.filter(end_date=None).order_by("-role").all() | ||||
|         ) | ||||
|         return super(ClubMembersView, self).dispatch(request, *args, **kwargs) | ||||
|         self.members = self.get_object().members.ongoing().order_by("-role") | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get_success_url(self, **kwargs): | ||||
|         return reverse_lazy( | ||||
| @@ -308,9 +292,7 @@ class ClubMembersView(ClubTabsMixin, CanViewMixin, DetailFormView): | ||||
|  | ||||
|  | ||||
| class ClubOldMembersView(ClubTabsMixin, CanViewMixin, DetailView): | ||||
|     """ | ||||
|     Old members of a club | ||||
|     """ | ||||
|     """Old members of a club.""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
| @@ -318,30 +300,42 @@ class ClubOldMembersView(ClubTabsMixin, CanViewMixin, DetailView): | ||||
|     current_tab = "elderlies" | ||||
|  | ||||
|  | ||||
| class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailView): | ||||
|     """ | ||||
|     Sellings of a club | ||||
|     """ | ||||
| class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|     """Sellings of a club.""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
|     template_name = "club/club_sellings.jinja" | ||||
|     current_tab = "sellings" | ||||
|     form_class = SellingsForm | ||||
|     paginate_by = 70 | ||||
|  | ||||
|     def get_form_class(self): | ||||
|         kwargs = { | ||||
|             "product": forms.ModelChoiceField( | ||||
|                 self.object.products.order_by("name").all(), | ||||
|                 label=_("Product"), | ||||
|                 required=False, | ||||
|             ) | ||||
|         } | ||||
|         return type("SellingsForm", (SellingsFormBase,), kwargs) | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         try: | ||||
|             self.asked_page = int(request.GET.get("page", 1)) | ||||
|         except ValueError as e: | ||||
|             raise Http404 from e | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get_form_kwargs(self): | ||||
|         kwargs = super().get_form_kwargs() | ||||
|         kwargs["club"] = self.object | ||||
|         return kwargs | ||||
|  | ||||
|     def post(self, request, *args, **kwargs): | ||||
|         return self.get(request, *args, **kwargs) | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super(ClubSellingView, self).get_context_data(**kwargs) | ||||
|         form = self.get_form_class()(self.request.GET) | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         qs = Selling.objects.filter(club=self.object) | ||||
|  | ||||
|         kwargs["result"] = qs[:0] | ||||
|         kwargs["paginated_result"] = kwargs["result"] | ||||
|         kwargs["total"] = 0 | ||||
|         kwargs["total_quantity"] = 0 | ||||
|         kwargs["benefit"] = 0 | ||||
|  | ||||
|         form = self.get_form() | ||||
|         if form.is_valid(): | ||||
|             if not len([v for v in form.cleaned_data.values() if v is not None]): | ||||
|                 qs = Selling.objects.filter(id=-1) | ||||
| @@ -349,37 +343,82 @@ class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailView): | ||||
|                 qs = qs.filter(date__gte=form.cleaned_data["begin_date"]) | ||||
|             if form.cleaned_data["end_date"]: | ||||
|                 qs = qs.filter(date__lte=form.cleaned_data["end_date"]) | ||||
|             if form.cleaned_data["counter"]: | ||||
|                 qs = qs.filter(counter=form.cleaned_data["counter"]) | ||||
|             if form.cleaned_data["product"]: | ||||
|                 qs = qs.filter(product__id=form.cleaned_data["product"].id) | ||||
|  | ||||
|             if form.cleaned_data["counters"]: | ||||
|                 qs = qs.filter(counter__in=form.cleaned_data["counters"]) | ||||
|  | ||||
|             selected_products = [] | ||||
|             if form.cleaned_data["products"]: | ||||
|                 selected_products.extend(form.cleaned_data["products"]) | ||||
|             if form.cleaned_data["archived_products"]: | ||||
|                 selected_products.extend(form.cleaned_data["archived_products"]) | ||||
|  | ||||
|             if len(selected_products) > 0: | ||||
|                 qs = qs.filter(product__in=selected_products) | ||||
|  | ||||
|             kwargs["result"] = qs.all().order_by("-id") | ||||
|             kwargs["total"] = sum([s.quantity * s.unit_price for s in qs.all()]) | ||||
|             kwargs["total_quantity"] = sum([s.quantity for s in qs.all()]) | ||||
|             kwargs["benefit"] = kwargs["total"] - sum( | ||||
|                 [s.product.purchase_price for s in qs.exclude(product=None)] | ||||
|             kwargs["total"] = sum([s.quantity * s.unit_price for s in kwargs["result"]]) | ||||
|             total_quantity = qs.all().aggregate(Sum("quantity")) | ||||
|             if total_quantity["quantity__sum"]: | ||||
|                 kwargs["total_quantity"] = total_quantity["quantity__sum"] | ||||
|             benefit = ( | ||||
|                 qs.exclude(product=None).all().aggregate(Sum("product__purchase_price")) | ||||
|             ) | ||||
|         else: | ||||
|             kwargs["result"] = qs[:0] | ||||
|         kwargs["form"] = form | ||||
|             if benefit["product__purchase_price__sum"]: | ||||
|                 kwargs["benefit"] = benefit["product__purchase_price__sum"] | ||||
|  | ||||
|         kwargs["paginator"] = Paginator(kwargs["result"], self.paginate_by) | ||||
|         try: | ||||
|             kwargs["paginated_result"] = kwargs["paginator"].page(self.asked_page) | ||||
|         except InvalidPage as e: | ||||
|             raise Http404 from e | ||||
|  | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class ClubSellingCSVView(ClubSellingView): | ||||
|     """ | ||||
|     Generate sellings in csv for a given period | ||||
|     """ | ||||
|     """Generate sellings in csv for a given period.""" | ||||
|  | ||||
|     class StreamWriter: | ||||
|         """Implements a file-like interface for streaming the CSV.""" | ||||
|  | ||||
|         def write(self, value): | ||||
|             """Write the value by returning it, instead of storing in a buffer.""" | ||||
|             return value | ||||
|  | ||||
|     def write_selling(self, selling): | ||||
|         row = [selling.date, selling.counter] | ||||
|         if selling.seller: | ||||
|             row.append(selling.seller.get_display_name()) | ||||
|         else: | ||||
|             row.append("") | ||||
|         if selling.customer: | ||||
|             row.append(selling.customer.user.get_display_name()) | ||||
|         else: | ||||
|             row.append("") | ||||
|         row = [ | ||||
|             *row, | ||||
|             selling.label, | ||||
|             selling.quantity, | ||||
|             selling.quantity * selling.unit_price, | ||||
|             selling.get_payment_method_display(), | ||||
|         ] | ||||
|         if selling.product: | ||||
|             row.append(selling.product.selling_price) | ||||
|             row.append(selling.product.purchase_price) | ||||
|             row.append(selling.product.selling_price - selling.product.purchase_price) | ||||
|         else: | ||||
|             row = [*row, "", "", ""] | ||||
|         return row | ||||
|  | ||||
|     def get(self, request, *args, **kwargs): | ||||
|         import csv | ||||
|  | ||||
|         response = HttpResponse(content_type="text/csv") | ||||
|         self.object = self.get_object() | ||||
|         name = _("Sellings") + "_" + self.object.name + ".csv" | ||||
|         response["Content-Disposition"] = "filename=" + name | ||||
|         kwargs = self.get_context_data(**kwargs) | ||||
|  | ||||
|         # Use the StreamWriter class instead of request for streaming | ||||
|         pseudo_buffer = self.StreamWriter() | ||||
|         writer = csv.writer( | ||||
|             response, delimiter=";", lineterminator="\n", quoting=csv.QUOTE_ALL | ||||
|             pseudo_buffer, delimiter=";", lineterminator="\n", quoting=csv.QUOTE_ALL | ||||
|         ) | ||||
|  | ||||
|         writer.writerow([_t("Quantity"), kwargs["total_quantity"]]) | ||||
| @@ -400,37 +439,23 @@ class ClubSellingCSVView(ClubSellingView): | ||||
|                 _t("Benefit"), | ||||
|             ] | ||||
|         ) | ||||
|         for o in kwargs["result"]: | ||||
|             row = [o.date, o.counter] | ||||
|             if o.seller: | ||||
|                 row.append(o.seller.get_display_name()) | ||||
|             else: | ||||
|                 row.append("") | ||||
|             if o.customer: | ||||
|                 row.append(o.customer.user.get_display_name()) | ||||
|             else: | ||||
|                 row.append("") | ||||
|             row = row + [ | ||||
|                 o.label, | ||||
|                 o.quantity, | ||||
|                 o.quantity * o.unit_price, | ||||
|                 o.get_payment_method_display(), | ||||
|             ] | ||||
|             if o.product: | ||||
|                 row.append(o.product.selling_price) | ||||
|                 row.append(o.product.purchase_price) | ||||
|                 row.append(o.product.selling_price - o.product.purchase_price) | ||||
|             else: | ||||
|                 row = row + ["", "", ""] | ||||
|             writer.writerow(row) | ||||
|  | ||||
|         # Stream response | ||||
|         response = StreamingHttpResponse( | ||||
|             ( | ||||
|                 writer.writerow(self.write_selling(selling)) | ||||
|                 for selling in kwargs["result"] | ||||
|             ), | ||||
|             content_type="text/csv", | ||||
|         ) | ||||
|         name = _("Sellings") + "_" + self.object.name + ".csv" | ||||
|         response["Content-Disposition"] = "filename=" + name | ||||
|  | ||||
|         return response | ||||
|  | ||||
|  | ||||
| class ClubEditView(ClubTabsMixin, CanEditMixin, UpdateView): | ||||
|     """ | ||||
|     Edit a Club's main informations (for the club's members) | ||||
|     """ | ||||
|     """Edit a Club's main informations (for the club's members).""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
| @@ -440,9 +465,7 @@ class ClubEditView(ClubTabsMixin, CanEditMixin, UpdateView): | ||||
|  | ||||
|  | ||||
| class ClubEditPropView(ClubTabsMixin, CanEditPropMixin, UpdateView): | ||||
|     """ | ||||
|     Edit the properties of a Club object (for the Sith admins) | ||||
|     """ | ||||
|     """Edit the properties of a Club object (for the Sith admins).""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
| @@ -451,10 +474,8 @@ class ClubEditPropView(ClubTabsMixin, CanEditPropMixin, UpdateView): | ||||
|     current_tab = "props" | ||||
|  | ||||
|  | ||||
| class ClubCreateView(CanEditPropMixin, CreateView): | ||||
|     """ | ||||
|     Create a club (for the Sith admin) | ||||
|     """ | ||||
| class ClubCreateView(CanCreateMixin, CreateView): | ||||
|     """Create a club (for the Sith admin).""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
| @@ -463,9 +484,7 @@ class ClubCreateView(CanEditPropMixin, CreateView): | ||||
|  | ||||
|  | ||||
| class MembershipSetOldView(CanEditMixin, DetailView): | ||||
|     """ | ||||
|     Set a membership as beeing old | ||||
|     """ | ||||
|     """Set a membership as beeing old.""" | ||||
|  | ||||
|     model = Membership | ||||
|     pk_url_kwarg = "membership_id" | ||||
| @@ -493,19 +512,28 @@ class MembershipSetOldView(CanEditMixin, DetailView): | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class MembershipDeleteView(UserIsRootMixin, DeleteView): | ||||
|     """Delete a membership (for admins only).""" | ||||
|  | ||||
|     model = Membership | ||||
|     pk_url_kwarg = "membership_id" | ||||
|     template_name = "core/delete_confirm.jinja" | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("core:user_clubs", kwargs={"user_id": self.object.user.id}) | ||||
|  | ||||
|  | ||||
| class ClubStatView(TemplateView): | ||||
|     template_name = "club/stats.jinja" | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super(ClubStatView, self).get_context_data(**kwargs) | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["club_list"] = Club.objects.all() | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class ClubMailingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|     """ | ||||
|     A list of mailing for a given club | ||||
|     """ | ||||
|     """A list of mailing for a given club.""" | ||||
|  | ||||
|     model = Club | ||||
|     form_class = MailingForm | ||||
| @@ -514,7 +542,7 @@ class ClubMailingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|     current_tab = "mailing" | ||||
|  | ||||
|     def get_form_kwargs(self): | ||||
|         kwargs = super(ClubMailingView, self).get_form_kwargs() | ||||
|         kwargs = super().get_form_kwargs() | ||||
|         kwargs["club_id"] = self.get_object().id | ||||
|         kwargs["user_id"] = self.request.user.id | ||||
|         kwargs["mailings"] = self.mailings | ||||
| @@ -522,10 +550,10 @@ class ClubMailingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         self.mailings = Mailing.objects.filter(club_id=self.get_object().id).all() | ||||
|         return super(ClubMailingView, self).dispatch(request, *args, **kwargs) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super(ClubMailingView, self).get_context_data(**kwargs) | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["club"] = self.get_object() | ||||
|         kwargs["user"] = self.request.user | ||||
|         kwargs["mailings"] = self.mailings | ||||
| @@ -542,10 +570,8 @@ class ClubMailingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|         } | ||||
|         return kwargs | ||||
|  | ||||
|     def add_new_mailing(self, cleaned_data) -> ValidationError: | ||||
|         """ | ||||
|         Create a new mailing list from the form | ||||
|         """ | ||||
|     def add_new_mailing(self, cleaned_data) -> ValidationError | None: | ||||
|         """Create a new mailing list from the form.""" | ||||
|         mailing = Mailing( | ||||
|             club=self.get_object(), | ||||
|             email=cleaned_data["mailing_email"], | ||||
| @@ -559,10 +585,8 @@ class ClubMailingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|         mailing.save() | ||||
|         return None | ||||
|  | ||||
|     def add_new_subscription(self, cleaned_data) -> ValidationError: | ||||
|         """ | ||||
|         Add mailing subscriptions for each user given and/or for the specified email in form | ||||
|         """ | ||||
|     def add_new_subscription(self, cleaned_data) -> ValidationError | None: | ||||
|         """Add mailing subscriptions for each user given and/or for the specified email in form.""" | ||||
|         users_to_save = [] | ||||
|  | ||||
|         for user in cleaned_data["subscription_users"]: | ||||
| @@ -574,7 +598,8 @@ class ClubMailingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|             except ValidationError as validation_error: | ||||
|                 return validation_error | ||||
|  | ||||
|             users_to_save.append(sub.save()) | ||||
|             sub.save() | ||||
|             users_to_save.append(sub) | ||||
|  | ||||
|         if cleaned_data["subscription_email"]: | ||||
|             sub = MailingSubscription( | ||||
| @@ -595,20 +620,16 @@ class ClubMailingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|         return None | ||||
|  | ||||
|     def remove_subscription(self, cleaned_data): | ||||
|         """ | ||||
|         Remove specified users from a mailing list | ||||
|         """ | ||||
|         """Remove specified users from a mailing list.""" | ||||
|         fields = [ | ||||
|             cleaned_data[key] | ||||
|             for key in cleaned_data.keys() | ||||
|             if key.startswith("removal_") | ||||
|             val for key, val in cleaned_data.items() if key.startswith("removal_") | ||||
|         ] | ||||
|         for field in fields: | ||||
|             for sub in field: | ||||
|                 sub.delete() | ||||
|  | ||||
|     def form_valid(self, form): | ||||
|         resp = super(ClubMailingView, self).form_valid(form) | ||||
|         resp = super().form_valid(form) | ||||
|  | ||||
|         cleaned_data = form.clean() | ||||
|         error = None | ||||
| @@ -633,7 +654,6 @@ class ClubMailingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|  | ||||
|  | ||||
| class MailingDeleteView(CanEditMixin, DeleteView): | ||||
|  | ||||
|     model = Mailing | ||||
|     template_name = "core/delete_confirm.jinja" | ||||
|     pk_url_kwarg = "mailing_id" | ||||
| @@ -641,7 +661,7 @@ class MailingDeleteView(CanEditMixin, DeleteView): | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         self.club_id = self.get_object().club.id | ||||
|         return super(MailingDeleteView, self).dispatch(request, *args, **kwargs) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get_success_url(self, **kwargs): | ||||
|         if self.redirect_page: | ||||
| @@ -651,16 +671,13 @@ class MailingDeleteView(CanEditMixin, DeleteView): | ||||
|  | ||||
|  | ||||
| class MailingSubscriptionDeleteView(CanEditMixin, DeleteView): | ||||
|  | ||||
|     model = MailingSubscription | ||||
|     template_name = "core/delete_confirm.jinja" | ||||
|     pk_url_kwarg = "mailing_subscription_id" | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         self.club_id = self.get_object().mailing.club.id | ||||
|         return super(MailingSubscriptionDeleteView, self).dispatch( | ||||
|             request, *args, **kwargs | ||||
|         ) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get_success_url(self, **kwargs): | ||||
|         return reverse_lazy("club:mailing", kwargs={"club_id": self.club_id}) | ||||
| @@ -671,7 +688,7 @@ class MailingAutoGenerationView(View): | ||||
|         self.mailing = get_object_or_404(Mailing, pk=kwargs["mailing_id"]) | ||||
|         if not request.user.can_edit(self.mailing): | ||||
|             raise PermissionDenied | ||||
|         return super(MailingAutoGenerationView, self).dispatch(request, *args, **kwargs) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get(self, request, *args, **kwargs): | ||||
|         club = self.mailing.club | ||||
| @@ -685,25 +702,25 @@ class MailingAutoGenerationView(View): | ||||
|  | ||||
|  | ||||
| class PosterListView(ClubTabsMixin, PosterListBaseView, CanViewMixin): | ||||
|     """List communication posters""" | ||||
|     """List communication posters.""" | ||||
|  | ||||
|     def get_object(self): | ||||
|         return self.club | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super(PosterListView, self).get_context_data(**kwargs) | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["app"] = "club" | ||||
|         kwargs["club"] = self.club | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class PosterCreateView(PosterCreateBaseView, CanCreateMixin): | ||||
|     """Create communication poster""" | ||||
|     """Create communication poster.""" | ||||
|  | ||||
|     pk_url_kwarg = "club_id" | ||||
|  | ||||
|     def get_object(self): | ||||
|         obj = super(PosterCreateView, self).get_object() | ||||
|         obj = super().get_object() | ||||
|         if not obj: | ||||
|             return self.club | ||||
|         return obj | ||||
| @@ -713,19 +730,19 @@ class PosterCreateView(PosterCreateBaseView, CanCreateMixin): | ||||
|  | ||||
|  | ||||
| class PosterEditView(ClubTabsMixin, PosterEditBaseView, CanEditMixin): | ||||
|     """Edit communication poster""" | ||||
|     """Edit communication poster.""" | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("club:poster_list", kwargs={"club_id": self.club.id}) | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super(PosterEditView, self).get_context_data(**kwargs) | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["app"] = "club" | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class PosterDeleteView(PosterDeleteBaseView, ClubTabsMixin, CanEditMixin): | ||||
|     """Delete communication poster""" | ||||
|     """Delete communication poster.""" | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("club:poster_list", kwargs={"club_id": self.club.id}) | ||||
|   | ||||
							
								
								
									
										23
									
								
								club/widgets/select.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								club/widgets/select.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| from pydantic import TypeAdapter | ||||
|  | ||||
| from club.models import Club | ||||
| from club.schemas import ClubSchema | ||||
| from core.views.widgets.select import AutoCompleteSelect, AutoCompleteSelectMultiple | ||||
|  | ||||
| _js = ["bundled/club/components/ajax-select-index.ts"] | ||||
|  | ||||
|  | ||||
| class AutoCompleteSelectClub(AutoCompleteSelect): | ||||
|     component_name = "club-ajax-select" | ||||
|     model = Club | ||||
|     adapter = TypeAdapter(list[ClubSchema]) | ||||
|  | ||||
|     js = _js | ||||
|  | ||||
|  | ||||
| class AutoCompleteSelectMultipleClub(AutoCompleteSelectMultiple): | ||||
|     component_name = "club-ajax-select" | ||||
|     model = Club | ||||
|     adapter = TypeAdapter(list[ClubSchema]) | ||||
|  | ||||
|     js = _js | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user