mirror of
https://gitlab.com/allianceauth/allianceauth.git
synced 2025-07-09 20:40:17 +02:00
Compare commits
713 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
34b94ae685 | ||
|
50fd900bdc | ||
|
1bf8ec5bc6 | ||
|
f849b75029 | ||
|
25c27793fe | ||
|
6e413772ad | ||
|
137a202e1b | ||
|
aaf718fe4d | ||
|
a193d9959b | ||
|
12250ef0c2 | ||
|
bde9802583 | ||
|
1b30b86d2b | ||
|
0707b9b98c | ||
|
b22a379db2 | ||
|
bb2e0aabbc | ||
|
449991d846 | ||
|
dd42c2b074 | ||
|
abff1b0add | ||
|
fc51f6bea2 | ||
|
6477c22308 | ||
|
329b3fecfb | ||
|
677505f22a | ||
|
f518166bd0 | ||
|
1f4c49f823 | ||
|
abcc4d47b5 | ||
|
3d4737df72 | ||
|
8f94885d8e | ||
|
993455d664 | ||
|
3cb0addee7 | ||
|
5530b76294 | ||
|
9fb51165ab | ||
|
a650f0730e | ||
|
63eb9edc9c | ||
|
d6e1eb9792 | ||
|
b459f96e6b | ||
|
bf32f2c1ef | ||
|
7ca67ebaae | ||
|
fa32f87a35 | ||
|
a630015451 | ||
|
bf43f59232 | ||
|
54910746e3 | ||
|
07ae68333d | ||
|
69e70a4c9b | ||
|
b55f11ee74 | ||
|
94ee3c0203 | ||
|
25cf329a50 | ||
|
b02827cb3f | ||
|
2bcc0570ad | ||
|
a3ea0c65a1 | ||
|
5e526da11c | ||
|
5c79265f90 | ||
|
eb0134e716 | ||
|
afde1f4729 | ||
|
5c6dda0eac | ||
|
af453bc772 | ||
|
e13674e886 | ||
|
e3e856b826 | ||
|
9d1cd23a8f | ||
|
148f7c116f | ||
|
33e7134d6f | ||
|
fb799551aa | ||
|
7b95051fe1 | ||
|
efb6a6db4f | ||
|
478aa1aa12 | ||
|
751e55ed6c | ||
|
9dad53f763 | ||
|
2d57064a7a | ||
|
833d12cf66 | ||
|
7b56caa4cb | ||
|
5752644122 | ||
|
cadc0cb534 | ||
|
dcdab5ae1f | ||
|
d64e896288 | ||
|
500d8ede32 | ||
|
f4c5c7f6db | ||
|
43e1be4032 | ||
|
702def2a4d | ||
|
a34baf4154 | ||
|
4de0774f15 | ||
|
83d2dfc7d9 | ||
|
c93afd2d68 | ||
|
b7bacd11af | ||
|
f26835fae0 | ||
|
4edb7cb678 | ||
|
2ce9ba997f | ||
|
055077fa77 | ||
|
f342ccbc6a | ||
|
37ffd0a1ac | ||
|
a1f705381e | ||
|
c0970ad4fa | ||
|
3818d0c6d1 | ||
|
95411c79cb | ||
|
eeccbbacfc | ||
|
f6c4180502 | ||
|
f4d3d6c0b1 | ||
|
e9d2d11297 | ||
|
96204b29e8 | ||
|
47842c1243 | ||
|
9b494106bc | ||
|
d51e730a7f | ||
|
363909c0c4 | ||
|
82273f68fe | ||
|
12fa38b446 | ||
|
c26af593ff | ||
|
8e9a53c494 | ||
|
5559ce5fbb | ||
|
faa529a55b | ||
|
4ccfe20c14 | ||
|
960c9625fe | ||
|
7b92d103d6 | ||
|
c1e2449084 | ||
|
3acb651650 | ||
|
2de57b334b | ||
|
0498f5bb1b | ||
|
929485a8f9 | ||
|
28cb62f373 | ||
|
02214b74d0 | ||
|
f497c18e5b | ||
|
cb57d922e6 | ||
|
805d138b09 | ||
|
09a583fb1d | ||
|
146c4c8d94 | ||
|
c2ae680f72 | ||
|
b5ad1c8a1a | ||
|
8be2760fc4 | ||
|
f047943eb7 | ||
|
43906f41b3 | ||
|
a18ec98877 | ||
|
14163d2c0c | ||
|
81d9c41cf6 | ||
|
58f5a5b41d | ||
|
6363bb706a | ||
|
baf3be4cb2 | ||
|
e69444fe79 | ||
|
7483fcb876 | ||
|
a57d55504d | ||
|
affb30e9f4 | ||
|
588cb1b7ca | ||
|
a609faa91b | ||
|
856e939c21 | ||
|
1b6cf98885 | ||
|
92c2af9975 | ||
|
5ef70bb031 | ||
|
60998bffc2 | ||
|
a5971314f5 | ||
|
a03c766840 | ||
|
ad47ff2c54 | ||
|
3efdb8f12b | ||
|
823fc82d19 | ||
|
a93e510895 | ||
|
d99f5858d8 | ||
|
4578ecf21d | ||
|
b737504d52 | ||
|
c6b6443901 | ||
|
f51523dc07 | ||
|
bd4dd60c98 | ||
|
a4ea48e14e | ||
|
646d3f5408 | ||
|
0f057ffa84 | ||
|
7033406ba6 | ||
|
6b395ca1d4 | ||
|
795a7e006f | ||
|
2a894cd62c | ||
|
9ada26e849 | ||
|
7120b3956c | ||
|
4da67cfaf6 | ||
|
0a940810bd | ||
|
a868438492 | ||
|
dc1ed8c570 | ||
|
8489f204dd | ||
|
1478588016 | ||
|
a16eb4b7f7 | ||
|
292fb7b29d | ||
|
c6890dd2c6 | ||
|
702564d15e | ||
|
cef2e86ea1 | ||
|
50681b023b | ||
|
2822775fb8 | ||
|
ef7c8be7b5 | ||
|
d639617eba | ||
|
2125192f72 | ||
|
8d63801b00 | ||
|
e053fb7d96 | ||
|
ae7ed5c297 | ||
|
d624ba4427 | ||
|
164cd4fbb2 | ||
|
94b52c850e | ||
|
4d19ceb388 | ||
|
9a64728311 | ||
|
dcc0c78daf | ||
|
5507c30af4 | ||
|
66b97835d4 | ||
|
fd66a7cb20 | ||
|
c0f0f8db73 | ||
|
4210b2eabc | ||
|
225e68647e | ||
|
7e2f864ebf | ||
|
f2384ba45b | ||
|
c5918b9b3c | ||
|
ffedc4103d | ||
|
0467b23a1a | ||
|
cda5ce739f | ||
|
e5c8426ea3 | ||
|
b2bd489ddc | ||
|
6397cf358a | ||
|
90fcc4a811 | ||
|
ec7472fe22 | ||
|
5148b1914d | ||
|
2e0716f5ae | ||
|
1fb091acb2 | ||
|
07c62ed32a | ||
|
660fe79d08 | ||
|
42ee06470c | ||
|
69aaa9652f | ||
|
1ccfff50e5 | ||
|
57a39557fd | ||
|
c36dea08e3 | ||
|
d3acd821b7 | ||
|
0a17427169 | ||
|
ce8935e621 | ||
|
efff946a56 | ||
|
1dea92ed76 | ||
|
3f54d49d8b | ||
|
77da6928b2 | ||
|
bd17b95cac | ||
|
4ed1c5b7c4 | ||
|
271fd8e8c4 | ||
|
9b4321281a | ||
|
052c35c8e5 | ||
|
0fcb517b0b | ||
|
dcfddf0add | ||
|
4a4258d0e6 | ||
|
dd15a221aa | ||
|
737e02293a | ||
|
c34efebacf | ||
|
4602097399 | ||
|
7051e06564 | ||
|
9767ce79d8 | ||
|
0c090f1486 | ||
|
618ee81f9b | ||
|
98efb9f887 | ||
|
cbe6c821cc | ||
|
de9d2b39a6 | ||
|
0d5f22288b | ||
|
e0d76dc268 | ||
|
ecc9e68330 | ||
|
710149ec21 | ||
|
3c2c137dad | ||
|
a8271c4189 | ||
|
3315ae7778 | ||
|
d2f048f8fe | ||
|
0fe2855faa | ||
|
79a1fa3d7c | ||
|
96fe88d5c7 | ||
|
59391ad3c5 | ||
|
94e9c08422 | ||
|
acff3695bc | ||
|
43ec8514aa | ||
|
4c629b193f | ||
|
c651da4011 | ||
|
da382cffd1 | ||
|
4ecfc3afd8 | ||
|
4eb7dbbe62 | ||
|
c96ba65296 | ||
|
ff2f60f7f3 | ||
|
3000545c98 | ||
|
f3ad092ef2 | ||
|
a012e7df2f | ||
|
1fa77412c0 | ||
|
e56caeb22b | ||
|
ceb07ebc67 | ||
|
237075d45c | ||
|
7099b1946d | ||
|
e416ab8ff2 | ||
|
2802ed03a5 | ||
|
4af73c76fe | ||
|
b6149979aa | ||
|
cb20288427 | ||
|
db6f4c91dc | ||
|
57ac7a5277 | ||
|
136438f9c2 | ||
|
e2be8b3440 | ||
|
04f3473ef3 | ||
|
255cb0da8d | ||
|
069352fb0f | ||
|
66e8ddb684 | ||
|
179c26975c | ||
|
e17f6e799b | ||
|
7cd8294104 | ||
|
ede5540335 | ||
|
747279b773 | ||
|
44f8b1c477 | ||
|
7c6ebd9bf6 | ||
|
430469b708 | ||
|
efbb3cee31 | ||
|
21094ed4dd | ||
|
5f326efc7e | ||
|
b6e34ace35 | ||
|
fe4a8965e3 | ||
|
23371c233d | ||
|
7a3bbf0d7f | ||
|
89a1bec9c1 | ||
|
1c1e70619a | ||
|
0ff4374efa | ||
|
18d0e58a48 | ||
|
84f44338dc | ||
|
2ba0412890 | ||
|
2326522b29 | ||
|
a7cb6ee434 | ||
|
2aeef63565 | ||
|
3c9e7335ef | ||
|
49067de325 | ||
|
471e7e29ae | ||
|
3ec5775406 | ||
|
e804d2b60d | ||
|
742438a95d | ||
|
5c60086baa | ||
|
e49041bb14 | ||
|
f3cbe91883 | ||
|
ea439a2176 | ||
|
56e1e76f11 | ||
|
634e7357be | ||
|
08dc88da1a | ||
|
3d206e445c | ||
|
64686cdad1 | ||
|
d7fe09bdf1 | ||
|
6da50da92f | ||
|
51e4dd986f | ||
|
bee6522182 | ||
|
1711a9dd33 | ||
|
3914626379 | ||
|
df276cb32d | ||
|
daad7d8b10 | ||
|
3bf5bc0fe3 | ||
|
96abae553a | ||
|
f9cbfb1562 | ||
|
8eaa94e179 | ||
|
4f876b648b | ||
|
cd738137c0 | ||
|
5605eb129d | ||
|
87ef0f21a3 | ||
|
a1c7ce827e | ||
|
97466bcdfb | ||
|
ff3096b106 | ||
|
98f0d77f3f | ||
|
92548ba402 | ||
|
c46741d311 | ||
|
7c7c1abf7c | ||
|
fc303b1b0a | ||
|
4e220a9679 | ||
|
b17b1f7504 | ||
|
7081fc0e76 | ||
|
68e4574f19 | ||
|
e6e0a70012 | ||
|
13e38da942 | ||
|
468c1de26b | ||
|
22ef5ac0e5 | ||
|
ef2dc08958 | ||
|
6b84ffa16c | ||
|
d7a1096413 | ||
|
93b94a8bc2 | ||
|
9a95716105 | ||
|
dbfcf5d87a | ||
|
105d7d53b3 | ||
|
01cefe1457 | ||
|
7fe3db8017 | ||
|
e0945fac80 | ||
|
40fa190820 | ||
|
670580f8f3 | ||
|
323a0bcf16 | ||
|
6e995edd80 | ||
|
8d86e45b7a | ||
|
2aa6df4461 | ||
|
cf6f989502 | ||
|
3e1d8ae334 | ||
|
bcfe9484b5 | ||
|
5e4d1b9cfd | ||
|
3b463e7305 | ||
|
eedf5082fa | ||
|
2ea5b15175 | ||
|
7a9808aad3 | ||
|
a1d712694c | ||
|
ca11c726a3 | ||
|
6e0f7a35bd | ||
|
7375b001ca | ||
|
0287086633 | ||
|
9eb2becbb5 | ||
|
12f1444fe7 | ||
|
d6372bd093 | ||
|
3935a9cdd2 | ||
|
49fb6c39d5 | ||
|
8821f18b21 | ||
|
f4d8ead54e | ||
|
7427e13505 | ||
|
445683c3d5 | ||
|
677c46c48a | ||
|
87b9e3f87a | ||
|
da2a5aff2f | ||
|
65d77743dc | ||
|
1c7f8256d0 | ||
|
61f0aae5d9 | ||
|
4d56030edf | ||
|
2dfe194a3b | ||
|
ebb40deb7f | ||
|
a3a2d3d35b | ||
|
d4dee519b8 | ||
|
dfcbad3476 | ||
|
e03c1307a3 | ||
|
054ef27fa4 | ||
|
97e224b8e6 | ||
|
3b8fa415bc | ||
|
b94fd7ed19 | ||
|
d1dac61135 | ||
|
d2a095217f | ||
|
3a95b89779 | ||
|
4f5b231bdf | ||
|
40c0b8d862 | ||
|
62c936f1c0 | ||
|
2a762df9b3 | ||
|
8fb5a488f7 | ||
|
dc239d5396 | ||
|
2815ebaa07 | ||
|
34dd4802a8 | ||
|
3ebf11308c | ||
|
5cd5180a91 | ||
|
ba213db493 | ||
|
8362d11714 | ||
|
7ba65968ed | ||
|
7fdbac20cb | ||
|
76efcb5266 | ||
|
6aacb8c2e3 | ||
|
465fba3a18 | ||
|
9ae6addc71 | ||
|
76ae9b8849 | ||
|
13a8b7678f | ||
|
c166fc0ef9 | ||
|
9da61588eb | ||
|
7425176b3f | ||
|
2bb518f7e0 | ||
|
84d5693583 | ||
|
867e2a1ded | ||
|
fd6c6991f5 | ||
|
333fe8497d | ||
|
8a3fb17147 | ||
|
f8baeb19a7 | ||
|
10096862e5 | ||
|
89193d2fcf | ||
|
fd08b987bd | ||
|
4e9e22cb4b | ||
|
f8fbbb5ba7 | ||
|
dd3ef41396 | ||
|
d5fda05dc9 | ||
|
d815fad0e6 | ||
|
e109198782 | ||
|
fbd4672454 | ||
|
a29bd567c2 | ||
|
960aef95ad | ||
|
4aae5497bb | ||
|
6081cbe900 | ||
|
5e9b47cf79 | ||
|
853826c140 | ||
|
ce0d8342e3 | ||
|
006785e592 | ||
|
df05070a55 | ||
|
e81450baf3 | ||
|
24b6c19aca | ||
|
9f4bf13cc9 | ||
|
2a156302f0 | ||
|
c4d3bde106 | ||
|
9c7de58989 | ||
|
0900806f68 | ||
|
a0d32d8c2d | ||
|
e16252842a | ||
|
950ae34093 | ||
|
0ba94c53aa | ||
|
96cc55d174 | ||
|
a9062c4389 | ||
|
51b86f88b9 | ||
|
8184461b48 | ||
|
ca0cdd6e15 | ||
|
036a17ad3b | ||
|
2418023ddd | ||
|
c602cf0b00 | ||
|
3de988369f | ||
|
b28b51916c | ||
|
0db0978d5f | ||
|
0531da1128 | ||
|
547d047f59 | ||
|
5cd92e29d3 | ||
|
bb44194cfc | ||
|
aa1cb96c8a | ||
|
40ae092306 | ||
|
b2e70c3702 | ||
|
eebeb26001 | ||
|
003a67224e | ||
|
d3a3810456 | ||
|
f55008559e | ||
|
dbc19c76c5 | ||
|
ced7972962 | ||
|
ad508bd880 | ||
|
5c128f2c78 | ||
|
1caaca86cf | ||
|
33ad1413d5 | ||
|
cbb5c80b94 | ||
|
d2edd288f9 | ||
|
21e80f6961 | ||
|
a747951d19 | ||
|
4cc7135ace | ||
|
6a990c11e6 | ||
|
3f47cecbfc | ||
|
77c126ea2f | ||
|
6e3219fd1b | ||
|
8aeb061635 | ||
|
84e2107b62 | ||
|
8b7e57494c | ||
|
20fcf5efa4 | ||
|
8ff3d854ba | ||
|
49ff355d50 | ||
|
c15b955d5e | ||
|
65e1545a66 | ||
|
c558a980e1 | ||
|
bd8ef84862 | ||
|
6f670da1db | ||
|
c558f3785f | ||
|
04cdd4c64f | ||
|
42e96d2f14 | ||
|
00fcebd8e3 | ||
|
995c84481c | ||
|
4dd42da993 | ||
|
8295cc51a3 | ||
|
f7e1d7c47e | ||
|
234191218a | ||
|
de12b49527 | ||
|
021b7b2edb | ||
|
20b959f273 | ||
|
e16bb2a737 | ||
|
0a6a6e0bf9 | ||
|
f0fe3929d4 | ||
|
9a62c729eb | ||
|
eaedcd5bb7 | ||
|
d742257c74 | ||
|
ad92d7a916 | ||
|
af22222bdf | ||
|
01f49bd125 | ||
|
af5930b9d0 | ||
|
95ba19a827 | ||
|
c75647301f | ||
|
adedf7534f | ||
|
d0684862fe | ||
|
2440e5d2b2 | ||
|
5f51ad4a6a | ||
|
0bd3acc230 | ||
|
11a32c90a8 | ||
|
6f0b853a60 | ||
|
583a6d4c7f | ||
|
155494afea | ||
|
b0aa58b910 | ||
|
1adce85422 | ||
|
cd47eadcdc | ||
|
247058a30f | ||
|
e0fa615e90 | ||
|
61ec67183c | ||
|
4369813478 | ||
|
8bb3d35252 | ||
|
398a980fb5 | ||
|
e14a295ce6 | ||
|
e0cd5c6fb9 | ||
|
c37ece49d5 | ||
|
8adab8bae0 | ||
|
a9c87bc25a | ||
|
a4901802c0 | ||
|
079dcc5d28 | ||
|
10c5a8906d | ||
|
e6306ea7aa | ||
|
20067c1133 | ||
|
dba3c651dc | ||
|
0b4d2b819b | ||
|
23a3dd1ab9 | ||
|
81e5bc5337 | ||
|
7eebf4d953 | ||
|
9e45d2eac7 | ||
|
e3017f1ec7 | ||
|
a8ef844fe7 | ||
|
12709b1b56 | ||
|
51ae604efd | ||
|
63071ec359 | ||
|
0032e4a01f | ||
|
25ab78a41e | ||
|
3aa0e323d2 | ||
|
cadbb7e61c | ||
|
115263eb5a | ||
|
18fec5f614 | ||
|
02ab064ec3 | ||
|
ba25f99cb4 | ||
|
28fd1b07ea | ||
|
5de19c43df | ||
|
9ce1939040 | ||
|
322131cd4f | ||
|
55e6e92da5 | ||
|
e5d29629a5 | ||
|
6a0ddc9a83 | ||
|
03be66d11f | ||
|
26e187e4c8 | ||
|
3480c4e0e8 | ||
|
1544f097e0 | ||
|
2477c31656 | ||
|
0dc631d69e | ||
|
2a9981cdb9 | ||
|
3d92008069 | ||
|
59e47c24c2 | ||
|
6d942555ff | ||
|
7e312bb95f | ||
|
c92fee78e2 | ||
|
004c48b8ad | ||
|
658a8cd6ce | ||
|
c1dc130766 | ||
|
35f5573b63 | ||
|
4d66b7d456 | ||
|
77e5747a23 | ||
|
a9ebecdec6 | ||
|
21f0a96422 | ||
|
9e47d19337 | ||
|
2c5972d0ab | ||
|
6118c0ddec | ||
|
ce25deeca1 | ||
|
60084de3db | ||
|
e16c68e255 | ||
|
bf14e9c7c3 | ||
|
98e91fe207 | ||
|
ee41d62c13 | ||
|
346b4014a9 | ||
|
9b56a441ed | ||
|
068bf1ae7a | ||
|
5be686e3ca | ||
|
a215b4411c | ||
|
e15cfa0fb1 | ||
|
46d51699f4 | ||
|
ff30a136d5 | ||
|
6dcf3304d5 | ||
|
beddeea338 | ||
|
69723937f7 | ||
|
c541f56ee2 | ||
|
7e887e5e34 | ||
|
072327c79f | ||
|
28af3ff11e | ||
|
e3b151f2fb | ||
|
f87d7dbdf8 | ||
|
a04e6ae3d0 | ||
|
15042f5e77 | ||
|
6e25361d5e | ||
|
9e639a0eeb | ||
|
257fbdef36 | ||
|
df003c8ec5 | ||
|
ba22685eb8 | ||
|
773288072a | ||
|
63afb13d25 | ||
|
5dd286bbe7 | ||
|
8aaa8172ca | ||
|
b68b401146 | ||
|
f7821b647f | ||
|
a6526d6f78 | ||
|
7898594909 | ||
|
cfd12ee3cc | ||
|
2c9177b19f | ||
|
abff26fb6e | ||
|
e8c3b5225c | ||
|
98fd1dcc4c | ||
|
7024552c4e | ||
|
a0719e4b86 | ||
|
906c589f14 | ||
|
cfe46e4ca5 | ||
|
4675193416 | ||
|
a84fa1ca69 | ||
|
8f6cb0b9bb | ||
|
1c8634f1c8 | ||
|
2a21599d45 | ||
|
ffb526ab0c | ||
|
b9d128259e | ||
|
13d866bd0d | ||
|
e379c01655 | ||
|
afa3d2e7cc | ||
|
e5ed33aeec | ||
|
b12471e775 | ||
|
ea1887b9ec | ||
|
d2f8c2a42f | ||
|
5e70dab11f | ||
|
f728c786b3 | ||
|
7056912d54 | ||
|
7efed950ca | ||
|
886acf2005 | ||
|
b2dec3bff2 | ||
|
f0a402e141 | ||
|
2e2afd7923 | ||
|
e9ea09bc56 | ||
|
186fa1be03 | ||
|
37d1d84fc3 | ||
|
ee24706e43 | ||
|
07e85727ea | ||
|
4912f0f8f0 | ||
|
424246df26 | ||
|
563e2210ef | ||
|
02a1078005 | ||
|
30107de44e | ||
|
200e8f2ff1 | ||
|
77a08cd218 | ||
|
e5a09027e5 | ||
|
24376262f0 | ||
|
efe0c6963b | ||
|
a4644028ae | ||
|
3a77b4a429 | ||
|
52b6c5d341 | ||
|
fa375a551c | ||
|
00a93e6fe9 |
@ -19,5 +19,6 @@ exclude_lines =
|
|||||||
if __name__ == .__main__.:
|
if __name__ == .__main__.:
|
||||||
def __repr__
|
def __repr__
|
||||||
raise AssertionError
|
raise AssertionError
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
|
||||||
ignore_errors = True
|
ignore_errors = True
|
||||||
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -73,3 +73,4 @@ celerybeat-schedule
|
|||||||
.flake8
|
.flake8
|
||||||
.pylintrc
|
.pylintrc
|
||||||
Makefile
|
Makefile
|
||||||
|
alliance_auth.sqlite3
|
||||||
|
@ -25,12 +25,12 @@ before_script:
|
|||||||
pre-commit-check:
|
pre-commit-check:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
stage: pre-commit
|
stage: pre-commit
|
||||||
image: python:3.11-bullseye
|
image: python:3.11-bookworm
|
||||||
variables:
|
# variables:
|
||||||
PRE_COMMIT_HOME: ${CI_PROJECT_DIR}/.cache/pre-commit
|
# PRE_COMMIT_HOME: ${CI_PROJECT_DIR}/.cache/pre-commit
|
||||||
cache:
|
# cache:
|
||||||
paths:
|
# paths:
|
||||||
- ${PRE_COMMIT_HOME}
|
# - ${PRE_COMMIT_HOME}
|
||||||
script:
|
script:
|
||||||
- pip install pre-commit
|
- pip install pre-commit
|
||||||
- pre-commit run --all-files
|
- pre-commit run --all-files
|
||||||
@ -53,7 +53,7 @@ secret_detection:
|
|||||||
|
|
||||||
test-3.8-core:
|
test-3.8-core:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.8-bullseye
|
image: python:3.8-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e py38-core
|
- tox -e py38-core
|
||||||
artifacts:
|
artifacts:
|
||||||
@ -65,7 +65,7 @@ test-3.8-core:
|
|||||||
|
|
||||||
test-3.9-core:
|
test-3.9-core:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.9-bullseye
|
image: python:3.9-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e py39-core
|
- tox -e py39-core
|
||||||
artifacts:
|
artifacts:
|
||||||
@ -77,7 +77,7 @@ test-3.9-core:
|
|||||||
|
|
||||||
test-3.10-core:
|
test-3.10-core:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.10-bullseye
|
image: python:3.10-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e py310-core
|
- tox -e py310-core
|
||||||
artifacts:
|
artifacts:
|
||||||
@ -89,7 +89,7 @@ test-3.10-core:
|
|||||||
|
|
||||||
test-3.11-core:
|
test-3.11-core:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.11-bullseye
|
image: python:3.11-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e py311-core
|
- tox -e py311-core
|
||||||
artifacts:
|
artifacts:
|
||||||
@ -99,22 +99,9 @@ test-3.11-core:
|
|||||||
coverage_format: cobertura
|
coverage_format: cobertura
|
||||||
path: coverage.xml
|
path: coverage.xml
|
||||||
|
|
||||||
test-pvpy-core:
|
|
||||||
<<: *only-default
|
|
||||||
image: pypy:3.9-bullseye
|
|
||||||
script:
|
|
||||||
- tox -e pypy-all
|
|
||||||
artifacts:
|
|
||||||
when: always
|
|
||||||
reports:
|
|
||||||
coverage_report:
|
|
||||||
coverage_format: cobertura
|
|
||||||
path: coverage.xml
|
|
||||||
allow_failure: true
|
|
||||||
|
|
||||||
test-3.12-core:
|
test-3.12-core:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.12-rc-bullseye
|
image: python:3.12-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e py312-core
|
- tox -e py312-core
|
||||||
artifacts:
|
artifacts:
|
||||||
@ -123,11 +110,10 @@ test-3.12-core:
|
|||||||
coverage_report:
|
coverage_report:
|
||||||
coverage_format: cobertura
|
coverage_format: cobertura
|
||||||
path: coverage.xml
|
path: coverage.xml
|
||||||
allow_failure: true
|
|
||||||
|
|
||||||
test-3.8-all:
|
test-3.8-all:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.8-bullseye
|
image: python:3.8-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e py38-all
|
- tox -e py38-all
|
||||||
artifacts:
|
artifacts:
|
||||||
@ -139,7 +125,7 @@ test-3.8-all:
|
|||||||
|
|
||||||
test-3.9-all:
|
test-3.9-all:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.9-bullseye
|
image: python:3.9-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e py39-all
|
- tox -e py39-all
|
||||||
artifacts:
|
artifacts:
|
||||||
@ -151,7 +137,7 @@ test-3.9-all:
|
|||||||
|
|
||||||
test-3.10-all:
|
test-3.10-all:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.10-bullseye
|
image: python:3.10-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e py310-all
|
- tox -e py310-all
|
||||||
artifacts:
|
artifacts:
|
||||||
@ -163,7 +149,7 @@ test-3.10-all:
|
|||||||
|
|
||||||
test-3.11-all:
|
test-3.11-all:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.11-bullseye
|
image: python:3.11-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e py311-all
|
- tox -e py311-all
|
||||||
artifacts:
|
artifacts:
|
||||||
@ -174,22 +160,9 @@ test-3.11-all:
|
|||||||
path: coverage.xml
|
path: coverage.xml
|
||||||
coverage: '/(?i)total.*? (100(?:\.0+)?\%|[1-9]?\d(?:\.\d+)?\%)$/'
|
coverage: '/(?i)total.*? (100(?:\.0+)?\%|[1-9]?\d(?:\.\d+)?\%)$/'
|
||||||
|
|
||||||
test-pvpy-all:
|
|
||||||
<<: *only-default
|
|
||||||
image: pypy:3.9-bullseye
|
|
||||||
script:
|
|
||||||
- tox -e pypy-all
|
|
||||||
artifacts:
|
|
||||||
when: always
|
|
||||||
reports:
|
|
||||||
coverage_report:
|
|
||||||
coverage_format: cobertura
|
|
||||||
path: coverage.xml
|
|
||||||
allow_failure: true
|
|
||||||
|
|
||||||
test-3.12-all:
|
test-3.12-all:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.12-rc-bullseye
|
image: python:3.12-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e py312-all
|
- tox -e py312-all
|
||||||
artifacts:
|
artifacts:
|
||||||
@ -198,11 +171,10 @@ test-3.12-all:
|
|||||||
coverage_report:
|
coverage_report:
|
||||||
coverage_format: cobertura
|
coverage_format: cobertura
|
||||||
path: coverage.xml
|
path: coverage.xml
|
||||||
allow_failure: true
|
|
||||||
|
|
||||||
build-test:
|
build-test:
|
||||||
stage: test
|
stage: test
|
||||||
image: python:3.11-bullseye
|
image: python:3.11-bookworm
|
||||||
|
|
||||||
before_script:
|
before_script:
|
||||||
- python -m pip install --upgrade pip
|
- python -m pip install --upgrade pip
|
||||||
@ -221,13 +193,13 @@ build-test:
|
|||||||
|
|
||||||
test-docs:
|
test-docs:
|
||||||
<<: *only-default
|
<<: *only-default
|
||||||
image: python:3.11-bullseye
|
image: python:3.11-bookworm
|
||||||
script:
|
script:
|
||||||
- tox -e docs
|
- tox -e docs
|
||||||
|
|
||||||
deploy_production:
|
deploy_production:
|
||||||
stage: deploy
|
stage: deploy
|
||||||
image: python:3.11-bullseye
|
image: python:3.11-bookworm
|
||||||
|
|
||||||
before_script:
|
before_script:
|
||||||
- python -m pip install --upgrade pip
|
- python -m pip install --upgrade pip
|
||||||
@ -259,7 +231,7 @@ build-image:
|
|||||||
docker run --privileged --rm tonistiigi/binfmt --uninstall qemu-*
|
docker run --privileged --rm tonistiigi/binfmt --uninstall qemu-*
|
||||||
docker run --privileged --rm tonistiigi/binfmt --install all
|
docker run --privileged --rm tonistiigi/binfmt --install all
|
||||||
docker buildx create --use --name new-builder
|
docker buildx create --use --name new-builder
|
||||||
docker buildx build . -t $IMAGE_TAG -t $MINOR_TAG -t $MAJOR_TAG -t $LATEST_TAG -f docker/Dockerfile --platform linux/amd64,linux/arm64 --push --build-arg AUTH_VERSION=$(echo $CI_COMMIT_TAG | cut -c 2-)
|
docker buildx build . --tag $IMAGE_TAG --tag $CURRENT_TAG --tag $MINOR_TAG --tag $MAJOR_TAG --tag $LATEST_TAG --file docker/Dockerfile --platform linux/amd64,linux/arm64 --push --build-arg AUTH_VERSION=$(echo $CI_COMMIT_TAG | cut -c 2-)
|
||||||
rules:
|
rules:
|
||||||
- if: $CI_COMMIT_TAG
|
- if: $CI_COMMIT_TAG
|
||||||
when: delayed
|
when: delayed
|
||||||
@ -279,7 +251,7 @@ build-image-dev:
|
|||||||
docker run --privileged --rm tonistiigi/binfmt --uninstall qemu-*
|
docker run --privileged --rm tonistiigi/binfmt --uninstall qemu-*
|
||||||
docker run --privileged --rm tonistiigi/binfmt --install all
|
docker run --privileged --rm tonistiigi/binfmt --install all
|
||||||
docker buildx create --use --name new-builder
|
docker buildx create --use --name new-builder
|
||||||
docker buildx build . -t $IMAGE_TAG -f docker/Dockerfile --platform linux/amd64,linux/arm64 --push --build-arg AUTH_PACKAGE=git+https://gitlab.com/allianceauth/allianceauth@$CI_COMMIT_BRANCH
|
docker buildx build . --tag $IMAGE_TAG --file docker/Dockerfile --platform linux/amd64,linux/arm64 --push --build-arg AUTH_PACKAGE=git+https://gitlab.com/allianceauth/allianceauth@$CI_COMMIT_BRANCH
|
||||||
rules:
|
rules:
|
||||||
- if: '$CI_MERGE_REQUEST_SOURCE_BRANCH_NAME == ""'
|
- if: '$CI_MERGE_REQUEST_SOURCE_BRANCH_NAME == ""'
|
||||||
when: manual
|
when: manual
|
||||||
@ -300,7 +272,7 @@ build-image-mr:
|
|||||||
docker run --privileged --rm tonistiigi/binfmt --uninstall qemu-*
|
docker run --privileged --rm tonistiigi/binfmt --uninstall qemu-*
|
||||||
docker run --privileged --rm tonistiigi/binfmt --install all
|
docker run --privileged --rm tonistiigi/binfmt --install all
|
||||||
docker buildx create --use --name new-builder
|
docker buildx create --use --name new-builder
|
||||||
docker buildx build . -t $IMAGE_TAG -f docker/Dockerfile --platform linux/amd64,linux/arm64 --push --build-arg AUTH_PACKAGE=git+$CI_MERGE_REQUEST_SOURCE_PROJECT_URL@$CI_MERGE_REQUEST_SOURCE_BRANCH_NAME
|
docker buildx build . --tag $IMAGE_TAG --file docker/Dockerfile --platform linux/amd64,linux/arm64 --push --build-arg AUTH_PACKAGE=git+$CI_MERGE_REQUEST_SOURCE_PROJECT_URL@$CI_MERGE_REQUEST_SOURCE_BRANCH_NAME
|
||||||
rules:
|
rules:
|
||||||
- if: '$CI_PIPELINE_SOURCE == "merge_request_event"'
|
- if: '$CI_PIPELINE_SOURCE == "merge_request_event"'
|
||||||
when: manual
|
when: manual
|
||||||
|
@ -3,9 +3,41 @@
|
|||||||
# Update this file:
|
# Update this file:
|
||||||
# pre-commit autoupdate
|
# pre-commit autoupdate
|
||||||
|
|
||||||
|
# Set the default language versions for the hooks
|
||||||
|
default_language_version:
|
||||||
|
python: python3 # Force all Python hooks to use Python 3
|
||||||
|
node: 22.12.0 # Force all Node hooks to use Node 22.12.0
|
||||||
|
|
||||||
|
# Globally exclude files
|
||||||
|
# https://pre-commit.com/#top_level-exclude
|
||||||
|
exclude: |
|
||||||
|
(?x)(
|
||||||
|
LICENSE|
|
||||||
|
allianceauth\/static\/allianceauth\/css\/themes\/bootstrap-locals.less|
|
||||||
|
\.min\.css|
|
||||||
|
\.min\.js|
|
||||||
|
\.po|
|
||||||
|
\.mo|
|
||||||
|
swagger\.json|
|
||||||
|
static/(.*)/libs/
|
||||||
|
)
|
||||||
|
|
||||||
repos:
|
repos:
|
||||||
|
# Code Upgrades
|
||||||
|
- repo: https://github.com/asottile/pyupgrade
|
||||||
|
rev: v3.20.0
|
||||||
|
hooks:
|
||||||
|
- id: pyupgrade
|
||||||
|
args: [--py38-plus]
|
||||||
|
- repo: https://github.com/adamchainz/django-upgrade
|
||||||
|
rev: 1.25.0
|
||||||
|
hooks:
|
||||||
|
- id: django-upgrade
|
||||||
|
args: [--target-version=4.2]
|
||||||
|
|
||||||
|
# Formatting
|
||||||
- repo: https://github.com/pre-commit/pre-commit-hooks
|
- repo: https://github.com/pre-commit/pre-commit-hooks
|
||||||
rev: v4.4.0
|
rev: v5.0.0
|
||||||
hooks:
|
hooks:
|
||||||
# Identify invalid files
|
# Identify invalid files
|
||||||
- id: check-ast
|
- id: check-ast
|
||||||
@ -13,73 +45,51 @@ repos:
|
|||||||
- id: check-json
|
- id: check-json
|
||||||
- id: check-toml
|
- id: check-toml
|
||||||
- id: check-xml
|
- id: check-xml
|
||||||
|
|
||||||
# git checks
|
# git checks
|
||||||
- id: check-merge-conflict
|
- id: check-merge-conflict
|
||||||
- id: check-added-large-files
|
- id: check-added-large-files
|
||||||
args: [ --maxkb=1000 ]
|
args: [--maxkb=1000]
|
||||||
- id: detect-private-key
|
- id: detect-private-key
|
||||||
- id: check-case-conflict
|
- id: check-case-conflict
|
||||||
|
|
||||||
# Python checks
|
# Python checks
|
||||||
# - id: check-docstring-first
|
# - id: check-docstring-first
|
||||||
- id: debug-statements
|
- id: debug-statements
|
||||||
# - id: requirements-txt-fixer
|
# - id: requirements-txt-fixer
|
||||||
- id: fix-encoding-pragma
|
- id: fix-encoding-pragma
|
||||||
args: [ --remove ]
|
args: [--remove]
|
||||||
- id: fix-byte-order-marker
|
- id: fix-byte-order-marker
|
||||||
|
|
||||||
# General quality checks
|
# General quality checks
|
||||||
- id: mixed-line-ending
|
- id: mixed-line-ending
|
||||||
args: [ --fix=lf ]
|
args: [--fix=lf]
|
||||||
- id: trailing-whitespace
|
- id: trailing-whitespace
|
||||||
args: [ --markdown-linebreak-ext=md ]
|
args: [--markdown-linebreak-ext=md]
|
||||||
exclude: |
|
|
||||||
(?x)(
|
|
||||||
\.min\.css|
|
|
||||||
\.min\.js|
|
|
||||||
\.po|
|
|
||||||
\.mo|
|
|
||||||
swagger\.json
|
|
||||||
)
|
|
||||||
- id: check-executables-have-shebangs
|
- id: check-executables-have-shebangs
|
||||||
- id: end-of-file-fixer
|
- id: end-of-file-fixer
|
||||||
exclude: |
|
|
||||||
(?x)(
|
|
||||||
\.min\.css|
|
|
||||||
\.min\.js|
|
|
||||||
\.po|
|
|
||||||
\.mo|
|
|
||||||
swagger\.json
|
|
||||||
)
|
|
||||||
|
|
||||||
- repo: https://github.com/editorconfig-checker/editorconfig-checker.python
|
- repo: https://github.com/editorconfig-checker/editorconfig-checker.python
|
||||||
rev: 2.7.2
|
rev: 3.2.1
|
||||||
hooks:
|
hooks:
|
||||||
- id: editorconfig-checker
|
- id: editorconfig-checker
|
||||||
exclude: |
|
- repo: https://github.com/igorshubovych/markdownlint-cli
|
||||||
(?x)(
|
rev: v0.45.0
|
||||||
LICENSE|
|
|
||||||
allianceauth\/static\/allianceauth\/css\/themes\/bootstrap-locals.less|
|
|
||||||
\.po|
|
|
||||||
\.mo|
|
|
||||||
swagger\.json
|
|
||||||
)
|
|
||||||
|
|
||||||
- repo: https://github.com/asottile/pyupgrade
|
|
||||||
rev: v3.10.1
|
|
||||||
hooks:
|
hooks:
|
||||||
- id: pyupgrade
|
- id: markdownlint
|
||||||
args: [ --py38-plus ]
|
language: node
|
||||||
|
args:
|
||||||
- repo: https://github.com/adamchainz/django-upgrade
|
- --disable=MD013
|
||||||
rev: 1.14.0
|
# Infrastructure
|
||||||
|
- repo: https://github.com/tox-dev/pyproject-fmt
|
||||||
|
rev: v2.6.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: django-upgrade
|
- id: pyproject-fmt
|
||||||
args: [--target-version=4.2]
|
name: pyproject.toml formatter
|
||||||
|
description: "Format the pyproject.toml file."
|
||||||
- repo: https://github.com/asottile/setup-cfg-fmt
|
args:
|
||||||
rev: v2.3.0
|
- --indent=4
|
||||||
|
additional_dependencies:
|
||||||
|
- tox==4.24.1 # https://github.com/tox-dev/tox/releases/latest
|
||||||
|
- repo: https://github.com/abravalheri/validate-pyproject
|
||||||
|
rev: v0.24.1
|
||||||
hooks:
|
hooks:
|
||||||
- id: setup-cfg-fmt
|
- id: validate-pyproject
|
||||||
args: [ --include-version-classifiers ]
|
name: Validate pyproject.toml
|
||||||
|
description: "Validate the pyproject.toml file."
|
||||||
|
@ -7,11 +7,14 @@ version: 2
|
|||||||
|
|
||||||
# Set the version of Python and other tools you might need
|
# Set the version of Python and other tools you might need
|
||||||
build:
|
build:
|
||||||
os: ubuntu-20.04
|
os: ubuntu-22.04
|
||||||
apt_packages:
|
apt_packages:
|
||||||
- redis
|
- redis
|
||||||
tools:
|
tools:
|
||||||
python: "3.8"
|
python: "3.11"
|
||||||
|
jobs:
|
||||||
|
post_system_dependencies:
|
||||||
|
- redis-server --daemonize yes
|
||||||
|
|
||||||
# Build documentation in the docs/ directory with Sphinx
|
# Build documentation in the docs/ directory with Sphinx
|
||||||
sphinx:
|
sphinx:
|
||||||
@ -20,7 +23,10 @@ sphinx:
|
|||||||
# Optionally build your docs in additional formats such as PDF and ePub
|
# Optionally build your docs in additional formats such as PDF and ePub
|
||||||
formats: all
|
formats: all
|
||||||
|
|
||||||
# Optionally set the version of Python and requirements required to build your docs
|
# Python requirements required to build your docs
|
||||||
python:
|
python:
|
||||||
install:
|
install:
|
||||||
- requirements: docs/requirements.txt
|
- method: pip
|
||||||
|
path: .
|
||||||
|
extra_requirements:
|
||||||
|
- docs
|
||||||
|
10
.tx/config
10
.tx/config
@ -1,10 +0,0 @@
|
|||||||
[main]
|
|
||||||
host = https://www.transifex.com
|
|
||||||
lang_map = zh-Hans: zh_Hans
|
|
||||||
|
|
||||||
[o:alliance-auth:p:alliance-auth:r:django-po]
|
|
||||||
file_filter = allianceauth/locale/<lang>/LC_MESSAGES/django.po
|
|
||||||
source_file = allianceauth/locale/en/LC_MESSAGES/django.po
|
|
||||||
source_lang = en
|
|
||||||
type = PO
|
|
||||||
minimum_perc = 0
|
|
@ -1,10 +0,0 @@
|
|||||||
[main]
|
|
||||||
host = https://www.transifex.com
|
|
||||||
lang_map = zh-Hans:zh_Hans
|
|
||||||
|
|
||||||
[alliance-auth.django-po]
|
|
||||||
file_filter = allianceauth/locale/<lang>/LC_MESSAGES/django.po
|
|
||||||
minimum_perc = 0
|
|
||||||
source_file = allianceauth/locale/en/LC_MESSAGES/django.po
|
|
||||||
source_lang = en
|
|
||||||
type = PO
|
|
10
.tx/transifex.yml
Normal file
10
.tx/transifex.yml
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
filters:
|
||||||
|
- filter_type: file
|
||||||
|
file_format: PO
|
||||||
|
source_file: allianceauth/locale/en/LC_MESSAGES/django.po
|
||||||
|
source_language: en
|
||||||
|
translation_files_expression: allianceauth/locale/<lang>/LC_MESSAGES/django.po
|
||||||
|
|
||||||
|
settings:
|
||||||
|
language_mapping:
|
||||||
|
zh-Hans: zh_Hans
|
40
README.md
40
README.md
@ -1,50 +1,56 @@
|
|||||||
# Alliance Auth
|
# Alliance Auth
|
||||||
|
|
||||||
[](https://pypi.org/project/allianceauth/)
|
[](https://pypi.org/project/allianceauth/)
|
||||||
[](https://pypi.org/project/allianceauth/)
|
[](https://pypi.org/project/allianceauth/)
|
||||||
[](https://pypi.org/project/allianceauth/)
|
[](https://pypi.org/project/allianceauth/)
|
||||||
[](https://pypi.org/project/allianceauth/)
|
[](https://pypi.org/project/allianceauth/)
|
||||||
[](https://gitlab.com/allianceauth/allianceauth/commits/master)
|
[](https://gitlab.com/allianceauth/allianceauth/commits/master)
|
||||||
[](http://allianceauth.readthedocs.io/?badge=latest)
|
[](https://allianceauth.readthedocs.io/?badge=latest)
|
||||||
[](https://gitlab.com/allianceauth/allianceauth/commits/master)
|
[](https://gitlab.com/allianceauth/allianceauth/commits/master)
|
||||||
[](https://discord.gg/fjnHAmk)
|
[](https://discord.gg/fjnHAmk)
|
||||||
|
|
||||||
An auth system for EVE Online to help in-game organizations manage online service access.
|
A flexible authentication platform for EVE Online to help in-game organizations manage access to applications and services. AA provides both, a stable core, and a robust framework for community development and custom applications.
|
||||||
|
|
||||||
## Content
|
## Content
|
||||||
|
|
||||||
- [Overview](#overview)
|
- [Overview](#overview)
|
||||||
- [Documentation](http://allianceauth.rtfd.io)
|
- [Documentation](https://allianceauth.rtfd.io)
|
||||||
- [Support](#support)
|
- [Support](#support)
|
||||||
- [Release Notes](https://gitlab.com/allianceauth/allianceauth/-/releases)
|
- [Release Notes](https://gitlab.com/allianceauth/allianceauth/-/releases)
|
||||||
- [Developer Team](#developer-team)
|
- [Developer Team](#development-team)
|
||||||
- [Contributing](#contributing)
|
- [Contributing](#contributing)
|
||||||
|
|
||||||
## Overview
|
## Overview
|
||||||
|
|
||||||
Alliance Auth (AA) is a web site that helps Eve Online organizations efficiently manage access to applications and services.
|
Alliance Auth (AA) is a platform that helps Eve Online organizations efficiently manage access to applications and services.
|
||||||
|
|
||||||
Main features:
|
Main features:
|
||||||
|
|
||||||
- Automatically grants or revokes user access to external services (e.g. Discord, Mumble) and web apps (e.g. SRP requests) based on the user's current membership to [in-game organizations](https://allianceauth.readthedocs.io/en/latest/features/core/states/) and [groups](https://allianceauth.readthedocs.io/en/latest/features/core/groups/)
|
- Automatically grants or revokes user access to external services (e.g.: Discord, Mumble) based on the user's current membership to [a variety of EVE Online affiliation](https://allianceauth.readthedocs.io/en/latest/features/core/states/) and [groups](https://allianceauth.readthedocs.io/en/latest/features/core/groups/)
|
||||||
|
|
||||||
- Provides a central web site where users can directly access web apps (e.g. SRP requests, Fleet Schedule) and manage their access to external services and groups.
|
- Provides a central web site where users can directly access web apps (e.g. SRP requests, Fleet Schedule) and manage their access to external services and groups.
|
||||||
|
|
||||||
- Includes a set of connectors (called ["services"](https://allianceauth.readthedocs.io/en/latest/features/services/)) for integrating access management with many popular external applications / services like Discord, Mumble, Teamspeak 3, SMF and others
|
- Includes a set of connectors (called ["Services"](https://allianceauth.readthedocs.io/en/latest/features/services/)) for integrating access management with many popular external applications / services like Discord, Mumble, Teamspeak 3, SMF and others
|
||||||
|
|
||||||
- Includes a set of web [apps](https://allianceauth.readthedocs.io/en/latest/features/apps/) which add many useful functions, e.g.: fleet schedule, timer board, SRP request management, fleet activity tracker
|
- Includes a set of web [Apps](https://allianceauth.readthedocs.io/en/latest/features/apps/) which add many useful functions, e.g.: fleet schedule, timer board, SRP request management, fleet activity tracker
|
||||||
|
|
||||||
- Can be easily extended with additional services and apps. Many are provided by the community and can be found here: [Community Creations](https://gitlab.com/allianceauth/community-creations)
|
- Can be easily extended with additional services and apps. Many are provided by the community and can be found here: [Community Creations](https://gitlab.com/allianceauth/community-creations)
|
||||||
|
|
||||||
- English :flag_gb:, Chinese :flag_cn:, German :flag_de:, Spanish :flag_es:, Korean :flag_kr:, Russian :flag_ru:, Italian :flag_it:, French :flag_fr:, Japanese :flag_jp: and Ukrainian :flag_ua: Localization
|
- English :flag_gb:, Chinese :flag_cn:, German :flag_de:, Spanish :flag_es:, Korean :flag_kr:, Russian :flag_ru:, Italian :flag_it:, French :flag_fr:, Japanese :flag_jp: and Ukrainian :flag_ua: Localization
|
||||||
|
|
||||||
For further details about AA - including an installation guide and a full list of included services and plugin apps - please see the [official documentation](http://allianceauth.rtfd.io).
|
For further details about AA - including an installation guide and a full list of included services and plugin apps - please see the [official documentation](https://allianceauth.rtfd.io).
|
||||||
|
|
||||||
## Screenshot
|
## Screenshot
|
||||||
|
|
||||||
Here is an example of the Alliance Auth web site with some plug-ins apps and services enabled:
|
Here is an example of the Alliance Auth web site with a mixture of Services, Apps and Community Creations enabled:
|
||||||
|
|
||||||

|
### Flatly Theme
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
### Darkly Theme
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
## Support
|
## Support
|
||||||
|
|
||||||
|
@ -5,7 +5,7 @@ manage online service access.
|
|||||||
# This will make sure the app is always imported when
|
# This will make sure the app is always imported when
|
||||||
# Django starts so that shared_task will use this app.
|
# Django starts so that shared_task will use this app.
|
||||||
|
|
||||||
__version__ = '4.0.0a1'
|
__version__ = '4.8.0'
|
||||||
__title__ = 'Alliance Auth'
|
__title__ = 'Alliance Auth'
|
||||||
__url__ = 'https://gitlab.com/allianceauth/allianceauth'
|
__url__ = 'https://gitlab.com/allianceauth/allianceauth'
|
||||||
NAME = f'{__title__} v{__version__}'
|
NAME = f'{__title__} v{__version__}'
|
||||||
|
@ -1,15 +1,16 @@
|
|||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
|
|
||||||
from .models import AnalyticsIdentifier, AnalyticsTokens
|
from .models import AnalyticsIdentifier, AnalyticsTokens
|
||||||
|
from solo.admin import SingletonModelAdmin
|
||||||
|
|
||||||
|
|
||||||
@admin.register(AnalyticsIdentifier)
|
@admin.register(AnalyticsIdentifier)
|
||||||
class AnalyticsIdentifierAdmin(admin.ModelAdmin):
|
class AnalyticsIdentifierAdmin(SingletonModelAdmin):
|
||||||
search_fields = ['identifier', ]
|
search_fields = ['identifier', ]
|
||||||
list_display = ('identifier',)
|
list_display = ['identifier', ]
|
||||||
|
|
||||||
|
|
||||||
@admin.register(AnalyticsTokens)
|
@admin.register(AnalyticsTokens)
|
||||||
class AnalyticsTokensAdmin(admin.ModelAdmin):
|
class AnalyticsTokensAdmin(admin.ModelAdmin):
|
||||||
search_fields = ['name', ]
|
search_fields = ['name', ]
|
||||||
list_display = ('name', 'type',)
|
list_display = ['name', 'type', ]
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
from django.apps import AppConfig
|
from django.apps import AppConfig
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
|
||||||
class AnalyticsConfig(AppConfig):
|
class AnalyticsConfig(AppConfig):
|
||||||
name = 'allianceauth.analytics'
|
name = 'allianceauth.analytics'
|
||||||
label = 'analytics'
|
label = 'analytics'
|
||||||
|
verbose_name = _('Analytics')
|
||||||
|
@ -0,0 +1,17 @@
|
|||||||
|
# Generated by Django 4.2.16 on 2024-12-11 02:17
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('analytics', '0009_remove_analyticstokens_ignore_paths_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='analyticsidentifier',
|
||||||
|
options={'verbose_name': 'Analytics Identifier'},
|
||||||
|
),
|
||||||
|
]
|
@ -1,23 +1,19 @@
|
|||||||
|
from typing import Literal
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.core.exceptions import ValidationError
|
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
from solo.models import SingletonModel
|
||||||
from uuid import uuid4
|
from uuid import uuid4
|
||||||
|
|
||||||
|
|
||||||
class AnalyticsIdentifier(models.Model):
|
class AnalyticsIdentifier(SingletonModel):
|
||||||
|
|
||||||
identifier = models.UUIDField(
|
identifier = models.UUIDField(default=uuid4, editable=False)
|
||||||
default=uuid4,
|
|
||||||
editable=False)
|
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
def __str__(self) -> Literal['Analytics Identifier']:
|
||||||
if not self.pk and AnalyticsIdentifier.objects.exists():
|
return "Analytics Identifier"
|
||||||
# Force a single object
|
|
||||||
raise ValidationError('There is can be only one \
|
class Meta:
|
||||||
AnalyticsIdentifier instance')
|
verbose_name = "Analytics Identifier"
|
||||||
self.pk = self.id = 1 # If this happens to be deleted and recreated, force it to be 1
|
|
||||||
return super().save(*args, **kwargs)
|
|
||||||
|
|
||||||
|
|
||||||
class AnalyticsTokens(models.Model):
|
class AnalyticsTokens(models.Model):
|
||||||
|
@ -5,10 +5,13 @@ from django.apps import apps
|
|||||||
from celery import shared_task
|
from celery import shared_task
|
||||||
from .models import AnalyticsTokens, AnalyticsIdentifier
|
from .models import AnalyticsTokens, AnalyticsIdentifier
|
||||||
from .utils import (
|
from .utils import (
|
||||||
|
existence_baremetal_or_docker,
|
||||||
install_stat_addons,
|
install_stat_addons,
|
||||||
install_stat_tokens,
|
install_stat_tokens,
|
||||||
install_stat_users)
|
install_stat_users)
|
||||||
|
|
||||||
|
from allianceauth import __version__
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
BASE_URL = "https://www.google-analytics.com"
|
BASE_URL = "https://www.google-analytics.com"
|
||||||
@ -65,8 +68,8 @@ def analytics_event(namespace: str,
|
|||||||
value=value).apply_async(priority=9)
|
value=value).apply_async(priority=9)
|
||||||
|
|
||||||
|
|
||||||
@shared_task()
|
@shared_task
|
||||||
def analytics_daily_stats():
|
def analytics_daily_stats() -> None:
|
||||||
"""Celery Task: Do not call directly
|
"""Celery Task: Do not call directly
|
||||||
|
|
||||||
Gathers a series of daily statistics
|
Gathers a series of daily statistics
|
||||||
@ -75,6 +78,7 @@ def analytics_daily_stats():
|
|||||||
users = install_stat_users()
|
users = install_stat_users()
|
||||||
tokens = install_stat_tokens()
|
tokens = install_stat_tokens()
|
||||||
addons = install_stat_addons()
|
addons = install_stat_addons()
|
||||||
|
existence_type = existence_baremetal_or_docker()
|
||||||
logger.debug("Running Daily Analytics Upload")
|
logger.debug("Running Daily Analytics Upload")
|
||||||
|
|
||||||
analytics_event(namespace='allianceauth.analytics',
|
analytics_event(namespace='allianceauth.analytics',
|
||||||
@ -82,6 +86,11 @@ def analytics_daily_stats():
|
|||||||
label='existence',
|
label='existence',
|
||||||
value=1,
|
value=1,
|
||||||
event_type='Stats')
|
event_type='Stats')
|
||||||
|
analytics_event(namespace='allianceauth.analytics',
|
||||||
|
task='send_install_stats',
|
||||||
|
label=existence_type,
|
||||||
|
value=1,
|
||||||
|
event_type='Stats')
|
||||||
analytics_event(namespace='allianceauth.analytics',
|
analytics_event(namespace='allianceauth.analytics',
|
||||||
task='send_install_stats',
|
task='send_install_stats',
|
||||||
label='users',
|
label='users',
|
||||||
@ -97,8 +106,34 @@ def analytics_daily_stats():
|
|||||||
label='addons',
|
label='addons',
|
||||||
value=addons,
|
value=addons,
|
||||||
event_type='Stats')
|
event_type='Stats')
|
||||||
|
|
||||||
for appconfig in apps.get_app_configs():
|
for appconfig in apps.get_app_configs():
|
||||||
|
if appconfig.label in [
|
||||||
|
"django_celery_beat",
|
||||||
|
"bootstrapform",
|
||||||
|
"messages",
|
||||||
|
"sessions",
|
||||||
|
"auth",
|
||||||
|
"staticfiles",
|
||||||
|
"users",
|
||||||
|
"addons",
|
||||||
|
"admin",
|
||||||
|
"humanize",
|
||||||
|
"contenttypes",
|
||||||
|
"sortedm2m",
|
||||||
|
"django_bootstrap5",
|
||||||
|
"tokens",
|
||||||
|
"authentication",
|
||||||
|
"services",
|
||||||
|
"framework",
|
||||||
|
"notifications"
|
||||||
|
"eveonline",
|
||||||
|
"navhelper",
|
||||||
|
"analytics",
|
||||||
|
"menu",
|
||||||
|
"theme"
|
||||||
|
]:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
analytics_event(namespace='allianceauth.analytics',
|
analytics_event(namespace='allianceauth.analytics',
|
||||||
task='send_extension_stats',
|
task='send_extension_stats',
|
||||||
label=appconfig.label,
|
label=appconfig.label,
|
||||||
@ -106,7 +141,7 @@ def analytics_daily_stats():
|
|||||||
event_type='Stats')
|
event_type='Stats')
|
||||||
|
|
||||||
|
|
||||||
@shared_task()
|
@shared_task
|
||||||
def send_ga_tracking_celery_event(
|
def send_ga_tracking_celery_event(
|
||||||
measurement_id: str,
|
measurement_id: str,
|
||||||
secret: str,
|
secret: str,
|
||||||
@ -136,10 +171,10 @@ def send_ga_tracking_celery_event(
|
|||||||
}
|
}
|
||||||
|
|
||||||
payload = {
|
payload = {
|
||||||
'client_id': AnalyticsIdentifier.objects.get(id=1).identifier.hex,
|
'client_id': AnalyticsIdentifier.get_solo().identifier.hex,
|
||||||
"user_properties": {
|
"user_properties": {
|
||||||
"allianceauth_version": {
|
"allianceauth_version": {
|
||||||
"value": "allianceauth_version"
|
"value": __version__
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'non_personalized_ads': True,
|
'non_personalized_ads': True,
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
from allianceauth.analytics.models import AnalyticsIdentifier
|
from allianceauth.analytics.models import AnalyticsIdentifier
|
||||||
from django.core.exceptions import ValidationError
|
|
||||||
|
|
||||||
from django.test.testcases import TestCase
|
from django.test.testcases import TestCase
|
||||||
|
|
||||||
from uuid import UUID, uuid4
|
from uuid import uuid4
|
||||||
|
|
||||||
|
|
||||||
# Identifiers
|
# Identifiers
|
||||||
@ -14,14 +13,4 @@ uuid_2 = "7aa6bd70701f44729af5e3095ff4b55c"
|
|||||||
class TestAnalyticsIdentifier(TestCase):
|
class TestAnalyticsIdentifier(TestCase):
|
||||||
|
|
||||||
def test_identifier_random(self):
|
def test_identifier_random(self):
|
||||||
self.assertNotEqual(AnalyticsIdentifier.objects.get(), uuid4)
|
self.assertNotEqual(AnalyticsIdentifier.get_solo(), uuid4)
|
||||||
|
|
||||||
def test_identifier_singular(self):
|
|
||||||
AnalyticsIdentifier.objects.all().delete()
|
|
||||||
AnalyticsIdentifier.objects.create(identifier=uuid_1)
|
|
||||||
# Yeah i have multiple asserts here, they all do the same thing
|
|
||||||
with self.assertRaises(ValidationError):
|
|
||||||
AnalyticsIdentifier.objects.create(identifier=uuid_2)
|
|
||||||
self.assertEqual(AnalyticsIdentifier.objects.count(), 1)
|
|
||||||
self.assertEqual(AnalyticsIdentifier.objects.get(
|
|
||||||
pk=1).identifier, UUID(uuid_1))
|
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import os
|
||||||
from django.apps import apps
|
from django.apps import apps
|
||||||
from allianceauth.authentication.models import User
|
from allianceauth.authentication.models import User
|
||||||
from esi.models import Token
|
from esi.models import Token
|
||||||
@ -34,3 +35,16 @@ def install_stat_addons() -> int:
|
|||||||
The Number of Installed Apps"""
|
The Number of Installed Apps"""
|
||||||
addons = len(list(apps.get_app_configs()))
|
addons = len(list(apps.get_app_configs()))
|
||||||
return addons
|
return addons
|
||||||
|
|
||||||
|
|
||||||
|
def existence_baremetal_or_docker() -> str:
|
||||||
|
"""Checks the Installation Type of an install
|
||||||
|
|
||||||
|
Returns
|
||||||
|
-------
|
||||||
|
str
|
||||||
|
existence_baremetal or existence_docker"""
|
||||||
|
docker_tag = os.getenv('AA_DOCKER_TAG')
|
||||||
|
if docker_tag:
|
||||||
|
return "existence_docker"
|
||||||
|
return "existence_baremetal"
|
||||||
|
@ -1,30 +1,8 @@
|
|||||||
from django.apps import AppConfig
|
from django.apps import AppConfig
|
||||||
from django.core.checks import Warning, Error, register
|
|
||||||
|
|
||||||
|
|
||||||
class AllianceAuthConfig(AppConfig):
|
class AllianceAuthConfig(AppConfig):
|
||||||
name = 'allianceauth'
|
name = 'allianceauth'
|
||||||
|
|
||||||
|
def ready(self) -> None:
|
||||||
@register()
|
import allianceauth.checks # noqa
|
||||||
def check_settings(app_configs, **kwargs):
|
|
||||||
from django.conf import settings
|
|
||||||
|
|
||||||
errors = []
|
|
||||||
if hasattr(settings, "SITE_URL"):
|
|
||||||
if settings.SITE_URL[-1] == "/":
|
|
||||||
errors.append(Warning(
|
|
||||||
"'SITE_URL' Has a trailing slash. This may lead to incorrect links being generated by Auth."))
|
|
||||||
else:
|
|
||||||
errors.append(Error(
|
|
||||||
"No 'SITE_URL' found is settings. This may lead to incorrect links being generated by Auth or Errors in 3rd party modules."))
|
|
||||||
if hasattr(settings, "CSRF_TRUSTED_ORIGINS"):
|
|
||||||
if hasattr(settings, "SITE_URL"):
|
|
||||||
if settings.SITE_URL not in settings.CSRF_TRUSTED_ORIGINS:
|
|
||||||
errors.append(Warning(
|
|
||||||
"'SITE_URL' not found in 'CSRF_TRUSTED_ORIGINS'. Auth may not load pages correctly until this is rectified."))
|
|
||||||
else:
|
|
||||||
errors.append(Error(
|
|
||||||
"No 'CSRF_TRUSTED_ORIGINS' found is settings, Auth may not load pages correctly until this is rectified"))
|
|
||||||
|
|
||||||
return errors
|
|
||||||
|
@ -1,10 +1,12 @@
|
|||||||
from django.apps import AppConfig
|
from django.apps import AppConfig
|
||||||
from django.core.checks import register, Tags
|
from django.core.checks import register, Tags
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
|
||||||
class AuthenticationConfig(AppConfig):
|
class AuthenticationConfig(AppConfig):
|
||||||
name = "allianceauth.authentication"
|
name = "allianceauth.authentication"
|
||||||
label = "authentication"
|
label = "authentication"
|
||||||
|
verbose_name = _("Authentication")
|
||||||
|
|
||||||
def ready(self):
|
def ready(self):
|
||||||
from allianceauth.authentication import checks, signals # noqa: F401
|
from allianceauth.authentication import checks, signals # noqa: F401
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
from allianceauth.hooks import DashboardItemHook
|
from allianceauth.hooks import DashboardItemHook
|
||||||
from allianceauth import hooks
|
from allianceauth import hooks
|
||||||
from .views import dashboard_characters, dashboard_groups, dashboard_admin
|
from .views import dashboard_characters, dashboard_esi_check, dashboard_groups, dashboard_admin
|
||||||
|
|
||||||
|
|
||||||
class UserCharactersHook(DashboardItemHook):
|
class UserCharactersHook(DashboardItemHook):
|
||||||
@ -26,6 +26,15 @@ class AdminHook(DashboardItemHook):
|
|||||||
DashboardItemHook.__init__(
|
DashboardItemHook.__init__(
|
||||||
self,
|
self,
|
||||||
dashboard_admin,
|
dashboard_admin,
|
||||||
|
1
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class ESICheckHook(DashboardItemHook):
|
||||||
|
def __init__(self):
|
||||||
|
DashboardItemHook.__init__(
|
||||||
|
self,
|
||||||
|
dashboard_esi_check,
|
||||||
0
|
0
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -43,3 +52,8 @@ def register_groups_hook():
|
|||||||
@hooks.register('dashboard_hook')
|
@hooks.register('dashboard_hook')
|
||||||
def register_admin_hook():
|
def register_admin_hook():
|
||||||
return AdminHook()
|
return AdminHook()
|
||||||
|
|
||||||
|
|
||||||
|
@hooks.register('dashboard_hook')
|
||||||
|
def register_esi_hook():
|
||||||
|
return ESICheckHook()
|
||||||
|
@ -2,7 +2,6 @@ import logging
|
|||||||
|
|
||||||
from django.contrib.auth.backends import ModelBackend
|
from django.contrib.auth.backends import ModelBackend
|
||||||
from django.contrib.auth.models import User, Permission
|
from django.contrib.auth.models import User, Permission
|
||||||
from django.contrib import messages
|
|
||||||
|
|
||||||
from .models import UserProfile, CharacterOwnership, OwnershipRecord
|
from .models import UserProfile, CharacterOwnership, OwnershipRecord
|
||||||
|
|
||||||
@ -41,9 +40,7 @@ class StateBackend(ModelBackend):
|
|||||||
if ownership.user.profile.main_character:
|
if ownership.user.profile.main_character:
|
||||||
if ownership.user.profile.main_character.character_id == token.character_id:
|
if ownership.user.profile.main_character.character_id == token.character_id:
|
||||||
return ownership.user
|
return ownership.user
|
||||||
else: ## this is an alt, enforce main only.
|
else: # this is an alt, enforce main only.
|
||||||
if request:
|
|
||||||
messages.error("Unable to authenticate with this Character, Please log in with the main character associated with this account.")
|
|
||||||
return None
|
return None
|
||||||
else:
|
else:
|
||||||
logger.debug(f'{token.character_name} has changed ownership. Creating new user account.')
|
logger.debug(f'{token.character_name} has changed ownership. Creating new user account.')
|
||||||
@ -66,9 +63,7 @@ class StateBackend(ModelBackend):
|
|||||||
user = records[0].user
|
user = records[0].user
|
||||||
if user.profile.main_character:
|
if user.profile.main_character:
|
||||||
if user.profile.main_character.character_id != token.character_id:
|
if user.profile.main_character.character_id != token.character_id:
|
||||||
## this is an alt, enforce main only due to trust issues in SSO.
|
# this is an alt, enforce main only due to trust issues in SSO.
|
||||||
if request:
|
|
||||||
messages.error("Unable to authenticate with this Character, Please log in with the main character associated with this account. Then add this character from the dashboard.")
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
token.user = user
|
token.user = user
|
||||||
|
12
allianceauth/authentication/constants.py
Normal file
12
allianceauth/authentication/constants.py
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
# Overide ESI messages in the dashboard widget
|
||||||
|
# when the returned messages are not helpful or out of date
|
||||||
|
ESI_ERROR_MESSAGE_OVERRIDES = {
|
||||||
|
420: _("This software has exceeded the error limit for ESI. "
|
||||||
|
"If you are a user, please contact the maintainer of this software."
|
||||||
|
" If you are a developer/maintainer, please make a greater "
|
||||||
|
"effort in the future to receive valid responses. For tips on how, "
|
||||||
|
"come have a chat with us in ##3rd-party-dev-and-esi on the EVE "
|
||||||
|
"Online Discord. https://www.eveonline.com/discord")
|
||||||
|
}
|
@ -44,6 +44,8 @@ class UserSettingsMiddleware(MiddlewareMixin):
|
|||||||
logger.exception(e)
|
logger.exception(e)
|
||||||
|
|
||||||
# AA v4 Themes
|
# AA v4 Themes
|
||||||
|
# Null = has not been set by the user ever, dont act
|
||||||
|
# DEFAULT_THEME or DEFAULT_THEME_DARK will be used in get_theme()
|
||||||
try:
|
try:
|
||||||
if request.user.profile.theme is not None:
|
if request.user.profile.theme is not None:
|
||||||
request.session["THEME"] = request.user.profile.theme
|
request.session["THEME"] = request.user.profile.theme
|
||||||
|
@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 4.2.13 on 2024-05-12 09:44
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('authentication', '0022_userprofile_theme'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='userprofile',
|
||||||
|
name='language',
|
||||||
|
field=models.CharField(blank=True, choices=[('en', 'English'), ('de', 'German'), ('es', 'Spanish'), ('zh-hans', 'Chinese Simplified'), ('ru', 'Russian'), ('ko', 'Korean'), ('fr', 'French'), ('ja', 'Japanese'), ('it', 'Italian'), ('uk', 'Ukrainian'), ('pl', 'Polish')], default='', max_length=10, verbose_name='Language'),
|
||||||
|
),
|
||||||
|
]
|
@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 4.2 on 2024-09-13 09:46
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('authentication', '0023_alter_userprofile_language'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='userprofile',
|
||||||
|
name='language',
|
||||||
|
field=models.CharField(blank=True, choices=[('en', 'English'), ('cs-cz', 'Czech'), ('de', 'German'), ('es', 'Spanish'), ('it-it', 'Italian'), ('ja', 'Japanese'), ('ko-kr', 'Korean'), ('fr-fr', 'French'), ('ru', 'Russian'), ('nl-nl', 'Dutch'), ('pl-pl', 'Polish'), ('uk', 'Ukrainian'), ('zh-hans', 'Simplified Chinese')], default='', max_length=10, verbose_name='Language'),
|
||||||
|
),
|
||||||
|
]
|
@ -1,4 +1,5 @@
|
|||||||
import logging
|
import logging
|
||||||
|
from typing import ClassVar
|
||||||
|
|
||||||
from django.contrib.auth.models import User, Permission
|
from django.contrib.auth.models import User, Permission
|
||||||
from django.db import models, transaction
|
from django.db import models, transaction
|
||||||
@ -27,7 +28,7 @@ class State(models.Model):
|
|||||||
help_text="Factions to whose members this state is available.")
|
help_text="Factions to whose members this state is available.")
|
||||||
public = models.BooleanField(default=False, help_text="Make this state available to any character.")
|
public = models.BooleanField(default=False, help_text="Make this state available to any character.")
|
||||||
|
|
||||||
objects = StateManager()
|
objects: ClassVar[StateManager] = StateManager()
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ['-priority']
|
ordering = ['-priority']
|
||||||
@ -67,17 +68,20 @@ class UserProfile(models.Model):
|
|||||||
"""
|
"""
|
||||||
Choices for UserProfile.language
|
Choices for UserProfile.language
|
||||||
"""
|
"""
|
||||||
|
# Sorted by Language Code alphabetical order + English at top
|
||||||
ENGLISH = 'en', _('English')
|
ENGLISH = 'en', _('English')
|
||||||
|
CZECH = 'cs-cz', _("Czech") # Not yet at 50% translated
|
||||||
GERMAN = 'de', _('German')
|
GERMAN = 'de', _('German')
|
||||||
SPANISH = 'es', _('Spanish')
|
SPANISH = 'es', _('Spanish')
|
||||||
CHINESE = 'zh-hans', _('Chinese Simplified')
|
ITALIAN = 'it-it', _('Italian')
|
||||||
RUSSIAN = 'ru', _('Russian')
|
|
||||||
KOREAN = 'ko', _('Korean')
|
|
||||||
FRENCH = 'fr', _('French')
|
|
||||||
JAPANESE = 'ja', _('Japanese')
|
JAPANESE = 'ja', _('Japanese')
|
||||||
ITALIAN = 'it', _('Italian')
|
KOREAN = 'ko-kr', _('Korean')
|
||||||
|
FRENCH = 'fr-fr', _('French')
|
||||||
|
RUSSIAN = 'ru', _('Russian')
|
||||||
|
DUTCH = 'nl-nl', _("Dutch")
|
||||||
|
POLISH = 'pl-pl', _("Polish")
|
||||||
UKRAINIAN = 'uk', _('Ukrainian')
|
UKRAINIAN = 'uk', _('Ukrainian')
|
||||||
|
CHINESE = 'zh-hans', _('Simplified Chinese')
|
||||||
|
|
||||||
user = models.OneToOneField(
|
user = models.OneToOneField(
|
||||||
User,
|
User,
|
||||||
@ -134,8 +138,10 @@ class UserProfile(models.Model):
|
|||||||
sender=self.__class__, user=self.user, state=self.state
|
sender=self.__class__, user=self.user, state=self.state
|
||||||
)
|
)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self) -> str:
|
||||||
return str(self.user)
|
return str(self.user)
|
||||||
|
|
||||||
|
|
||||||
class CharacterOwnership(models.Model):
|
class CharacterOwnership(models.Model):
|
||||||
class Meta:
|
class Meta:
|
||||||
default_permissions = ('change', 'delete')
|
default_permissions = ('change', 'delete')
|
||||||
@ -145,7 +151,7 @@ class CharacterOwnership(models.Model):
|
|||||||
owner_hash = models.CharField(max_length=28, unique=True)
|
owner_hash = models.CharField(max_length=28, unique=True)
|
||||||
user = models.ForeignKey(User, on_delete=models.CASCADE, related_name='character_ownerships')
|
user = models.ForeignKey(User, on_delete=models.CASCADE, related_name='character_ownerships')
|
||||||
|
|
||||||
objects = CharacterOwnershipManager()
|
objects: ClassVar[CharacterOwnershipManager] = CharacterOwnershipManager()
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return f"{self.user}: {self.character}"
|
return f"{self.user}: {self.character}"
|
||||||
|
@ -1,41 +0,0 @@
|
|||||||
{% load i18n %}
|
|
||||||
<div class="col-12 col-xl-8 align-self-stretch p-2">
|
|
||||||
<div class="card">
|
|
||||||
<div class="card-body">
|
|
||||||
<div class="d-flex align-items-center">
|
|
||||||
<h4 class="ms-auto me-auto">
|
|
||||||
{% translate "Characters" %}
|
|
||||||
</h4>
|
|
||||||
</div>
|
|
||||||
<div class="card-body">
|
|
||||||
<div style="height: 300px;overflow:-moz-scrollbars-vertical;overflow-y:auto;">
|
|
||||||
<div class="d-flex">
|
|
||||||
<a href="{% url 'authentication:add_character' %}" class="btn btn-primary flex-fill m-1" title="{% translate 'Add Character' %}"><span class="d-md-inline m-2">{% translate 'Add Character' %}</span></a>
|
|
||||||
<a href="{% url 'authentication:change_main_character' %}" class="btn btn-primary flex-fill m-1" title="{% translate 'Change Main' %}"></span><span class="d-md-inline m-2">{% translate 'Change Main' %}</span></a>
|
|
||||||
</div>
|
|
||||||
<table class="table" style="--bs-table-bg: transparent;">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th class="text-center"></th>
|
|
||||||
<th class="text-center">{% translate "Name" %}</th>
|
|
||||||
<th class="text-center">{% translate "Corp" %}</th>
|
|
||||||
<th class="text-center">{% translate "Alliance" %}</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
{% for char in characters %}
|
|
||||||
<tr>
|
|
||||||
<td class="text-center"><img class="ra-avatar rounded-circle" src="{{ char.portrait_url_32 }}">
|
|
||||||
</td>
|
|
||||||
<td class="text-center">{{ char.character_name }}</td>
|
|
||||||
<td class="text-center">{{ char.corporation_name }}</td>
|
|
||||||
<td class="text-center">{{ char.alliance_name|default_if_none:"" }}</td>
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
@ -1,20 +0,0 @@
|
|||||||
{% load i18n %}
|
|
||||||
<div class="col-12 col-xl-4 align-self-stretch p-2">
|
|
||||||
<div class="card h-100">
|
|
||||||
<div class="card-body">
|
|
||||||
<h4 class="card-title text-center">{% translate "Membership" %}</h4>
|
|
||||||
<div class="card-body">
|
|
||||||
<div style="height: 300px;overflow:-moz-scrollbars-vertical;overflow-y:auto;">
|
|
||||||
<h6 class="text-center">{% translate "State:" %} {{ request.user.profile.state }}</h6>
|
|
||||||
<table class="table" style="--bs-table-bg: transparent;">
|
|
||||||
{% for group in groups %}
|
|
||||||
<tr>
|
|
||||||
<td class="text-center">{{ group.name }}</td>
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
@ -7,7 +7,7 @@
|
|||||||
{% translate "Dashboard" %}
|
{% translate "Dashboard" %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="d-flex justify-content-around align-self-center flex-wrap">
|
<div class="row">
|
||||||
{% for dash in views %}
|
{% for dash in views %}
|
||||||
{{ dash | safe }}
|
{{ dash | safe }}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
@ -0,0 +1,10 @@
|
|||||||
|
{% extends 'allianceauth/base.html' %}
|
||||||
|
|
||||||
|
|
||||||
|
{% block page_title %}Dashboard{% endblock page_title %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<div>
|
||||||
|
<h1>Dashboard Dummy</h1>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
@ -0,0 +1,44 @@
|
|||||||
|
{% load i18n %}
|
||||||
|
<div id="aa-dashboard-panel-characters" class="col-12 col-xl-8 mb-3">
|
||||||
|
<div class="card h-100">
|
||||||
|
<div class="card-body">
|
||||||
|
{% translate "Characters" as widget_title %}
|
||||||
|
{% include "framework/dashboard/widget-title.html" with title=widget_title %}
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<div style="height: 300px; overflow-y:auto;">
|
||||||
|
<div class="d-flex">
|
||||||
|
<a href="{% url 'authentication:add_character' %}" class="btn btn-primary flex-fill m-1" title="{% translate 'Add Character' %}">
|
||||||
|
<span class="d-md-inline m-2">{% translate 'Add Character' %}</span>
|
||||||
|
</a>
|
||||||
|
<a href="{% url 'authentication:change_main_character' %}" class="btn btn-primary flex-fill m-1" title="{% translate 'Change Main' %}">
|
||||||
|
<span class="d-md-inline m-2">{% translate 'Change Main' %}</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<table class="table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th class="text-center"></th>
|
||||||
|
<th class="text-center">{% translate "Name" %}</th>
|
||||||
|
<th class="text-center">{% translate "Corp" %}</th>
|
||||||
|
<th class="text-center">{% translate "Alliance" %}</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{% for char in characters %}
|
||||||
|
<tr>
|
||||||
|
<td class="text-center">
|
||||||
|
<img class="ra-avatar rounded-circle" src="{{ char.portrait_url_32 }}" alt="{{ char.character_name }}">
|
||||||
|
</td>
|
||||||
|
<td class="text-center">{{ char.character_name }}</td>
|
||||||
|
<td class="text-center">{{ char.corporation_name }}</td>
|
||||||
|
<td class="text-center">{{ char.alliance_name|default_if_none:"" }}</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
@ -0,0 +1,22 @@
|
|||||||
|
{% load i18n %}
|
||||||
|
<div id="aa-dashboard-panel-membership" class="col-12 col-xl-4 mb-3">
|
||||||
|
<div class="card h-100">
|
||||||
|
<div class="card-body">
|
||||||
|
{% translate "Membership" as widget_title %}
|
||||||
|
{% include "framework/dashboard/widget-title.html" with title=widget_title %}
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<div style="height: 300px; overflow-y:auto;">
|
||||||
|
<h5 class="text-center">{% translate "State:" %} {{ request.user.profile.state }}</h5>
|
||||||
|
<table class="table">
|
||||||
|
{% for group in groups %}
|
||||||
|
<tr>
|
||||||
|
<td class="text-center">{{ group.name }}</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
@ -1,62 +1,89 @@
|
|||||||
{% extends "allianceauth/base-bs5.html" %}
|
{% extends "allianceauth/base-bs5.html" %}
|
||||||
|
|
||||||
|
{% load aa_i18n %}
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block page_title %}{% translate "Dashboard" %}{% endblock page_title %}
|
{% block page_title %}
|
||||||
|
{% translate "Token Management" %}
|
||||||
|
{% endblock page_title %}
|
||||||
|
|
||||||
|
{% block header_nav_brand %}
|
||||||
|
{% translate "Token Management" %}
|
||||||
|
{% endblock header_nav_brand %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<h1 class="page-header text-center">{% translate "Token Management" %}</h1>
|
<div>
|
||||||
<div class="col-sm-12">
|
<p class="mb-3">
|
||||||
<table class="table table-aa" id="table_tokens" style="width:100%">
|
{% translate "This page is a best attempt, but backups or database logs can still contain your tokens. Always revoke tokens on https://developers.eveonline.com/authorized-apps where possible."|urlize %}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<table class="table w-100" id="table_tokens">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th>{% translate "Scopes" %}</th>
|
<th>{% translate "Scopes" %}</th>
|
||||||
<th class="text-end">{% translate "Actions" %}</th>
|
<th class="text-end">{% translate "Actions" %}</th>
|
||||||
<th>{% translate "Character" %}</th>
|
<th>{% translate "Character" %}</th>
|
||||||
|
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
|
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for t in tokens %}
|
{% for t in tokens %}
|
||||||
<tr>
|
<tr>
|
||||||
<td styl="white-space:initial;">{% for s in t.scopes.all %}<span class="badge badge-secondar">{{ s.name }}</span> {% endfor %}</td>
|
<td style="white-space:initial;">
|
||||||
<td nowrap class="text-end"><a href="{% url 'authentication:token_delete' t.id %}" class="btn btn-danger"><i class="fas fa-trash"></i></a> <a href="{% url 'authentication:token_refresh' t.id %}" class="btn btn-success"><i class="fas fa-sync-alt"></i></a></td>
|
{% for s in t.scopes.all %}
|
||||||
|
<span class="badge text-bg-secondary">{{ s.name }}</span>
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
|
||||||
|
<td nowrap class="text-end">
|
||||||
|
<a href="{% url 'authentication:token_delete' t.id %}" class="btn btn-danger"><i class="fa-solid fa-trash-can"></i></a>
|
||||||
|
<a href="{% url 'authentication:token_refresh' t.id %}" class="btn btn-success"><i class="fa-solid fa-rotate"></i></a>
|
||||||
|
</td>
|
||||||
|
|
||||||
<td>{{ t.character_name }}</td>
|
<td>{{ t.character_name }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
{% translate "This page is a best attempt, but backups or database logs can still contain your tokens. Always revoke tokens on https://community.eveonline.com/support/third-party-applications/ where possible."|urlize %}
|
|
||||||
</div>
|
</div>
|
||||||
{% endblock content %}
|
{% endblock content %}
|
||||||
|
|
||||||
{% block extra_javascript %}
|
{% block extra_javascript %}
|
||||||
{% include "bundles/datatables-js-bs5.html" %}
|
{% include "bundles/datatables-js-bs5.html" %}
|
||||||
|
|
||||||
|
{% get_datatables_language_static LANGUAGE_CODE as DT_LANG_PATH %}
|
||||||
|
|
||||||
|
<script>
|
||||||
|
$(document).ready(() => {
|
||||||
|
let grp = 2;
|
||||||
|
|
||||||
|
$('#table_tokens').DataTable({
|
||||||
|
"language": {"url": '{{ DT_LANG_PATH }}'},
|
||||||
|
'columnDefs': [{orderable: false, targets: [0, 1]}, {
|
||||||
|
'visible': false,
|
||||||
|
'targets': grp
|
||||||
|
}],
|
||||||
|
'order': [[grp, 'asc']],
|
||||||
|
'drawCallback': function (settings) {
|
||||||
|
const api = this.api();
|
||||||
|
const rows = api.rows({page: 'current'}).nodes();
|
||||||
|
let last = null;
|
||||||
|
api.column(grp, {page: 'current'})
|
||||||
|
.data()
|
||||||
|
.each((group, i) => {
|
||||||
|
if (last !== group) {
|
||||||
|
$(rows).eq(i).before(`<tr class="h5 table-primary"><td colspan="3">${group}</td></tr>`);
|
||||||
|
|
||||||
|
last = group;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
'stateSave': true
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
{% endblock extra_javascript %}
|
{% endblock extra_javascript %}
|
||||||
|
|
||||||
{% block extra_css %}
|
{% block extra_css %}
|
||||||
{% include "bundles/datatables-css-bs5.html" %}
|
{% include "bundles/datatables-css-bs5.html" %}
|
||||||
{% endblock extra_css %}
|
{% endblock extra_css %}
|
||||||
|
|
||||||
{% block extra_script %}
|
|
||||||
$(document).ready(function(){
|
|
||||||
let grp = 2;
|
|
||||||
var table = $('#table_tokens').DataTable({
|
|
||||||
"columnDefs": [{ orderable: false, targets: [0,1] },{ "visible": false, "targets": grp }],
|
|
||||||
"order": [[grp, 'asc']],
|
|
||||||
"drawCallback": function (settings) {
|
|
||||||
var api = this.api();
|
|
||||||
var rows = api.rows({ page: 'current' }).nodes();
|
|
||||||
var last = null;
|
|
||||||
api.column(grp, { page: 'current' })
|
|
||||||
.data()
|
|
||||||
.each(function (group, i) {
|
|
||||||
if (last !== group) {
|
|
||||||
$(rows).eq(i).before('<tr class="info"><td colspan="3">' + group + '</td></tr>');
|
|
||||||
last = group;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
},
|
|
||||||
"stateSave": true,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
{% endblock extra_script %}
|
|
||||||
|
@ -1,23 +1,25 @@
|
|||||||
|
{% load theme_tags %}
|
||||||
{% load static %}
|
{% load static %}
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en" {% theme_html_tags %}>
|
||||||
<head>
|
<head>
|
||||||
|
<!-- Required meta tags -->
|
||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
<meta name="description" content="">
|
<!-- End Required meta tags -->
|
||||||
<meta name="author" content="">
|
|
||||||
<!-- TODO Bundle all the site specific stuff up into its own template for easy overide -->
|
|
||||||
<meta property="og:title" content="{{ SITE_NAME }}">
|
|
||||||
<meta property="og:image" content="{{ SITE_URL }}{% static 'allianceauth/icons/apple-touch-icon.png' %}">
|
|
||||||
<meta property="og:description" content="Alliance Auth - An auth system for EVE Online to help in-game organizations manage online service access.">
|
|
||||||
|
|
||||||
|
<!-- Meta tags -->
|
||||||
|
{% include 'allianceauth/opengraph.html' %}
|
||||||
{% include 'allianceauth/icons.html' %}
|
{% include 'allianceauth/icons.html' %}
|
||||||
|
<!-- Meta tags -->
|
||||||
|
|
||||||
<title>{% block title %}{% block page_title %}{% endblock page_title %} - {{ SITE_NAME }}{% endblock title %}</title>
|
<title>{% block title %}{% block page_title %}{% endblock page_title %} - {{ SITE_NAME }}{% endblock title %}</title>
|
||||||
|
|
||||||
{% include 'bundles/bootstrap-css.html' %}
|
{% theme_css %}
|
||||||
|
|
||||||
{% include 'bundles/fontawesome.html' %}
|
{% include 'bundles/fontawesome.html' %}
|
||||||
|
{% include 'bundles/auth-framework-css.html' %}
|
||||||
|
|
||||||
{% block extra_include %}
|
{% block extra_include %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
@ -30,25 +32,23 @@
|
|||||||
background-size: cover;
|
background-size: cover;
|
||||||
}
|
}
|
||||||
|
|
||||||
.panel-transparent {
|
.card-login {
|
||||||
background: rgba(48, 48, 48, 0.7);
|
background: rgba(48 48 48 / 0.7);
|
||||||
color: #ffffff;
|
color: rgb(255 255 255);
|
||||||
padding-bottom: 21px;
|
padding-bottom: 21px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.panel-body {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
#lang-select {
|
#lang-select {
|
||||||
width: 40%;
|
width: 40%;
|
||||||
margin-left: auto;
|
margin-left: auto;
|
||||||
margin-right: auto;
|
margin-right: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
{% block extra_style %}
|
{% block extra_style %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
<div class="container" style="margin-top:150px;">
|
<div class="container" style="margin-top:150px;">
|
||||||
{% block content %}
|
{% block content %}
|
||||||
|
@ -1,14 +1,15 @@
|
|||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
<div class="dropdown">
|
|
||||||
<form action="{% url 'set_language' %}" method="post">
|
<form class="dropdown-item" action="{% url 'set_language' %}" method="post">
|
||||||
{% csrf_token %}
|
{% csrf_token %}
|
||||||
<select onchange="this.form.submit()" class="form-control" id="lang-select" name="language">
|
|
||||||
{% get_language_info_list for LANGUAGES as languages %}
|
<select class="form-select" onchange="this.form.submit()" class="form-control" id="lang-select" name="language">
|
||||||
{% for language in languages %}
|
{% get_available_languages as LANGUAGES %}
|
||||||
<option lang="{{ language.code }}" value="{{ language.code }}"{% if language.code == LANGUAGE_CODE %} selected="selected"{% endif %}>
|
|
||||||
{{ language.name_local|capfirst }} ({{ language.code }})
|
{% for lang_code, lang_name in LANGUAGES %}
|
||||||
|
<option lang="{{ lang_code }}" value="{{ lang_code }}"{% if lang_code == LANGUAGE_CODE %} selected{% endif %}>
|
||||||
|
{{ lang_code|language_name_local|capfirst }} ({{ lang_code }})
|
||||||
</option>
|
</option>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</select>
|
</select>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
|
||||||
|
@ -3,16 +3,17 @@
|
|||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="col-md-4 col-md-offset-4">
|
<div class="row justify-content-center">
|
||||||
|
<div class="col-md-6">
|
||||||
{% if messages %}
|
{% if messages %}
|
||||||
{% for message in messages %}
|
{% for message in messages %}
|
||||||
<div class="alert alert-{{ message.level_tag}}">{{ message }}</div>
|
<div class="alert alert-{{ message.level_tag}}">{{ message }}</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
<div class="panel panel-default panel-transparent">
|
<div class="card card-login border-secondary p-3">
|
||||||
<div class="panel-body">
|
<div class="card-body">
|
||||||
<div class="col-md-12">
|
<div class="text-center">
|
||||||
{% block middle_box_content %}
|
{% block middle_box_content %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
</div>
|
</div>
|
||||||
@ -20,22 +21,23 @@
|
|||||||
|
|
||||||
{% include 'public/lang_select.html' %}
|
{% include 'public/lang_select.html' %}
|
||||||
|
|
||||||
<p class="text-center" style="margin-top: 2rem;">
|
<p class="text-center mt-3">
|
||||||
{% translate "For information on SSO, ESI and security read the CCP Dev Blog" %}<br>
|
{% translate "For information on SSO, ESI and security read the CCP Dev Blog" %}<br>
|
||||||
<a href="https://www.eveonline.com/article/introducing-esi" target="_blank" rel="noopener noreferrer">
|
<a class="text-reset" href="https://www.eveonline.com/news/view/introducing-esi" target="_blank" rel="noopener noreferrer">
|
||||||
{% translate "Introducing ESI - A New API For Eve Online" %}
|
{% translate "Introducing ESI - A New API For EVE Online" %}
|
||||||
</a>
|
</a>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p class="text-center">
|
<p class="text-center">
|
||||||
<a href="https://community.eveonline.com/support/third-party-applications/" target="_blank" rel="noopener noreferrer">
|
<a class="text-reset" href="https://developers.eveonline.com/authorized-apps" target="_blank" rel="noopener noreferrer">
|
||||||
{% translate "Manage ESI Applications" %}
|
{% translate "Manage ESI Applications" %}
|
||||||
</a>
|
</a>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block extra_include %}
|
{% block extra_include %}
|
||||||
{% include 'bundles/bootstrap-js.html' %}
|
{% include 'bundles/bootstrap-js-bs5.html' %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
@ -1,27 +1,31 @@
|
|||||||
{% extends 'public/base.html' %}
|
{% extends 'public/base.html' %}
|
||||||
|
|
||||||
{% load bootstrap %}
|
{% load django_bootstrap5 %}
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block page_title %}{% translate "Registration" %}{% endblock %}
|
{% block page_title %}{% translate "Registration" %}{% endblock %}
|
||||||
|
|
||||||
{% block extra_include %}
|
{% block extra_include %}
|
||||||
{% include 'bundles/bootstrap-css.html' %}
|
{% include 'bundles/bootstrap-css-bs5.html' %}
|
||||||
{% include 'bundles/fontawesome.html' %}
|
{% include 'bundles/fontawesome.html' %}
|
||||||
{% include 'bundles/bootstrap-js.html' %}
|
{% include 'bundles/bootstrap-js-bs5.html' %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="col-md-4 col-md-offset-4">
|
<div class="row justify-content-center">
|
||||||
<div class="panel panel-default panel-transparent">
|
<div class="col-md-6">
|
||||||
<div class="panel-body">
|
<div class="card card-login border-secondary p-3">
|
||||||
|
<div class="card-body">
|
||||||
<form method="POST">
|
<form method="POST">
|
||||||
{% csrf_token %}
|
{% csrf_token %}
|
||||||
{{ form|bootstrap }}
|
{% bootstrap_form form %}
|
||||||
<button class="btn btn-lg btn-primary btn-block" type="submit">{% translate "Register" %}</button>
|
|
||||||
|
<button class="btn btn-primary btn-block" type="submit">{% translate "Register" %}</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% include 'public/lang_select.html' %}
|
{% include 'public/lang_select.html' %}
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
{% extends 'public/middle_box.html' %}
|
{% extends 'public/middle_box.html' %}
|
||||||
|
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block middle_box_content %}
|
{% block middle_box_content %}
|
||||||
<div class="alert alert-danger">{% translate 'Invalid or expired activation link.' %}</div>
|
<div class="alert alert-danger">{% translate 'Invalid or expired activation link.' %}</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
@ -1,10 +1,12 @@
|
|||||||
import json
|
import json
|
||||||
|
import requests_mock
|
||||||
from unittest.mock import patch
|
from unittest.mock import patch
|
||||||
|
|
||||||
from django.test import RequestFactory, TestCase
|
from django.test import RequestFactory, TestCase
|
||||||
|
|
||||||
from allianceauth.authentication.views import task_counts
|
from allianceauth.authentication.views import task_counts, esi_check
|
||||||
from allianceauth.tests.auth_utils import AuthUtils
|
from allianceauth.tests.auth_utils import AuthUtils
|
||||||
|
from allianceauth.authentication.constants import ESI_ERROR_MESSAGE_OVERRIDES
|
||||||
|
|
||||||
MODULE_PATH = "allianceauth.authentication.views"
|
MODULE_PATH = "allianceauth.authentication.views"
|
||||||
|
|
||||||
@ -21,6 +23,8 @@ class TestRunningTasksCount(TestCase):
|
|||||||
super().setUpClass()
|
super().setUpClass()
|
||||||
cls.factory = RequestFactory()
|
cls.factory = RequestFactory()
|
||||||
cls.user = AuthUtils.create_user("bruce_wayne")
|
cls.user = AuthUtils.create_user("bruce_wayne")
|
||||||
|
cls.user.is_superuser = True
|
||||||
|
cls.user.save()
|
||||||
|
|
||||||
def test_should_return_data(
|
def test_should_return_data(
|
||||||
self, mock_active_tasks_count, mock_queued_tasks_count
|
self, mock_active_tasks_count, mock_queued_tasks_count
|
||||||
@ -35,5 +39,164 @@ class TestRunningTasksCount(TestCase):
|
|||||||
# then
|
# then
|
||||||
self.assertEqual(response.status_code, 200)
|
self.assertEqual(response.status_code, 200)
|
||||||
self.assertDictEqual(
|
self.assertDictEqual(
|
||||||
jsonresponse_to_dict(response), {"tasks_running": 2, "tasks_queued": 3}
|
jsonresponse_to_dict(response), {
|
||||||
|
"tasks_running": 2, "tasks_queued": 3}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_su_only(
|
||||||
|
self, mock_active_tasks_count, mock_queued_tasks_count
|
||||||
|
):
|
||||||
|
self.user.is_superuser = False
|
||||||
|
self.user.save()
|
||||||
|
self.user.refresh_from_db()
|
||||||
|
# given
|
||||||
|
mock_active_tasks_count.return_value = 2
|
||||||
|
mock_queued_tasks_count.return_value = 3
|
||||||
|
request = self.factory.get("/")
|
||||||
|
request.user = self.user
|
||||||
|
# when
|
||||||
|
response = task_counts(request)
|
||||||
|
# then
|
||||||
|
self.assertEqual(response.status_code, 302)
|
||||||
|
|
||||||
|
|
||||||
|
class TestEsiCheck(TestCase):
|
||||||
|
@classmethod
|
||||||
|
def setUpClass(cls) -> None:
|
||||||
|
super().setUpClass()
|
||||||
|
cls.factory = RequestFactory()
|
||||||
|
cls.user = AuthUtils.create_user("bruce_wayne")
|
||||||
|
cls.user.is_superuser = True
|
||||||
|
cls.user.save()
|
||||||
|
|
||||||
|
@requests_mock.Mocker()
|
||||||
|
def test_401_data_returns_200(
|
||||||
|
self, m
|
||||||
|
):
|
||||||
|
error_json = {
|
||||||
|
"error": "You have been banned from using ESI. Please contact Technical Support. (support@eveonline.com)"
|
||||||
|
}
|
||||||
|
status_code = 401
|
||||||
|
m.get(
|
||||||
|
"https://esi.evetech.net/latest/status/?datasource=tranquility",
|
||||||
|
text=json.dumps(error_json),
|
||||||
|
status_code=status_code
|
||||||
|
)
|
||||||
|
# given
|
||||||
|
request = self.factory.get("/")
|
||||||
|
request.user = self.user
|
||||||
|
# when
|
||||||
|
response = esi_check(request)
|
||||||
|
# then
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
self.assertDictEqual(
|
||||||
|
jsonresponse_to_dict(response), {
|
||||||
|
"status": status_code,
|
||||||
|
"data": error_json
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
@requests_mock.Mocker()
|
||||||
|
def test_504_data_returns_200(
|
||||||
|
self, m
|
||||||
|
):
|
||||||
|
error_json = {
|
||||||
|
"error": "Gateway timeout message",
|
||||||
|
"timeout": 5000
|
||||||
|
}
|
||||||
|
status_code = 504
|
||||||
|
m.get(
|
||||||
|
"https://esi.evetech.net/latest/status/?datasource=tranquility",
|
||||||
|
text=json.dumps(error_json),
|
||||||
|
status_code=status_code
|
||||||
|
)
|
||||||
|
# given
|
||||||
|
request = self.factory.get("/")
|
||||||
|
request.user = self.user
|
||||||
|
# when
|
||||||
|
response = esi_check(request)
|
||||||
|
# then
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
self.assertDictEqual(
|
||||||
|
jsonresponse_to_dict(response), {
|
||||||
|
"status": status_code,
|
||||||
|
"data": error_json
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
@requests_mock.Mocker()
|
||||||
|
def test_420_data_override(
|
||||||
|
self, m
|
||||||
|
):
|
||||||
|
error_json = {
|
||||||
|
"error": "message from CCP",
|
||||||
|
}
|
||||||
|
status_code = 420
|
||||||
|
m.get(
|
||||||
|
"https://esi.evetech.net/latest/status/?datasource=tranquility",
|
||||||
|
text=json.dumps(error_json),
|
||||||
|
status_code=status_code
|
||||||
|
)
|
||||||
|
# given
|
||||||
|
request = self.factory.get("/")
|
||||||
|
request.user = self.user
|
||||||
|
# when
|
||||||
|
response = esi_check(request)
|
||||||
|
# then
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
self.assertNotEqual(
|
||||||
|
jsonresponse_to_dict(response)["data"],
|
||||||
|
error_json
|
||||||
|
)
|
||||||
|
self.assertDictEqual(
|
||||||
|
jsonresponse_to_dict(response), {
|
||||||
|
"status": status_code,
|
||||||
|
"data": {
|
||||||
|
"error": ESI_ERROR_MESSAGE_OVERRIDES.get(status_code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
@requests_mock.Mocker()
|
||||||
|
def test_200_data_returns_200(
|
||||||
|
self, m
|
||||||
|
):
|
||||||
|
good_json = {
|
||||||
|
"players": 5,
|
||||||
|
"server_version": "69420",
|
||||||
|
"start_time": "2030-01-01T23:59:59Z"
|
||||||
|
}
|
||||||
|
status_code = 200
|
||||||
|
|
||||||
|
m.get(
|
||||||
|
"https://esi.evetech.net/latest/status/?datasource=tranquility",
|
||||||
|
text=json.dumps(good_json),
|
||||||
|
status_code=status_code
|
||||||
|
)
|
||||||
|
# given
|
||||||
|
request = self.factory.get("/")
|
||||||
|
request.user = self.user
|
||||||
|
# when
|
||||||
|
response = esi_check(request)
|
||||||
|
# then
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
self.assertDictEqual(
|
||||||
|
jsonresponse_to_dict(response), {
|
||||||
|
"status": status_code,
|
||||||
|
"data": good_json
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_su_only(
|
||||||
|
self,
|
||||||
|
):
|
||||||
|
self.user.is_superuser = False
|
||||||
|
self.user.save()
|
||||||
|
self.user.refresh_from_db()
|
||||||
|
# given
|
||||||
|
request = self.factory.get("/")
|
||||||
|
request.user = self.user
|
||||||
|
# when
|
||||||
|
response = esi_check(request)
|
||||||
|
# then
|
||||||
|
self.assertEqual(response.status_code, 302)
|
||||||
|
@ -38,5 +38,7 @@ urlpatterns = [
|
|||||||
name='token_refresh'
|
name='token_refresh'
|
||||||
),
|
),
|
||||||
path('dashboard/', views.dashboard, name='dashboard'),
|
path('dashboard/', views.dashboard, name='dashboard'),
|
||||||
|
path('dashboard_bs3/', views.dashboard_bs3, name='dashboard_bs3'),
|
||||||
path('task-counts/', views.task_counts, name='task_counts'),
|
path('task-counts/', views.task_counts, name='task_counts'),
|
||||||
|
path('esi-check/', views.esi_check, name='esi_check'),
|
||||||
]
|
]
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import logging
|
import logging
|
||||||
from allianceauth.hooks import get_hooks
|
|
||||||
|
|
||||||
|
import requests
|
||||||
from django_registration.backends.activation.views import (
|
from django_registration.backends.activation.views import (
|
||||||
REGISTRATION_SALT, ActivationView as BaseActivationView,
|
REGISTRATION_SALT, ActivationView as BaseActivationView,
|
||||||
RegistrationView as BaseRegistrationView,
|
RegistrationView as BaseRegistrationView,
|
||||||
@ -10,7 +10,7 @@ from django_registration.signals import user_registered
|
|||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.contrib import messages
|
from django.contrib import messages
|
||||||
from django.contrib.auth import authenticate, login
|
from django.contrib.auth import authenticate, login
|
||||||
from django.contrib.auth.decorators import login_required
|
from django.contrib.auth.decorators import login_required, user_passes_test
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
from django.core import signing
|
from django.core import signing
|
||||||
from django.http import JsonResponse
|
from django.http import JsonResponse
|
||||||
@ -23,14 +23,16 @@ from esi.decorators import token_required
|
|||||||
from esi.models import Token
|
from esi.models import Token
|
||||||
|
|
||||||
from allianceauth.eveonline.models import EveCharacter
|
from allianceauth.eveonline.models import EveCharacter
|
||||||
|
from allianceauth.hooks import get_hooks
|
||||||
|
|
||||||
|
from .constants import ESI_ERROR_MESSAGE_OVERRIDES
|
||||||
from .core.celery_workers import active_tasks_count, queued_tasks_count
|
from .core.celery_workers import active_tasks_count, queued_tasks_count
|
||||||
from .forms import RegistrationForm
|
from .forms import RegistrationForm
|
||||||
from .models import CharacterOwnership
|
from .models import CharacterOwnership
|
||||||
|
|
||||||
if 'allianceauth.eveonline.autogroups' in settings.INSTALLED_APPS:
|
if 'allianceauth.eveonline.autogroups' in settings.INSTALLED_APPS:
|
||||||
_has_auto_groups = True
|
_has_auto_groups = True
|
||||||
from allianceauth.eveonline.autogroups.models import *
|
from allianceauth.eveonline.autogroups.models import * # noqa: F401, F403
|
||||||
else:
|
else:
|
||||||
_has_auto_groups = False
|
_has_auto_groups = False
|
||||||
|
|
||||||
@ -54,7 +56,7 @@ def dashboard_groups(request):
|
|||||||
context = {
|
context = {
|
||||||
'groups': groups,
|
'groups': groups,
|
||||||
}
|
}
|
||||||
return render_to_string('authentication/dashboard.groups.html', context=context, request=request)
|
return render_to_string('authentication/dashboard_groups.html', context=context, request=request)
|
||||||
|
|
||||||
|
|
||||||
def dashboard_characters(request):
|
def dashboard_characters(request):
|
||||||
@ -66,7 +68,7 @@ def dashboard_characters(request):
|
|||||||
context = {
|
context = {
|
||||||
'characters': characters
|
'characters': characters
|
||||||
}
|
}
|
||||||
return render_to_string('authentication/dashboard.characters.html', context=context, request=request)
|
return render_to_string('authentication/dashboard_characters.html', context=context, request=request)
|
||||||
|
|
||||||
|
|
||||||
def dashboard_admin(request):
|
def dashboard_admin(request):
|
||||||
@ -76,6 +78,13 @@ def dashboard_admin(request):
|
|||||||
return ""
|
return ""
|
||||||
|
|
||||||
|
|
||||||
|
def dashboard_esi_check(request):
|
||||||
|
if request.user.is_superuser:
|
||||||
|
return render_to_string('allianceauth/admin-status/esi_check.html', request=request)
|
||||||
|
else:
|
||||||
|
return ""
|
||||||
|
|
||||||
|
|
||||||
@login_required
|
@login_required
|
||||||
def dashboard(request):
|
def dashboard(request):
|
||||||
_dash_items = list()
|
_dash_items = list()
|
||||||
@ -135,23 +144,30 @@ def token_refresh(request, token_id=None):
|
|||||||
@login_required
|
@login_required
|
||||||
@token_required(scopes=settings.LOGIN_TOKEN_SCOPES)
|
@token_required(scopes=settings.LOGIN_TOKEN_SCOPES)
|
||||||
def main_character_change(request, token):
|
def main_character_change(request, token):
|
||||||
logger.debug(f"main_character_change called by user {request.user} for character {token.character_name}")
|
logger.debug(
|
||||||
|
f"main_character_change called by user {request.user} for character {token.character_name}")
|
||||||
try:
|
try:
|
||||||
co = CharacterOwnership.objects.get(character__character_id=token.character_id, user=request.user)
|
co = CharacterOwnership.objects.get(
|
||||||
|
character__character_id=token.character_id, user=request.user)
|
||||||
except CharacterOwnership.DoesNotExist:
|
except CharacterOwnership.DoesNotExist:
|
||||||
if not CharacterOwnership.objects.filter(character__character_id=token.character_id).exists():
|
if not CharacterOwnership.objects.filter(character__character_id=token.character_id).exists():
|
||||||
co = CharacterOwnership.objects.create_by_token(token)
|
co = CharacterOwnership.objects.create_by_token(token)
|
||||||
else:
|
else:
|
||||||
messages.error(
|
messages.error(
|
||||||
request,
|
request,
|
||||||
_('Cannot change main character to %(char)s: character owned by a different account.') % ({'char': token.character_name})
|
_('Cannot change main character to %(char)s: character owned by a different account.') % (
|
||||||
|
{'char': token.character_name})
|
||||||
)
|
)
|
||||||
co = None
|
co = None
|
||||||
if co:
|
if co:
|
||||||
request.user.profile.main_character = co.character
|
request.user.profile.main_character = co.character
|
||||||
request.user.profile.save(update_fields=['main_character'])
|
request.user.profile.save(update_fields=['main_character'])
|
||||||
messages.success(request, _('Changed main character to %(char)s') % {"char": co.character})
|
messages.success(request, _('Changed main character to %s') % co.character)
|
||||||
logger.info('Changed user %(user)s main character to %(char)s' % ({'user': request.user, 'char': co.character}))
|
logger.info(
|
||||||
|
'Changed user {user} main character to {char}'.format(
|
||||||
|
user=request.user, char=co.character
|
||||||
|
)
|
||||||
|
)
|
||||||
return redirect("authentication:dashboard")
|
return redirect("authentication:dashboard")
|
||||||
|
|
||||||
|
|
||||||
@ -159,9 +175,11 @@ def main_character_change(request, token):
|
|||||||
def add_character(request, token):
|
def add_character(request, token):
|
||||||
if CharacterOwnership.objects.filter(character__character_id=token.character_id).filter(
|
if CharacterOwnership.objects.filter(character__character_id=token.character_id).filter(
|
||||||
owner_hash=token.character_owner_hash).filter(user=request.user).exists():
|
owner_hash=token.character_owner_hash).filter(user=request.user).exists():
|
||||||
messages.success(request, _('Added %(name)s to your account.' % ({'name': token.character_name})))
|
messages.success(request, _(
|
||||||
|
'Added %(name)s to your account.' % ({'name': token.character_name})))
|
||||||
else:
|
else:
|
||||||
messages.error(request, _('Failed to add %(name)s to your account: they already have an account.' % ({'name': token.character_name})))
|
messages.error(request, _('Failed to add %(name)s to your account: they already have an account.' % (
|
||||||
|
{'name': token.character_name})))
|
||||||
return redirect('authentication:dashboard')
|
return redirect('authentication:dashboard')
|
||||||
|
|
||||||
|
|
||||||
@ -200,7 +218,15 @@ def sso_login(request, token):
|
|||||||
request.session['registration_uid'] = user.pk
|
request.session['registration_uid'] = user.pk
|
||||||
# Go to Step 2
|
# Go to Step 2
|
||||||
return redirect('registration_register')
|
return redirect('registration_register')
|
||||||
messages.error(request, _('Unable to authenticate as the selected character.'))
|
# Logging in with an alt is not allowed due to security concerns.
|
||||||
|
token.delete()
|
||||||
|
messages.error(
|
||||||
|
request,
|
||||||
|
_(
|
||||||
|
'Unable to authenticate as the selected character. '
|
||||||
|
'Please log in with the main character associated with this account.'
|
||||||
|
)
|
||||||
|
)
|
||||||
return redirect(settings.LOGIN_URL)
|
return redirect(settings.LOGIN_URL)
|
||||||
|
|
||||||
|
|
||||||
@ -272,7 +298,8 @@ class RegistrationView(BaseRegistrationView):
|
|||||||
return super().dispatch(request, *args, **kwargs)
|
return super().dispatch(request, *args, **kwargs)
|
||||||
|
|
||||||
def register(self, form):
|
def register(self, form):
|
||||||
user = User.objects.get(pk=self.request.session.get('registration_uid'))
|
user = User.objects.get(
|
||||||
|
pk=self.request.session.get('registration_uid'))
|
||||||
user.email = form.cleaned_data['email']
|
user.email = form.cleaned_data['email']
|
||||||
user_registered.send(self.__class__, user=user, request=self.request)
|
user_registered.send(self.__class__, user=user, request=self.request)
|
||||||
if getattr(settings, 'REGISTRATION_VERIFY_EMAIL', True):
|
if getattr(settings, 'REGISTRATION_VERIFY_EMAIL', True):
|
||||||
@ -289,7 +316,8 @@ class RegistrationView(BaseRegistrationView):
|
|||||||
|
|
||||||
def get_email_context(self, activation_key):
|
def get_email_context(self, activation_key):
|
||||||
context = super().get_email_context(activation_key)
|
context = super().get_email_context(activation_key)
|
||||||
context['url'] = context['site'].domain + reverse('registration_activate', args=[activation_key])
|
context['url'] = context['site'].domain + \
|
||||||
|
reverse('registration_activate', args=[activation_key])
|
||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
@ -322,20 +350,24 @@ class ActivationView(BaseActivationView):
|
|||||||
|
|
||||||
|
|
||||||
def registration_complete(request):
|
def registration_complete(request):
|
||||||
messages.success(request, _('Sent confirmation email. Please follow the link to confirm your email address.'))
|
messages.success(request, _(
|
||||||
|
'Sent confirmation email. Please follow the link to confirm your email address.'))
|
||||||
return redirect('authentication:login')
|
return redirect('authentication:login')
|
||||||
|
|
||||||
|
|
||||||
def activation_complete(request):
|
def activation_complete(request):
|
||||||
messages.success(request, _('Confirmed your email address. Please login to continue.'))
|
messages.success(request, _(
|
||||||
|
'Confirmed your email address. Please login to continue.'))
|
||||||
return redirect('authentication:dashboard')
|
return redirect('authentication:dashboard')
|
||||||
|
|
||||||
|
|
||||||
def registration_closed(request):
|
def registration_closed(request):
|
||||||
messages.error(request, _('Registration of new accounts is not allowed at this time.'))
|
messages.error(request, _(
|
||||||
|
'Registration of new accounts is not allowed at this time.'))
|
||||||
return redirect('authentication:login')
|
return redirect('authentication:login')
|
||||||
|
|
||||||
|
|
||||||
|
@user_passes_test(lambda u: u.is_superuser)
|
||||||
def task_counts(request) -> JsonResponse:
|
def task_counts(request) -> JsonResponse:
|
||||||
"""Return task counts as JSON for an AJAX call."""
|
"""Return task counts as JSON for an AJAX call."""
|
||||||
data = {
|
data = {
|
||||||
@ -343,3 +375,31 @@ def task_counts(request) -> JsonResponse:
|
|||||||
"tasks_queued": queued_tasks_count()
|
"tasks_queued": queued_tasks_count()
|
||||||
}
|
}
|
||||||
return JsonResponse(data)
|
return JsonResponse(data)
|
||||||
|
|
||||||
|
|
||||||
|
def check_for_override_esi_error_message(response):
|
||||||
|
if response.status_code in ESI_ERROR_MESSAGE_OVERRIDES:
|
||||||
|
return {"error": ESI_ERROR_MESSAGE_OVERRIDES.get(response.status_code)}
|
||||||
|
else:
|
||||||
|
return response.json()
|
||||||
|
|
||||||
|
|
||||||
|
@user_passes_test(lambda u: u.is_superuser)
|
||||||
|
def esi_check(request) -> JsonResponse:
|
||||||
|
"""Return if ESI ok With error messages and codes as JSON"""
|
||||||
|
_r = requests.get("https://esi.evetech.net/latest/status/?datasource=tranquility")
|
||||||
|
|
||||||
|
data = {
|
||||||
|
"status": _r.status_code,
|
||||||
|
"data": check_for_override_esi_error_message(_r)
|
||||||
|
}
|
||||||
|
return JsonResponse(data)
|
||||||
|
|
||||||
|
|
||||||
|
@login_required
|
||||||
|
def dashboard_bs3(request):
|
||||||
|
"""Render dashboard view with BS3 theme.
|
||||||
|
|
||||||
|
This is an internal view used for testing BS3 backward compatibility in AA4 only.
|
||||||
|
"""
|
||||||
|
return render(request, 'authentication/dashboard_bs3.html')
|
||||||
|
@ -12,13 +12,14 @@ class StartProject(BaseStartProject):
|
|||||||
parser.add_argument('--python', help='The path to the python executable.')
|
parser.add_argument('--python', help='The path to the python executable.')
|
||||||
parser.add_argument('--celery', help='The path to the celery executable.')
|
parser.add_argument('--celery', help='The path to the celery executable.')
|
||||||
parser.add_argument('--gunicorn', help='The path to the gunicorn executable.')
|
parser.add_argument('--gunicorn', help='The path to the gunicorn executable.')
|
||||||
|
parser.add_argument('--memmon', help='The path to the memmon executable.')
|
||||||
|
|
||||||
|
|
||||||
def create_project(parser, options, args):
|
def create_project(parser, options, args):
|
||||||
# Validate args
|
# Validate args
|
||||||
if len(args) < 2:
|
if len(args) < 2:
|
||||||
parser.error("Please specify a name for your Alliance Auth installation.")
|
parser.error("Please specify a name for your Alliance Auth installation.")
|
||||||
elif len(args) > 3:
|
elif len(args) > 4:
|
||||||
parser.error("Too many arguments.")
|
parser.error("Too many arguments.")
|
||||||
|
|
||||||
# First find the path to Alliance Auth
|
# First find the path to Alliance Auth
|
||||||
@ -32,6 +33,7 @@ def create_project(parser, options, args):
|
|||||||
'python': shutil.which('python'),
|
'python': shutil.which('python'),
|
||||||
'gunicorn': shutil.which('gunicorn'),
|
'gunicorn': shutil.which('gunicorn'),
|
||||||
'celery': shutil.which('celery'),
|
'celery': shutil.which('celery'),
|
||||||
|
'memmon': shutil.which('memmon'),
|
||||||
'extensions': ['py', 'conf', 'json'],
|
'extensions': ['py', 'conf', 'json'],
|
||||||
}
|
}
|
||||||
|
|
||||||
|
562
allianceauth/checks.py
Normal file
562
allianceauth/checks.py
Normal file
@ -0,0 +1,562 @@
|
|||||||
|
"""
|
||||||
|
Django system checks for Alliance Auth
|
||||||
|
"""
|
||||||
|
|
||||||
|
from typing import List
|
||||||
|
from django import db
|
||||||
|
from django.core.checks import CheckMessage, Error, register, Warning
|
||||||
|
from allianceauth.utils.cache import get_redis_client
|
||||||
|
from django.utils import timezone
|
||||||
|
from packaging.version import InvalidVersion, Version as Pep440Version
|
||||||
|
from celery import current_app
|
||||||
|
from django.conf import settings
|
||||||
|
from sqlite3.dbapi2 import sqlite_version_info
|
||||||
|
|
||||||
|
"""
|
||||||
|
A = System Packages
|
||||||
|
B = Configuration
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
@register()
|
||||||
|
def django_settings(app_configs, **kwargs) -> List[CheckMessage]:
|
||||||
|
"""
|
||||||
|
Check that Django settings are correctly configured
|
||||||
|
|
||||||
|
:param app_configs:
|
||||||
|
:type app_configs:
|
||||||
|
:param kwargs:
|
||||||
|
:type kwargs:
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
errors: List[CheckMessage] = []
|
||||||
|
|
||||||
|
# Check for SITE_URL
|
||||||
|
if hasattr(settings, "SITE_URL"):
|
||||||
|
# Check if SITE_URL is empty
|
||||||
|
if settings.SITE_URL == "":
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg="'SITE_URL' is empty.",
|
||||||
|
hint="Make sure to set 'SITE_URL' to the URL of your Auth instance. (Without trailing slash)",
|
||||||
|
id="allianceauth.checks.B011",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
# Check if SITE_URL has a trailing slash
|
||||||
|
elif settings.SITE_URL[-1] == "/":
|
||||||
|
errors.append(
|
||||||
|
Warning(
|
||||||
|
msg="'SITE_URL' has a trailing slash. This may lead to incorrect links being generated by Auth.",
|
||||||
|
hint="",
|
||||||
|
id="allianceauth.checks.B005",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
# SITE_URL not found
|
||||||
|
else:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg="No 'SITE_URL' found is settings. This may lead to incorrect links being generated by Auth or Errors in 3rd party modules.",
|
||||||
|
hint="",
|
||||||
|
id="allianceauth.checks.B006",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
# Check for CSRF_TRUSTED_ORIGINS
|
||||||
|
if hasattr(settings, "CSRF_TRUSTED_ORIGINS") and hasattr(settings, "SITE_URL"):
|
||||||
|
# Check if SITE_URL is not in CSRF_TRUSTED_ORIGINS
|
||||||
|
if settings.SITE_URL not in settings.CSRF_TRUSTED_ORIGINS:
|
||||||
|
errors.append(
|
||||||
|
Warning(
|
||||||
|
msg="'SITE_URL' not found in 'CSRF_TRUSTED_ORIGINS'. Auth may not load pages correctly until this is rectified.",
|
||||||
|
hint="",
|
||||||
|
id="allianceauth.checks.B007",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
# CSRF_TRUSTED_ORIGINS not found
|
||||||
|
else:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg="No 'CSRF_TRUSTED_ORIGINS' found is settings, Auth may not load pages correctly until this is rectified",
|
||||||
|
hint="",
|
||||||
|
id="allianceauth.checks.B008",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
# Check for ESI_USER_CONTACT_EMAIL
|
||||||
|
if hasattr(settings, "ESI_USER_CONTACT_EMAIL"):
|
||||||
|
# Check if ESI_USER_CONTACT_EMAIL is empty
|
||||||
|
if settings.ESI_USER_CONTACT_EMAIL == "":
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg="'ESI_USER_CONTACT_EMAIL' is empty. A valid email is required as maintainer contact for CCP.",
|
||||||
|
hint="",
|
||||||
|
id="allianceauth.checks.B009",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
# ESI_USER_CONTACT_EMAIL not found
|
||||||
|
else:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg="No 'ESI_USER_CONTACT_EMAIL' found is settings. A valid email is required as maintainer contact for CCP.",
|
||||||
|
hint="",
|
||||||
|
id="allianceauth.checks.B010",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
|
|
||||||
|
@register()
|
||||||
|
def system_package_redis(app_configs, **kwargs) -> List[CheckMessage]:
|
||||||
|
"""
|
||||||
|
Check that Redis is a supported version
|
||||||
|
|
||||||
|
:param app_configs:
|
||||||
|
:type app_configs:
|
||||||
|
:param kwargs:
|
||||||
|
:type kwargs:
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
allianceauth_redis_install_link = "https://allianceauth.readthedocs.io/en/latest/installation/allianceauth.html#redis-and-other-tools"
|
||||||
|
|
||||||
|
errors: List[CheckMessage] = []
|
||||||
|
|
||||||
|
try:
|
||||||
|
redis_version = Pep440Version(get_redis_client().info()["redis_version"])
|
||||||
|
except InvalidVersion:
|
||||||
|
errors.append(Warning("Unable to confirm Redis Version"))
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
|
if (
|
||||||
|
redis_version.major == 7
|
||||||
|
and redis_version.minor == 2
|
||||||
|
and timezone.now()
|
||||||
|
> timezone.datetime(year=2025, month=8, day=31, tzinfo=timezone.utc)
|
||||||
|
):
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"Redis {redis_version.public} in Security Support only, Updating Suggested",
|
||||||
|
hint=allianceauth_redis_install_link,
|
||||||
|
id="allianceauth.checks.A001",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif redis_version.major == 7 and redis_version.minor == 0:
|
||||||
|
errors.append(
|
||||||
|
Warning(
|
||||||
|
msg=f"Redis {redis_version.public} in Security Support only, Updating Suggested",
|
||||||
|
hint=allianceauth_redis_install_link,
|
||||||
|
id="allianceauth.checks.A002",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif redis_version.major == 6 and redis_version.minor == 2:
|
||||||
|
errors.append(
|
||||||
|
Warning(
|
||||||
|
msg=f"Redis {redis_version.public} in Security Support only, Updating Suggested",
|
||||||
|
hint=allianceauth_redis_install_link,
|
||||||
|
id="allianceauth.checks.A018",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif redis_version.major in [6, 5]:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"Redis {redis_version.public} EOL",
|
||||||
|
hint=allianceauth_redis_install_link,
|
||||||
|
id="allianceauth.checks.A003",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
|
|
||||||
|
@register()
|
||||||
|
def system_package_mysql(app_configs, **kwargs) -> List[CheckMessage]:
|
||||||
|
"""
|
||||||
|
Check that MySQL is a supported version
|
||||||
|
|
||||||
|
:param app_configs:
|
||||||
|
:type app_configs:
|
||||||
|
:param kwargs:
|
||||||
|
:type kwargs:
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
mysql_quick_guide_link = "https://dev.mysql.com/doc/mysql-apt-repo-quick-guide/en/"
|
||||||
|
|
||||||
|
errors: List[CheckMessage] = []
|
||||||
|
|
||||||
|
for connection in db.connections.all():
|
||||||
|
if connection.vendor == "mysql":
|
||||||
|
try:
|
||||||
|
mysql_version = Pep440Version(
|
||||||
|
".".join(str(i) for i in connection.mysql_version)
|
||||||
|
)
|
||||||
|
except InvalidVersion:
|
||||||
|
errors.append(Warning("Unable to confirm MySQL Version"))
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
|
# MySQL 8
|
||||||
|
if mysql_version.major == 8:
|
||||||
|
if mysql_version.minor == 4 and timezone.now() > timezone.datetime(
|
||||||
|
year=2032, month=4, day=30, tzinfo=timezone.utc
|
||||||
|
):
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MySQL {mysql_version.public} EOL",
|
||||||
|
hint=mysql_quick_guide_link,
|
||||||
|
id="allianceauth.checks.A004",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif mysql_version.minor == 3:
|
||||||
|
errors.append(
|
||||||
|
Warning(
|
||||||
|
msg=f"MySQL {mysql_version.public} Non LTS",
|
||||||
|
hint=mysql_quick_guide_link,
|
||||||
|
id="allianceauth.checks.A005",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif mysql_version.minor == 2:
|
||||||
|
errors.append(
|
||||||
|
Warning(
|
||||||
|
msg=f"MySQL {mysql_version.public} Non LTS",
|
||||||
|
hint=mysql_quick_guide_link,
|
||||||
|
id="allianceauth.checks.A006",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif mysql_version.minor == 1:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MySQL {mysql_version.public} EOL",
|
||||||
|
hint=mysql_quick_guide_link,
|
||||||
|
id="allianceauth.checks.A007",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif mysql_version.minor == 0 and timezone.now() > timezone.datetime(
|
||||||
|
year=2026, month=4, day=30, tzinfo=timezone.utc
|
||||||
|
):
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MySQL {mysql_version.public} EOL",
|
||||||
|
hint=mysql_quick_guide_link,
|
||||||
|
id="allianceauth.checks.A008",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
# MySQL below 8
|
||||||
|
# This will also catch Mariadb 5.x
|
||||||
|
elif mysql_version.major < 8:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MySQL or MariaDB {mysql_version.public} EOL",
|
||||||
|
hint=mysql_quick_guide_link,
|
||||||
|
id="allianceauth.checks.A009",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
|
|
||||||
|
@register()
|
||||||
|
def system_package_mariadb(app_configs, **kwargs) -> List[CheckMessage]:
|
||||||
|
"""
|
||||||
|
Check that MariaDB is a supported version
|
||||||
|
|
||||||
|
:param app_configs:
|
||||||
|
:type app_configs:
|
||||||
|
:param kwargs:
|
||||||
|
:type kwargs:
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
mariadb_download_link = "https://mariadb.org/download/?t=repo-config"
|
||||||
|
|
||||||
|
errors: List[CheckMessage] = []
|
||||||
|
|
||||||
|
for connection in db.connections.all():
|
||||||
|
# TODO: Find a way to determine MySQL vs. MariaDB
|
||||||
|
if connection.vendor == "mysql":
|
||||||
|
try:
|
||||||
|
mariadb_version = Pep440Version(
|
||||||
|
".".join(str(i) for i in connection.mysql_version)
|
||||||
|
)
|
||||||
|
except InvalidVersion:
|
||||||
|
errors.append(Warning("Unable to confirm MariaDB Version"))
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
|
# MariaDB 11
|
||||||
|
if mariadb_version.major == 11:
|
||||||
|
if mariadb_version.minor == 4 and timezone.now() > timezone.datetime(
|
||||||
|
year=2029, month=5, day=19, tzinfo=timezone.utc
|
||||||
|
):
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MariaDB {mariadb_version.public} EOL",
|
||||||
|
hint=mariadb_download_link,
|
||||||
|
id="allianceauth.checks.A010",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif mariadb_version.minor == 2:
|
||||||
|
errors.append(
|
||||||
|
Warning(
|
||||||
|
msg=f"MariaDB {mariadb_version.public} Non LTS",
|
||||||
|
hint=mariadb_download_link,
|
||||||
|
id="allianceauth.checks.A018",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
if timezone.now() > timezone.datetime(
|
||||||
|
year=2024, month=11, day=21, tzinfo=timezone.utc
|
||||||
|
):
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MariaDB {mariadb_version.public} EOL",
|
||||||
|
hint=mariadb_download_link,
|
||||||
|
id="allianceauth.checks.A011",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif mariadb_version.minor == 1:
|
||||||
|
errors.append(
|
||||||
|
Warning(
|
||||||
|
msg=f"MariaDB {mariadb_version.public} Non LTS",
|
||||||
|
hint=mariadb_download_link,
|
||||||
|
id="allianceauth.checks.A019",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
if timezone.now() > timezone.datetime(
|
||||||
|
year=2024, month=8, day=21, tzinfo=timezone.utc
|
||||||
|
):
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MariaDB {mariadb_version.public} EOL",
|
||||||
|
hint=mariadb_download_link,
|
||||||
|
id="allianceauth.checks.A012",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
# Demote versions down here once EOL
|
||||||
|
elif mariadb_version.minor in [0, 3]:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MariaDB {mariadb_version.public} EOL",
|
||||||
|
hint=mariadb_download_link,
|
||||||
|
id="allianceauth.checks.A013",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
# MariaDB 10
|
||||||
|
elif mariadb_version.major == 10:
|
||||||
|
if mariadb_version.minor == 11 and timezone.now() > timezone.datetime(
|
||||||
|
year=2028, month=2, day=10, tzinfo=timezone.utc
|
||||||
|
):
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MariaDB {mariadb_version.public} EOL",
|
||||||
|
hint=mariadb_download_link,
|
||||||
|
id="allianceauth.checks.A014",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif mariadb_version.minor == 6 and timezone.now() > timezone.datetime(
|
||||||
|
year=2026, month=7, day=6, tzinfo=timezone.utc
|
||||||
|
):
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MariaDB {mariadb_version.public} EOL",
|
||||||
|
hint=mariadb_download_link,
|
||||||
|
id="allianceauth.checks.A0015",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
elif mariadb_version.minor == 5 and timezone.now() > timezone.datetime(
|
||||||
|
year=2025, month=6, day=24, tzinfo=timezone.utc
|
||||||
|
):
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MariaDB {mariadb_version.public} EOL",
|
||||||
|
hint=mariadb_download_link,
|
||||||
|
id="allianceauth.checks.A016",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
# Demote versions down here once EOL
|
||||||
|
elif mariadb_version.minor in [0, 1, 2, 3, 4, 7, 9, 10]:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"MariaDB {mariadb_version.public} EOL",
|
||||||
|
hint=mariadb_download_link,
|
||||||
|
id="allianceauth.checks.A017",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
|
|
||||||
|
@register()
|
||||||
|
def system_package_sqlite(app_configs, **kwargs) -> List[CheckMessage]:
|
||||||
|
"""
|
||||||
|
Check that SQLite is a supported version
|
||||||
|
|
||||||
|
:param app_configs:
|
||||||
|
:type app_configs:
|
||||||
|
:param kwargs:
|
||||||
|
:type kwargs:
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
errors: List[CheckMessage] = []
|
||||||
|
|
||||||
|
for connection in db.connections.all():
|
||||||
|
if connection.vendor == "sqlite":
|
||||||
|
try:
|
||||||
|
sqlite_version = Pep440Version(
|
||||||
|
".".join(str(i) for i in sqlite_version_info)
|
||||||
|
)
|
||||||
|
except InvalidVersion:
|
||||||
|
errors.append(Warning("Unable to confirm SQLite Version"))
|
||||||
|
|
||||||
|
return errors
|
||||||
|
if sqlite_version.major == 3 and sqlite_version.minor < 27:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"SQLite {sqlite_version.public} Unsupported by Django",
|
||||||
|
hint="https://pkgs.org/download/sqlite3",
|
||||||
|
id="allianceauth.checks.A020",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
|
|
||||||
|
@register()
|
||||||
|
def sql_settings(app_configs, **kwargs) -> List[CheckMessage]:
|
||||||
|
"""
|
||||||
|
Check that SQL settings are correctly configured
|
||||||
|
|
||||||
|
:param app_configs:
|
||||||
|
:type app_configs:
|
||||||
|
:param kwargs:
|
||||||
|
:type kwargs:
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
errors: List[CheckMessage] = []
|
||||||
|
|
||||||
|
for connection in db.connections.all():
|
||||||
|
if connection.vendor == "mysql":
|
||||||
|
try:
|
||||||
|
if connection.settings_dict["OPTIONS"]["charset"] != "utf8mb4":
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"SQL Charset is not set to utf8mb4 DB: {connection.alias}",
|
||||||
|
hint="https://gitlab.com/allianceauth/allianceauth/-/commit/89be2456fb2d741b86417e889da9b6129525bec8",
|
||||||
|
id="allianceauth.checks.B001",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
except KeyError:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg=f"SQL Charset is not set to utf8mb4 DB: {connection.alias}",
|
||||||
|
hint="https://gitlab.com/allianceauth/allianceauth/-/commit/89be2456fb2d741b86417e889da9b6129525bec8",
|
||||||
|
id="allianceauth.checks.B001",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
# This hasn't actually been set on AA yet
|
||||||
|
# try:
|
||||||
|
# if (
|
||||||
|
# connection.settings_dict["OPTIONS"]["collation"]
|
||||||
|
# != "utf8mb4_unicode_ci"
|
||||||
|
# ):
|
||||||
|
# errors.append(
|
||||||
|
# Error(
|
||||||
|
# msg=f"SQL Collation is not set to utf8mb4_unicode_ci DB:{connection.alias}",
|
||||||
|
# hint="https://gitlab.com/allianceauth/allianceauth/-/commit/89be2456fb2d741b86417e889da9b6129525bec8",
|
||||||
|
# id="allianceauth.checks.B001",
|
||||||
|
# )
|
||||||
|
# )
|
||||||
|
# except KeyError:
|
||||||
|
# errors.append(
|
||||||
|
# Error(
|
||||||
|
# msg=f"SQL Collation is not set to utf8mb4_unicode_ci DB:{connection.alias}",
|
||||||
|
# hint="https://gitlab.com/allianceauth/allianceauth/-/commit/89be2456fb2d741b86417e889da9b6129525bec8",
|
||||||
|
# id="allianceauth.checks.B001",
|
||||||
|
# )
|
||||||
|
# )
|
||||||
|
|
||||||
|
# if connection.vendor == "sqlite":
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
|
|
||||||
|
@register()
|
||||||
|
def celery_settings(app_configs, **kwargs) -> List[CheckMessage]:
|
||||||
|
"""
|
||||||
|
Check that Celery settings are correctly configured
|
||||||
|
|
||||||
|
:param app_configs:
|
||||||
|
:type app_configs:
|
||||||
|
:param kwargs:
|
||||||
|
:type kwargs:
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
errors: List[CheckMessage] = []
|
||||||
|
|
||||||
|
try:
|
||||||
|
if current_app.conf.broker_transport_options != {
|
||||||
|
"priority_steps": [0, 1, 2, 3, 4, 5, 6, 7, 8, 9],
|
||||||
|
"queue_order_strategy": "priority",
|
||||||
|
}:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg="Celery Priorities are not set correctly",
|
||||||
|
hint="https://gitlab.com/allianceauth/allianceauth/-/commit/8861ec0a61790eca0261f1adc1cc04ca5f243cbc",
|
||||||
|
id="allianceauth.checks.B003",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
except KeyError:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg="Celery Priorities are not set",
|
||||||
|
hint="https://gitlab.com/allianceauth/allianceauth/-/commit/8861ec0a61790eca0261f1adc1cc04ca5f243cbc",
|
||||||
|
id="allianceauth.checks.B003",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
if not current_app.conf.broker_connection_retry_on_startup:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg="Celery broker_connection_retry_on_startup not set correctly",
|
||||||
|
hint="https://gitlab.com/allianceauth/allianceauth/-/commit/380c41400b535447839e5552df2410af35a75280",
|
||||||
|
id="allianceauth.checks.B004",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
except KeyError:
|
||||||
|
errors.append(
|
||||||
|
Error(
|
||||||
|
msg="Celery broker_connection_retry_on_startup not set",
|
||||||
|
hint="https://gitlab.com/allianceauth/allianceauth/-/commit/380c41400b535447839e5552df2410af35a75280",
|
||||||
|
id="allianceauth.checks.B004",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
|
|
||||||
|
# IDEAS
|
||||||
|
|
||||||
|
# Any other celery things weve manually changed over the years
|
||||||
|
# I'd be happy to add Community App checks, old versions the owners dont want to support etc.
|
||||||
|
|
||||||
|
|
||||||
|
# Check Default Collation on DB
|
||||||
|
# Check Charset Collation on all tables
|
@ -1,5 +1,5 @@
|
|||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from .views import NightModeRedirectView, ThemeRedirectView
|
from .views import NightModeRedirectView
|
||||||
|
|
||||||
|
|
||||||
def auth_settings(request):
|
def auth_settings(request):
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
from django.apps import AppConfig
|
from django.apps import AppConfig
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
|
||||||
class CorpUtilsConfig(AppConfig):
|
class CorpUtilsConfig(AppConfig):
|
||||||
name = 'allianceauth.corputils'
|
name = 'allianceauth.corputils'
|
||||||
label = 'corputils'
|
label = 'corputils'
|
||||||
|
verbose_name = _('Corporation Stats')
|
||||||
|
@ -10,7 +10,7 @@ class CorpStats(MenuItemHook):
|
|||||||
MenuItemHook.__init__(
|
MenuItemHook.__init__(
|
||||||
self,
|
self,
|
||||||
_('Corporation Stats'),
|
_('Corporation Stats'),
|
||||||
'fas fa-share-alt fa-fw',
|
'fa-solid fa-share-nodes',
|
||||||
'corputils:view',
|
'corputils:view',
|
||||||
navactive=['corputils:']
|
navactive=['corputils:']
|
||||||
)
|
)
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
|
from typing import ClassVar
|
||||||
|
|
||||||
from allianceauth.authentication.models import CharacterOwnership, UserProfile
|
from allianceauth.authentication.models import CharacterOwnership, UserProfile
|
||||||
from bravado.exception import HTTPForbidden
|
from bravado.exception import HTTPForbidden
|
||||||
@ -40,9 +41,9 @@ class CorpStats(models.Model):
|
|||||||
verbose_name = "corp stats"
|
verbose_name = "corp stats"
|
||||||
verbose_name_plural = "corp stats"
|
verbose_name_plural = "corp stats"
|
||||||
|
|
||||||
objects = CorpStatsManager()
|
objects: ClassVar[CorpStatsManager] = CorpStatsManager()
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self) -> str:
|
||||||
return f"{self.__class__.__name__} for {self.corp}"
|
return f"{self.__class__.__name__} for {self.corp}"
|
||||||
|
|
||||||
def update(self):
|
def update(self):
|
||||||
|
@ -1,40 +1,63 @@
|
|||||||
{% extends "allianceauth/base-bs5.html" %}
|
{% extends "allianceauth/base-bs5.html" %}
|
||||||
|
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block page_title %}
|
{% block page_title %}
|
||||||
{% translate "Corporation Member Data" %}
|
{% translate "Corporation Member Data" %}
|
||||||
{% endblock page_title %}
|
{% endblock page_title %}
|
||||||
{% block content %}
|
|
||||||
<div class="col-lg-12">
|
{% block header_nav_brand %}
|
||||||
<h1 class="page-header text-center">{% translate "Corporation Member Data" %}</h1>
|
{% translate "Corporation Member Data" %}
|
||||||
<div class="col-lg-10 col-lg-offset-1 container">
|
{% endblock header_nav_brand %}
|
||||||
<nav class="navbar navbar-default">
|
|
||||||
<div class="container-fluid">
|
{% block header_nav_collapse_left %}
|
||||||
<ul class="nav navbar-nav">
|
<li class="nav-item dropdown">
|
||||||
<li class="dropdown">
|
<a class="nav-link dropdown-toggle" data-bs-toggle="dropdown" href="#" role="button" aria-expanded="false">
|
||||||
<a href="#" id="dLabel" class="dropdown-toggle" role="button" data-toggle="dropdown" aria-haspopup="false" aria-expanded="false">{% translate "Corporations" %}<span class="caret"></span></a>
|
{% translate "Corporations" %}
|
||||||
<ul class="dropdown-menu" role="menu" aria-labelledby="dLabel">
|
</a>
|
||||||
|
|
||||||
|
<ul class="dropdown-menu">
|
||||||
{% for corpstat in available %}
|
{% for corpstat in available %}
|
||||||
<li>
|
<li>
|
||||||
<a href="{% url 'corputils:view_corp' corpstat.corp.corporation_id %}">{{ corpstat.corp.corporation_name }}</a>
|
<a class="dropdown-item" href="{% url 'corputils:view_corp' corpstat.corp.corporation_id %}">
|
||||||
|
{{ corpstat.corp.corporation_name }}
|
||||||
|
</a>
|
||||||
</li>
|
</li>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</ul>
|
|
||||||
</li>
|
|
||||||
{% if perms.corputils.add_corpstats %}
|
{% if perms.corputils.add_corpstats %}
|
||||||
|
{% if available.count >= 1 %}
|
||||||
|
<li> </li>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
<li>
|
<li>
|
||||||
<a href="{% url 'corputils:add' %}">{% translate "Add" %}</a>
|
<a class="dropdown-item" href="{% url 'corputils:add' %}">
|
||||||
|
{% translate "Add corporation" %}
|
||||||
|
</a>
|
||||||
</li>
|
</li>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</ul>
|
</ul>
|
||||||
|
</li>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block header_nav_collapse_right %}
|
||||||
|
<li class="nav-item">
|
||||||
<form class="navbar-form navbar-right" role="search" action="{% url 'corputils:search' %}" method="GET">
|
<form class="navbar-form navbar-right" role="search" action="{% url 'corputils:search' %}" method="GET">
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<input type="text" class="form-control" name="search_string" placeholder="{% if search_string %}{{ search_string }}{% else %}{% translate 'Search all corporations...' %}{% endif %}">
|
<input
|
||||||
|
type="text"
|
||||||
|
class="form-control"
|
||||||
|
name="search_string"
|
||||||
|
placeholder="{% if search_string %}{{ search_string }}{% else %}{% translate 'Search all corporations...' %}{% endif %}"
|
||||||
|
>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</li>
|
||||||
</nav>
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<div>
|
||||||
{% block member_data %}
|
{% block member_data %}
|
||||||
{% endblock member_data %}
|
{% endblock member_data %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
|
||||||
{% endblock content %}
|
{% endblock content %}
|
||||||
|
@ -1,93 +1,144 @@
|
|||||||
{% extends 'corputils/base.html' %}
|
{% extends 'corputils/base.html' %}
|
||||||
|
|
||||||
|
{% load aa_i18n %}
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
{% load humanize %}
|
{% load humanize %}
|
||||||
|
|
||||||
{% block member_data %}
|
{% block member_data %}
|
||||||
{% if corpstats %}
|
{% if corpstats %}
|
||||||
<div class="row">
|
<div>
|
||||||
<div class="col-lg-12 text-center">
|
<table class="table text-center">
|
||||||
<table class="table">
|
|
||||||
<tr>
|
<tr>
|
||||||
<td class="text-center col-lg-6{% if corpstats.corp.alliance %}{% else %}col-lg-offset-3{% endif %}">
|
<td>
|
||||||
<img class="ra-avatar" src="{{ corpstats.corp.logo_url_64 }}" alt="{{ corpstats.corp.corporation_name }}">
|
<img class="ra-avatar" src="{{ corpstats.corp.logo_url_64 }}" alt="{{ corpstats.corp.corporation_name }}">
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
{% if corpstats.corp.alliance %}
|
{% if corpstats.corp.alliance %}
|
||||||
<td class="text-center col-lg-6">
|
<td>
|
||||||
<img class="ra-avatar" src="{{ corpstats.corp.alliance.logo_url_64 }}" alt="{{ corpstats.corp.alliance.alliance_name }}">
|
<img class="ra-avatar" src="{{ corpstats.corp.alliance.logo_url_64 }}" alt="{{ corpstats.corp.alliance.alliance_name }}">
|
||||||
</td>
|
</td>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<td class="text-center"><h4>{{ corpstats.corp.corporation_name }}</h4></td>
|
<td><p class="h4">{{ corpstats.corp.corporation_name }}</p></td>
|
||||||
|
|
||||||
{% if corpstats.corp.alliance %}
|
{% if corpstats.corp.alliance %}
|
||||||
<td class="text-center"><h4>{{ corpstats.corp.alliance.alliance_name }}</h4></td>
|
<td><p class="h4">{{ corpstats.corp.alliance.alliance_name }}</p></td>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
|
||||||
<div class="row">
|
<div class="card card-default mt-4">
|
||||||
<div class="col-lg-12">
|
<div class="card-header clearfix" role="tablist">
|
||||||
<div class="panel panel-default">
|
<ul class="nav nav-pills float-start">
|
||||||
<div class="panel-heading">
|
<li class="nav-item" role="presentation">
|
||||||
<ul class="nav nav-pills pull-left">
|
<a
|
||||||
<li class="active"><a href="#mains" data-toggle="pill">{% translate 'Mains' %} ({{ total_mains }})</a></li>
|
class="nav-link active"
|
||||||
<li><a href="#members" data-toggle="pill">{% translate 'Members' %} ({{ corpstats.member_count }})</a></li>
|
id="mains"
|
||||||
<li><a href="#unregistered" data-toggle="pill">{% translate 'Unregistered' %} ({{ unregistered.count }})</a></li>
|
data-bs-toggle="tab"
|
||||||
|
href="#tab-mains"
|
||||||
|
role="tab"
|
||||||
|
aria-controls="tab-mains"
|
||||||
|
aria-selected="true"
|
||||||
|
>
|
||||||
|
{% translate 'Mains' %} ({{ total_mains }})
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
|
||||||
|
<li class="nav-item" role="presentation">
|
||||||
|
<a
|
||||||
|
class="nav-link"
|
||||||
|
id="members"
|
||||||
|
data-bs-toggle="tab"
|
||||||
|
href="#tab-members"
|
||||||
|
role="tab"
|
||||||
|
aria-controls="tab-members"
|
||||||
|
aria-selected="false"
|
||||||
|
>
|
||||||
|
{% translate 'Members' %} ({{ corpstats.member_count }})
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
|
||||||
|
<li class="nav-item">
|
||||||
|
<a
|
||||||
|
class="nav-link"
|
||||||
|
id="unregistered"
|
||||||
|
data-bs-toggle="tab"
|
||||||
|
href="#tab-unregistered"
|
||||||
|
role="tab"
|
||||||
|
aria-controls="tab-unregistered"
|
||||||
|
aria-selected="false"
|
||||||
|
>
|
||||||
|
{% translate 'Unregistered' %} ({{ unregistered.count }})
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
<div class="pull-right hidden-xs">
|
|
||||||
{% translate "Last update:" %} {{ corpstats.last_update|naturaltime }}
|
<div class="float-end d-none d-sm-block">
|
||||||
<a class="btn btn-success" type="button" href="{% url 'corputils:update' corpstats.corp.corporation_id %}" title="Update Now">
|
{% translate "Last update:" %} {{ corpstats.last_update|naturaltime }}
|
||||||
<span class="glyphicon glyphicon-refresh"></span>
|
|
||||||
|
<a
|
||||||
|
class="btn btn-success btn-sm ms-2"
|
||||||
|
type="button"
|
||||||
|
href="{% url 'corputils:update' corpstats.corp.corporation_id %}"
|
||||||
|
title="{% translate 'Update Now' %}"
|
||||||
|
>
|
||||||
|
<i class="fa-solid fa-rotate"></i>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="clearfix"></div>
|
|
||||||
</div>
|
</div>
|
||||||
<div class="panel-body">
|
|
||||||
|
<div class="card-body">
|
||||||
<div class="tab-content">
|
<div class="tab-content">
|
||||||
<div class="tab-pane fade in active" id="mains">
|
<div class="tab-pane fade show active" id="tab-mains" role="tabpanel" aria-labelledby="tab-mains">
|
||||||
{% if mains %}
|
{% if mains %}
|
||||||
<div class="table-responsive">
|
<div class="table-responsive">
|
||||||
<table class="table table-hover" id="table-mains">
|
<table class="table table-hover" id="table-mains">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th style="height:1em;"><!-- Must have text or height to prevent clipping --></th>
|
<th>{% translate "Main character" %}</th>
|
||||||
<th></th>
|
<th>{% translate "Registered characters" %}</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
|
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for id, main in mains.items %}
|
{% for id, main in mains.items %}
|
||||||
<tr>
|
<tr>
|
||||||
<td class="text-center" style="vertical-align:middle">
|
<td class="text-center" style="vertical-align: middle;">
|
||||||
<div class="thumbnail" style="border: 0 none; box-shadow: none; background: transparent;">
|
<div class="thumbnail" style="border: 0 none; box-shadow: none; background: transparent;">
|
||||||
<img src="{{ main.main.portrait_url_64 }}" class="img-circle" alt="{{ main.main }}">
|
<img src="{{ main.main.portrait_url_64 }}" class="img-circle" alt="{{ main.main }}">
|
||||||
<div class="caption text-center">
|
<div class="caption">
|
||||||
{{ main.main }}
|
{{ main.main }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td>
|
<td>
|
||||||
<table class="table table-hover">
|
<table class="table table-hover">
|
||||||
{% for alt in main.alts %}
|
{% for alt in main.alts|dictsort:"character_name" %}
|
||||||
{% if forloop.first %}
|
{% if forloop.first %}
|
||||||
<tr>
|
<tr>
|
||||||
<th></th>
|
<th></th>
|
||||||
<th class="text-center">{% translate "Character" %}</th>
|
<th>{% translate "Character" %}</th>
|
||||||
<th class="text-center">{% translate "Corporation" %}</th>
|
<th>{% translate "Corporation" %}</th>
|
||||||
<th class="text-center">{% translate "Alliance" %}</th>
|
<th>{% translate "Alliance" %}</th>
|
||||||
<th class="text-center"></th>
|
<th></th>
|
||||||
</tr>
|
</tr>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<td class="text-center" style="width:5%">
|
<td style="width: 5%;">
|
||||||
<div class="thumbnail" style="border: 0 none; box-shadow: none; background: transparent;">
|
<div class="thumbnail" style="border: 0 none; box-shadow: none; background: transparent;">
|
||||||
<img src="{{ alt.portrait_url_32 }}" class="img-circle" alt="{{ alt.character_name }}">
|
<img src="{{ alt.portrait_url_32 }}" class="img-circle" alt="{{ alt.character_name }}">
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
<td class="text-center" style="width:30%">{{ alt.character_name }}</td>
|
<td style="width: 30%;">{{ alt.character_name }}</td>
|
||||||
<td class="text-center" style="width:30%">{{ alt.corporation_name }}</td>
|
<td style="width: 30%;">{{ alt.corporation_name }}</td>
|
||||||
<td class="text-center" style="width:30%">{{ alt.alliance_name }}</td>
|
<td style="width: 30%;">{{ alt.alliance_name|default_if_none:"" }}</td>
|
||||||
<td class="text-center" style="width:5%">
|
<td style="width: 5%;">
|
||||||
<a href="https://zkillboard.com/character/{{ alt.character_id }}/" class="badge badge-danger" target="_blank">
|
<a href="https://zkillboard.com/character/{{ alt.character_id }}/" class="badge text-bg-danger" target="_blank">
|
||||||
{% translate "Killboard" %}
|
{% translate "Killboard" %}
|
||||||
</a>
|
</a>
|
||||||
</td>
|
</td>
|
||||||
@ -102,43 +153,46 @@
|
|||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
<div class="tab-pane fade" id="members">
|
|
||||||
|
<div class="tab-pane fade" id="tab-members" role="tabpanel" aria-labelledby="tab-members">
|
||||||
{% if members %}
|
{% if members %}
|
||||||
<div class="table-responsive">
|
<div class="table-responsive">
|
||||||
<table class="table table-hover" id="table-members">
|
<table class="table table-hover" id="table-members">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th></th>
|
<th></th>
|
||||||
<th class="text-center">{% translate "Character" %}</th>
|
<th>{% translate "Character" %}</th>
|
||||||
<th class="text-center"></th>
|
<th></th>
|
||||||
<th class="text-center">{% translate "Main Character" %}</th>
|
<th>{% translate "Main Character" %}</th>
|
||||||
<th class="text-center">{% translate "Main Corporation" %}</th>
|
<th>{% translate "Main Corporation" %}</th>
|
||||||
<th class="text-center">{% translate "Main Alliance" %}</th>
|
<th>{% translate "Main Alliance" %}</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
|
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for member in members %}
|
{% for member in members %}
|
||||||
<tr>
|
<tr>
|
||||||
<td><img src="{{ member.portrait_url }}" class="img-circle" alt="{{ member }}"></td>
|
<td><img src="{{ member.portrait_url }}" class="img-circle" alt="{{ member }}"></td>
|
||||||
<td class="text-center">{{ member }}</td>
|
<td>{{ member }}</td>
|
||||||
<td class="text-center">
|
<td>
|
||||||
<a href="https://zkillboard.com/character/{{ member.character_id }}/" class="badge badge-danger" target="_blank">{% translate "Killboard" %}</a>
|
<a href="https://zkillboard.com/character/{{ member.character_id }}/" class="badge text-bg-danger" target="_blank">{% translate "Killboard" %}</a>
|
||||||
</td>
|
</td>
|
||||||
<td class="text-center">{{ member.character_ownership.user.profile.main_character.character_name }}</td>
|
<td>{{ member.character_ownership.user.profile.main_character.character_name }}</td>
|
||||||
<td class="text-center">{{ member.character_ownership.user.profile.main_character.corporation_name }}</td>
|
<td>{{ member.character_ownership.user.profile.main_character.corporation_name }}</td>
|
||||||
<td class="text-center">{{ member.character_ownership.user.profile.main_character.alliance_name }}</td>
|
<td>{{ member.character_ownership.user.profile.main_character.alliance_name|default_if_none:"" }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
{% for member in unregistered %}
|
{% for member in unregistered %}
|
||||||
<tr class="danger">
|
<tr class="table-danger">
|
||||||
<td><img src="{{ member.portrait_url }}" class="img-circle" alt="{{ member.character_name }}"></td>
|
<td><img src="{{ member.portrait_url }}" class="img-circle" alt="{{ member.character_name }}"></td>
|
||||||
<td class="text-center">{{ member.character_name }}</td>
|
<td>{{ member.character_name }}</td>
|
||||||
<td class="text-center">
|
<td>
|
||||||
<a href="https://zkillboard.com/character/{{ member.character_id }}/" class="badge badge-danger" target="_blank">{% translate "Killboard" %}</a>
|
<a href="https://zkillboard.com/character/{{ member.character_id }}/" class="badge text-bg-danger" target="_blank">{% translate "Killboard" %}</a>
|
||||||
</td>
|
</td>
|
||||||
<td class="text-center"></td>
|
<td></td>
|
||||||
<td class="text-center"></td>
|
<td></td>
|
||||||
<td class="text-center"></td>
|
<td></td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
@ -146,24 +200,26 @@
|
|||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
<div class="tab-pane fade" id="unregistered">
|
|
||||||
|
<div class="tab-pane fade" id="tab-unregistered" role="tabpanel" aria-labelledby="tab-unregistered">
|
||||||
{% if unregistered %}
|
{% if unregistered %}
|
||||||
<div class="table-responsive">
|
<div class="table-responsive">
|
||||||
<table class="table table-hover" id="table-unregistered">
|
<table class="table table-hover" id="table-unregistered">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th></th>
|
<th></th>
|
||||||
<th class="text-center">{% translate "Character" %}</th>
|
<th>{% translate "Character" %}</th>
|
||||||
<th class="text-center"></th>
|
<th></th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
|
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for member in unregistered %}
|
{% for member in unregistered %}
|
||||||
<tr class="danger">
|
<tr class="table-danger">
|
||||||
<td><img src="{{ member.portrait_url }}" class="img-circle" alt="{{ member.character_name }}"></td>
|
<td><img src="{{ member.portrait_url }}" class="img-circle" alt="{{ member.character_name }}"></td>
|
||||||
<td class="text-center">{{ member.character_name }}</td>
|
<td>{{ member.character_name }}</td>
|
||||||
<td class="text-center">
|
<td>
|
||||||
<a href="https://zkillboard.com/character/{{ member.character_id }}/" class="badge badge-danger" target="_blank">
|
<a href="https://zkillboard.com/character/{{ member.character_id }}/" class="badge text-bg-danger" target="_blank">
|
||||||
{% translate "Killboard" %}
|
{% translate "Killboard" %}
|
||||||
</a>
|
</a>
|
||||||
</td>
|
</td>
|
||||||
@ -177,35 +233,27 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block extra_javascript %}
|
{% block extra_javascript %}
|
||||||
{% include 'bundles/datatables-js.html' %}
|
{% include 'bundles/datatables-js-bs5.html' %}
|
||||||
{% endblock %}
|
|
||||||
{% block extra_css %}
|
{% get_datatables_language_static LANGUAGE_CODE as DT_LANG_PATH %}
|
||||||
{% include 'bundles/datatables-css.html' %}
|
|
||||||
{% endblock %}
|
<script>
|
||||||
{% block extra_script %}
|
$(document).ready(() => {
|
||||||
$(document).ready(function(){
|
|
||||||
$('#table-mains').DataTable({
|
$('#table-mains').DataTable({
|
||||||
|
"language": {"url": '{{ DT_LANG_PATH }}'},
|
||||||
"columnDefs": [
|
"columnDefs": [
|
||||||
{ "sortable": false, "targets": [1] },
|
{ "sortable": false, "targets": [1] },
|
||||||
],
|
],
|
||||||
"stateSave": true,
|
"stateSave": true,
|
||||||
"stateDuration": 0
|
"stateDuration": 0
|
||||||
});
|
});
|
||||||
|
|
||||||
$('#table-members').DataTable({
|
$('#table-members').DataTable({
|
||||||
"columnDefs": [
|
"language": {"url": '{{ DT_LANG_PATH }}'},
|
||||||
{ "searchable": false, "targets": [0, 2] },
|
|
||||||
{ "sortable": false, "targets": [0, 2] },
|
|
||||||
],
|
|
||||||
"order": [[ 1, "asc" ]],
|
|
||||||
"stateSave": true,
|
|
||||||
"stateDuration": 0
|
|
||||||
});
|
|
||||||
$('#table-unregistered').DataTable({
|
|
||||||
"columnDefs": [
|
"columnDefs": [
|
||||||
{ "searchable": false, "targets": [0, 2] },
|
{ "searchable": false, "targets": [0, 2] },
|
||||||
{ "sortable": false, "targets": [0, 2] },
|
{ "sortable": false, "targets": [0, 2] },
|
||||||
@ -215,5 +263,20 @@
|
|||||||
"stateDuration": 0
|
"stateDuration": 0
|
||||||
});
|
});
|
||||||
|
|
||||||
|
$('#table-unregistered').DataTable({
|
||||||
|
"language": {"url": '{{ DT_LANG_PATH }}'},
|
||||||
|
"columnDefs": [
|
||||||
|
{ "searchable": false, "targets": [0, 2] },
|
||||||
|
{ "sortable": false, "targets": [0, 2] },
|
||||||
|
],
|
||||||
|
"order": [[ 1, "asc" ]],
|
||||||
|
"stateSave": true,
|
||||||
|
"stateDuration": 0
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block extra_css %}
|
||||||
|
{% include 'bundles/datatables-css-bs5.html' %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
@ -1,33 +1,37 @@
|
|||||||
{% extends "corputils/base.html" %}
|
{% extends "corputils/base.html" %}
|
||||||
|
|
||||||
|
{% load aa_i18n %}
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block member_data %}
|
{% block member_data %}
|
||||||
<div class="panel panel-default">
|
<div class="card card-default">
|
||||||
<div class="panel-heading clearfix">
|
<div class="card-header clearfix">
|
||||||
<div class="panel-title pull-left">{% translate "Search Results" %}</div>
|
<div class="card-title">{% translate "Search Results" %}</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="panel-body">
|
|
||||||
|
<div class="card-body mt-2">
|
||||||
<table class="table table-hover" id="table-search">
|
<table class="table table-hover" id="table-search">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th class="text-center"></th>
|
<th></th>
|
||||||
<th class="text-center">{% translate "Character" %}</th>
|
<th>{% translate "Character" %}</th>
|
||||||
<th class="text-center">{% translate "Corporation" %}</th>
|
<th>{% translate "Corporation" %}</th>
|
||||||
<th class="text-center">{% translate "zKillboard" %}</th>
|
<th>{% translate "zKillboard" %}</th>
|
||||||
<th class="text-center">{% translate "Main Character" %}</th>
|
<th>{% translate "Main Character" %}</th>
|
||||||
<th class="text-center">{% translate "Main Corporation" %}</th>
|
<th>{% translate "Main Corporation" %}</th>
|
||||||
<th class="text-center">{% translate "Main Alliance" %}</th>
|
<th>{% translate "Main Alliance" %}</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for result in results %}
|
{% for result in results %}
|
||||||
<tr {% if not result.1.registered %}class="danger"{% endif %}>
|
<tr {% if not result.1.registered %}class="danger"{% endif %}>
|
||||||
<td class="text-center"><img src="{{ result.1.portrait_url }}" class="img-circle" alt="{{ result.1.character_name }}"></td>
|
<td><img src="{{ result.1.portrait_url }}" class="img-circle" alt="{{ result.1.character_name }}"></td>
|
||||||
<td class="text-center">{{ result.1.character_name }}</td>
|
<td>{{ result.1.character_name }}</td>
|
||||||
<td class="text-center">{{ result.0.corp.corporation_name }}</td>
|
<td >{{ result.0.corp.corporation_name }}</td>
|
||||||
<td class="text-center"><a href="https://zkillboard.com/character/{{ result.1.character_id }}/" class="badge badge-danger" target="_blank">{% translate "Killboard" %}</a></td>
|
<td><a href="https://zkillboard.com/character/{{ result.1.character_id }}/" class="badge text-bg-danger" target="_blank">{% translate "Killboard" %}</a></td>
|
||||||
<td class="text-center">{{ result.1.main_character.character_name }}</td>
|
<td>{{ result.1.main_character.character_name }}</td>
|
||||||
<td class="text-center">{{ result.1.main_character.corporation_name }}</td>
|
<td>{{ result.1.main_character.corporation_name }}</td>
|
||||||
<td class="text-center">{{ result.1.main_character.alliance_name }}</td>
|
<td>{{ result.1.main_character.alliance_name }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
@ -35,17 +39,23 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block extra_javascript %}
|
{% block extra_javascript %}
|
||||||
{% include 'bundles/datatables-js.html' %}
|
{% include 'bundles/datatables-js-bs5.html' %}
|
||||||
{% endblock %}
|
|
||||||
{% block extra_css %}
|
{% get_datatables_language_static LANGUAGE_CODE as DT_LANG_PATH %}
|
||||||
{% include 'bundles/datatables-css.html' %}
|
|
||||||
{% endblock %}
|
<script>
|
||||||
{% block extra_script %}
|
$(document).ready(() => {
|
||||||
$(document).ready(function(){
|
|
||||||
$('#table-search').DataTable({
|
$('#table-search').DataTable({
|
||||||
|
"language": {"url": '{{ DT_LANG_PATH }}'},
|
||||||
"stateSave": true,
|
"stateSave": true,
|
||||||
"stateDuration": 0
|
"stateDuration": 0
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
</script>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block extra_css %}
|
||||||
|
{% include 'bundles/datatables-css-bs5.html' %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
3
allianceauth/crontab/__init__.py
Normal file
3
allianceauth/crontab/__init__.py
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
"""
|
||||||
|
Alliance Auth Crontab Utilities
|
||||||
|
"""
|
16
allianceauth/crontab/apps.py
Normal file
16
allianceauth/crontab/apps.py
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
"""
|
||||||
|
Crontab App Config
|
||||||
|
"""
|
||||||
|
|
||||||
|
from django.apps import AppConfig
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
|
||||||
|
class CrontabConfig(AppConfig):
|
||||||
|
"""
|
||||||
|
Crontab App Config
|
||||||
|
"""
|
||||||
|
|
||||||
|
name = "allianceauth.crontab"
|
||||||
|
label = "crontab"
|
||||||
|
verbose_name = _("Crontab")
|
29
allianceauth/crontab/migrations/0001_initial.py
Normal file
29
allianceauth/crontab/migrations/0001_initial.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
# Generated by Django 4.2.16 on 2025-01-20 06:16
|
||||||
|
|
||||||
|
import allianceauth.crontab.models
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
initial = True
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='CronOffset',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('minute', models.FloatField(default=allianceauth.crontab.models.random_default, verbose_name='Minute Offset')),
|
||||||
|
('hour', models.FloatField(default=allianceauth.crontab.models.random_default, verbose_name='Hour Offset')),
|
||||||
|
('day_of_month', models.FloatField(default=allianceauth.crontab.models.random_default, verbose_name='Day of Month Offset')),
|
||||||
|
('month_of_year', models.FloatField(default=allianceauth.crontab.models.random_default, verbose_name='Month of Year Offset')),
|
||||||
|
('day_of_week', models.FloatField(default=allianceauth.crontab.models.random_default, verbose_name='Day of Week Offset')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': 'Cron Offsets',
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
23
allianceauth/crontab/models.py
Normal file
23
allianceauth/crontab/models.py
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
from random import random
|
||||||
|
from django.db import models
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
from solo.models import SingletonModel
|
||||||
|
|
||||||
|
|
||||||
|
def random_default() -> float:
|
||||||
|
return random()
|
||||||
|
|
||||||
|
|
||||||
|
class CronOffset(SingletonModel):
|
||||||
|
|
||||||
|
minute = models.FloatField(_("Minute Offset"), default=random_default)
|
||||||
|
hour = models.FloatField(_("Hour Offset"), default=random_default)
|
||||||
|
day_of_month = models.FloatField(_("Day of Month Offset"), default=random_default)
|
||||||
|
month_of_year = models.FloatField(_("Month of Year Offset"), default=random_default)
|
||||||
|
day_of_week = models.FloatField(_("Day of Week Offset"), default=random_default)
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return "Cron Offsets"
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = "Cron Offsets"
|
70
allianceauth/crontab/schedulers.py
Normal file
70
allianceauth/crontab/schedulers.py
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
from django.core.exceptions import ObjectDoesNotExist
|
||||||
|
from django_celery_beat.schedulers import (
|
||||||
|
DatabaseScheduler
|
||||||
|
)
|
||||||
|
from django_celery_beat.models import CrontabSchedule
|
||||||
|
from django.db.utils import OperationalError, ProgrammingError
|
||||||
|
|
||||||
|
from celery import schedules
|
||||||
|
from celery.utils.log import get_logger
|
||||||
|
|
||||||
|
from allianceauth.crontab.models import CronOffset
|
||||||
|
from allianceauth.crontab.utils import offset_cron
|
||||||
|
|
||||||
|
logger = get_logger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class OffsetDatabaseScheduler(DatabaseScheduler):
|
||||||
|
"""
|
||||||
|
Customization of Django Celery Beat, Database Scheduler
|
||||||
|
Takes the Celery Schedule from local.py and applies our AA Framework Cron Offset, if apply_offset is true
|
||||||
|
Otherwise it passes it through as normal
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs) -> None:
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
|
||||||
|
def update_from_dict(self, mapping):
|
||||||
|
s = {}
|
||||||
|
|
||||||
|
try:
|
||||||
|
cron_offset = CronOffset.get_solo()
|
||||||
|
except (OperationalError, ProgrammingError, ObjectDoesNotExist) as exc:
|
||||||
|
# This is just incase we haven't migrated yet or something
|
||||||
|
logger.warning(
|
||||||
|
"OffsetDatabaseScheduler: Could not fetch CronOffset (%r). "
|
||||||
|
"Defering to DatabaseScheduler",
|
||||||
|
exc
|
||||||
|
)
|
||||||
|
return super().update_from_dict(mapping)
|
||||||
|
|
||||||
|
for name, entry_fields in mapping.items():
|
||||||
|
try:
|
||||||
|
apply_offset = entry_fields.pop("apply_offset", False) # Ensure this pops before django tries to save to ORM
|
||||||
|
entry = self.Entry.from_entry(name, app=self.app, **entry_fields)
|
||||||
|
|
||||||
|
if apply_offset:
|
||||||
|
entry_fields.update({"apply_offset": apply_offset}) # Reapply this as its gets pulled from config inconsistently.
|
||||||
|
schedule_obj = entry.schedule
|
||||||
|
if isinstance(schedule_obj, schedules.crontab):
|
||||||
|
offset_cs = CrontabSchedule.from_schedule(offset_cron(schedule_obj))
|
||||||
|
offset_cs, created = CrontabSchedule.objects.get_or_create(
|
||||||
|
minute=offset_cs.minute,
|
||||||
|
hour=offset_cs.hour,
|
||||||
|
day_of_month=offset_cs.day_of_month,
|
||||||
|
month_of_year=offset_cs.month_of_year,
|
||||||
|
day_of_week=offset_cs.day_of_week,
|
||||||
|
timezone=offset_cs.timezone,
|
||||||
|
)
|
||||||
|
entry.schedule = offset_cron(schedule_obj) # This gets passed into Celery Beats Memory, important to keep it in sync with the model/DB
|
||||||
|
entry.model.crontab = offset_cs
|
||||||
|
entry.model.save()
|
||||||
|
logger.debug(f"Offset applied for '{name}' due to 'apply_offset' = True.")
|
||||||
|
|
||||||
|
if entry.model.enabled:
|
||||||
|
s[name] = entry
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logger.exception("Error updating schedule for %s: %r", name, e)
|
||||||
|
|
||||||
|
self.schedule.update(s)
|
63
allianceauth/crontab/tests/test_models.py
Normal file
63
allianceauth/crontab/tests/test_models.py
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
from unittest.mock import patch
|
||||||
|
from django.test import TestCase
|
||||||
|
|
||||||
|
from allianceauth.crontab.models import CronOffset
|
||||||
|
|
||||||
|
|
||||||
|
class CronOffsetModelTest(TestCase):
|
||||||
|
def test_cron_offset_is_singleton(self):
|
||||||
|
"""
|
||||||
|
Test that CronOffset is indeed a singleton and that
|
||||||
|
multiple calls to get_solo() return the same instance.
|
||||||
|
"""
|
||||||
|
offset1 = CronOffset.get_solo()
|
||||||
|
offset2 = CronOffset.get_solo()
|
||||||
|
|
||||||
|
# They should be the exact same object in memory
|
||||||
|
self.assertEqual(offset1.pk, offset2.pk)
|
||||||
|
|
||||||
|
def test_default_values_random(self):
|
||||||
|
"""
|
||||||
|
Test that the default values are set via random_default() when
|
||||||
|
no explicit value is provided. We'll patch 'random.random' to
|
||||||
|
produce predictable output.
|
||||||
|
"""
|
||||||
|
with patch('allianceauth.crontab.models.random', return_value=0.1234):
|
||||||
|
# Force creation of a new CronOffset by clearing the existing one
|
||||||
|
CronOffset.objects.all().delete()
|
||||||
|
|
||||||
|
offset = CronOffset.get_solo() # This triggers creation
|
||||||
|
|
||||||
|
# All fields should be 0.1234, because we patched random()
|
||||||
|
self.assertAlmostEqual(offset.minute, 0.1234)
|
||||||
|
self.assertAlmostEqual(offset.hour, 0.1234)
|
||||||
|
self.assertAlmostEqual(offset.day_of_month, 0.1234)
|
||||||
|
self.assertAlmostEqual(offset.month_of_year, 0.1234)
|
||||||
|
self.assertAlmostEqual(offset.day_of_week, 0.1234)
|
||||||
|
|
||||||
|
def test_update_offset_values(self):
|
||||||
|
"""
|
||||||
|
Test that we can update the offsets and retrieve them.
|
||||||
|
"""
|
||||||
|
offset = CronOffset.get_solo()
|
||||||
|
offset.minute = 0.5
|
||||||
|
offset.hour = 0.25
|
||||||
|
offset.day_of_month = 0.75
|
||||||
|
offset.month_of_year = 0.99
|
||||||
|
offset.day_of_week = 0.33
|
||||||
|
offset.save()
|
||||||
|
|
||||||
|
# Retrieve again to ensure changes persist
|
||||||
|
saved_offset = CronOffset.get_solo()
|
||||||
|
self.assertEqual(saved_offset.minute, 0.5)
|
||||||
|
self.assertEqual(saved_offset.hour, 0.25)
|
||||||
|
self.assertEqual(saved_offset.day_of_month, 0.75)
|
||||||
|
self.assertEqual(saved_offset.month_of_year, 0.99)
|
||||||
|
self.assertEqual(saved_offset.day_of_week, 0.33)
|
||||||
|
|
||||||
|
def test_str_representation(self):
|
||||||
|
"""
|
||||||
|
Verify the __str__ method returns 'Cron Offsets'.
|
||||||
|
"""
|
||||||
|
offset = CronOffset.get_solo()
|
||||||
|
self.assertEqual(str(offset), "Cron Offsets")
|
80
allianceauth/crontab/tests/test_utils.py
Normal file
80
allianceauth/crontab/tests/test_utils.py
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
# myapp/tests/test_tasks.py
|
||||||
|
|
||||||
|
import logging
|
||||||
|
from unittest.mock import patch
|
||||||
|
from django.test import TestCase
|
||||||
|
from django.db import ProgrammingError
|
||||||
|
from celery.schedules import crontab
|
||||||
|
|
||||||
|
from allianceauth.crontab.utils import offset_cron
|
||||||
|
from allianceauth.crontab.models import CronOffset
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class TestOffsetCron(TestCase):
|
||||||
|
|
||||||
|
def test_offset_cron_normal(self):
|
||||||
|
"""
|
||||||
|
Test that offset_cron modifies the minute/hour fields
|
||||||
|
based on the CronOffset values when everything is normal.
|
||||||
|
"""
|
||||||
|
# We'll create a mock CronOffset instance
|
||||||
|
mock_offset = CronOffset(minute=0.5, hour=0.5)
|
||||||
|
|
||||||
|
# Our initial crontab schedule
|
||||||
|
original_schedule = crontab(
|
||||||
|
minute=[0, 5, 55],
|
||||||
|
hour=[0, 3, 23],
|
||||||
|
day_of_month='*',
|
||||||
|
month_of_year='*',
|
||||||
|
day_of_week='*'
|
||||||
|
)
|
||||||
|
|
||||||
|
# Patch CronOffset.get_solo to return our mock offset
|
||||||
|
with patch('allianceauth.crontab.models.CronOffset.get_solo', return_value=mock_offset):
|
||||||
|
new_schedule = offset_cron(original_schedule)
|
||||||
|
|
||||||
|
# Check the new minute/hour
|
||||||
|
# minute 0 -> 0 + round(60 * 0.5) = 30 % 60 = 30
|
||||||
|
# minute 5 -> 5 + 30 = 35 % 60 = 35
|
||||||
|
# minute 55 -> 55 + 30 = 85 % 60 = 25 --> sorted => 25,30,35
|
||||||
|
self.assertEqual(new_schedule._orig_minute, '25,30,35')
|
||||||
|
|
||||||
|
# hour 0 -> 0 + round(24 * 0.5) = 12 % 24 = 12
|
||||||
|
# hour 3 -> 3 + 12 = 15 % 24 = 15
|
||||||
|
# hour 23 -> 23 + 12 = 35 % 24 = 11 --> sorted => 11,12,15
|
||||||
|
self.assertEqual(new_schedule._orig_hour, '11,12,15')
|
||||||
|
|
||||||
|
# Check that other fields are unchanged
|
||||||
|
self.assertEqual(new_schedule._orig_day_of_month, '*')
|
||||||
|
self.assertEqual(new_schedule._orig_month_of_year, '*')
|
||||||
|
self.assertEqual(new_schedule._orig_day_of_week, '*')
|
||||||
|
|
||||||
|
def test_offset_cron_programming_error(self):
|
||||||
|
"""
|
||||||
|
Test that if a ProgrammingError is raised (e.g. before migrations),
|
||||||
|
offset_cron just returns the original schedule.
|
||||||
|
"""
|
||||||
|
original_schedule = crontab(minute=[0, 15, 30], hour=[1, 2, 3])
|
||||||
|
|
||||||
|
# Force get_solo to raise ProgrammingError
|
||||||
|
with patch('allianceauth.crontab.models.CronOffset.get_solo', side_effect=ProgrammingError()):
|
||||||
|
new_schedule = offset_cron(original_schedule)
|
||||||
|
|
||||||
|
# Should return the original schedule unchanged
|
||||||
|
self.assertEqual(new_schedule, original_schedule)
|
||||||
|
|
||||||
|
def test_offset_cron_unexpected_exception(self):
|
||||||
|
"""
|
||||||
|
Test that if any other exception is raised, offset_cron
|
||||||
|
also returns the original schedule, and logs the error.
|
||||||
|
"""
|
||||||
|
original_schedule = crontab(minute='0', hour='0')
|
||||||
|
|
||||||
|
# Force get_solo to raise a generic Exception
|
||||||
|
with patch('allianceauth.crontab.models.CronOffset.get_solo', side_effect=Exception("Something bad")):
|
||||||
|
new_schedule = offset_cron(original_schedule)
|
||||||
|
|
||||||
|
# Should return the original schedule unchanged
|
||||||
|
self.assertEqual(new_schedule, original_schedule)
|
50
allianceauth/crontab/utils.py
Normal file
50
allianceauth/crontab/utils.py
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
from celery.schedules import crontab
|
||||||
|
import logging
|
||||||
|
from allianceauth.crontab.models import CronOffset
|
||||||
|
from django.db import ProgrammingError
|
||||||
|
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def offset_cron(schedule: crontab) -> crontab:
|
||||||
|
"""Take a crontab and apply a series of precalculated offsets to spread out tasks execution on remote resources
|
||||||
|
|
||||||
|
Args:
|
||||||
|
schedule (crontab): celery.schedules.crontab()
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
crontab: A crontab with offsetted Minute and Hour fields
|
||||||
|
"""
|
||||||
|
|
||||||
|
try:
|
||||||
|
cron_offset = CronOffset.get_solo()
|
||||||
|
|
||||||
|
# Stops this shit from happening 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23
|
||||||
|
# It is only cosmetic, but still annoying
|
||||||
|
if schedule._orig_minute == '*':
|
||||||
|
new_minute = '*'
|
||||||
|
else:
|
||||||
|
new_minute = [(m + (round(60 * cron_offset.minute))) % 60 for m in schedule.minute]
|
||||||
|
if schedule._orig_hour == '*':
|
||||||
|
new_hour = '*'
|
||||||
|
else:
|
||||||
|
new_hour = [(m + (round(24 * cron_offset.hour))) % 24 for m in schedule.hour]
|
||||||
|
|
||||||
|
return crontab(
|
||||||
|
minute=",".join(str(m) for m in sorted(new_minute)),
|
||||||
|
hour=",".join(str(h) for h in sorted(new_hour)),
|
||||||
|
day_of_month=schedule._orig_day_of_month,
|
||||||
|
month_of_year=schedule._orig_month_of_year,
|
||||||
|
day_of_week=schedule._orig_day_of_week)
|
||||||
|
|
||||||
|
except ProgrammingError as e:
|
||||||
|
# If this is called before migrations are run hand back the default schedule
|
||||||
|
# These offsets are stored in a Singleton Model,
|
||||||
|
logger.error(e)
|
||||||
|
return schedule
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
# We absolutely cant fail to hand back a schedule
|
||||||
|
logger.error(e)
|
||||||
|
return schedule
|
3
allianceauth/custom_css/__init__.py
Normal file
3
allianceauth/custom_css/__init__.py
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
"""
|
||||||
|
Initializes the custom_css module.
|
||||||
|
"""
|
25
allianceauth/custom_css/admin.py
Normal file
25
allianceauth/custom_css/admin.py
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
"""
|
||||||
|
Admin classes for custom_css app
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Django
|
||||||
|
from django.contrib import admin
|
||||||
|
|
||||||
|
# Django Solos
|
||||||
|
from solo.admin import SingletonModelAdmin
|
||||||
|
|
||||||
|
# Alliance Auth Custom CSS
|
||||||
|
from allianceauth.custom_css.models import CustomCSS
|
||||||
|
from allianceauth.custom_css.forms import CustomCSSAdminForm
|
||||||
|
|
||||||
|
|
||||||
|
@admin.register(CustomCSS)
|
||||||
|
class CustomCSSAdmin(SingletonModelAdmin):
|
||||||
|
"""
|
||||||
|
Custom CSS Admin
|
||||||
|
"""
|
||||||
|
|
||||||
|
form = CustomCSSAdminForm
|
||||||
|
|
||||||
|
# Leave this here for when we decide to add syntax highlighting to the CSS editor
|
||||||
|
# change_form_template = 'custom_css/admin/change_form.html'
|
13
allianceauth/custom_css/apps.py
Normal file
13
allianceauth/custom_css/apps.py
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
"""
|
||||||
|
Django app configuration for custom_css
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Django
|
||||||
|
from django.apps import AppConfig
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
|
||||||
|
class CustomCSSConfig(AppConfig):
|
||||||
|
name = "allianceauth.custom_css"
|
||||||
|
label = "custom_css"
|
||||||
|
verbose_name = _("Custom CSS")
|
29
allianceauth/custom_css/forms.py
Normal file
29
allianceauth/custom_css/forms.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
"""
|
||||||
|
Forms for custom_css app
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Alliance Auth Custom CSS
|
||||||
|
from allianceauth.custom_css.models import CustomCSS
|
||||||
|
from allianceauth.custom_css.widgets import CssEditorWidget
|
||||||
|
|
||||||
|
# Django
|
||||||
|
from django import forms
|
||||||
|
|
||||||
|
|
||||||
|
class CustomCSSAdminForm(forms.ModelForm):
|
||||||
|
"""
|
||||||
|
Form for editing custom CSS
|
||||||
|
"""
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = CustomCSS
|
||||||
|
fields = ("css",)
|
||||||
|
widgets = {
|
||||||
|
"css": CssEditorWidget(
|
||||||
|
attrs={
|
||||||
|
"style": "width: 90%; height: 100%;",
|
||||||
|
"data-editor": "code-highlight",
|
||||||
|
"data-language": "css",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
42
allianceauth/custom_css/migrations/0001_initial.py
Normal file
42
allianceauth/custom_css/migrations/0001_initial.py
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
# Generated by Django 4.2.15 on 2024-08-14 11:25
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
initial = True
|
||||||
|
|
||||||
|
dependencies = []
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name="CustomCSS",
|
||||||
|
fields=[
|
||||||
|
(
|
||||||
|
"id",
|
||||||
|
models.AutoField(
|
||||||
|
auto_created=True,
|
||||||
|
primary_key=True,
|
||||||
|
serialize=False,
|
||||||
|
verbose_name="ID",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"css",
|
||||||
|
models.TextField(
|
||||||
|
blank=True,
|
||||||
|
help_text="This CSS will be added to the site after the default CSS.",
|
||||||
|
null=True,
|
||||||
|
verbose_name="Your custom CSS",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
("timestamp", models.DateTimeField(auto_now=True)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
"verbose_name": "Custom CSS",
|
||||||
|
"verbose_name_plural": "Custom CSS",
|
||||||
|
"default_permissions": (),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
0
allianceauth/custom_css/migrations/__init__.py
Normal file
0
allianceauth/custom_css/migrations/__init__.py
Normal file
143
allianceauth/custom_css/models.py
Normal file
143
allianceauth/custom_css/models.py
Normal file
@ -0,0 +1,143 @@
|
|||||||
|
"""
|
||||||
|
Models for the custom_css app
|
||||||
|
"""
|
||||||
|
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
|
||||||
|
# Django Solo
|
||||||
|
from solo.models import SingletonModel
|
||||||
|
|
||||||
|
# Django
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import models
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
|
||||||
|
class CustomCSS(SingletonModel):
|
||||||
|
"""
|
||||||
|
Model for storing custom CSS for the site
|
||||||
|
"""
|
||||||
|
|
||||||
|
css = models.TextField(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
verbose_name=_("Your custom CSS"),
|
||||||
|
help_text=_("This CSS will be added to the site after the default CSS."),
|
||||||
|
)
|
||||||
|
timestamp = models.DateTimeField(auto_now=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
"""
|
||||||
|
Meta for CustomCSS
|
||||||
|
"""
|
||||||
|
|
||||||
|
default_permissions = ()
|
||||||
|
verbose_name = _("Custom CSS")
|
||||||
|
verbose_name_plural = _("Custom CSS")
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
"""
|
||||||
|
String representation of CustomCSS
|
||||||
|
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
return str(_("Custom CSS"))
|
||||||
|
|
||||||
|
def save(self, *args, **kwargs):
|
||||||
|
"""
|
||||||
|
Save method for CustomCSS
|
||||||
|
|
||||||
|
:param args:
|
||||||
|
:type args:
|
||||||
|
:param kwargs:
|
||||||
|
:type kwargs:
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.pk = 1
|
||||||
|
|
||||||
|
if self.css and len(self.css.replace(" ", "")) > 0:
|
||||||
|
# Write the custom CSS to a file
|
||||||
|
custom_css_file = open(
|
||||||
|
f"{settings.STATIC_ROOT}allianceauth/custom-styles.css", "w+"
|
||||||
|
)
|
||||||
|
custom_css_file.write(self.compress_css())
|
||||||
|
custom_css_file.close()
|
||||||
|
else:
|
||||||
|
# Remove the custom CSS file
|
||||||
|
try:
|
||||||
|
os.remove(f"{settings.STATIC_ROOT}allianceauth/custom-styles.css")
|
||||||
|
except FileNotFoundError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
super().save(*args, **kwargs)
|
||||||
|
|
||||||
|
def compress_css(self) -> str:
|
||||||
|
"""
|
||||||
|
Compress CSS
|
||||||
|
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
css = self.css
|
||||||
|
new_css = ""
|
||||||
|
|
||||||
|
# Remove comments
|
||||||
|
css = re.sub(pattern=r"\s*/\*\s*\*/", repl="$$HACK1$$", string=css)
|
||||||
|
css = re.sub(pattern=r"/\*[\s\S]*?\*/", repl="", string=css)
|
||||||
|
css = css.replace("$$HACK1$$", "/**/")
|
||||||
|
|
||||||
|
# url() doesn't need quotes
|
||||||
|
css = re.sub(pattern=r'url\((["\'])([^)]*)\1\)', repl=r"url(\2)", string=css)
|
||||||
|
|
||||||
|
# Spaces may be safely collapsed as generated content will collapse them anyway.
|
||||||
|
css = re.sub(pattern=r"\s+", repl=" ", string=css)
|
||||||
|
|
||||||
|
# Shorten collapsable colors: #aabbcc to #abc
|
||||||
|
css = re.sub(
|
||||||
|
pattern=r"#([0-9a-f])\1([0-9a-f])\2([0-9a-f])\3(\s|;)",
|
||||||
|
repl=r"#\1\2\3\4",
|
||||||
|
string=css,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Fragment values can loose zeros
|
||||||
|
css = re.sub(
|
||||||
|
pattern=r":\s*0(\.\d+([cm]m|e[mx]|in|p[ctx]))\s*;", repl=r":\1;", string=css
|
||||||
|
)
|
||||||
|
|
||||||
|
for rule in re.findall(pattern=r"([^{]+){([^}]*)}", string=css):
|
||||||
|
# We don't need spaces around operators
|
||||||
|
selectors = [
|
||||||
|
re.sub(
|
||||||
|
pattern=r"(?<=[\[\(>+=])\s+|\s+(?=[=~^$*|>+\]\)])",
|
||||||
|
repl=r"",
|
||||||
|
string=selector.strip(),
|
||||||
|
)
|
||||||
|
for selector in rule[0].split(",")
|
||||||
|
]
|
||||||
|
|
||||||
|
# Order is important, but we still want to discard repetitions
|
||||||
|
properties = {}
|
||||||
|
porder = []
|
||||||
|
|
||||||
|
for prop in re.findall(pattern="(.*?):(.*?)(;|$)", string=rule[1]):
|
||||||
|
key = prop[0].strip().lower()
|
||||||
|
|
||||||
|
if key not in porder:
|
||||||
|
porder.append(key)
|
||||||
|
|
||||||
|
properties[key] = prop[1].strip()
|
||||||
|
|
||||||
|
# output rule if it contains any declarations
|
||||||
|
if properties:
|
||||||
|
new_css += "{}{{{}}}".format(
|
||||||
|
",".join(selectors),
|
||||||
|
"".join([f"{key}:{properties[key]};" for key in porder])[:-1],
|
||||||
|
)
|
||||||
|
|
||||||
|
return new_css
|
@ -0,0 +1,48 @@
|
|||||||
|
{% extends "admin/change_form.html" %}
|
||||||
|
|
||||||
|
{% block field_sets %}
|
||||||
|
{% for fieldset in adminform %}
|
||||||
|
<fieldset class="module aligned {{ fieldset.classes }}">
|
||||||
|
{% if fieldset.name %}<h2>{{ fieldset.name }}</h2>{% endif %}
|
||||||
|
|
||||||
|
{% if fieldset.description %}
|
||||||
|
<div class="description">{{ fieldset.description|safe }}</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% for line in fieldset %}
|
||||||
|
<div class="form-row{% if line.fields|length == 1 and line.errors %} errors{% endif %}{% if not line.has_visible_field %} hidden{% endif %}{% for field in line %}{% if field.field.name %} field-{{ field.field.name }}{% endif %}{% endfor %}">
|
||||||
|
{% if line.fields|length == 1 %}{{ line.errors }}{% else %}<div class="flex-container form-multiline">{% endif %}
|
||||||
|
|
||||||
|
{% for field in line %}
|
||||||
|
<div>
|
||||||
|
{% if not line.fields|length == 1 and not field.is_readonly %}{{ field.errors }}{% endif %}
|
||||||
|
|
||||||
|
<div class="flex-container{% if not line.fields|length == 1 %} fieldBox{% if field.field.name %} field-{{ field.field.name }}{% endif %}{% if not field.is_readonly and field.errors %} errors{% endif %}{% if field.field.is_hidden %} hidden{% endif %}{% elif field.is_checkbox %} checkbox-row{% endif %}">
|
||||||
|
{% if field.is_checkbox %}
|
||||||
|
{{ field.field }}{{ field.label_tag }}
|
||||||
|
{% else %}
|
||||||
|
{{ field.label_tag }}
|
||||||
|
{% if field.is_readonly %}
|
||||||
|
<div class="readonly">{{ field.contents }}</div>
|
||||||
|
{% else %}
|
||||||
|
{{ field.field }}
|
||||||
|
{% endif %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% if field.field.help_text %}
|
||||||
|
<div class="help"{% if field.field.id_for_label %} id="{{ field.field.id_for_label }}_helptext"{% endif %}>
|
||||||
|
<div>{{ field.field.help_text|safe }}</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
|
||||||
|
{% if not line.fields|length == 1 %}</div>{% endif %}
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</fieldset>
|
||||||
|
{% endfor %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block after_field_sets %}{% endblock %}
|
@ -0,0 +1,3 @@
|
|||||||
|
{% load custom_css %}
|
||||||
|
|
||||||
|
{% custom_css_static 'allianceauth/custom-styles.css' %}
|
3
allianceauth/custom_css/templatetags/__init__.py
Normal file
3
allianceauth/custom_css/templatetags/__init__.py
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
"""
|
||||||
|
Init file for custom_css templatetags
|
||||||
|
"""
|
48
allianceauth/custom_css/templatetags/custom_css.py
Normal file
48
allianceauth/custom_css/templatetags/custom_css.py
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
"""
|
||||||
|
Custom template tags for custom_css app
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Alliance Auth Custom CSS
|
||||||
|
from allianceauth.custom_css.models import CustomCSS
|
||||||
|
|
||||||
|
# Django
|
||||||
|
from django.conf import settings
|
||||||
|
from django.template.defaulttags import register
|
||||||
|
from django.templatetags.static import static
|
||||||
|
from django.utils.safestring import mark_safe
|
||||||
|
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
|
@register.simple_tag
|
||||||
|
def custom_css_static(path: str) -> str:
|
||||||
|
"""
|
||||||
|
Versioned static URL
|
||||||
|
This is to make sure to break the browser cache on CSS updates.
|
||||||
|
|
||||||
|
Example: /static/allianceauth/custom-styles.css?v=1234567890
|
||||||
|
|
||||||
|
:param path:
|
||||||
|
:type path:
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
try:
|
||||||
|
Path(f"{settings.STATIC_ROOT}{path}").resolve(strict=True)
|
||||||
|
except FileNotFoundError:
|
||||||
|
return ""
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
custom_css = CustomCSS.objects.get(pk=1)
|
||||||
|
except CustomCSS.DoesNotExist:
|
||||||
|
return ""
|
||||||
|
else:
|
||||||
|
custom_css_changed = custom_css.timestamp.timestamp()
|
||||||
|
custom_css_version = (
|
||||||
|
str(custom_css_changed).replace(" ", "").replace(":", "").replace("-", "")
|
||||||
|
) # remove spaces, colons, and dashes
|
||||||
|
static_url = static(path)
|
||||||
|
versioned_url = static_url + "?v=" + custom_css_version
|
||||||
|
|
||||||
|
return mark_safe(f'<link rel="stylesheet" href="{versioned_url}">')
|
38
allianceauth/custom_css/widgets.py
Normal file
38
allianceauth/custom_css/widgets.py
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
"""
|
||||||
|
Form widgets for custom_css app
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Django
|
||||||
|
from django import forms
|
||||||
|
|
||||||
|
# Alliance Auth
|
||||||
|
from allianceauth.custom_css.models import CustomCSS
|
||||||
|
|
||||||
|
|
||||||
|
class CssEditorWidget(forms.Textarea):
|
||||||
|
"""
|
||||||
|
Widget for editing CSS
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, attrs=None):
|
||||||
|
default_attrs = {"class": "custom-css-editor"}
|
||||||
|
|
||||||
|
if attrs:
|
||||||
|
default_attrs.update(attrs)
|
||||||
|
|
||||||
|
super().__init__(default_attrs)
|
||||||
|
|
||||||
|
# For when we want to add some sort of syntax highlight to it, which is not that
|
||||||
|
# easy to do on a textarea field though.
|
||||||
|
# `highlight.js` is just used as an example here, and doesn't work on a textarea field.
|
||||||
|
# class Media:
|
||||||
|
# css = {
|
||||||
|
# "all": (
|
||||||
|
# "/static/custom_css/libs/highlight.js/11.10.0/styles/github.min.css",
|
||||||
|
# )
|
||||||
|
# }
|
||||||
|
# js = (
|
||||||
|
# "/static/custom_css/libs/highlight.js/11.10.0/highlight.min.js",
|
||||||
|
# "/static/custom_css/libs/highlight.js/11.10.0/languages/css.min.js",
|
||||||
|
# "/static/custom_css/javascript/custom-css.min.js",
|
||||||
|
# )
|
@ -1,6 +1,8 @@
|
|||||||
from django.apps import AppConfig
|
from django.apps import AppConfig
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
|
||||||
class EveonlineConfig(AppConfig):
|
class EveonlineConfig(AppConfig):
|
||||||
name = 'allianceauth.eveonline'
|
name = 'allianceauth.eveonline'
|
||||||
label = 'eveonline'
|
label = 'eveonline'
|
||||||
|
verbose_name = _('EVE Online')
|
||||||
|
@ -1,9 +1,11 @@
|
|||||||
from django.apps import AppConfig
|
from django.apps import AppConfig
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
|
||||||
class EveAutogroupsConfig(AppConfig):
|
class EveAutogroupsConfig(AppConfig):
|
||||||
name = 'allianceauth.eveonline.autogroups'
|
name = 'allianceauth.eveonline.autogroups'
|
||||||
label = 'eve_autogroups'
|
label = 'eve_autogroups'
|
||||||
|
verbose_name = _('EVE Online Autogroups')
|
||||||
|
|
||||||
def ready(self):
|
def ready(self):
|
||||||
import allianceauth.eveonline.autogroups.signals
|
import allianceauth.eveonline.autogroups.signals
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import logging
|
import logging
|
||||||
|
from typing import ClassVar
|
||||||
from django.db import models, transaction
|
from django.db import models, transaction
|
||||||
from django.contrib.auth.models import Group, User
|
from django.contrib.auth.models import Group, User
|
||||||
from django.core.exceptions import ObjectDoesNotExist
|
from django.core.exceptions import ObjectDoesNotExist
|
||||||
@ -38,13 +39,13 @@ class AutogroupsConfigManager(models.Manager):
|
|||||||
"""
|
"""
|
||||||
if state is None:
|
if state is None:
|
||||||
state = user.profile.state
|
state = user.profile.state
|
||||||
for config in self.filter(states=state):
|
|
||||||
# grant user new groups for their state
|
|
||||||
config.update_group_membership_for_user(user)
|
|
||||||
for config in self.exclude(states=state):
|
for config in self.exclude(states=state):
|
||||||
# ensure user does not have groups from previous state
|
# ensure user does not have groups from previous state
|
||||||
config.remove_user_from_alliance_groups(user)
|
config.remove_user_from_alliance_groups(user)
|
||||||
config.remove_user_from_corp_groups(user)
|
config.remove_user_from_corp_groups(user)
|
||||||
|
for config in self.filter(states=state):
|
||||||
|
# grant user new groups for their state
|
||||||
|
config.update_group_membership_for_user(user)
|
||||||
|
|
||||||
|
|
||||||
class AutogroupsConfig(models.Model):
|
class AutogroupsConfig(models.Model):
|
||||||
@ -78,7 +79,7 @@ class AutogroupsConfig(models.Model):
|
|||||||
max_length=10, default='', blank=True,
|
max_length=10, default='', blank=True,
|
||||||
help_text='Any spaces in the group name will be replaced with this.')
|
help_text='Any spaces in the group name will be replaced with this.')
|
||||||
|
|
||||||
objects = AutogroupsConfigManager()
|
objects: ClassVar[AutogroupsConfigManager] = AutogroupsConfigManager()
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, **kwargs)
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
from allianceauth.eveonline.models import EveCorporationInfo
|
||||||
from django.test import TestCase
|
from django.test import TestCase
|
||||||
from allianceauth.tests.auth_utils import AuthUtils
|
from allianceauth.tests.auth_utils import AuthUtils
|
||||||
|
|
||||||
@ -73,3 +74,51 @@ class AutogroupsConfigManagerTestCase(TestCase):
|
|||||||
AutogroupsConfig.objects.update_groups_for_user(member)
|
AutogroupsConfig.objects.update_groups_for_user(member)
|
||||||
|
|
||||||
self.assertTrue(update_groups.called)
|
self.assertTrue(update_groups.called)
|
||||||
|
|
||||||
|
def test_update_group_membership_corp_in_two_configs(self):
|
||||||
|
# given
|
||||||
|
member = AuthUtils.create_member('test member')
|
||||||
|
AuthUtils.add_main_character_2(
|
||||||
|
member,
|
||||||
|
character_id='1234',
|
||||||
|
name='test character',
|
||||||
|
corp_id='2345',
|
||||||
|
corp_name='corp name',
|
||||||
|
corp_ticker='TIKK',
|
||||||
|
|
||||||
|
)
|
||||||
|
|
||||||
|
corp = EveCorporationInfo.objects.create(
|
||||||
|
corporation_id='2345',
|
||||||
|
corporation_name='corp name',
|
||||||
|
corporation_ticker='TIKK',
|
||||||
|
member_count=10,
|
||||||
|
)
|
||||||
|
|
||||||
|
member_state = AuthUtils.get_member_state()
|
||||||
|
member_config = AutogroupsConfig.objects.create(corp_groups=True)
|
||||||
|
member_config.states.add(member_state)
|
||||||
|
blue_state = AuthUtils.get_blue_state()
|
||||||
|
blue_state.member_corporations.add(corp)
|
||||||
|
blue_config = AutogroupsConfig.objects.create(corp_groups=True)
|
||||||
|
blue_config.states.add(blue_state)
|
||||||
|
|
||||||
|
member.profile.state = blue_state
|
||||||
|
member.profile.save()
|
||||||
|
|
||||||
|
AutogroupsConfig.objects.update_groups_for_user(member)
|
||||||
|
|
||||||
|
# Checks before test that the role is correctly applied
|
||||||
|
group = blue_config.get_corp_group(corp)
|
||||||
|
self.assertIn(group, member.groups.all())
|
||||||
|
|
||||||
|
# when
|
||||||
|
blue_state.member_corporations.remove(corp)
|
||||||
|
member_state.member_corporations.add(corp)
|
||||||
|
member.profile.state = member_state
|
||||||
|
member.profile.save()
|
||||||
|
|
||||||
|
# then
|
||||||
|
AutogroupsConfig.objects.update_groups_for_user(member)
|
||||||
|
group = member_config.get_corp_group(corp)
|
||||||
|
self.assertIn(group, member.groups.all())
|
||||||
|
@ -10,7 +10,7 @@ from . import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
_BASE_URL = 'http://evemaps.dotlan.net'
|
_BASE_URL = 'https://evemaps.dotlan.net'
|
||||||
|
|
||||||
|
|
||||||
def _build_url(category: str, name: str) -> str:
|
def _build_url(category: str, name: str) -> str:
|
||||||
|
@ -31,29 +31,29 @@ class TestDotlan(TestCase):
|
|||||||
def test_alliance_url(self):
|
def test_alliance_url(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
dotlan.alliance_url('Wayne Enterprices'),
|
dotlan.alliance_url('Wayne Enterprices'),
|
||||||
'http://evemaps.dotlan.net/alliance/Wayne_Enterprices'
|
'https://evemaps.dotlan.net/alliance/Wayne_Enterprices'
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_corporation_url(self):
|
def test_corporation_url(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
dotlan.corporation_url('Wayne Technology'),
|
dotlan.corporation_url('Wayne Technology'),
|
||||||
'http://evemaps.dotlan.net/corp/Wayne_Technology'
|
'https://evemaps.dotlan.net/corp/Wayne_Technology'
|
||||||
)
|
)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
dotlan.corporation_url('Crédit Agricole'),
|
dotlan.corporation_url('Crédit Agricole'),
|
||||||
'http://evemaps.dotlan.net/corp/Cr%C3%A9dit_Agricole'
|
'https://evemaps.dotlan.net/corp/Cr%C3%A9dit_Agricole'
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_region_url(self):
|
def test_region_url(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
dotlan.region_url('Black Rise'),
|
dotlan.region_url('Black Rise'),
|
||||||
'http://evemaps.dotlan.net/map/Black_Rise'
|
'https://evemaps.dotlan.net/map/Black_Rise'
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_solar_system_url(self):
|
def test_solar_system_url(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
dotlan.solar_system_url('Jita'),
|
dotlan.solar_system_url('Jita'),
|
||||||
'http://evemaps.dotlan.net/system/Jita'
|
'https://evemaps.dotlan.net/system/Jita'
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -14,10 +14,20 @@ class EveCharacterProviderManager:
|
|||||||
class EveCharacterManager(models.Manager):
|
class EveCharacterManager(models.Manager):
|
||||||
provider = EveCharacterProviderManager()
|
provider = EveCharacterProviderManager()
|
||||||
|
|
||||||
def create_character(self, character_id):
|
def exclude_biomassed(self):
|
||||||
|
"""
|
||||||
|
Get a queryset of EveCharacter objects, excluding the "Doomheim" corporation (1000001).
|
||||||
|
|
||||||
|
:return:
|
||||||
|
:rtype:
|
||||||
|
"""
|
||||||
|
|
||||||
|
return self.exclude(corporation_id=1000001)
|
||||||
|
|
||||||
|
def create_character(self, character_id) -> models.Model:
|
||||||
return self.create_character_obj(self.provider.get_character(character_id))
|
return self.create_character_obj(self.provider.get_character(character_id))
|
||||||
|
|
||||||
def create_character_obj(self, character: providers.Character):
|
def create_character_obj(self, character: providers.Character) -> models.Model:
|
||||||
return self.create(
|
return self.create(
|
||||||
character_id=character.id,
|
character_id=character.id,
|
||||||
character_name=character.name,
|
character_name=character.name,
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import logging
|
import logging
|
||||||
from typing import Union
|
from typing import ClassVar, Union
|
||||||
|
|
||||||
from django.core.exceptions import ObjectDoesNotExist
|
from django.core.exceptions import ObjectDoesNotExist
|
||||||
from django.db import models
|
from django.db import models
|
||||||
@ -75,8 +75,8 @@ class EveAllianceInfo(models.Model):
|
|||||||
alliance_ticker = models.CharField(max_length=254)
|
alliance_ticker = models.CharField(max_length=254)
|
||||||
executor_corp_id = models.PositiveIntegerField()
|
executor_corp_id = models.PositiveIntegerField()
|
||||||
|
|
||||||
objects = EveAllianceManager()
|
objects: ClassVar[EveAllianceManager] = EveAllianceManager()
|
||||||
provider = EveAllianceProviderManager()
|
provider: ClassVar[EveAllianceProviderManager] = EveAllianceProviderManager()
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
indexes = [models.Index(fields=['executor_corp_id',])]
|
indexes = [models.Index(fields=['executor_corp_id',])]
|
||||||
@ -147,7 +147,7 @@ class EveCorporationInfo(models.Model):
|
|||||||
EveAllianceInfo, blank=True, null=True, on_delete=models.SET_NULL
|
EveAllianceInfo, blank=True, null=True, on_delete=models.SET_NULL
|
||||||
)
|
)
|
||||||
|
|
||||||
objects = EveCorporationManager()
|
objects: ClassVar[EveCorporationManager] = EveCorporationManager()
|
||||||
provider = EveCorporationProviderManager()
|
provider = EveCorporationProviderManager()
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
@ -214,7 +214,7 @@ class EveCharacter(models.Model):
|
|||||||
faction_id = models.PositiveIntegerField(blank=True, null=True, default=None)
|
faction_id = models.PositiveIntegerField(blank=True, null=True, default=None)
|
||||||
faction_name = models.CharField(max_length=254, blank=True, null=True, default='')
|
faction_name = models.CharField(max_length=254, blank=True, null=True, default='')
|
||||||
|
|
||||||
objects = EveCharacterManager()
|
objects: ClassVar[EveCharacterManager] = EveCharacterManager()
|
||||||
provider = EveCharacterProviderManager()
|
provider = EveCharacterProviderManager()
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import logging
|
import logging
|
||||||
|
from random import randint
|
||||||
|
|
||||||
from celery import shared_task
|
from celery import shared_task
|
||||||
|
|
||||||
@ -9,7 +10,8 @@ from . import providers
|
|||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
TASK_PRIORITY = 7
|
TASK_PRIORITY = 7
|
||||||
CHUNK_SIZE = 500
|
CHARACTER_AFFILIATION_CHUNK_SIZE = 500
|
||||||
|
EVEONLINE_TASK_JITTER = 600
|
||||||
|
|
||||||
|
|
||||||
def chunks(lst, n):
|
def chunks(lst, n):
|
||||||
@ -19,13 +21,13 @@ def chunks(lst, n):
|
|||||||
|
|
||||||
|
|
||||||
@shared_task
|
@shared_task
|
||||||
def update_corp(corp_id):
|
def update_corp(corp_id: int) -> None:
|
||||||
"""Update given corporation from ESI"""
|
"""Update given corporation from ESI"""
|
||||||
EveCorporationInfo.objects.update_corporation(corp_id)
|
EveCorporationInfo.objects.update_corporation(corp_id)
|
||||||
|
|
||||||
|
|
||||||
@shared_task
|
@shared_task
|
||||||
def update_alliance(alliance_id):
|
def update_alliance(alliance_id: int) -> None:
|
||||||
"""Update given alliance from ESI"""
|
"""Update given alliance from ESI"""
|
||||||
EveAllianceInfo.objects.update_alliance(alliance_id).populate_alliance()
|
EveAllianceInfo.objects.update_alliance(alliance_id).populate_alliance()
|
||||||
|
|
||||||
@ -37,23 +39,30 @@ def update_character(character_id: int) -> None:
|
|||||||
|
|
||||||
|
|
||||||
@shared_task
|
@shared_task
|
||||||
def run_model_update():
|
def run_model_update() -> None:
|
||||||
"""Update all alliances, corporations and characters from ESI"""
|
"""Update all alliances, corporations and characters from ESI"""
|
||||||
|
|
||||||
#update existing corp models
|
# Queue update tasks for Known Corporation Models
|
||||||
for corp in EveCorporationInfo.objects.all().values('corporation_id'):
|
for corp in EveCorporationInfo.objects.all().values('corporation_id'):
|
||||||
update_corp.apply_async(args=[corp['corporation_id']], priority=TASK_PRIORITY)
|
update_corp.apply_async(
|
||||||
|
args=[corp['corporation_id']],
|
||||||
|
priority=TASK_PRIORITY,
|
||||||
|
countdown=randint(1, EVEONLINE_TASK_JITTER))
|
||||||
|
|
||||||
# update existing alliance models
|
# Queue update tasks for Known Alliance Models
|
||||||
for alliance in EveAllianceInfo.objects.all().values('alliance_id'):
|
for alliance in EveAllianceInfo.objects.all().values('alliance_id'):
|
||||||
update_alliance.apply_async(args=[alliance['alliance_id']], priority=TASK_PRIORITY)
|
update_alliance.apply_async(
|
||||||
|
args=[alliance['alliance_id']],
|
||||||
|
priority=TASK_PRIORITY,
|
||||||
|
countdown=randint(1, EVEONLINE_TASK_JITTER))
|
||||||
|
|
||||||
# update existing character models
|
# Queue update tasks for Known Character Models
|
||||||
character_ids = EveCharacter.objects.all().values_list('character_id', flat=True)
|
character_ids = EveCharacter.objects.all().values_list('character_id', flat=True)
|
||||||
for character_ids_chunk in chunks(character_ids, CHUNK_SIZE):
|
for character_ids_chunk in chunks(character_ids, CHARACTER_AFFILIATION_CHUNK_SIZE):
|
||||||
update_character_chunk.apply_async(
|
update_character_chunk.apply_async(
|
||||||
args=[character_ids_chunk], priority=TASK_PRIORITY
|
args=[character_ids_chunk],
|
||||||
)
|
priority=TASK_PRIORITY,
|
||||||
|
countdown=randint(1, EVEONLINE_TASK_JITTER))
|
||||||
|
|
||||||
|
|
||||||
@shared_task
|
@shared_task
|
||||||
@ -68,8 +77,9 @@ def update_character_chunk(character_ids_chunk: list):
|
|||||||
logger.info("Failed to bulk update characters. Attempting single updates")
|
logger.info("Failed to bulk update characters. Attempting single updates")
|
||||||
for character_id in character_ids_chunk:
|
for character_id in character_ids_chunk:
|
||||||
update_character.apply_async(
|
update_character.apply_async(
|
||||||
args=[character_id], priority=TASK_PRIORITY
|
args=[character_id],
|
||||||
)
|
priority=TASK_PRIORITY,
|
||||||
|
countdown=randint(1, EVEONLINE_TASK_JITTER))
|
||||||
return
|
return
|
||||||
|
|
||||||
affiliations = {
|
affiliations = {
|
||||||
@ -107,5 +117,5 @@ def update_character_chunk(character_ids_chunk: list):
|
|||||||
|
|
||||||
if corp_changed or alliance_changed or name_changed:
|
if corp_changed or alliance_changed or name_changed:
|
||||||
update_character.apply_async(
|
update_character.apply_async(
|
||||||
args=[character.get('character_id')], priority=TASK_PRIORITY
|
args=[character.get('character_id')],
|
||||||
)
|
priority=TASK_PRIORITY)
|
||||||
|
@ -14,15 +14,13 @@ Needs to be called with a context containing three objects:
|
|||||||
{% block page_title %}Evelinks Examples{% endblock page_title %}
|
{% block page_title %}Evelinks Examples{% endblock page_title %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
|
<div>
|
||||||
|
{% include "framework/header/page-header.html" with title="Evelinks templatetags examples" %}
|
||||||
|
|
||||||
<div class="col-lg-12">
|
|
||||||
<h1 class="page-header text-center">Evelinks templatetags examples</h1>
|
|
||||||
<div class="col-lg-12 container">
|
<div class="col-lg-12 container">
|
||||||
|
|
||||||
<h2>profile URLs</h2>
|
<h2>profile URLs</h2>
|
||||||
|
|
||||||
<div class="rows">
|
<div class="rows">
|
||||||
|
|
||||||
<div class="col-md-4">
|
<div class="col-md-4">
|
||||||
<h3>evewho</h3>
|
<h3>evewho</h3>
|
||||||
<p><a href="{{ my_character|evewho_character_url }}">character from character object</a></p>
|
<p><a href="{{ my_character|evewho_character_url }}">character from character object</a></p>
|
||||||
@ -57,7 +55,6 @@ Needs to be called with a context containing three objects:
|
|||||||
<h2>image URLs</h2>
|
<h2>image URLs</h2>
|
||||||
|
|
||||||
<div class="rows">
|
<div class="rows">
|
||||||
|
|
||||||
<div class="col-md-4">
|
<div class="col-md-4">
|
||||||
<p>character from ID: <img src="{{ my_character.character_id|character_portrait_url:128 }}"></p>
|
<p>character from ID: <img src="{{ my_character.character_id|character_portrait_url:128 }}"></p>
|
||||||
<p>character from character object: <img src="{{ my_character|character_portrait_url:128 }}"></p>
|
<p>character from character object: <img src="{{ my_character|character_portrait_url:128 }}"></p>
|
||||||
@ -77,5 +74,4 @@ Needs to be called with a context containing three objects:
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{% endblock content %}
|
{% endblock content %}
|
||||||
|
@ -723,5 +723,5 @@ class TestEveSwaggerProvider(TestCase):
|
|||||||
my_client = my_provider.client
|
my_client = my_provider.client
|
||||||
operation = my_client.Universe.get_universe_factions()
|
operation = my_client.Universe.get_universe_factions()
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
operation.future.request.headers['User-Agent'], 'allianceauth v1.0.0'
|
operation.future.request.headers['User-Agent'], 'allianceauth v1.0.0 dummy@example.net'
|
||||||
)
|
)
|
||||||
|
@ -84,7 +84,7 @@ class TestUpdateTasks(TestCase):
|
|||||||
@override_settings(CELERY_ALWAYS_EAGER=True)
|
@override_settings(CELERY_ALWAYS_EAGER=True)
|
||||||
@patch('allianceauth.eveonline.providers.esi_client_factory')
|
@patch('allianceauth.eveonline.providers.esi_client_factory')
|
||||||
@patch('allianceauth.eveonline.tasks.providers')
|
@patch('allianceauth.eveonline.tasks.providers')
|
||||||
@patch('allianceauth.eveonline.tasks.CHUNK_SIZE', 2)
|
@patch('allianceauth.eveonline.tasks.CHARACTER_AFFILIATION_CHUNK_SIZE', 2)
|
||||||
class TestRunModelUpdate(TransactionTestCase):
|
class TestRunModelUpdate(TransactionTestCase):
|
||||||
def test_should_run_updates(self, mock_providers, mock_esi_client_factory):
|
def test_should_run_updates(self, mock_providers, mock_esi_client_factory):
|
||||||
# given
|
# given
|
||||||
@ -139,7 +139,7 @@ class TestRunModelUpdate(TransactionTestCase):
|
|||||||
@patch('allianceauth.eveonline.tasks.update_character', wraps=update_character)
|
@patch('allianceauth.eveonline.tasks.update_character', wraps=update_character)
|
||||||
@patch('allianceauth.eveonline.providers.esi_client_factory')
|
@patch('allianceauth.eveonline.providers.esi_client_factory')
|
||||||
@patch('allianceauth.eveonline.tasks.providers')
|
@patch('allianceauth.eveonline.tasks.providers')
|
||||||
@patch('allianceauth.eveonline.tasks.CHUNK_SIZE', 2)
|
@patch('allianceauth.eveonline.tasks.CHARACTER_AFFILIATION_CHUNK_SIZE', 2)
|
||||||
class TestUpdateCharacterChunk(TestCase):
|
class TestUpdateCharacterChunk(TestCase):
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _updated_character_ids(spy_update_character) -> set:
|
def _updated_character_ids(spy_update_character) -> set:
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
from django.apps import AppConfig
|
from django.apps import AppConfig
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
|
||||||
class FatConfig(AppConfig):
|
class FatConfig(AppConfig):
|
||||||
name = 'allianceauth.fleetactivitytracking'
|
name = 'allianceauth.fleetactivitytracking'
|
||||||
label = 'fleetactivitytracking'
|
label = 'fleetactivitytracking'
|
||||||
|
verbose_name = _('Fleet Activity Tracking')
|
||||||
|
@ -7,7 +7,7 @@ from allianceauth.services.hooks import UrlHook
|
|||||||
|
|
||||||
@hooks.register('menu_item_hook')
|
@hooks.register('menu_item_hook')
|
||||||
def register_menu():
|
def register_menu():
|
||||||
return MenuItemHook(_('Fleet Activity Tracking'), 'fas fa-users fa-fw', 'fatlink:view',
|
return MenuItemHook(_('Fleet Activity Tracking'), 'fa-solid fa-users', 'fatlink:view',
|
||||||
navactive=['fatlink:'])
|
navactive=['fatlink:'])
|
||||||
|
|
||||||
|
|
||||||
|
@ -4,4 +4,4 @@ from django.utils.translation import gettext_lazy as _
|
|||||||
|
|
||||||
class FatlinkForm(forms.Form):
|
class FatlinkForm(forms.Form):
|
||||||
fleet = forms.CharField(label=_("Fleet Name"), max_length=50)
|
fleet = forms.CharField(label=_("Fleet Name"), max_length=50)
|
||||||
duration = forms.IntegerField(label=_("Duration of fat-link"), required=True, initial=30, min_value=1, max_value=2147483647, help_text=_('minutes'))
|
duration = forms.IntegerField(label=_("Duration of fat-link"), required=True, initial=30, min_value=1, max_value=2147483647, help_text=_('Duration of the fat-link in minutes'))
|
||||||
|
@ -0,0 +1,26 @@
|
|||||||
|
"""
|
||||||
|
Migration to AA Framework API method
|
||||||
|
"""
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
import allianceauth.framework.api.user
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("fleetactivitytracking", "0006_auto_20180803_0430"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="fatlink",
|
||||||
|
name="creator",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
on_delete=models.SET(allianceauth.framework.api.user.get_sentinel_user),
|
||||||
|
to=settings.AUTH_USER_MODEL
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
@ -3,10 +3,7 @@ from django.db import models
|
|||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
|
|
||||||
from allianceauth.eveonline.models import EveCharacter
|
from allianceauth.eveonline.models import EveCharacter
|
||||||
|
from allianceauth.framework.api.user import get_sentinel_user
|
||||||
|
|
||||||
def get_sentinel_user():
|
|
||||||
return User.objects.get_or_create(username='deleted')[0]
|
|
||||||
|
|
||||||
|
|
||||||
class Fatlink(models.Model):
|
class Fatlink(models.Model):
|
||||||
|
@ -1,23 +1,43 @@
|
|||||||
{% extends 'allianceauth/base-bs5.html' %}
|
{% extends 'allianceauth/base-bs5.html' %}
|
||||||
|
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block page_title %}
|
{% block page_title %}
|
||||||
{% translate "Fleet Participation" %}
|
{% translate "Fleet Participation" %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block header_nav_brand %}
|
||||||
|
{% translate "Fleet Activity Tracking" %}
|
||||||
|
{% endblock header_nav_brand %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="col-lg-12">
|
<div>
|
||||||
<h1 class="page-header text-center">{% translate "Character not found!" %}</h1>
|
{% translate "Character not found!" as page_header %}
|
||||||
<div class="col-lg-12 container" id="example">
|
{% include "framework/header/page-header.html" with title=page_header %}
|
||||||
|
|
||||||
|
<div class="col-lg-12 container">
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<div class="panel panel-default">
|
<div class="card card-default">
|
||||||
<div class="panel-heading">{{ character_name }}</div>
|
<div class="card-header">
|
||||||
<div class="panel-body">
|
<div class="card-title mb-0">
|
||||||
|
{{ character_name }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="card-body">
|
||||||
<div class="col-lg-2 col-sm-2">
|
<div class="col-lg-2 col-sm-2">
|
||||||
<img class="ra-avatar img-responsive" src="{{ character_portrait_url }}" alt="{{ character_name }}">
|
<img class="ra-avatar img-responsive" src="{{ character_portrait_url }}" alt="{{ character_name }}">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="col-lg-10 col-sm-2">
|
<div class="col-lg-10 col-sm-2">
|
||||||
<div class="alert alert-danger" role="alert">{% translate "Character not registered!" %}</div>
|
<div class="alert alert-danger" role="alert">
|
||||||
{% translate "This character is not associated with an auth account." %} <a href=" {% url 'authentication:add_character' %}">{% translate "Add it here" %}</a> {% translate "before attempting to click fleet attendance links." %}
|
{% translate "Character not registered!" %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% translate "This character is not associated with an auth account." %}
|
||||||
|
<a href="{% url 'authentication:add_character' %}">{% translate "Add it here" %}</a>
|
||||||
|
{% translate "before attempting to click fleet attendance links." %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -0,0 +1,53 @@
|
|||||||
|
{% extends "allianceauth/base-bs5.html" %}
|
||||||
|
|
||||||
|
{% load django_bootstrap5 %}
|
||||||
|
{% load i18n %}
|
||||||
|
|
||||||
|
{% block page_title %}
|
||||||
|
{% translate "Create Fatlink" %}
|
||||||
|
{% endblock page_title %}
|
||||||
|
|
||||||
|
{% block header_nav_brand %}
|
||||||
|
{% translate "Fleet Activity Tracking" %}
|
||||||
|
{% endblock header_nav_brand %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<div>
|
||||||
|
{% translate "Create Fatlink" as page_header %}
|
||||||
|
{% include "framework/header/page-header.html" with title=page_header %}
|
||||||
|
|
||||||
|
<div>
|
||||||
|
{% if badrequest %}
|
||||||
|
<div class="alert alert-danger" role="alert">{% translate "Bad request!" %}</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% for message in errormessages %}
|
||||||
|
<div class="alert alert-danger" role="alert">{{ message }}</div>
|
||||||
|
{% endfor %}
|
||||||
|
|
||||||
|
<div class="card card-primary border-0">
|
||||||
|
<div class="card-header">
|
||||||
|
<div class="card-title mb-0">
|
||||||
|
{% translate "Fatlink details" %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="card-body">
|
||||||
|
<div class="row justify-content-center">
|
||||||
|
<div class="col-md-6">
|
||||||
|
<form role="form" action="" method="POST">
|
||||||
|
{% csrf_token %}
|
||||||
|
{% bootstrap_form form %}
|
||||||
|
|
||||||
|
<div class="form-group mt-3 clearfix">
|
||||||
|
{% translate "Create fatlink" as button_text %}
|
||||||
|
{% bootstrap_button button_class="btn btn-primary" content=button_text name="submit_fat" id="submit_fat" %}
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endblock content %}
|
@ -1,31 +0,0 @@
|
|||||||
{% extends "allianceauth/base-bs5.html" %}
|
|
||||||
{% load bootstrap %}
|
|
||||||
{% load i18n %}
|
|
||||||
{% block page_title %}
|
|
||||||
{% translate "Create Fatlink" %}
|
|
||||||
{% endblock page_title %}
|
|
||||||
{% block content %}
|
|
||||||
<div class="col-lg-12">
|
|
||||||
<h1 class="page-header text-center">{% translate "Create Fleet Operation" %}</h1>
|
|
||||||
<div class="container-fluid">
|
|
||||||
{% if badrequest %}
|
|
||||||
<div class="alert alert-danger" role="alert">{% translate "Bad request!" %}</div>
|
|
||||||
{% endif %}
|
|
||||||
{% for message in errormessages %}<div class="alert alert-danger" role="alert">{{ message }}</div>{% endfor %}
|
|
||||||
<div class="col-md-4 offset-md-4">
|
|
||||||
<div class="row">
|
|
||||||
<form class="form-signin" role="form" action="" method="POST">
|
|
||||||
{% csrf_token %}
|
|
||||||
{{ form|bootstrap }}
|
|
||||||
<br/>
|
|
||||||
<button class="btn btn-lg btn-primary btn-block"
|
|
||||||
type="submit"
|
|
||||||
name="submit_fat">
|
|
||||||
{% translate "Create fatlink" %}
|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endblock content %}
|
|
@ -1,45 +1,61 @@
|
|||||||
{% extends "allianceauth/base-bs5.html" %}
|
{% extends "allianceauth/base-bs5.html" %}
|
||||||
|
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
{% block page_title %}{% translate "Fatlink view" %}{% endblock page_title %}
|
|
||||||
|
{% block page_title %}
|
||||||
|
{% translate "Fatlink view" %}
|
||||||
|
{% endblock page_title %}
|
||||||
|
|
||||||
|
{% block header_nav_brand %}
|
||||||
|
{% translate "Fleet Activity Tracking" %}
|
||||||
|
{% endblock header_nav_brand %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="col-lg-12">
|
<div>
|
||||||
<h1 class="page-header text-center">{% translate "Edit fatlink" %} "{{ fatlink }}"
|
<h1 class="page-header text-center mb-3">
|
||||||
<div class="text-end">
|
{% translate "Edit fatlink" %} "{{ fatlink }}"
|
||||||
|
</h1>
|
||||||
|
|
||||||
|
<div class="text-end mb-3">
|
||||||
<form>
|
<form>
|
||||||
<button type="submit" onclick="return confirm('Are you sure?')" class="btn btn-danger" name="deletefat" value="True">
|
<button type="submit" onclick="return confirm('{% translate "Are you sure?" %}')" class="btn btn-danger" name="deletefat" value="True">
|
||||||
{% translate "Delete fat" %}
|
{% translate "Delete fat" %}
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
</h1>
|
|
||||||
<div class="panel panel-default">
|
<div class="card card-default">
|
||||||
<div class="panel-heading">{% translate "Registered characters" %}</div>
|
<div class="card-header">
|
||||||
<div class="panel-body">
|
<div class="card-title mb-0">{% translate "Registered characters" %}</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="card-body">
|
||||||
<table class="table table-responsive table-hover">
|
<table class="table table-responsive table-hover">
|
||||||
<tr>
|
<tr>
|
||||||
<th class="text-center">{% translate "User" %}</th>
|
<th class="text-center">{% translate "User" %}</th>
|
||||||
<th class="text-center">{% translate "Character" %}</th>
|
<th class="text-center">{% translate "Character" %}</th>
|
||||||
<th class="text-center">{% translate "System" %}</th>
|
<th class="text-center">{% translate "System" %}</th>
|
||||||
<th class="text-center">{% translate "Ship" %}</th>
|
<th class="text-center">{% translate "Ship" %}</th>
|
||||||
<th class="text-center">{% translate "Eve Time" %}</th>
|
<th class="text-center">{% translate "EVE time" %}</th>
|
||||||
<th></th>
|
<th></th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
{% for fat in registered_fats %}
|
{% for fat in registered_fats %}
|
||||||
<tr>
|
<tr>
|
||||||
<td class="text-center">{{ fat.user }}</td>
|
<td class="text-center">{{ fat.user }}</td>
|
||||||
<td class="text-center">{{ fat.character.character_name }}</td>
|
<td class="text-center">{{ fat.character.character_name }}</td>
|
||||||
|
<td class="text-center">
|
||||||
{% if fat.station != "No Station" %}
|
{% if fat.station != "No Station" %}
|
||||||
<td class="text-center">{% blocktranslate %}Docked in {% endblocktranslate %}{{ fat.system }}</td>
|
{% translate "Docked in" %}
|
||||||
{% else %}
|
|
||||||
<td class="text-center">{{ fat.system }}</td>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
{{ fat.system }}
|
||||||
|
</td>
|
||||||
<td class="text-center">{{ fat.shiptype }}</td>
|
<td class="text-center">{{ fat.shiptype }}</td>
|
||||||
<td class="text-center">{{ fat.fatlink.fatdatetime }}</td>
|
<td class="text-center">{{ fat.fatlink.fatdatetime }}</td>
|
||||||
<td class="text-center">
|
<td class="text-center">
|
||||||
<form>
|
<form>
|
||||||
<button type="submit" class="btn btn-warning" name="removechar" value="{{ fat.character.character_id }}">
|
<button type="submit" class="btn btn-warning" name="removechar" value="{{ fat.character.character_id }}">
|
||||||
<span class="glyphicon glyphicon-remove"></span>
|
<i class="fa-solid fa-trash-can fa-fw"></i>
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</td>
|
</td>
|
||||||
|
@ -1,30 +1,50 @@
|
|||||||
{% extends "allianceauth/base-bs5.html" %}
|
{% extends "allianceauth/base-bs5.html" %}
|
||||||
|
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block page_title %}{% translate "Personal fatlink statistics" %}{% endblock page_title %}
|
{% block page_title %}
|
||||||
|
{% translate "Personal fatlink statistics" %}
|
||||||
|
{% endblock page_title %}
|
||||||
|
|
||||||
|
{% block header_nav_brand %}
|
||||||
|
{% translate "Fleet Activity Tracking" %}
|
||||||
|
{% endblock header_nav_brand %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="col-lg-12">
|
<div>
|
||||||
<h1 class="page-header text-center">{% blocktranslate %}Participation data statistics for {{ month }}, {{ year }}{% endblocktranslate %}
|
<h1 class="page-header text-center mb-3">
|
||||||
|
{% blocktranslate %}Participation data statistics for {{ month }}, {{ year }}{% endblocktranslate %}
|
||||||
|
</h1>
|
||||||
|
|
||||||
{% if char_id %}
|
{% if char_id %}
|
||||||
<div class="text-end">
|
<div class="text-end mb-3">
|
||||||
<a href="{% url 'fatlink:user_statistics_month' char_id previous_month|date:'Y' previous_month|date:'m' %}" class="btn btn-info">{% translate "Previous month" %}</a>
|
<a href="{% url 'fatlink:user_statistics_month' char_id previous_month|date:'Y' previous_month|date:'m' %}" class="btn btn-info">
|
||||||
<a href="{% url 'fatlink:user_statistics_month' char_id next_month|date:'Y' next_month|date:'m' %}" class="btn btn-info">{% translate "Next month" %}</a>
|
{% translate "Previous month" %}
|
||||||
|
</a>
|
||||||
|
<a href="{% url 'fatlink:user_statistics_month' char_id next_month|date:'Y' next_month|date:'m' %}" class="btn btn-info">
|
||||||
|
{% translate "Next month" %}
|
||||||
|
</a>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</h1>
|
|
||||||
<h2>
|
<div class="card card-default mb-3">
|
||||||
|
<div class="card-header">
|
||||||
|
<div class="card-title mb-0">
|
||||||
{% blocktranslate count links=n_fats trimmed %}
|
{% blocktranslate count links=n_fats trimmed %}
|
||||||
{{ user }} has collected one link this month.
|
{{ user }} has collected one link this month.
|
||||||
{% plural %}
|
{% plural %}
|
||||||
{{ user }} has collected {{ links }} links this month.
|
{{ user }} has collected {{ links }} links this month.
|
||||||
{% endblocktranslate %}
|
{% endblocktranslate %}
|
||||||
</h2>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="card-body">
|
||||||
<table class="table table-responsive">
|
<table class="table table-responsive">
|
||||||
<tr>
|
<tr>
|
||||||
<th class="col-md-2 text-center">{% translate "Ship" %}</th>
|
<th class="col-md-2 text-center">{% translate "Ship" %}</th>
|
||||||
<th class="col-md-2 text-center">{% translate "Times used" %}</th>
|
<th class="col-md-2 text-center">{% translate "Times used" %}</th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
{% for ship, n_fats in shipStats %}
|
{% for ship, n_fats in shipStats %}
|
||||||
<tr>
|
<tr>
|
||||||
<td class="text-center">{{ ship }}</td>
|
<td class="text-center">{{ ship }}</td>
|
||||||
@ -32,40 +52,53 @@
|
|||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</table>
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
{% if created_fats %}
|
{% if created_fats %}
|
||||||
<h2>
|
<div class="card card-default">
|
||||||
|
<div class="card-header">
|
||||||
|
<div class="card-title mb-0">
|
||||||
{% blocktranslate count links=n_created_fats trimmed %}
|
{% blocktranslate count links=n_created_fats trimmed %}
|
||||||
{{ user }} has created one link this month.
|
{{ user }} has created one link this month.
|
||||||
{% plural %}
|
{% plural %}
|
||||||
{{ user }} has created {{ links }} links this month.
|
{{ user }} has created {{ links }} links this month.
|
||||||
{% endblocktranslate %}
|
{% endblocktranslate %}
|
||||||
</h2>
|
</div>
|
||||||
{% if created_fats %}
|
</div>
|
||||||
|
|
||||||
|
<div class="card-body">
|
||||||
<table class="table">
|
<table class="table">
|
||||||
<tr>
|
<tr>
|
||||||
<th class="text-center">{% translate "Fleet" %}</th>
|
<th class="text-center">{% translate "Fleet" %}</th>
|
||||||
<th class="text-center">{% translate "Creator" %}</th>
|
<th class="text-center">{% translate "Creator" %}</th>
|
||||||
<th class="text-center">{% translate "Eve Time" %}</th>
|
<th class="text-center">{% translate "EVE time" %}</th>
|
||||||
<th class="text-center">{% translate "Duration" %}</th>
|
<th class="text-center">{% translate "Duration" %}</th>
|
||||||
<th class="text-center">{% translate "Edit" %}</th>
|
<th class="text-center">{% translate "Edit" %}</th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
{% for link in created_fats %}
|
{% for link in created_fats %}
|
||||||
<tr>
|
<tr>
|
||||||
<td class="text-center"><a href="{% url 'fatlink:click' link.hash %}" class="badge badge-primary">{{ link.fleet }}</a></td>
|
<td class="text-center">
|
||||||
|
<a href="{% url 'fatlink:click' link.hash %}" class="badge text-bg-primary">
|
||||||
|
{{ link.fleet }}
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
<td class="text-center">{{ link.creator.username }}</td>
|
<td class="text-center">{{ link.creator.username }}</td>
|
||||||
<td class="text-center">{{ link.fatdatetime }}</td>
|
<td class="text-center">{{ link.fatdatetime }}</td>
|
||||||
<td class="text-center">{{ link.duration }}</td>
|
<td class="text-center">{{ link.duration }}</td>
|
||||||
<td class="text-center">
|
<td class="text-center">
|
||||||
<a href="{% url 'fatlink:modify' link.hash %}">
|
<a href="{% url 'fatlink:modify' link.hash %}">
|
||||||
<button type="button" class="btn btn-info"><span
|
<button type="button" class="btn btn-info">
|
||||||
class="glyphicon glyphicon-edit"></span></button>
|
<i class="fa-solid fa-pen-to-square fa-fw"></i>
|
||||||
|
</button>
|
||||||
</a>
|
</a>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
</table>
|
</table>
|
||||||
{% endif %}
|
</div>
|
||||||
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
{% endblock content %}
|
{% endblock content %}
|
||||||
|
@ -1,25 +1,36 @@
|
|||||||
{% extends "allianceauth/base-bs5.html" %}
|
{% extends "allianceauth/base-bs5.html" %}
|
||||||
|
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block page_title %}
|
{% block page_title %}
|
||||||
{% translate "Personal fatlink statistics" %}
|
{% translate "Personal fatlink statistics" %}
|
||||||
{% endblock page_title %}
|
{% endblock page_title %}
|
||||||
|
|
||||||
|
{% block header_nav_brand %}
|
||||||
|
{% translate "Fleet Activity Tracking" %}
|
||||||
|
{% endblock header_nav_brand %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="col-lg-12">
|
<div>
|
||||||
<h1 class="page-header text-center">
|
<h1 class="page-header text-center mb-3">
|
||||||
{% blocktranslate %}Participation data statistics for {{ year }}{% endblocktranslate %}
|
{% blocktranslate %}Participation data statistics for {{ year }}{% endblocktranslate %}
|
||||||
<div class="text-end">
|
</h1>
|
||||||
|
|
||||||
|
<div class="text-end mb-3">
|
||||||
<a href="{% url "fatlink:personal_statistics_year" previous_year %}" class="btn btn-info"><i class="fa-solid fa-chevron-left"></i> {% translate "Previous year" %}</a>
|
<a href="{% url "fatlink:personal_statistics_year" previous_year %}" class="btn btn-info"><i class="fa-solid fa-chevron-left"></i> {% translate "Previous year" %}</a>
|
||||||
|
|
||||||
{% if next_year %}
|
{% if next_year %}
|
||||||
<a href="{% url "fatlink:personal_statistics_year" next_year %}" class="btn btn-info">{% translate "Next year" %} <i class="fa-solid fa-chevron-right"></i></a>
|
<a href="{% url "fatlink:personal_statistics_year" next_year %}" class="btn btn-info">{% translate "Next year" %} <i class="fa-solid fa-chevron-right"></i></a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
</h1>
|
|
||||||
<div class="col-lg-2 offset-lg-5">
|
<div class="col-lg-2 offset-lg-5">
|
||||||
<table class="table table-responsive">
|
<table class="table table-responsive">
|
||||||
<tr>
|
<tr>
|
||||||
<th scope="col" class="col-md-2 text-center">{% translate "Month" %}</th>
|
<th scope="col" class="col-md-2 text-center">{% translate "Month" %}</th>
|
||||||
<th scope="col" class="col-md-2 text-center">{% translate "Fats" %}</th>
|
<th scope="col" class="col-md-2 text-center">{% translate "Fats" %}</th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
{% for monthnr, month, n_fats in monthlystats %}
|
{% for monthnr, month, n_fats in monthlystats %}
|
||||||
<tr>
|
<tr>
|
||||||
<td class="text-center">
|
<td class="text-center">
|
||||||
|
@ -1,19 +1,29 @@
|
|||||||
{% extends "allianceauth/base-bs5.html" %}
|
{% extends "allianceauth/base-bs5.html" %}
|
||||||
|
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block page_title %}
|
{% block page_title %}
|
||||||
{% translate "Fatlink Corp Statistics" %}
|
{% translate "Fatlink Corp Statistics" %}
|
||||||
{% endblock page_title %}
|
{% endblock page_title %}
|
||||||
|
|
||||||
|
{% block header_nav_brand %}
|
||||||
|
{% translate "Fleet Activity Tracking" %}
|
||||||
|
{% endblock header_nav_brand %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="col-lg-12">
|
<div>
|
||||||
<h1 class="page-header text-center">
|
<h1 class="page-header text-center mb-3">
|
||||||
{% blocktranslate %}Participation data statistics for {{ month }}, {{ year }}{% endblocktranslate %}
|
{% blocktranslate %}Participation data statistics for {{ month }}, {{ year }}{% endblocktranslate %}
|
||||||
<div class="text-end">
|
</h1>
|
||||||
|
|
||||||
|
<div class="text-end mb-3">
|
||||||
<a href="{% url "fatlink:statistics_corp_month" corpid previous_month|date:"Y" previous_month|date:"m" %}" class="btn btn-info">{% translate "Previous month" %}</a>
|
<a href="{% url "fatlink:statistics_corp_month" corpid previous_month|date:"Y" previous_month|date:"m" %}" class="btn btn-info">{% translate "Previous month" %}</a>
|
||||||
|
|
||||||
{% if next_month %}
|
{% if next_month %}
|
||||||
<a href="{% url "fatlink:statistics_corp_month" corpid next_month|date:"Y" next_month|date:"m" %}" class="btn btn-info">{% translate "Next month" %}</a>
|
<a href="{% url "fatlink:statistics_corp_month" corpid next_month|date:"Y" next_month|date:"m" %}" class="btn btn-info">{% translate "Next month" %}</a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
</h1>
|
|
||||||
{% if fatStats %}
|
{% if fatStats %}
|
||||||
<div class="table-responsive">
|
<div class="table-responsive">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
@ -43,8 +53,11 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
{% endblock content %}
|
{% endblock content %}
|
||||||
{% block extra_script %}
|
|
||||||
$(document).ready(function () {
|
{% block extra_javascript %}
|
||||||
|
<script>
|
||||||
|
$(document).ready(() => {
|
||||||
$("[rel=tooltip]").tooltip();
|
$("[rel=tooltip]").tooltip();
|
||||||
});
|
});
|
||||||
{% endblock extra_script %}
|
</script>
|
||||||
|
{% endblock extra_javascript %}
|
||||||
|
@ -1,19 +1,29 @@
|
|||||||
{% extends "allianceauth/base-bs5.html" %}
|
{% extends "allianceauth/base-bs5.html" %}
|
||||||
|
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block page_title %}
|
{% block page_title %}
|
||||||
{% translate "Fatlink Statistics" %}
|
{% translate "Fatlink Statistics" %}
|
||||||
{% endblock page_title %}
|
{% endblock page_title %}
|
||||||
|
|
||||||
|
{% block header_nav_brand %}
|
||||||
|
{% translate "Fleet Activity Tracking" %}
|
||||||
|
{% endblock header_nav_brand %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="col-lg-12">
|
<div>
|
||||||
<h1 class="page-header text-center">
|
<h1 class="page-header text-center mb-3">
|
||||||
{% blocktranslate %}Participation data statistics for {{ month }}, {{ year }}{% endblocktranslate %}
|
{% blocktranslate %}Participation data statistics for {{ month }}, {{ year }}{% endblocktranslate %}
|
||||||
<div class="text-end">
|
</h1>
|
||||||
|
|
||||||
|
<div class="text-end mb-3">
|
||||||
<a href="{% url "fatlink:statistics_month" previous_month|date:"Y" previous_month|date:"m" %}" class="btn btn-info">{% translate "Previous month" %}</a>
|
<a href="{% url "fatlink:statistics_month" previous_month|date:"Y" previous_month|date:"m" %}" class="btn btn-info">{% translate "Previous month" %}</a>
|
||||||
|
|
||||||
{% if next_month %}
|
{% if next_month %}
|
||||||
<a href="{% url 'fatlink:statistics_month' next_month|date:"Y" next_month|date:"m" %}" wclass="btn btn-info">{% translate "Next month" %}</a>
|
<a href="{% url 'fatlink:statistics_month' next_month|date:"Y" next_month|date:"m" %}" class="btn btn-info">{% translate "Next month" %}</a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
</h1>
|
|
||||||
{% if fatStats %}
|
{% if fatStats %}
|
||||||
<div class="table-responsive">
|
<div class="table-responsive">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
@ -31,7 +41,7 @@
|
|||||||
{% for corpStat in fatStats %}
|
{% for corpStat in fatStats %}
|
||||||
<tr>
|
<tr>
|
||||||
<td>
|
<td>
|
||||||
<img src="{{ corpStat.corp.logo_url_32 }}" class="ra-avatar img-responsive" alt="{{ corpStat.corp.corporation_name }}"/>
|
<img src="{{ corpStat.corp.logo_url_32 }}" class="ra-avatar img-responsive" alt="{{ corpStat.corp.corporation_name }}">
|
||||||
</td>
|
</td>
|
||||||
<td class="text-center">
|
<td class="text-center">
|
||||||
<a href="{% url 'fatlink:statistics_corp' corpStat.corp.corporation_id %}">[{{ corpStat.corp.corporation_ticker }}]</a>
|
<a href="{% url 'fatlink:statistics_corp' corpStat.corp.corporation_id %}">[{{ corpStat.corp.corporation_ticker }}]</a>
|
||||||
@ -47,8 +57,11 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
{% endblock content %}
|
{% endblock content %}
|
||||||
{% block extra_script %}
|
|
||||||
$(document).ready(function () {
|
{% block extra_javascript %}
|
||||||
|
<script>
|
||||||
|
$(document).ready(() => {
|
||||||
$("[rel=tooltip]").tooltip();
|
$("[rel=tooltip]").tooltip();
|
||||||
});
|
});
|
||||||
{% endblock extra_script %}
|
</script>
|
||||||
|
{% endblock extra_javascript %}
|
||||||
|
@ -1,11 +1,20 @@
|
|||||||
{% extends "allianceauth/base-bs5.html" %}
|
{% extends "allianceauth/base-bs5.html" %}
|
||||||
|
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block page_title %}
|
{% block page_title %}
|
||||||
{% translate "Fatlink view" %}
|
{% translate "Fatlink view" %}
|
||||||
{% endblock page_title %}
|
{% endblock page_title %}
|
||||||
|
|
||||||
|
{% block header_nav_brand %}
|
||||||
|
{% translate "Fleet Activity Tracking" %}
|
||||||
|
{% endblock header_nav_brand %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="col-lg-12">
|
<div>
|
||||||
<h1 class="page-header text-center">{% translate "Participation data" %}</h1>
|
{% translate "Participation data" as page_header %}
|
||||||
|
{% include "framework/header/page-header.html" with title=page_header %}
|
||||||
|
|
||||||
<div class="table-responsive">
|
<div class="table-responsive">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
<tr>
|
<tr>
|
||||||
@ -15,11 +24,15 @@
|
|||||||
</h4>
|
</h4>
|
||||||
</th>
|
</th>
|
||||||
<th class="col-md-2 align-self-end">
|
<th class="col-md-2 align-self-end">
|
||||||
<a href="{% url 'fatlink:personal_statistics' %}" class="btn btn-info"><i class="fa-solid fa-circle-info fa-fw"></i>{% translate "Personal statistics" %}</a>
|
<a href="{% url 'fatlink:personal_statistics' %}" class="btn btn-info">
|
||||||
|
<i class="fa-solid fa-circle-info fa-fw"></i>
|
||||||
|
{% translate "Personal statistics" %}
|
||||||
|
</a>
|
||||||
</th>
|
</th>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{% if fats %}
|
{% if fats %}
|
||||||
<div class="table-responsive">
|
<div class="table-responsive">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
@ -28,8 +41,9 @@
|
|||||||
<th scope="col" class="text-center">{% translate "Character" %}</th>
|
<th scope="col" class="text-center">{% translate "Character" %}</th>
|
||||||
<th scope="col" class="text-center">{% translate "System" %}</th>
|
<th scope="col" class="text-center">{% translate "System" %}</th>
|
||||||
<th scope="col" class="text-center">{% translate "Ship" %}</th>
|
<th scope="col" class="text-center">{% translate "Ship" %}</th>
|
||||||
<th scope="col" class="text-center">{% translate "Eve Time" %}</th>
|
<th scope="col" class="text-center">{% translate "EVE time" %}</th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
{% for fat in fats %}
|
{% for fat in fats %}
|
||||||
<tr>
|
<tr>
|
||||||
<td class="text-center">{{ fat.fatlink.fleet }}</td>
|
<td class="text-center">{{ fat.fatlink.fleet }}</td>
|
||||||
@ -48,6 +62,7 @@
|
|||||||
{% else %}
|
{% else %}
|
||||||
<div class="alert alert-warning text-center">{% translate "No fleet activity on record." %}</div>
|
<div class="alert alert-warning text-center">{% translate "No fleet activity on record." %}</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% if perms.auth.fleetactivitytracking %}
|
{% if perms.auth.fleetactivitytracking %}
|
||||||
<div class="table-responsive">
|
<div class="table-responsive">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
@ -66,6 +81,7 @@
|
|||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{% if fatlinks %}
|
{% if fatlinks %}
|
||||||
<div class="table-responsive">
|
<div class="table-responsive">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
@ -73,14 +89,15 @@
|
|||||||
<th scope="col" class="text-center">{% translate "Name" %}</th>
|
<th scope="col" class="text-center">{% translate "Name" %}</th>
|
||||||
<th scope="col" class="text-center">{% translate "Creator" %}</th>
|
<th scope="col" class="text-center">{% translate "Creator" %}</th>
|
||||||
<th scope="col" class="text-center">{% translate "Fleet" %}</th>
|
<th scope="col" class="text-center">{% translate "Fleet" %}</th>
|
||||||
<th scope="col" class="text-center">{% translate "Eve Time" %}</th>
|
<th scope="col" class="text-center">{% translate "EVE time" %}</th>
|
||||||
<th scope="col" class="text-center">{% translate "Duration" %}</th>
|
<th scope="col" class="text-center">{% translate "Duration" %}</th>
|
||||||
<th scope="col" class="text-center">{% translate "Edit" %}</th>
|
<th scope="col" class="text-center">{% translate "Edit" %}</th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
{% for link in fatlinks %}
|
{% for link in fatlinks %}
|
||||||
<tr>
|
<tr>
|
||||||
<td class="text-center">
|
<td class="text-center">
|
||||||
<a href="{% url 'fatlink:click' link.hash %}" class="badge badge-primary">{{ link.fleet }}</a>
|
<a href="{% url 'fatlink:click' link.hash %}" class="badge text-bg-primary">{{ link.fleet }}</a>
|
||||||
</td>
|
</td>
|
||||||
<td class="text-center">{{ link.creator.username }}</td>
|
<td class="text-center">{{ link.creator.username }}</td>
|
||||||
<td class="text-center">{{ link.fleet }}</td>
|
<td class="text-center">{{ link.fleet }}</td>
|
||||||
|
@ -352,11 +352,11 @@ def create_fatlink_view(request):
|
|||||||
for errorname, message in e.message_dict.items():
|
for errorname, message in e.message_dict.items():
|
||||||
messages.append(message[0].decode())
|
messages.append(message[0].decode())
|
||||||
context = {'form': form, 'errormessages': messages}
|
context = {'form': form, 'errormessages': messages}
|
||||||
return render(request, 'fleetactivitytracking/fatlinkformatter.html', context=context)
|
return render(request, 'fleetactivitytracking/fatlinkcreate.html', context=context)
|
||||||
else:
|
else:
|
||||||
form = FatlinkForm()
|
form = FatlinkForm()
|
||||||
context = {'form': form, 'badrequest': True}
|
context = {'form': form, 'badrequest': True}
|
||||||
return render(request, 'fleetactivitytracking/fatlinkformatter.html', context=context)
|
return render(request, 'fleetactivitytracking/fatlinkcreate.html', context=context)
|
||||||
return redirect('fatlink:view')
|
return redirect('fatlink:view')
|
||||||
|
|
||||||
else:
|
else:
|
||||||
@ -365,7 +365,7 @@ def create_fatlink_view(request):
|
|||||||
|
|
||||||
context = {'form': form}
|
context = {'form': form}
|
||||||
|
|
||||||
return render(request, 'fleetactivitytracking/fatlinkformatter.html', context=context)
|
return render(request, 'fleetactivitytracking/fatlinkcreate.html', context=context)
|
||||||
|
|
||||||
|
|
||||||
@login_required
|
@login_required
|
||||||
|
3
allianceauth/framework/__init__.py
Normal file
3
allianceauth/framework/__init__.py
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
"""
|
||||||
|
Alliance Auth Framework
|
||||||
|
"""
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user