mirror of
https://github.com/9001/copyparty.git
synced 2025-11-04 13:53:18 +00:00
Compare commits
814 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ecdec75b4e | ||
|
|
5cb2e33353 | ||
|
|
43ff2e531a | ||
|
|
1c2c9db8f0 | ||
|
|
7ea183baef | ||
|
|
ab87fac6d8 | ||
|
|
1e3b7eee3b | ||
|
|
4de028fc3b | ||
|
|
604e5dfaaf | ||
|
|
05e0c2ec9e | ||
|
|
76bd005bdc | ||
|
|
5effaed352 | ||
|
|
cedaf4809f | ||
|
|
6deaf5c268 | ||
|
|
9dc6a26472 | ||
|
|
14ad5916fc | ||
|
|
1a46738649 | ||
|
|
9e5e3b099a | ||
|
|
292ce75cc2 | ||
|
|
ce7df7afd4 | ||
|
|
e28e793f81 | ||
|
|
3e561976db | ||
|
|
273a4eb7d0 | ||
|
|
6175f85bb6 | ||
|
|
a80579f63a | ||
|
|
96d6bcf26e | ||
|
|
49e8df25ac | ||
|
|
6a05850f21 | ||
|
|
5e7c3defe3 | ||
|
|
6c0987d4d0 | ||
|
|
6eba9feffe | ||
|
|
8adfcf5950 | ||
|
|
36d6fa512a | ||
|
|
79b6e9b393 | ||
|
|
dc2e2cbd4b | ||
|
|
5c12dac30f | ||
|
|
641929191e | ||
|
|
617321631a | ||
|
|
ddc0c899f8 | ||
|
|
cdec42c1ae | ||
|
|
c48f469e39 | ||
|
|
44909cc7b8 | ||
|
|
8f61e1568c | ||
|
|
b7be7a0fd8 | ||
|
|
1526a4e084 | ||
|
|
dbdb9574b1 | ||
|
|
853ae6386c | ||
|
|
a4b56c74c7 | ||
|
|
d7f1951e44 | ||
|
|
7e2ff9825e | ||
|
|
9b423396ec | ||
|
|
781146b2fb | ||
|
|
84937d1ce0 | ||
|
|
98cce66aa4 | ||
|
|
043c2d4858 | ||
|
|
99cc434779 | ||
|
|
5095d17e81 | ||
|
|
87d835ae37 | ||
|
|
6939ca768b | ||
|
|
e3957e8239 | ||
|
|
4ad6e45216 | ||
|
|
76e5eeea3f | ||
|
|
eb17f57761 | ||
|
|
b0db14d8b0 | ||
|
|
2b644fa81b | ||
|
|
190ccee820 | ||
|
|
4e7dd32e78 | ||
|
|
5817fb66ae | ||
|
|
9cb04eef93 | ||
|
|
0019fe7f04 | ||
|
|
852c6f2de1 | ||
|
|
c4191de2e7 | ||
|
|
4de61defc9 | ||
|
|
0aa88590d0 | ||
|
|
405f3ee5fe | ||
|
|
bc339f774a | ||
|
|
e67b695b23 | ||
|
|
4a7633ab99 | ||
|
|
c58f2ef61f | ||
|
|
3866e6a3f2 | ||
|
|
381686fc66 | ||
|
|
a918c285bf | ||
|
|
1e20eafbe0 | ||
|
|
39399934ee | ||
|
|
b47635150a | ||
|
|
78d2f69ed5 | ||
|
|
7a98dc669e | ||
|
|
2f15bb5085 | ||
|
|
712a578e6c | ||
|
|
d8dfc4ccb2 | ||
|
|
e413007eb0 | ||
|
|
6d1d3e48d8 | ||
|
|
04966164ce | ||
|
|
8b62aa7cc7 | ||
|
|
1088e8c6a5 | ||
|
|
8c54c2226f | ||
|
|
f74ac1f18b | ||
|
|
25931e62fd | ||
|
|
707a940399 | ||
|
|
87ef50d384 | ||
|
|
dcadf2b11c | ||
|
|
37a690a4c3 | ||
|
|
87ad23fb93 | ||
|
|
5f54d534e3 | ||
|
|
aecae552a4 | ||
|
|
eaa6b3d0be | ||
|
|
c2ace91e52 | ||
|
|
0bac87c36f | ||
|
|
e650d05939 | ||
|
|
85a96e4446 | ||
|
|
2569005139 | ||
|
|
c50cb66aef | ||
|
|
d4c5fca15b | ||
|
|
75cea4f684 | ||
|
|
68c6794d33 | ||
|
|
82f98dd54d | ||
|
|
741d781c18 | ||
|
|
0be1e43451 | ||
|
|
5366bf22bb | ||
|
|
bcd91b1809 | ||
|
|
9bd5738e6f | ||
|
|
bab4aa4c0a | ||
|
|
e965b9b9e2 | ||
|
|
31101427d3 | ||
|
|
a083dc36ba | ||
|
|
9b7b9262aa | ||
|
|
660011fa6e | ||
|
|
ead31b6823 | ||
|
|
4310580cd4 | ||
|
|
b005acbfda | ||
|
|
460709e6f3 | ||
|
|
a8768d05a9 | ||
|
|
f8e3e87a52 | ||
|
|
70f1642d0d | ||
|
|
3fc7561da4 | ||
|
|
9065226c3d | ||
|
|
b7e321fa47 | ||
|
|
664665b86b | ||
|
|
f4f362b7a4 | ||
|
|
577d23f460 | ||
|
|
504e168486 | ||
|
|
f2f9640371 | ||
|
|
ee46f832b1 | ||
|
|
b0e755d410 | ||
|
|
cfd24604d5 | ||
|
|
264894e595 | ||
|
|
5bb9f56247 | ||
|
|
18942ed066 | ||
|
|
85321a6f31 | ||
|
|
baf641396d | ||
|
|
17c91e7014 | ||
|
|
010770684d | ||
|
|
b4c503657b | ||
|
|
71bd306268 | ||
|
|
dd7fab1352 | ||
|
|
dacca18863 | ||
|
|
53d92cc0a6 | ||
|
|
434823f6f0 | ||
|
|
2cb1f50370 | ||
|
|
03f53f6392 | ||
|
|
a70ecd7af0 | ||
|
|
8b81e58205 | ||
|
|
4500c04edf | ||
|
|
6222ddd720 | ||
|
|
8a7135cf41 | ||
|
|
b4c7282956 | ||
|
|
8491a40a04 | ||
|
|
343d38b693 | ||
|
|
6cf53d7364 | ||
|
|
b070d44de7 | ||
|
|
79aa40fdea | ||
|
|
dcaff2785f | ||
|
|
497f5b4307 | ||
|
|
be32ad0da6 | ||
|
|
8ee2bf810b | ||
|
|
28232656a9 | ||
|
|
fbc2424e8f | ||
|
|
94cd13e8b8 | ||
|
|
447ed5ab37 | ||
|
|
af59808611 | ||
|
|
e3406a9f86 | ||
|
|
7fd1d6a4e8 | ||
|
|
0ab2a665de | ||
|
|
3895575bc2 | ||
|
|
138c2bbcbb | ||
|
|
bc7af1d1c8 | ||
|
|
19cd96e392 | ||
|
|
db194ab519 | ||
|
|
02ad4bfab2 | ||
|
|
56b73dcc8a | ||
|
|
7704b9c8a2 | ||
|
|
999b7ae919 | ||
|
|
252b5a88b1 | ||
|
|
01e2681a07 | ||
|
|
aa32f30202 | ||
|
|
195eb53995 | ||
|
|
06fa78f54a | ||
|
|
7a57c9dbf1 | ||
|
|
bb657bfa85 | ||
|
|
87181726b0 | ||
|
|
f1477a1c14 | ||
|
|
4f94a9e38b | ||
|
|
fbed322d3b | ||
|
|
9b0f519e4e | ||
|
|
6cd6dadd06 | ||
|
|
9a28afcb48 | ||
|
|
45b701801d | ||
|
|
062246fb12 | ||
|
|
416ebfdd68 | ||
|
|
731eb92f33 | ||
|
|
dbe2aec79c | ||
|
|
cd9cafe3a1 | ||
|
|
067cc23346 | ||
|
|
c573a780e9 | ||
|
|
8ef4a0aa71 | ||
|
|
89ba12065c | ||
|
|
99efc290df | ||
|
|
2fbdc0a85e | ||
|
|
4242422898 | ||
|
|
008d9b1834 | ||
|
|
7c76d08958 | ||
|
|
89c9f45fd0 | ||
|
|
f107497a94 | ||
|
|
b5dcf30e53 | ||
|
|
0cef062084 | ||
|
|
5c30148be4 | ||
|
|
3a800585bc | ||
|
|
29c212a60e | ||
|
|
2997baa7cb | ||
|
|
dc6bde594d | ||
|
|
e357aa546c | ||
|
|
d3fe19c5aa | ||
|
|
bd24bf9bae | ||
|
|
ee141544aa | ||
|
|
db6f6e6a23 | ||
|
|
c7d950dd5e | ||
|
|
6a96c62fde | ||
|
|
36dc8cd686 | ||
|
|
7622601a77 | ||
|
|
cfd41fcf41 | ||
|
|
f39e370e2a | ||
|
|
c1315a3b39 | ||
|
|
53b32f97e8 | ||
|
|
6c962ec7d3 | ||
|
|
6bc1bc542f | ||
|
|
f0e78a6826 | ||
|
|
e53531a9fb | ||
|
|
5cd9d11329 | ||
|
|
5a3e504ec4 | ||
|
|
d6e09c3880 | ||
|
|
04f44c3c7c | ||
|
|
ec587423e8 | ||
|
|
f57b31146d | ||
|
|
35175fd685 | ||
|
|
d326ba9723 | ||
|
|
ab655a56af | ||
|
|
d1eb113ea8 | ||
|
|
74effa9b8d | ||
|
|
bba4b1c663 | ||
|
|
8709d4dba0 | ||
|
|
4ad4657774 | ||
|
|
5abe0c955c | ||
|
|
0cedaf4fa9 | ||
|
|
0aa7d12704 | ||
|
|
a234aa1f7e | ||
|
|
9f68287846 | ||
|
|
cd2513ec16 | ||
|
|
91d132c2b4 | ||
|
|
97ff0ebd06 | ||
|
|
8829f56d4c | ||
|
|
37c1cab726 | ||
|
|
b3eb117e87 | ||
|
|
fc0a941508 | ||
|
|
c72753c5da | ||
|
|
e442cb677a | ||
|
|
450121eac9 | ||
|
|
b2ab8f971e | ||
|
|
e9c6268568 | ||
|
|
2170ee8da4 | ||
|
|
357e7333cc | ||
|
|
8bb4f02601 | ||
|
|
4213efc7a6 | ||
|
|
67a744c3e8 | ||
|
|
98818e7d63 | ||
|
|
8650ce1295 | ||
|
|
9638267b4c | ||
|
|
304e053155 | ||
|
|
89d1f52235 | ||
|
|
3312c6f5bd | ||
|
|
d4ba644d07 | ||
|
|
b9a504fd3a | ||
|
|
cebac523dc | ||
|
|
c2f4090318 | ||
|
|
d562956809 | ||
|
|
62499f9b71 | ||
|
|
89cf7608f9 | ||
|
|
dd26b8f183 | ||
|
|
79303dac6d | ||
|
|
4203fc161b | ||
|
|
f8a31cc24f | ||
|
|
fc5bfe81a0 | ||
|
|
aae14de796 | ||
|
|
54e1c8d261 | ||
|
|
a0cc4ca4b7 | ||
|
|
2701108c5b | ||
|
|
73bd2df2c6 | ||
|
|
0063021012 | ||
|
|
1c3e4750b3 | ||
|
|
edad3246e0 | ||
|
|
3411b0993f | ||
|
|
097b5609dc | ||
|
|
a42af7655e | ||
|
|
69f78b86af | ||
|
|
5f60c509c6 | ||
|
|
75e5e53276 | ||
|
|
4b2b4ed52d | ||
|
|
fb21bfd6d6 | ||
|
|
f14369e038 | ||
|
|
ff04b72f62 | ||
|
|
4535a81617 | ||
|
|
cce57b700b | ||
|
|
5b6194d131 | ||
|
|
2701238cea | ||
|
|
835f8a20e6 | ||
|
|
f3a501db30 | ||
|
|
4bcd30da6b | ||
|
|
947dbb6f8a | ||
|
|
1c2fedd2bf | ||
|
|
32e826efbc | ||
|
|
138b932c6a | ||
|
|
6da2f53aad | ||
|
|
20eeacaac3 | ||
|
|
81d896be9f | ||
|
|
c003dfab03 | ||
|
|
20c6b82bec | ||
|
|
046b494b53 | ||
|
|
f0e98d6e0d | ||
|
|
fe57321853 | ||
|
|
8510804e57 | ||
|
|
acd32abac5 | ||
|
|
2b47c96cf2 | ||
|
|
1027378bda | ||
|
|
e979d30659 | ||
|
|
574db704cc | ||
|
|
fdb969ea89 | ||
|
|
08977854b3 | ||
|
|
cecac64b68 | ||
|
|
7dabdade2a | ||
|
|
e788f098e2 | ||
|
|
69406d4344 | ||
|
|
d16dd26c65 | ||
|
|
12219c1bea | ||
|
|
118bdcc26e | ||
|
|
78fa96f0f4 | ||
|
|
c7deb63a04 | ||
|
|
4f811eb9e9 | ||
|
|
0b265bd673 | ||
|
|
ee67fabbeb | ||
|
|
b213de7e62 | ||
|
|
7c01505750 | ||
|
|
ae28dfd020 | ||
|
|
2a5a4e785f | ||
|
|
d8bddede6a | ||
|
|
b8a93e74bf | ||
|
|
e60ec94d35 | ||
|
|
84af5fd0a3 | ||
|
|
dbb3edec77 | ||
|
|
d284b46a3e | ||
|
|
9fcb4d222b | ||
|
|
d0bb1ad141 | ||
|
|
b299aaed93 | ||
|
|
abb3224cc5 | ||
|
|
1c66d06702 | ||
|
|
e00e80ae39 | ||
|
|
4f4f106c48 | ||
|
|
a286cc9d55 | ||
|
|
53bb1c719b | ||
|
|
98d5aa17e2 | ||
|
|
aaaa80e4b8 | ||
|
|
e70e926a40 | ||
|
|
e80c1f6d59 | ||
|
|
24de360325 | ||
|
|
e0039bc1e6 | ||
|
|
ae5c4a0109 | ||
|
|
1d367a0da0 | ||
|
|
d285f7ee4a | ||
|
|
37c84021a2 | ||
|
|
8ee9de4291 | ||
|
|
249b63453b | ||
|
|
1c0017d763 | ||
|
|
df51e23639 | ||
|
|
32e71a43b8 | ||
|
|
47a1e6ddfa | ||
|
|
c5f41457bb | ||
|
|
f1e0c44bdd | ||
|
|
9d2e390b6a | ||
|
|
75a58b435d | ||
|
|
f5474d34ac | ||
|
|
c962d2544f | ||
|
|
0b87a4a810 | ||
|
|
1882afb8b6 | ||
|
|
2270c8737a | ||
|
|
d6794955a4 | ||
|
|
f5520f45ef | ||
|
|
9401b5ae13 | ||
|
|
df64a62a03 | ||
|
|
09cea66aa8 | ||
|
|
13cc33e0a5 | ||
|
|
ab36c8c9de | ||
|
|
f85d4ce82f | ||
|
|
6bec4c28ba | ||
|
|
fad1449259 | ||
|
|
86b3b57137 | ||
|
|
b235037dd3 | ||
|
|
3108139d51 | ||
|
|
2ae99ecfa0 | ||
|
|
e8ab53c270 | ||
|
|
5e9bc1127d | ||
|
|
415e61c3c9 | ||
|
|
5152f37ec8 | ||
|
|
0dbeb010cf | ||
|
|
17c465bed7 | ||
|
|
add04478e5 | ||
|
|
6db72d7166 | ||
|
|
868103a9c5 | ||
|
|
0f37718671 | ||
|
|
fa1445df86 | ||
|
|
a783e7071e | ||
|
|
a9919df5af | ||
|
|
b0af31ac35 | ||
|
|
c4c964a685 | ||
|
|
348ec71398 | ||
|
|
a257ccc8b3 | ||
|
|
fcc4296040 | ||
|
|
1684d05d49 | ||
|
|
0006f933a2 | ||
|
|
0484f97c9c | ||
|
|
e430b2567a | ||
|
|
fbc8ee15da | ||
|
|
68a9c05947 | ||
|
|
0a81aba899 | ||
|
|
d2ae822e15 | ||
|
|
fac4b08526 | ||
|
|
3a7b43c663 | ||
|
|
8fcb2d1554 | ||
|
|
590c763659 | ||
|
|
11d1267f8c | ||
|
|
8f5bae95ce | ||
|
|
e6b12ef14c | ||
|
|
b65674618b | ||
|
|
20dca2bea5 | ||
|
|
059e93cdcf | ||
|
|
635ab25013 | ||
|
|
995cd10df8 | ||
|
|
50f3820a6d | ||
|
|
617f3ea861 | ||
|
|
788db47b95 | ||
|
|
5fa8aaabb9 | ||
|
|
89d1af7f33 | ||
|
|
799cf27c5d | ||
|
|
c930d8f773 | ||
|
|
a7f921abb9 | ||
|
|
bc6234e032 | ||
|
|
558bfa4e1e | ||
|
|
5d19f23372 | ||
|
|
27f08cdbfa | ||
|
|
993213e2c0 | ||
|
|
49470c05fa | ||
|
|
ee0a060b79 | ||
|
|
500e3157b9 | ||
|
|
eba86b1d23 | ||
|
|
b69a563fc2 | ||
|
|
a900c36395 | ||
|
|
1d9b324d3e | ||
|
|
539e7b8efe | ||
|
|
50a477ee47 | ||
|
|
7000123a8b | ||
|
|
d48a7d2398 | ||
|
|
389a00ce59 | ||
|
|
7a460de3c2 | ||
|
|
8ea1f4a751 | ||
|
|
1c69ccc6cd | ||
|
|
84b5bbd3b6 | ||
|
|
9ccd327298 | ||
|
|
11df36f3cf | ||
|
|
f62dd0e3cc | ||
|
|
ad18b6e15e | ||
|
|
c00b80ca29 | ||
|
|
92ed4ba3f8 | ||
|
|
7de9775dd9 | ||
|
|
5ce9060e5c | ||
|
|
f727d5cb5a | ||
|
|
4735fb1ebb | ||
|
|
c7d05cc13d | ||
|
|
51c152ff4a | ||
|
|
eeed2a840c | ||
|
|
4aaa111925 | ||
|
|
e31248f018 | ||
|
|
8b4cf022f2 | ||
|
|
4e7455268a | ||
|
|
680f8ae814 | ||
|
|
90555a4cea | ||
|
|
56a62db591 | ||
|
|
cf51997680 | ||
|
|
f05cc18d61 | ||
|
|
5384c2e0f5 | ||
|
|
9bfbf80a0e | ||
|
|
f874d7754f | ||
|
|
a669f79480 | ||
|
|
1c3894743a | ||
|
|
75cdf17df4 | ||
|
|
de7dd1e60a | ||
|
|
0ee574a718 | ||
|
|
faac894706 | ||
|
|
dac2fad48e | ||
|
|
77f624b01e | ||
|
|
e24ffebfc8 | ||
|
|
70d07d1609 | ||
|
|
bfb3303d87 | ||
|
|
660705a436 | ||
|
|
74a3f97671 | ||
|
|
b3e35bb494 | ||
|
|
76adac7c72 | ||
|
|
5dc75ebb67 | ||
|
|
d686ce12b6 | ||
|
|
d3c40a423e | ||
|
|
2fb1e6dab8 | ||
|
|
10430b347f | ||
|
|
e0e3f6ac3e | ||
|
|
c694cbffdc | ||
|
|
bdd0e5d771 | ||
|
|
aa98e427f0 | ||
|
|
daa6f4c94c | ||
|
|
4a76663fb2 | ||
|
|
cebda5028a | ||
|
|
3fa377a580 | ||
|
|
a11c1005a8 | ||
|
|
4a6aea9328 | ||
|
|
4ca041e93e | ||
|
|
52a866a405 | ||
|
|
8b6bd0e6ac | ||
|
|
780fc4639a | ||
|
|
3692fc9d83 | ||
|
|
c2a0b1b4c6 | ||
|
|
21bbdb5419 | ||
|
|
aa1c08962c | ||
|
|
8a5d0399dd | ||
|
|
f2cd0b0c4a | ||
|
|
c2b66bbe73 | ||
|
|
48b957f1d5 | ||
|
|
3683984c8d | ||
|
|
a3431512d8 | ||
|
|
d832b787e7 | ||
|
|
6f75b02723 | ||
|
|
b8241710bd | ||
|
|
d638404b6a | ||
|
|
9362ca3ed9 | ||
|
|
d1a03c6d17 | ||
|
|
c6c31702c2 | ||
|
|
bd2d88c96e | ||
|
|
76b1857e4e | ||
|
|
095bd17d10 | ||
|
|
204bfac3fa | ||
|
|
ac49b0ca93 | ||
|
|
c5b04f6fef | ||
|
|
5c58fda46d | ||
|
|
062730c70c | ||
|
|
cade1990ce | ||
|
|
59b6e61816 | ||
|
|
daff7ff158 | ||
|
|
0862860961 | ||
|
|
1cb24045a0 | ||
|
|
622358b172 | ||
|
|
7998884a9d | ||
|
|
51ddecd101 | ||
|
|
7a35ab1d1e | ||
|
|
48564ba52a | ||
|
|
49efffd740 | ||
|
|
d6ac224c8f | ||
|
|
a772b8c3f2 | ||
|
|
b580953dcd | ||
|
|
d86653c763 | ||
|
|
dded4fca76 | ||
|
|
36365ffa6b | ||
|
|
0f9aeeaa27 | ||
|
|
d8ebcd0ef7 | ||
|
|
6e445487b1 | ||
|
|
6605e461c7 | ||
|
|
40ce4e2275 | ||
|
|
8fef9e363e | ||
|
|
4792c2770d | ||
|
|
87bb49da36 | ||
|
|
1c0071d9ce | ||
|
|
efded35c2e | ||
|
|
1d74240b9a | ||
|
|
098184ff7b | ||
|
|
4083533916 | ||
|
|
feb1acd43a | ||
|
|
a9591db734 | ||
|
|
9ebf148cbe | ||
|
|
a473e5e19a | ||
|
|
5d3034c231 | ||
|
|
c3a895af64 | ||
|
|
cea5aecbf2 | ||
|
|
0e61e70670 | ||
|
|
1e333c0939 | ||
|
|
917b6ec03c | ||
|
|
fe67c52ead | ||
|
|
909c7bee3e | ||
|
|
27ca54d138 | ||
|
|
2147c3a646 | ||
|
|
a99120116f | ||
|
|
802efeaff2 | ||
|
|
9ad3af1ef6 | ||
|
|
715727b811 | ||
|
|
c6eaa7b836 | ||
|
|
c2fceea2a5 | ||
|
|
190e11f7ea | ||
|
|
ad7413a5ff | ||
|
|
903b9e627a | ||
|
|
c5c1e96cf8 | ||
|
|
62fbb04c9d | ||
|
|
728dc62d0b | ||
|
|
2dfe1b1c6b | ||
|
|
35d4a1a6af | ||
|
|
eb3fa5aa6b | ||
|
|
438384425a | ||
|
|
0b6f102436 | ||
|
|
c9b7ec72d8 | ||
|
|
256c7f1789 | ||
|
|
4e5a323c62 | ||
|
|
f4a3bbd237 | ||
|
|
fe73f2d579 | ||
|
|
f79fcc7073 | ||
|
|
4c4b3790c7 | ||
|
|
bd60b464bb | ||
|
|
6bce852765 | ||
|
|
3b19a5a59d | ||
|
|
f024583011 | ||
|
|
1111baacb2 | ||
|
|
1b9c913efb | ||
|
|
3524c36e1b | ||
|
|
cf87cea9f8 | ||
|
|
bfa34404b8 | ||
|
|
0aba5f35bf | ||
|
|
663bc0842a | ||
|
|
7d10c96e73 | ||
|
|
6b2720fab0 | ||
|
|
e74ad5132a | ||
|
|
1f6f89c1fd | ||
|
|
4d55e60980 | ||
|
|
ddaaccd5af | ||
|
|
c20b7dac3d | ||
|
|
1f779d5094 | ||
|
|
715401ca8e | ||
|
|
e7cd922d8b | ||
|
|
187feee0c1 | ||
|
|
49e962a7dc | ||
|
|
633ff601e5 | ||
|
|
331cf37054 | ||
|
|
23e4b9002f | ||
|
|
c0de3c8053 | ||
|
|
a82a3b084a | ||
|
|
67c298e66b | ||
|
|
c110ccb9ae | ||
|
|
0143380306 | ||
|
|
af9000d3c8 | ||
|
|
097d798e5e | ||
|
|
1d9f9f221a | ||
|
|
214a367f48 | ||
|
|
2fb46551a2 | ||
|
|
6bcf330ae0 | ||
|
|
2075a8b18c | ||
|
|
1275ac6c42 | ||
|
|
708f20b7af | ||
|
|
a2c0c708e8 | ||
|
|
2f2c65d91e | ||
|
|
cd5fcc7ca7 | ||
|
|
aa29e7be48 | ||
|
|
93febe34b0 | ||
|
|
f086e6d3c1 | ||
|
|
22e51e1c96 | ||
|
|
63a5336f31 | ||
|
|
bfc6c53cc5 | ||
|
|
236017f310 | ||
|
|
0a1d9b4dfd | ||
|
|
b50d090946 | ||
|
|
00b5db52cf | ||
|
|
24cb30e2c5 | ||
|
|
4549145ab5 | ||
|
|
67b0217754 | ||
|
|
ccae9efdf0 | ||
|
|
59d596b222 | ||
|
|
4878eb2c45 | ||
|
|
7755392f57 | ||
|
|
dc2ea20959 | ||
|
|
8eaea2bd17 | ||
|
|
58e559918f | ||
|
|
f38a3fca5b | ||
|
|
1ea145b384 | ||
|
|
0d9567575a | ||
|
|
e82f176289 | ||
|
|
d4b51c040e | ||
|
|
125d0efbd8 | ||
|
|
3215afc504 | ||
|
|
c73ff3ce1b | ||
|
|
f9c159a051 | ||
|
|
2ab1325c90 | ||
|
|
5b0f7ff506 | ||
|
|
9269bc84f2 | ||
|
|
4e8b651e18 | ||
|
|
65b4f79534 | ||
|
|
5dd43dbc45 | ||
|
|
5f73074c7e | ||
|
|
f5d6ba27b2 | ||
|
|
73fa70b41f | ||
|
|
2a1cda42e7 | ||
|
|
1bd7e31466 | ||
|
|
eb49e1fb4a | ||
|
|
9838c2f0ce | ||
|
|
6041df8370 | ||
|
|
2933dce3ef | ||
|
|
dab377d37b | ||
|
|
f35e41baf1 | ||
|
|
c4083a2942 | ||
|
|
36c20bbe53 | ||
|
|
e34634f5af | ||
|
|
cba9e5b669 | ||
|
|
1f3c46a6b0 | ||
|
|
799a5ffa47 | ||
|
|
b000707c10 | ||
|
|
feba4de1d6 | ||
|
|
951fdb27ca | ||
|
|
9697fb3d84 | ||
|
|
2dbed4500a | ||
|
|
fd9d0e433d | ||
|
|
f096f3ef81 | ||
|
|
cc4a063695 | ||
|
|
b64cabc3c9 | ||
|
|
3dd460717c | ||
|
|
bf658a522b | ||
|
|
e9be7e712d | ||
|
|
e40cd2a809 | ||
|
|
dbabeb9692 | ||
|
|
8dd37d76b0 | ||
|
|
fd475aa358 | ||
|
|
f0988c0e32 | ||
|
|
0632f09bff | ||
|
|
ba599aaca0 | ||
|
|
ff05919e89 | ||
|
|
52e63fa101 | ||
|
|
96ceccd12a | ||
|
|
87994fe006 | ||
|
|
fa12c81a03 | ||
|
|
344ce63455 | ||
|
|
ec4daacf9e | ||
|
|
f3e8308718 | ||
|
|
515ac5d941 | ||
|
|
954c7e7e50 | ||
|
|
67ff57f3a3 | ||
|
|
c10c70c1e5 | ||
|
|
04592a98d2 | ||
|
|
c9c4aac6cf | ||
|
|
8b2c7586ce | ||
|
|
32e22dfe84 | ||
|
|
d70b885722 | ||
|
|
ac6c4b13f5 | ||
|
|
ececdad22d | ||
|
|
bf659781b0 | ||
|
|
2c6bb195a4 | ||
|
|
c032cd08b3 | ||
|
|
39e7a7a231 | ||
|
|
6e14cd2c39 | ||
|
|
aab3baaea7 | ||
|
|
b8453c3b4f | ||
|
|
6ce0e2cd5b | ||
|
|
76beaae7f2 | ||
|
|
c1a7f9edbe | ||
|
|
b5f2fe2f0a | ||
|
|
98a90d49cb | ||
|
|
f55e982cb5 | ||
|
|
686c7defeb | ||
|
|
0b1e483c53 | ||
|
|
457d7df129 | ||
|
|
ce776a547c | ||
|
|
ded0567cbf | ||
|
|
c9cac83d09 | ||
|
|
4fbe6b01a8 | ||
|
|
ee9585264e | ||
|
|
c9ffead7bf | ||
|
|
ed69d42005 | ||
|
|
0b47ee306b | ||
|
|
e4e63619d4 | ||
|
|
f32cca292a | ||
|
|
e87ea19ff1 | ||
|
|
0214793740 | ||
|
|
fc9dd5d743 | ||
|
|
9e6d5dd2b9 | ||
|
|
bdad197e2c | ||
|
|
7e139288a6 | ||
|
|
6e7935abaf | ||
|
|
3ba0cc20f1 | ||
|
|
dd28de1796 | ||
|
|
9eecc9e19a | ||
|
|
6530cb6b05 | ||
|
|
41ce613379 | ||
|
|
5e2785caba | ||
|
|
d7cc000976 | ||
|
|
50d8ff95ae | ||
|
|
b2de1459b6 | ||
|
|
f0ffbea0b2 | ||
|
|
199ccca0fe | ||
|
|
1d9b355743 | ||
|
|
f0437fbb07 |
21
.gitignore
vendored
21
.gitignore
vendored
@@ -5,13 +5,16 @@ __pycache__/
|
|||||||
MANIFEST.in
|
MANIFEST.in
|
||||||
MANIFEST
|
MANIFEST
|
||||||
copyparty.egg-info/
|
copyparty.egg-info/
|
||||||
buildenv/
|
|
||||||
build/
|
|
||||||
dist/
|
|
||||||
sfx/
|
|
||||||
py2/
|
|
||||||
.venv/
|
.venv/
|
||||||
|
|
||||||
|
/buildenv/
|
||||||
|
/build/
|
||||||
|
/dist/
|
||||||
|
/py2/
|
||||||
|
/sfx*
|
||||||
|
/unt/
|
||||||
|
/log/
|
||||||
|
|
||||||
# ide
|
# ide
|
||||||
*.sublime-workspace
|
*.sublime-workspace
|
||||||
|
|
||||||
@@ -19,9 +22,15 @@ py2/
|
|||||||
*.bak
|
*.bak
|
||||||
|
|
||||||
# derived
|
# derived
|
||||||
|
copyparty/res/COPYING.txt
|
||||||
copyparty/web/deps/
|
copyparty/web/deps/
|
||||||
srv/
|
srv/
|
||||||
|
scripts/docker/i/
|
||||||
|
contrib/package/arch/pkg/
|
||||||
|
contrib/package/arch/src/
|
||||||
|
|
||||||
# state/logs
|
# state/logs
|
||||||
up.*.txt
|
up.*.txt
|
||||||
.hist/
|
.hist/
|
||||||
|
scripts/docker/*.out
|
||||||
|
scripts/docker/*.err
|
||||||
|
|||||||
1
.vscode/launch.json
vendored
1
.vscode/launch.json
vendored
@@ -8,6 +8,7 @@
|
|||||||
"module": "copyparty",
|
"module": "copyparty",
|
||||||
"console": "integratedTerminal",
|
"console": "integratedTerminal",
|
||||||
"cwd": "${workspaceFolder}",
|
"cwd": "${workspaceFolder}",
|
||||||
|
"justMyCode": false,
|
||||||
"args": [
|
"args": [
|
||||||
//"-nw",
|
//"-nw",
|
||||||
"-ed",
|
"-ed",
|
||||||
|
|||||||
8
.vscode/launch.py
vendored
Normal file → Executable file
8
.vscode/launch.py
vendored
Normal file → Executable file
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# takes arguments from launch.json
|
# takes arguments from launch.json
|
||||||
# is used by no_dbg in tasks.json
|
# is used by no_dbg in tasks.json
|
||||||
# launches 10x faster than mspython debugpy
|
# launches 10x faster than mspython debugpy
|
||||||
@@ -9,15 +11,15 @@ import sys
|
|||||||
|
|
||||||
print(sys.executable)
|
print(sys.executable)
|
||||||
|
|
||||||
|
import json5
|
||||||
import shlex
|
import shlex
|
||||||
import jstyleson
|
|
||||||
import subprocess as sp
|
import subprocess as sp
|
||||||
|
|
||||||
|
|
||||||
with open(".vscode/launch.json", "r", encoding="utf-8") as f:
|
with open(".vscode/launch.json", "r", encoding="utf-8") as f:
|
||||||
tj = f.read()
|
tj = f.read()
|
||||||
|
|
||||||
oj = jstyleson.loads(tj)
|
oj = json5.loads(tj)
|
||||||
argv = oj["configurations"][0]["args"]
|
argv = oj["configurations"][0]["args"]
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@@ -28,6 +30,8 @@ except:
|
|||||||
|
|
||||||
argv = [os.path.expanduser(x) if x.startswith("~") else x for x in argv]
|
argv = [os.path.expanduser(x) if x.startswith("~") else x for x in argv]
|
||||||
|
|
||||||
|
argv += sys.argv[1:]
|
||||||
|
|
||||||
if re.search(" -j ?[0-9]", " ".join(argv)):
|
if re.search(" -j ?[0-9]", " ".join(argv)):
|
||||||
argv = [sys.executable, "-m", "copyparty"] + argv
|
argv = [sys.executable, "-m", "copyparty"] + argv
|
||||||
sp.check_call(argv)
|
sp.check_call(argv)
|
||||||
|
|||||||
28
.vscode/settings.json
vendored
28
.vscode/settings.json
vendored
@@ -23,7 +23,6 @@
|
|||||||
"terminal.ansiBrightWhite": "#ffffff",
|
"terminal.ansiBrightWhite": "#ffffff",
|
||||||
},
|
},
|
||||||
"python.testing.pytestEnabled": false,
|
"python.testing.pytestEnabled": false,
|
||||||
"python.testing.nosetestsEnabled": false,
|
|
||||||
"python.testing.unittestEnabled": true,
|
"python.testing.unittestEnabled": true,
|
||||||
"python.testing.unittestArgs": [
|
"python.testing.unittestArgs": [
|
||||||
"-v",
|
"-v",
|
||||||
@@ -35,17 +34,42 @@
|
|||||||
"python.linting.pylintEnabled": true,
|
"python.linting.pylintEnabled": true,
|
||||||
"python.linting.flake8Enabled": true,
|
"python.linting.flake8Enabled": true,
|
||||||
"python.linting.banditEnabled": true,
|
"python.linting.banditEnabled": true,
|
||||||
|
"python.linting.mypyEnabled": true,
|
||||||
|
"python.linting.mypyArgs": [
|
||||||
|
"--ignore-missing-imports",
|
||||||
|
"--follow-imports=silent",
|
||||||
|
"--show-column-numbers",
|
||||||
|
"--strict"
|
||||||
|
],
|
||||||
"python.linting.flake8Args": [
|
"python.linting.flake8Args": [
|
||||||
"--max-line-length=120",
|
"--max-line-length=120",
|
||||||
"--ignore=E722,F405,E203,W503,W293,E402",
|
"--ignore=E722,F405,E203,W503,W293,E402,E501,E128",
|
||||||
],
|
],
|
||||||
"python.linting.banditArgs": [
|
"python.linting.banditArgs": [
|
||||||
"--ignore=B104"
|
"--ignore=B104"
|
||||||
],
|
],
|
||||||
|
"python.linting.pylintArgs": [
|
||||||
|
"--disable=missing-module-docstring",
|
||||||
|
"--disable=missing-class-docstring",
|
||||||
|
"--disable=missing-function-docstring",
|
||||||
|
"--disable=import-outside-toplevel",
|
||||||
|
"--disable=wrong-import-position",
|
||||||
|
"--disable=raise-missing-from",
|
||||||
|
"--disable=bare-except",
|
||||||
|
"--disable=broad-except",
|
||||||
|
"--disable=invalid-name",
|
||||||
|
"--disable=line-too-long",
|
||||||
|
"--disable=consider-using-f-string"
|
||||||
|
],
|
||||||
|
// python3 -m isort --py=27 --profile=black copyparty/
|
||||||
"python.formatting.provider": "black",
|
"python.formatting.provider": "black",
|
||||||
"editor.formatOnSave": true,
|
"editor.formatOnSave": true,
|
||||||
"[html]": {
|
"[html]": {
|
||||||
"editor.formatOnSave": false,
|
"editor.formatOnSave": false,
|
||||||
|
"editor.autoIndent": "keep",
|
||||||
|
},
|
||||||
|
"[css]": {
|
||||||
|
"editor.formatOnSave": false,
|
||||||
},
|
},
|
||||||
"files.associations": {
|
"files.associations": {
|
||||||
"*.makefile": "makefile"
|
"*.makefile": "makefile"
|
||||||
|
|||||||
9
SECURITY.md
Normal file
9
SECURITY.md
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
# Security Policy
|
||||||
|
|
||||||
|
if you hit something extra juicy pls let me know on either of the following
|
||||||
|
* email -- `copyparty@ocv.ze` except `ze` should be `me`
|
||||||
|
* [mastodon dm](https://layer8.space/@tripflag) -- `@tripflag@layer8.space`
|
||||||
|
* [github private vulnerability report](https://github.com/9001/copyparty/security/advisories/new), wow that form is complicated
|
||||||
|
* [twitter dm](https://twitter.com/tripflag) (if im somehow not banned yet)
|
||||||
|
|
||||||
|
no bug bounties sorry! all i can offer is greetz in the release notes
|
||||||
@@ -1,12 +1,18 @@
|
|||||||
# [`up2k.py`](up2k.py)
|
# [`up2k.py`](up2k.py)
|
||||||
* command-line up2k client [(webm)](https://ocv.me/stuff/u2cli.webm)
|
* command-line up2k client [(webm)](https://ocv.me/stuff/u2cli.webm)
|
||||||
* file uploads, file-search, autoresume of aborted/broken uploads
|
* file uploads, file-search, autoresume of aborted/broken uploads
|
||||||
* faster than browsers
|
* sync local folder to server
|
||||||
|
* generally faster than browsers
|
||||||
* if something breaks just restart it
|
* if something breaks just restart it
|
||||||
|
|
||||||
|
|
||||||
|
# [`partyjournal.py`](partyjournal.py)
|
||||||
|
produces a chronological list of all uploads by collecting info from up2k databases and the filesystem
|
||||||
|
* outputs a standalone html file
|
||||||
|
* optional mapping from IP-addresses to nicknames
|
||||||
|
|
||||||
# [`copyparty-fuse.py`](copyparty-fuse.py)
|
|
||||||
|
# [`partyfuse.py`](partyfuse.py)
|
||||||
* mount a copyparty server as a local filesystem (read-only)
|
* mount a copyparty server as a local filesystem (read-only)
|
||||||
* **supports Windows!** -- expect `194 MiB/s` sequential read
|
* **supports Windows!** -- expect `194 MiB/s` sequential read
|
||||||
* **supports Linux** -- expect `117 MiB/s` sequential read
|
* **supports Linux** -- expect `117 MiB/s` sequential read
|
||||||
@@ -25,19 +31,19 @@ also consider using [../docs/rclone.md](../docs/rclone.md) instead for 5x perfor
|
|||||||
* install [winfsp](https://github.com/billziss-gh/winfsp/releases/latest) and [python 3](https://www.python.org/downloads/)
|
* install [winfsp](https://github.com/billziss-gh/winfsp/releases/latest) and [python 3](https://www.python.org/downloads/)
|
||||||
* [x] add python 3.x to PATH (it asks during install)
|
* [x] add python 3.x to PATH (it asks during install)
|
||||||
* `python -m pip install --user fusepy`
|
* `python -m pip install --user fusepy`
|
||||||
* `python ./copyparty-fuse.py n: http://192.168.1.69:3923/`
|
* `python ./partyfuse.py n: http://192.168.1.69:3923/`
|
||||||
|
|
||||||
10% faster in [msys2](https://www.msys2.org/), 700% faster if debug prints are enabled:
|
10% faster in [msys2](https://www.msys2.org/), 700% faster if debug prints are enabled:
|
||||||
* `pacman -S mingw64/mingw-w64-x86_64-python{,-pip}`
|
* `pacman -S mingw64/mingw-w64-x86_64-python{,-pip}`
|
||||||
* `/mingw64/bin/python3 -m pip install --user fusepy`
|
* `/mingw64/bin/python3 -m pip install --user fusepy`
|
||||||
* `/mingw64/bin/python3 ./copyparty-fuse.py [...]`
|
* `/mingw64/bin/python3 ./partyfuse.py [...]`
|
||||||
|
|
||||||
you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releases/latest), let me know if you [figure out how](https://github.com/dokan-dev/dokany/wiki/FUSE)
|
you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releases/latest), let me know if you [figure out how](https://github.com/dokan-dev/dokany/wiki/FUSE)
|
||||||
(winfsp's sshfs leaks, doesn't look like winfsp itself does, should be fine)
|
(winfsp's sshfs leaks, doesn't look like winfsp itself does, should be fine)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# [`copyparty-fuse🅱️.py`](copyparty-fuseb.py)
|
# [`partyfuse2.py`](partyfuse2.py)
|
||||||
* mount a copyparty server as a local filesystem (read-only)
|
* mount a copyparty server as a local filesystem (read-only)
|
||||||
* does the same thing except more correct, `samba` approves
|
* does the same thing except more correct, `samba` approves
|
||||||
* **supports Linux** -- expect `18 MiB/s` (wait what)
|
* **supports Linux** -- expect `18 MiB/s` (wait what)
|
||||||
@@ -45,7 +51,7 @@ you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releas
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
# [`copyparty-fuse-streaming.py`](copyparty-fuse-streaming.py)
|
# [`partyfuse-streaming.py`](partyfuse-streaming.py)
|
||||||
* pretend this doesn't exist
|
* pretend this doesn't exist
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
111
bin/dbtool.py
111
bin/dbtool.py
@@ -8,7 +8,10 @@ import sqlite3
|
|||||||
import argparse
|
import argparse
|
||||||
|
|
||||||
DB_VER1 = 3
|
DB_VER1 = 3
|
||||||
DB_VER2 = 4
|
DB_VER2 = 5
|
||||||
|
|
||||||
|
BY_PATH = None
|
||||||
|
NC = None
|
||||||
|
|
||||||
|
|
||||||
def die(msg):
|
def die(msg):
|
||||||
@@ -57,8 +60,13 @@ def compare(n1, d1, n2, d2, verbose):
|
|||||||
if rd.split("/", 1)[0] == ".hist":
|
if rd.split("/", 1)[0] == ".hist":
|
||||||
continue
|
continue
|
||||||
|
|
||||||
q = "select w from up where rd = ? and fn = ?"
|
if BY_PATH:
|
||||||
hit = d2.execute(q, (rd, fn)).fetchone()
|
q = "select w from up where rd = ? and fn = ?"
|
||||||
|
hit = d2.execute(q, (rd, fn)).fetchone()
|
||||||
|
else:
|
||||||
|
q = "select w from up where substr(w,1,16) = ? and +w = ?"
|
||||||
|
hit = d2.execute(q, (w1[:16], w1)).fetchone()
|
||||||
|
|
||||||
if not hit:
|
if not hit:
|
||||||
miss += 1
|
miss += 1
|
||||||
if verbose:
|
if verbose:
|
||||||
@@ -70,27 +78,32 @@ def compare(n1, d1, n2, d2, verbose):
|
|||||||
n = 0
|
n = 0
|
||||||
miss = {}
|
miss = {}
|
||||||
nmiss = 0
|
nmiss = 0
|
||||||
for w1, k, v in d1.execute("select * from mt"):
|
for w1s, k, v in d1.execute("select * from mt"):
|
||||||
|
|
||||||
n += 1
|
n += 1
|
||||||
if n % 100_000 == 0:
|
if n % 100_000 == 0:
|
||||||
m = f"\033[36mchecked {n:,} of {nt:,} tags in {n1} against {n2}, so far {nmiss} missing tags\033[0m"
|
m = f"\033[36mchecked {n:,} of {nt:,} tags in {n1} against {n2}, so far {nmiss} missing tags\033[0m"
|
||||||
print(m)
|
print(m)
|
||||||
|
|
||||||
q = "select rd, fn from up where substr(w,1,16) = ?"
|
q = "select w, rd, fn from up where substr(w,1,16) = ?"
|
||||||
rd, fn = d1.execute(q, (w1,)).fetchone()
|
w1, rd, fn = d1.execute(q, (w1s,)).fetchone()
|
||||||
if rd.split("/", 1)[0] == ".hist":
|
if rd.split("/", 1)[0] == ".hist":
|
||||||
continue
|
continue
|
||||||
|
|
||||||
q = "select substr(w,1,16) from up where rd = ? and fn = ?"
|
if BY_PATH:
|
||||||
w2 = d2.execute(q, (rd, fn)).fetchone()
|
q = "select w from up where rd = ? and fn = ?"
|
||||||
|
w2 = d2.execute(q, (rd, fn)).fetchone()
|
||||||
|
else:
|
||||||
|
q = "select w from up where substr(w,1,16) = ? and +w = ?"
|
||||||
|
w2 = d2.execute(q, (w1s, w1)).fetchone()
|
||||||
|
|
||||||
if w2:
|
if w2:
|
||||||
w2 = w2[0]
|
w2 = w2[0]
|
||||||
|
|
||||||
v2 = None
|
v2 = None
|
||||||
if w2:
|
if w2:
|
||||||
v2 = d2.execute(
|
v2 = d2.execute(
|
||||||
"select v from mt where w = ? and +k = ?", (w2, k)
|
"select v from mt where w = ? and +k = ?", (w2[:16], k)
|
||||||
).fetchone()
|
).fetchone()
|
||||||
if v2:
|
if v2:
|
||||||
v2 = v2[0]
|
v2 = v2[0]
|
||||||
@@ -124,7 +137,7 @@ def compare(n1, d1, n2, d2, verbose):
|
|||||||
|
|
||||||
for k, v in sorted(miss.items()):
|
for k, v in sorted(miss.items()):
|
||||||
if v:
|
if v:
|
||||||
print(f"{n1} has {v:6} more {k:<6} tags than {n2}")
|
print(f"{n1} has {v:7} more {k:<7} tags than {n2}")
|
||||||
|
|
||||||
print(f"in total, {nmiss} missing tags in {n2}\n")
|
print(f"in total, {nmiss} missing tags in {n2}\n")
|
||||||
|
|
||||||
@@ -132,47 +145,75 @@ def compare(n1, d1, n2, d2, verbose):
|
|||||||
def copy_mtp(d1, d2, tag, rm):
|
def copy_mtp(d1, d2, tag, rm):
|
||||||
nt = next(d1.execute("select count(w) from mt where k = ?", (tag,)))[0]
|
nt = next(d1.execute("select count(w) from mt where k = ?", (tag,)))[0]
|
||||||
n = 0
|
n = 0
|
||||||
ndone = 0
|
ncopy = 0
|
||||||
for w1, k, v in d1.execute("select * from mt where k = ?", (tag,)):
|
nskip = 0
|
||||||
|
for w1s, k, v in d1.execute("select * from mt where k = ?", (tag,)):
|
||||||
n += 1
|
n += 1
|
||||||
if n % 25_000 == 0:
|
if n % 25_000 == 0:
|
||||||
m = f"\033[36m{n:,} of {nt:,} tags checked, so far {ndone} copied\033[0m"
|
m = f"\033[36m{n:,} of {nt:,} tags checked, so far {ncopy} copied, {nskip} skipped\033[0m"
|
||||||
print(m)
|
print(m)
|
||||||
|
|
||||||
q = "select rd, fn from up where substr(w,1,16) = ?"
|
q = "select w, rd, fn from up where substr(w,1,16) = ?"
|
||||||
rd, fn = d1.execute(q, (w1,)).fetchone()
|
w1, rd, fn = d1.execute(q, (w1s,)).fetchone()
|
||||||
if rd.split("/", 1)[0] == ".hist":
|
if rd.split("/", 1)[0] == ".hist":
|
||||||
continue
|
continue
|
||||||
|
|
||||||
q = "select substr(w,1,16) from up where rd = ? and fn = ?"
|
if BY_PATH:
|
||||||
w2 = d2.execute(q, (rd, fn)).fetchone()
|
q = "select w from up where rd = ? and fn = ?"
|
||||||
|
w2 = d2.execute(q, (rd, fn)).fetchone()
|
||||||
|
else:
|
||||||
|
q = "select w from up where substr(w,1,16) = ? and +w = ?"
|
||||||
|
w2 = d2.execute(q, (w1s, w1)).fetchone()
|
||||||
|
|
||||||
if not w2:
|
if not w2:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
w2 = w2[0]
|
w2s = w2[0][:16]
|
||||||
hit = d2.execute("select v from mt where w = ? and +k = ?", (w2, k)).fetchone()
|
hit = d2.execute("select v from mt where w = ? and +k = ?", (w2s, k)).fetchone()
|
||||||
if hit:
|
if hit:
|
||||||
hit = hit[0]
|
hit = hit[0]
|
||||||
|
|
||||||
if hit != v:
|
if hit != v:
|
||||||
ndone += 1
|
if NC and hit is not None:
|
||||||
if hit is not None:
|
nskip += 1
|
||||||
d2.execute("delete from mt where w = ? and +k = ?", (w2, k))
|
continue
|
||||||
|
|
||||||
d2.execute("insert into mt values (?,?,?)", (w2, k, v))
|
ncopy += 1
|
||||||
|
if hit is not None:
|
||||||
|
d2.execute("delete from mt where w = ? and +k = ?", (w2s, k))
|
||||||
|
|
||||||
|
d2.execute("insert into mt values (?,?,?)", (w2s, k, v))
|
||||||
if rm:
|
if rm:
|
||||||
d2.execute("delete from mt where w = ? and +k = 't:mtp'", (w2,))
|
d2.execute("delete from mt where w = ? and +k = 't:mtp'", (w2s,))
|
||||||
|
|
||||||
d2.commit()
|
d2.commit()
|
||||||
print(f"copied {ndone} {tag} tags over")
|
print(f"copied {ncopy} {tag} tags over, skipped {nskip}")
|
||||||
|
|
||||||
|
|
||||||
|
def examples():
|
||||||
|
print(
|
||||||
|
"""
|
||||||
|
# clearing the journal
|
||||||
|
./dbtool.py up2k.db
|
||||||
|
|
||||||
|
# copy tags ".bpm" and "key" from old.db to up2k.db, and remove the mtp flag from matching files (so copyparty won't run any mtps on it)
|
||||||
|
./dbtool.py -ls up2k.db
|
||||||
|
./dbtool.py -src old.db up2k.db -cmp
|
||||||
|
./dbtool.py -src old.v3 up2k.db -rm-mtp-flag -copy key
|
||||||
|
./dbtool.py -src old.v3 up2k.db -rm-mtp-flag -copy .bpm -vac
|
||||||
|
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
|
global NC, BY_PATH
|
||||||
os.system("")
|
os.system("")
|
||||||
print()
|
print()
|
||||||
|
|
||||||
ap = argparse.ArgumentParser()
|
ap = argparse.ArgumentParser()
|
||||||
ap.add_argument("db", help="database to work on")
|
ap.add_argument("db", help="database to work on")
|
||||||
|
ap.add_argument("-h2", action="store_true", help="show examples")
|
||||||
ap.add_argument("-src", metavar="DB", type=str, help="database to copy from")
|
ap.add_argument("-src", metavar="DB", type=str, help="database to copy from")
|
||||||
|
|
||||||
ap2 = ap.add_argument_group("informational / read-only stuff")
|
ap2 = ap.add_argument_group("informational / read-only stuff")
|
||||||
@@ -185,11 +226,29 @@ def main():
|
|||||||
ap2.add_argument(
|
ap2.add_argument(
|
||||||
"-rm-mtp-flag",
|
"-rm-mtp-flag",
|
||||||
action="store_true",
|
action="store_true",
|
||||||
help="when an mtp tag is copied over, also mark that as done, so copyparty won't run mtp on it",
|
help="when an mtp tag is copied over, also mark that file as done, so copyparty won't run any mtps on those files",
|
||||||
)
|
)
|
||||||
ap2.add_argument("-vac", action="store_true", help="optimize DB")
|
ap2.add_argument("-vac", action="store_true", help="optimize DB")
|
||||||
|
|
||||||
|
ap2 = ap.add_argument_group("behavior modifiers")
|
||||||
|
ap2.add_argument(
|
||||||
|
"-nc",
|
||||||
|
action="store_true",
|
||||||
|
help="no-clobber; don't replace/overwrite existing tags",
|
||||||
|
)
|
||||||
|
ap2.add_argument(
|
||||||
|
"-by-path",
|
||||||
|
action="store_true",
|
||||||
|
help="match files based on location rather than warks (content-hash), use this if the databases have different wark salts",
|
||||||
|
)
|
||||||
|
|
||||||
ar = ap.parse_args()
|
ar = ap.parse_args()
|
||||||
|
if ar.h2:
|
||||||
|
examples()
|
||||||
|
return
|
||||||
|
|
||||||
|
NC = ar.nc
|
||||||
|
BY_PATH = ar.by_path
|
||||||
|
|
||||||
for v in [ar.db, ar.src]:
|
for v in [ar.db, ar.src]:
|
||||||
if v and not os.path.exists(v):
|
if v and not os.path.exists(v):
|
||||||
|
|||||||
28
bin/hooks/README.md
Normal file
28
bin/hooks/README.md
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
standalone programs which are executed by copyparty when an event happens (upload, file rename, delete, ...)
|
||||||
|
|
||||||
|
these programs either take zero arguments, or a filepath (the affected file), or a json message with filepath + additional info
|
||||||
|
|
||||||
|
run copyparty with `--help-hooks` for usage details / hook type explanations (xbu/xau/xiu/xbr/xar/xbd/xad)
|
||||||
|
|
||||||
|
> **note:** in addition to event hooks (the stuff described here), copyparty has another api to run your programs/scripts while providing way more information such as audio tags / video codecs / etc and optionally daisychaining data between scripts in a processing pipeline; if that's what you want then see [mtp plugins](../mtag/) instead
|
||||||
|
|
||||||
|
|
||||||
|
# after upload
|
||||||
|
* [notify.py](notify.py) shows a desktop notification ([example](https://user-images.githubusercontent.com/241032/215335767-9c91ed24-d36e-4b6b-9766-fb95d12d163f.png))
|
||||||
|
* [notify2.py](notify2.py) uses the json API to show more context
|
||||||
|
* [discord-announce.py](discord-announce.py) announces new uploads on discord using webhooks ([example](https://user-images.githubusercontent.com/241032/215304439-1c1cb3c8-ec6f-4c17-9f27-81f969b1811a.png))
|
||||||
|
* [reject-mimetype.py](reject-mimetype.py) rejects uploads unless the mimetype is acceptable
|
||||||
|
|
||||||
|
|
||||||
|
# upload batches
|
||||||
|
these are `--xiu` hooks; unlike `xbu` and `xau` (which get executed on every single file), `xiu` hooks are given a list of recent uploads on STDIN after the server has gone idle for N seconds, reducing server load + providing more context
|
||||||
|
* [xiu.py](xiu.py) is a "minimal" example showing a list of filenames + total filesize
|
||||||
|
* [xiu-sha.py](xiu-sha.py) produces a sha512 checksum list in the volume root
|
||||||
|
|
||||||
|
|
||||||
|
# before upload
|
||||||
|
* [reject-extension.py](reject-extension.py) rejects uploads if they match a list of file extensions
|
||||||
|
|
||||||
|
|
||||||
|
# on message
|
||||||
|
* [wget.py](wget.py) lets you download files by POSTing URLs to copyparty
|
||||||
61
bin/hooks/discord-announce.py
Executable file
61
bin/hooks/discord-announce.py
Executable file
@@ -0,0 +1,61 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import json
|
||||||
|
import requests
|
||||||
|
from copyparty.util import humansize, quotep
|
||||||
|
|
||||||
|
|
||||||
|
_ = r"""
|
||||||
|
announces a new upload on discord
|
||||||
|
|
||||||
|
example usage as global config:
|
||||||
|
--xau f,t5,j,bin/hooks/discord-announce.py
|
||||||
|
|
||||||
|
example usage as a volflag (per-volume config):
|
||||||
|
-v srv/inc:inc:c,xau=f,t5,j,bin/hooks/discord-announce.py
|
||||||
|
|
||||||
|
parameters explained,
|
||||||
|
f = fork; don't wait for it to finish
|
||||||
|
t5 = timeout if it's still running after 5 sec
|
||||||
|
j = provide upload information as json; not just the filename
|
||||||
|
|
||||||
|
replace "xau" with "xbu" to announce Before upload starts instead of After completion
|
||||||
|
|
||||||
|
# how to discord:
|
||||||
|
first create the webhook url; https://support.discord.com/hc/en-us/articles/228383668-Intro-to-Webhooks
|
||||||
|
then use this to design your message: https://discohook.org/
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
WEBHOOK = "https://discord.com/api/webhooks/1234/base64"
|
||||||
|
|
||||||
|
# read info from copyparty
|
||||||
|
inf = json.loads(sys.argv[1])
|
||||||
|
vpath = inf["vp"]
|
||||||
|
filename = vpath.split("/")[-1]
|
||||||
|
url = f"https://{inf['host']}/{quotep(vpath)}"
|
||||||
|
|
||||||
|
# compose the message to discord
|
||||||
|
j = {
|
||||||
|
"title": filename,
|
||||||
|
"url": url,
|
||||||
|
"description": url.rsplit("/", 1)[0],
|
||||||
|
"color": 0x449900,
|
||||||
|
"fields": [
|
||||||
|
{"name": "Size", "value": humansize(inf["sz"])},
|
||||||
|
{"name": "User", "value": inf["user"]},
|
||||||
|
{"name": "IP", "value": inf["ip"]},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
|
||||||
|
for v in j["fields"]:
|
||||||
|
v["inline"] = True
|
||||||
|
|
||||||
|
r = requests.post(WEBHOOK, json={"embeds": [j]})
|
||||||
|
print(f"discord: {r}\n", end="")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
62
bin/hooks/notify.py
Executable file
62
bin/hooks/notify.py
Executable file
@@ -0,0 +1,62 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import subprocess as sp
|
||||||
|
from plyer import notification
|
||||||
|
|
||||||
|
|
||||||
|
_ = r"""
|
||||||
|
show os notification on upload; works on windows, linux, macos, android
|
||||||
|
|
||||||
|
depdencies:
|
||||||
|
windows: python3 -m pip install --user -U plyer
|
||||||
|
linux: python3 -m pip install --user -U plyer
|
||||||
|
macos: python3 -m pip install --user -U plyer pyobjus
|
||||||
|
android: just termux and termux-api
|
||||||
|
|
||||||
|
example usages; either as global config (all volumes) or as volflag:
|
||||||
|
--xau f,bin/hooks/notify.py
|
||||||
|
-v srv/inc:inc:c,xau=f,bin/hooks/notify.py
|
||||||
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
||||||
|
parameters explained,
|
||||||
|
xau = execute after upload
|
||||||
|
f = fork so it doesn't block uploads
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
try:
|
||||||
|
from copyparty.util import humansize
|
||||||
|
except:
|
||||||
|
|
||||||
|
def humansize(n):
|
||||||
|
return n
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
fp = sys.argv[1]
|
||||||
|
dp, fn = os.path.split(fp)
|
||||||
|
try:
|
||||||
|
sz = humansize(os.path.getsize(fp))
|
||||||
|
except:
|
||||||
|
sz = "?"
|
||||||
|
|
||||||
|
msg = "{} ({})\n📁 {}".format(fn, sz, dp)
|
||||||
|
title = "File received"
|
||||||
|
|
||||||
|
if "com.termux" in sys.executable:
|
||||||
|
sp.run(["termux-notification", "-t", title, "-c", msg])
|
||||||
|
return
|
||||||
|
|
||||||
|
icon = "emblem-documents-symbolic" if sys.platform == "linux" else ""
|
||||||
|
notification.notify(
|
||||||
|
title=title,
|
||||||
|
message=msg,
|
||||||
|
app_icon=icon,
|
||||||
|
timeout=10,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
68
bin/hooks/notify2.py
Executable file
68
bin/hooks/notify2.py
Executable file
@@ -0,0 +1,68 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import subprocess as sp
|
||||||
|
from datetime import datetime
|
||||||
|
from plyer import notification
|
||||||
|
|
||||||
|
|
||||||
|
_ = r"""
|
||||||
|
same as notify.py but with additional info (uploader, ...)
|
||||||
|
and also supports --xm (notify on 📟 message)
|
||||||
|
|
||||||
|
example usages; either as global config (all volumes) or as volflag:
|
||||||
|
--xm f,j,bin/hooks/notify2.py
|
||||||
|
--xau f,j,bin/hooks/notify2.py
|
||||||
|
-v srv/inc:inc:c,xm=f,j,bin/hooks/notify2.py
|
||||||
|
-v srv/inc:inc:c,xau=f,j,bin/hooks/notify2.py
|
||||||
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
||||||
|
parameters explained,
|
||||||
|
xau = execute after upload
|
||||||
|
f = fork so it doesn't block uploads
|
||||||
|
j = provide json instead of filepath list
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
try:
|
||||||
|
from copyparty.util import humansize
|
||||||
|
except:
|
||||||
|
|
||||||
|
def humansize(n):
|
||||||
|
return n
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
inf = json.loads(sys.argv[1])
|
||||||
|
fp = inf["ap"]
|
||||||
|
sz = humansize(inf["sz"])
|
||||||
|
dp, fn = os.path.split(fp)
|
||||||
|
mt = datetime.utcfromtimestamp(inf["mt"]).strftime("%Y-%m-%d %H:%M:%S")
|
||||||
|
|
||||||
|
msg = f"{fn} ({sz})\n📁 {dp}"
|
||||||
|
title = "File received"
|
||||||
|
icon = "emblem-documents-symbolic" if sys.platform == "linux" else ""
|
||||||
|
|
||||||
|
if inf.get("txt"):
|
||||||
|
msg = inf["txt"]
|
||||||
|
title = "Message received"
|
||||||
|
icon = "mail-unread-symbolic" if sys.platform == "linux" else ""
|
||||||
|
|
||||||
|
msg += f"\n👤 {inf['user']} ({inf['ip']})\n🕒 {mt}"
|
||||||
|
|
||||||
|
if "com.termux" in sys.executable:
|
||||||
|
sp.run(["termux-notification", "-t", title, "-c", msg])
|
||||||
|
return
|
||||||
|
|
||||||
|
notification.notify(
|
||||||
|
title=title,
|
||||||
|
message=msg,
|
||||||
|
app_icon=icon,
|
||||||
|
timeout=10,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
30
bin/hooks/reject-extension.py
Executable file
30
bin/hooks/reject-extension.py
Executable file
@@ -0,0 +1,30 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
|
|
||||||
|
_ = r"""
|
||||||
|
reject file uploads by file extension
|
||||||
|
|
||||||
|
example usage as global config:
|
||||||
|
--xbu c,bin/hooks/reject-extension.py
|
||||||
|
|
||||||
|
example usage as a volflag (per-volume config):
|
||||||
|
-v srv/inc:inc:c,xbu=c,bin/hooks/reject-extension.py
|
||||||
|
|
||||||
|
parameters explained,
|
||||||
|
xbu = execute before upload
|
||||||
|
c = check result, reject upload if error
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
bad = "exe scr com pif bat ps1 jar msi"
|
||||||
|
|
||||||
|
ext = sys.argv[1].split(".")[-1]
|
||||||
|
|
||||||
|
sys.exit(1 if ext in bad.split() else 0)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
39
bin/hooks/reject-mimetype.py
Executable file
39
bin/hooks/reject-mimetype.py
Executable file
@@ -0,0 +1,39 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import magic
|
||||||
|
|
||||||
|
|
||||||
|
_ = r"""
|
||||||
|
reject file uploads by mimetype
|
||||||
|
|
||||||
|
dependencies (linux, macos):
|
||||||
|
python3 -m pip install --user -U python-magic
|
||||||
|
|
||||||
|
dependencies (windows):
|
||||||
|
python3 -m pip install --user -U python-magic-bin
|
||||||
|
|
||||||
|
example usage as global config:
|
||||||
|
--xau c,bin/hooks/reject-mimetype.py
|
||||||
|
|
||||||
|
example usage as a volflag (per-volume config):
|
||||||
|
-v srv/inc:inc:c,xau=c,bin/hooks/reject-mimetype.py
|
||||||
|
|
||||||
|
parameters explained,
|
||||||
|
xau = execute after upload
|
||||||
|
c = check result, reject upload if error
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
ok = ["image/jpeg", "image/png"]
|
||||||
|
|
||||||
|
mt = magic.from_file(sys.argv[1], mime=True)
|
||||||
|
|
||||||
|
print(mt)
|
||||||
|
|
||||||
|
sys.exit(1 if mt not in ok else 0)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
54
bin/hooks/wget.py
Executable file
54
bin/hooks/wget.py
Executable file
@@ -0,0 +1,54 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import json
|
||||||
|
import subprocess as sp
|
||||||
|
|
||||||
|
|
||||||
|
_ = r"""
|
||||||
|
use copyparty as a file downloader by POSTing URLs as
|
||||||
|
application/x-www-form-urlencoded (for example using the
|
||||||
|
message/pager function on the website)
|
||||||
|
|
||||||
|
example usage as global config:
|
||||||
|
--xm f,j,t3600,bin/hooks/wget.py
|
||||||
|
|
||||||
|
example usage as a volflag (per-volume config):
|
||||||
|
-v srv/inc:inc:c,xm=f,j,t3600,bin/hooks/wget.py
|
||||||
|
|
||||||
|
parameters explained,
|
||||||
|
f = fork so it doesn't block uploads
|
||||||
|
j = provide message information as json; not just the text
|
||||||
|
c3 = mute all output
|
||||||
|
t3600 = timeout and kill download after 1 hour
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
inf = json.loads(sys.argv[1])
|
||||||
|
url = inf["txt"]
|
||||||
|
if "://" not in url:
|
||||||
|
url = "https://" + url
|
||||||
|
|
||||||
|
os.chdir(inf["ap"])
|
||||||
|
|
||||||
|
name = url.split("?")[0].split("/")[-1]
|
||||||
|
tfn = "-- DOWNLOADING " + name
|
||||||
|
print(f"{tfn}\n", end="")
|
||||||
|
open(tfn, "wb").close()
|
||||||
|
|
||||||
|
cmd = ["wget", "--trust-server-names", "-nv", "--", url]
|
||||||
|
|
||||||
|
try:
|
||||||
|
sp.check_call(cmd)
|
||||||
|
except:
|
||||||
|
t = "-- FAILED TO DONWLOAD " + name
|
||||||
|
print(f"{t}\n", end="")
|
||||||
|
open(t, "wb").close()
|
||||||
|
|
||||||
|
os.unlink(tfn)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
103
bin/hooks/xiu-sha.py
Executable file
103
bin/hooks/xiu-sha.py
Executable file
@@ -0,0 +1,103 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import hashlib
|
||||||
|
import json
|
||||||
|
import sys
|
||||||
|
from datetime import datetime
|
||||||
|
|
||||||
|
|
||||||
|
_ = r"""
|
||||||
|
this hook will produce a single sha512 file which
|
||||||
|
covers all recent uploads (plus metadata comments)
|
||||||
|
|
||||||
|
use this with --xiu, which makes copyparty buffer
|
||||||
|
uploads until server is idle, providing file infos
|
||||||
|
on stdin (filepaths or json)
|
||||||
|
|
||||||
|
example usage as global config:
|
||||||
|
--xiu i5,j,bin/hooks/xiu-sha.py
|
||||||
|
|
||||||
|
example usage as a volflag (per-volume config):
|
||||||
|
-v srv/inc:inc:c,xiu=i5,j,bin/hooks/xiu-sha.py
|
||||||
|
|
||||||
|
parameters explained,
|
||||||
|
xiu = execute after uploads...
|
||||||
|
i5 = ...after volume has been idle for 5sec
|
||||||
|
j = provide json instead of filepath list
|
||||||
|
|
||||||
|
note the "f" (fork) flag is not set, so this xiu
|
||||||
|
will block other xiu hooks while it's running
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
try:
|
||||||
|
from copyparty.util import fsenc
|
||||||
|
except:
|
||||||
|
|
||||||
|
def fsenc(p):
|
||||||
|
return p
|
||||||
|
|
||||||
|
|
||||||
|
def humantime(ts):
|
||||||
|
return datetime.utcfromtimestamp(ts).strftime("%Y-%m-%d %H:%M:%S")
|
||||||
|
|
||||||
|
|
||||||
|
def find_files_root(inf):
|
||||||
|
di = 9000
|
||||||
|
for f1, f2 in zip(inf, inf[1:]):
|
||||||
|
p1 = f1["ap"].replace("\\", "/").rsplit("/", 1)[0]
|
||||||
|
p2 = f2["ap"].replace("\\", "/").rsplit("/", 1)[0]
|
||||||
|
di = min(len(p1), len(p2), di)
|
||||||
|
di = next((i for i in range(di) if p1[i] != p2[i]), di)
|
||||||
|
|
||||||
|
return di + 1
|
||||||
|
|
||||||
|
|
||||||
|
def find_vol_root(inf):
|
||||||
|
return len(inf[0]["ap"][: -len(inf[0]["vp"])])
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
zb = sys.stdin.buffer.read()
|
||||||
|
zs = zb.decode("utf-8", "replace")
|
||||||
|
inf = json.loads(zs)
|
||||||
|
|
||||||
|
# root directory (where to put the sha512 file);
|
||||||
|
# di = find_files_root(inf) # next to the file closest to volume root
|
||||||
|
di = find_vol_root(inf) # top of the entire volume
|
||||||
|
|
||||||
|
ret = []
|
||||||
|
total_sz = 0
|
||||||
|
for md in inf:
|
||||||
|
ap = md["ap"]
|
||||||
|
rp = ap[di:]
|
||||||
|
total_sz += md["sz"]
|
||||||
|
fsize = "{:,}".format(md["sz"])
|
||||||
|
mtime = humantime(md["mt"])
|
||||||
|
up_ts = humantime(md["at"])
|
||||||
|
|
||||||
|
h = hashlib.sha512()
|
||||||
|
with open(fsenc(md["ap"]), "rb", 512 * 1024) as f:
|
||||||
|
while True:
|
||||||
|
buf = f.read(512 * 1024)
|
||||||
|
if not buf:
|
||||||
|
break
|
||||||
|
|
||||||
|
h.update(buf)
|
||||||
|
|
||||||
|
cksum = h.hexdigest()
|
||||||
|
meta = " | ".join([md["wark"], up_ts, mtime, fsize, md["ip"]])
|
||||||
|
ret.append("# {}\n{} *{}".format(meta, cksum, rp))
|
||||||
|
|
||||||
|
ret.append("# {} files, {} bytes total".format(len(inf), total_sz))
|
||||||
|
ret.append("")
|
||||||
|
ftime = datetime.utcnow().strftime("%Y-%m%d-%H%M%S.%f")
|
||||||
|
fp = "{}xfer-{}.sha512".format(inf[0]["ap"][:di], ftime)
|
||||||
|
with open(fsenc(fp), "wb") as f:
|
||||||
|
f.write("\n".join(ret).encode("utf-8", "replace"))
|
||||||
|
|
||||||
|
print("wrote checksums to {}".format(fp))
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
45
bin/hooks/xiu.py
Executable file
45
bin/hooks/xiu.py
Executable file
@@ -0,0 +1,45 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import json
|
||||||
|
import sys
|
||||||
|
|
||||||
|
|
||||||
|
_ = r"""
|
||||||
|
this hook prints absolute filepaths + total size
|
||||||
|
|
||||||
|
use this with --xiu, which makes copyparty buffer
|
||||||
|
uploads until server is idle, providing file infos
|
||||||
|
on stdin (filepaths or json)
|
||||||
|
|
||||||
|
example usage as global config:
|
||||||
|
--xiu i1,j,bin/hooks/xiu.py
|
||||||
|
|
||||||
|
example usage as a volflag (per-volume config):
|
||||||
|
-v srv/inc:inc:c,xiu=i1,j,bin/hooks/xiu.py
|
||||||
|
|
||||||
|
parameters explained,
|
||||||
|
xiu = execute after uploads...
|
||||||
|
i1 = ...after volume has been idle for 1sec
|
||||||
|
j = provide json instead of filepath list
|
||||||
|
|
||||||
|
note the "f" (fork) flag is not set, so this xiu
|
||||||
|
will block other xiu hooks while it's running
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
zb = sys.stdin.buffer.read()
|
||||||
|
zs = zb.decode("utf-8", "replace")
|
||||||
|
inf = json.loads(zs)
|
||||||
|
|
||||||
|
total_sz = 0
|
||||||
|
for upload in inf:
|
||||||
|
sz = upload["sz"]
|
||||||
|
total_sz += sz
|
||||||
|
print("{:9} {}".format(sz, upload["ap"]))
|
||||||
|
|
||||||
|
print("{} files, {} bytes total".format(len(inf), total_sz))
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
@@ -1,5 +1,9 @@
|
|||||||
standalone programs which take an audio file as argument
|
standalone programs which take an audio file as argument
|
||||||
|
|
||||||
|
you may want to forget about all this fancy complicated stuff and just use [event hooks](../hooks/) instead (which doesn't need `-e2ts` or ffmpeg)
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
**NOTE:** these all require `-e2ts` to be functional, meaning you need to do at least one of these: `apt install ffmpeg` or `pip3 install mutagen`
|
**NOTE:** these all require `-e2ts` to be functional, meaning you need to do at least one of these: `apt install ffmpeg` or `pip3 install mutagen`
|
||||||
|
|
||||||
some of these rely on libraries which are not MIT-compatible
|
some of these rely on libraries which are not MIT-compatible
|
||||||
@@ -17,6 +21,7 @@ these do not have any problematic dependencies at all:
|
|||||||
* [cksum.py](./cksum.py) computes various checksums
|
* [cksum.py](./cksum.py) computes various checksums
|
||||||
* [exe.py](./exe.py) grabs metadata from .exe and .dll files (example for retrieving multiple tags with one parser)
|
* [exe.py](./exe.py) grabs metadata from .exe and .dll files (example for retrieving multiple tags with one parser)
|
||||||
* [wget.py](./wget.py) lets you download files by POSTing URLs to copyparty
|
* [wget.py](./wget.py) lets you download files by POSTing URLs to copyparty
|
||||||
|
* also available as an [event hook](../hooks/wget.py)
|
||||||
|
|
||||||
|
|
||||||
# dependencies
|
# dependencies
|
||||||
@@ -42,7 +47,7 @@ run [`install-deps.sh`](install-deps.sh) to build/install most dependencies requ
|
|||||||
* `mtp` modules will not run if a file has existing tags in the db, so clear out the tags with `-e2tsr` the first time you launch with new `mtp` options
|
* `mtp` modules will not run if a file has existing tags in the db, so clear out the tags with `-e2tsr` the first time you launch with new `mtp` options
|
||||||
|
|
||||||
|
|
||||||
## usage with volume-flags
|
## usage with volflags
|
||||||
|
|
||||||
instead of affecting all volumes, you can set the options for just one volume like so:
|
instead of affecting all volumes, you can set the options for just one volume like so:
|
||||||
|
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ except:
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
calculates various checksums for uploads,
|
calculates various checksums for uploads,
|
||||||
usage: -mtp crc32,md5,sha1,sha256b=bin/mtag/cksum.py
|
usage: -mtp crc32,md5,sha1,sha256b=ad,bin/mtag/cksum.py
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
61
bin/mtag/guestbook-read.py
Executable file
61
bin/mtag/guestbook-read.py
Executable file
@@ -0,0 +1,61 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
"""
|
||||||
|
fetch latest msg from guestbook and return as tag
|
||||||
|
|
||||||
|
example copyparty config to use this:
|
||||||
|
--urlform save,get -vsrv/hello:hello:w:c,e2ts,mtp=guestbook=t10,ad,p,bin/mtag/guestbook-read.py:mte=+guestbook
|
||||||
|
|
||||||
|
explained:
|
||||||
|
for realpath srv/hello (served at /hello), write-only for eveyrone,
|
||||||
|
enable file analysis on upload (e2ts),
|
||||||
|
use mtp plugin "bin/mtag/guestbook-read.py" to provide metadata tag "guestbook",
|
||||||
|
do this on all uploads regardless of extension,
|
||||||
|
t10 = 10 seconds timeout for each dwonload,
|
||||||
|
ad = parse file regardless if FFmpeg thinks it is audio or not
|
||||||
|
p = request upload info as json on stdin (need ip)
|
||||||
|
mte=+guestbook enabled indexing of that tag for this volume
|
||||||
|
|
||||||
|
PS: this requires e2ts to be functional,
|
||||||
|
meaning you need to do at least one of these:
|
||||||
|
* apt install ffmpeg
|
||||||
|
* pip3 install mutagen
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
import sqlite3
|
||||||
|
import sys
|
||||||
|
|
||||||
|
|
||||||
|
# set 0 to allow infinite msgs from one IP,
|
||||||
|
# other values delete older messages to make space,
|
||||||
|
# so 1 only keeps latest msg
|
||||||
|
NUM_MSGS_TO_KEEP = 1
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
fp = os.path.abspath(sys.argv[1])
|
||||||
|
fdir = os.path.dirname(fp)
|
||||||
|
|
||||||
|
zb = sys.stdin.buffer.read()
|
||||||
|
zs = zb.decode("utf-8", "replace")
|
||||||
|
md = json.loads(zs)
|
||||||
|
|
||||||
|
ip = md["up_ip"]
|
||||||
|
|
||||||
|
# can put the database inside `fdir` if you'd like,
|
||||||
|
# by default it saves to PWD:
|
||||||
|
# os.chdir(fdir)
|
||||||
|
|
||||||
|
db = sqlite3.connect("guestbook.db3")
|
||||||
|
with db:
|
||||||
|
t = "select msg from gb where ip = ? order by ts desc"
|
||||||
|
r = db.execute(t, (ip,)).fetchone()
|
||||||
|
if r:
|
||||||
|
print(r[0])
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
111
bin/mtag/guestbook.py
Normal file
111
bin/mtag/guestbook.py
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
"""
|
||||||
|
store messages from users in an sqlite database
|
||||||
|
which can be read from another mtp for example
|
||||||
|
|
||||||
|
takes input from application/x-www-form-urlencoded POSTs,
|
||||||
|
for example using the message/pager function on the website
|
||||||
|
|
||||||
|
example copyparty config to use this:
|
||||||
|
--urlform save,get -vsrv/hello:hello:w:c,e2ts,mtp=xgb=ebin,t10,ad,p,bin/mtag/guestbook.py:mte=+xgb
|
||||||
|
|
||||||
|
explained:
|
||||||
|
for realpath srv/hello (served at /hello),write-only for eveyrone,
|
||||||
|
enable file analysis on upload (e2ts),
|
||||||
|
use mtp plugin "bin/mtag/guestbook.py" to provide metadata tag "xgb",
|
||||||
|
do this on all uploads with the file extension "bin",
|
||||||
|
t300 = 300 seconds timeout for each dwonload,
|
||||||
|
ad = parse file regardless if FFmpeg thinks it is audio or not
|
||||||
|
p = request upload info as json on stdin
|
||||||
|
mte=+xgb enabled indexing of that tag for this volume
|
||||||
|
|
||||||
|
PS: this requires e2ts to be functional,
|
||||||
|
meaning you need to do at least one of these:
|
||||||
|
* apt install ffmpeg
|
||||||
|
* pip3 install mutagen
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
import sqlite3
|
||||||
|
import sys
|
||||||
|
from urllib.parse import unquote_to_bytes as unquote
|
||||||
|
|
||||||
|
|
||||||
|
# set 0 to allow infinite msgs from one IP,
|
||||||
|
# other values delete older messages to make space,
|
||||||
|
# so 1 only keeps latest msg
|
||||||
|
NUM_MSGS_TO_KEEP = 1
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
fp = os.path.abspath(sys.argv[1])
|
||||||
|
fdir = os.path.dirname(fp)
|
||||||
|
fname = os.path.basename(fp)
|
||||||
|
if not fname.startswith("put-") or not fname.endswith(".bin"):
|
||||||
|
raise Exception("not a post file")
|
||||||
|
|
||||||
|
zb = sys.stdin.buffer.read()
|
||||||
|
zs = zb.decode("utf-8", "replace")
|
||||||
|
md = json.loads(zs)
|
||||||
|
|
||||||
|
buf = b""
|
||||||
|
with open(fp, "rb") as f:
|
||||||
|
while True:
|
||||||
|
b = f.read(4096)
|
||||||
|
buf += b
|
||||||
|
if len(buf) > 4096:
|
||||||
|
raise Exception("too big")
|
||||||
|
|
||||||
|
if not b:
|
||||||
|
break
|
||||||
|
|
||||||
|
if not buf:
|
||||||
|
raise Exception("file is empty")
|
||||||
|
|
||||||
|
buf = unquote(buf.replace(b"+", b" "))
|
||||||
|
txt = buf.decode("utf-8")
|
||||||
|
|
||||||
|
if not txt.startswith("msg="):
|
||||||
|
raise Exception("does not start with msg=")
|
||||||
|
|
||||||
|
ip = md["up_ip"]
|
||||||
|
ts = md["up_at"]
|
||||||
|
txt = txt[4:]
|
||||||
|
|
||||||
|
# can put the database inside `fdir` if you'd like,
|
||||||
|
# by default it saves to PWD:
|
||||||
|
# os.chdir(fdir)
|
||||||
|
|
||||||
|
db = sqlite3.connect("guestbook.db3")
|
||||||
|
try:
|
||||||
|
db.execute("select 1 from gb").fetchone()
|
||||||
|
except:
|
||||||
|
with db:
|
||||||
|
db.execute("create table gb (ip text, ts real, msg text)")
|
||||||
|
db.execute("create index gb_ip on gb(ip)")
|
||||||
|
|
||||||
|
with db:
|
||||||
|
if NUM_MSGS_TO_KEEP == 1:
|
||||||
|
t = "delete from gb where ip = ?"
|
||||||
|
db.execute(t, (ip,))
|
||||||
|
|
||||||
|
t = "insert into gb values (?,?,?)"
|
||||||
|
db.execute(t, (ip, ts, txt))
|
||||||
|
|
||||||
|
if NUM_MSGS_TO_KEEP > 1:
|
||||||
|
t = "select ts from gb where ip = ? order by ts desc"
|
||||||
|
hits = db.execute(t, (ip,)).fetchall()
|
||||||
|
|
||||||
|
if len(hits) > NUM_MSGS_TO_KEEP:
|
||||||
|
lim = hits[NUM_MSGS_TO_KEEP][0]
|
||||||
|
t = "delete from gb where ip = ? and ts <= ?"
|
||||||
|
db.execute(t, (ip, lim))
|
||||||
|
|
||||||
|
print(txt)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
@@ -43,7 +43,6 @@ PS: this requires e2ts to be functional,
|
|||||||
|
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
import time
|
|
||||||
import filecmp
|
import filecmp
|
||||||
import subprocess as sp
|
import subprocess as sp
|
||||||
|
|
||||||
@@ -62,7 +61,7 @@ def main():
|
|||||||
|
|
||||||
os.chdir(cwd)
|
os.chdir(cwd)
|
||||||
f1 = fsenc(fn)
|
f1 = fsenc(fn)
|
||||||
f2 = os.path.join(b"noexif", f1)
|
f2 = fsenc(os.path.join(b"noexif", fn))
|
||||||
cmd = [
|
cmd = [
|
||||||
b"exiftool",
|
b"exiftool",
|
||||||
b"-exif:all=",
|
b"-exif:all=",
|
||||||
@@ -90,4 +89,7 @@ def main():
|
|||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
main()
|
try:
|
||||||
|
main()
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|||||||
@@ -4,8 +4,9 @@ set -e
|
|||||||
|
|
||||||
# install dependencies for audio-*.py
|
# install dependencies for audio-*.py
|
||||||
#
|
#
|
||||||
# linux/alpine: requires {python3,ffmpeg,fftw}-dev py3-{wheel,pip} py3-numpy{,-dev} vamp-sdk-dev patchelf cmake
|
# linux/alpine: requires gcc g++ make cmake patchelf {python3,ffmpeg,fftw,libsndfile}-dev py3-{wheel,pip} py3-numpy{,-dev}
|
||||||
# linux/debian: requires libav{codec,device,filter,format,resample,util}-dev {libfftw3,python3}-dev python3-{numpy,pip} vamp-{plugin-sdk,examples} patchelf cmake
|
# linux/debian: requires libav{codec,device,filter,format,resample,util}-dev {libfftw3,python3,libsndfile1}-dev python3-{numpy,pip} vamp-{plugin-sdk,examples} patchelf cmake
|
||||||
|
# linux/fedora: requires gcc gcc-c++ make cmake patchelf {python3,ffmpeg,fftw,libsndfile}-devel python3-numpy vamp-plugin-sdk qm-vamp-plugins
|
||||||
# win64: requires msys2-mingw64 environment
|
# win64: requires msys2-mingw64 environment
|
||||||
# macos: requires macports
|
# macos: requires macports
|
||||||
#
|
#
|
||||||
@@ -56,6 +57,7 @@ hash -r
|
|||||||
command -v python3 && pybin=python3 || pybin=python
|
command -v python3 && pybin=python3 || pybin=python
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$pybin -c 'import numpy' ||
|
||||||
$pybin -m pip install --user numpy
|
$pybin -m pip install --user numpy
|
||||||
|
|
||||||
|
|
||||||
@@ -101,8 +103,11 @@ export -f dl_files
|
|||||||
|
|
||||||
|
|
||||||
github_tarball() {
|
github_tarball() {
|
||||||
|
rm -rf g
|
||||||
|
mkdir g
|
||||||
|
cd g
|
||||||
dl_text "$1" |
|
dl_text "$1" |
|
||||||
tee json |
|
tee ../json |
|
||||||
(
|
(
|
||||||
# prefer jq if available
|
# prefer jq if available
|
||||||
jq -r '.tarball_url' ||
|
jq -r '.tarball_url' ||
|
||||||
@@ -111,8 +116,11 @@ github_tarball() {
|
|||||||
awk -F\" '/"tarball_url": "/ {print$4}'
|
awk -F\" '/"tarball_url": "/ {print$4}'
|
||||||
) |
|
) |
|
||||||
tee /dev/stderr |
|
tee /dev/stderr |
|
||||||
|
head -n 1 |
|
||||||
tr -d '\r' | tr '\n' '\0' |
|
tr -d '\r' | tr '\n' '\0' |
|
||||||
xargs -0 bash -c 'dl_files "$@"' _
|
xargs -0 bash -c 'dl_files "$@"' _
|
||||||
|
mv * ../tgz
|
||||||
|
cd ..
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -127,6 +135,7 @@ gitlab_tarball() {
|
|||||||
tr \" '\n' | grep -E '\.tar\.gz$' | head -n 1
|
tr \" '\n' | grep -E '\.tar\.gz$' | head -n 1
|
||||||
) |
|
) |
|
||||||
tee /dev/stderr |
|
tee /dev/stderr |
|
||||||
|
head -n 1 |
|
||||||
tr -d '\r' | tr '\n' '\0' |
|
tr -d '\r' | tr '\n' '\0' |
|
||||||
tee links |
|
tee links |
|
||||||
xargs -0 bash -c 'dl_files "$@"' _
|
xargs -0 bash -c 'dl_files "$@"' _
|
||||||
@@ -138,20 +147,27 @@ install_keyfinder() {
|
|||||||
# use msys2 in mingw-w64 mode
|
# use msys2 in mingw-w64 mode
|
||||||
# pacman -S --needed mingw-w64-x86_64-{ffmpeg,python}
|
# pacman -S --needed mingw-w64-x86_64-{ffmpeg,python}
|
||||||
|
|
||||||
github_tarball https://api.github.com/repos/mixxxdj/libkeyfinder/releases/latest
|
[ -e $HOME/pe/keyfinder ] && {
|
||||||
|
echo found a keyfinder build in ~/pe, skipping
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
tar -xf mixxxdj-libkeyfinder-*
|
cd "$td"
|
||||||
rm -- *.tar.gz
|
github_tarball https://api.github.com/repos/mixxxdj/libkeyfinder/releases/latest
|
||||||
|
ls -al
|
||||||
|
|
||||||
|
tar -xf tgz
|
||||||
|
rm tgz
|
||||||
cd mixxxdj-libkeyfinder*
|
cd mixxxdj-libkeyfinder*
|
||||||
|
|
||||||
h="$HOME"
|
h="$HOME"
|
||||||
so="lib/libkeyfinder.so"
|
so="lib/libkeyfinder.so"
|
||||||
memes=()
|
memes=(-DBUILD_TESTING=OFF)
|
||||||
|
|
||||||
[ $win ] &&
|
[ $win ] &&
|
||||||
so="bin/libkeyfinder.dll" &&
|
so="bin/libkeyfinder.dll" &&
|
||||||
h="$(printf '%s\n' "$USERPROFILE" | tr '\\' '/')" &&
|
h="$(printf '%s\n' "$USERPROFILE" | tr '\\' '/')" &&
|
||||||
memes+=(-G "MinGW Makefiles" -DBUILD_TESTING=OFF)
|
memes+=(-G "MinGW Makefiles")
|
||||||
|
|
||||||
[ $mac ] &&
|
[ $mac ] &&
|
||||||
so="lib/libkeyfinder.dylib"
|
so="lib/libkeyfinder.dylib"
|
||||||
@@ -171,7 +187,7 @@ install_keyfinder() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
# rm -rf /Users/ed/Library/Python/3.9/lib/python/site-packages/*keyfinder*
|
# rm -rf /Users/ed/Library/Python/3.9/lib/python/site-packages/*keyfinder*
|
||||||
CFLAGS="-I$h/pe/keyfinder/include -I/opt/local/include" \
|
CFLAGS="-I$h/pe/keyfinder/include -I/opt/local/include -I/usr/include/ffmpeg" \
|
||||||
LDFLAGS="-L$h/pe/keyfinder/lib -L$h/pe/keyfinder/lib64 -L/opt/local/lib" \
|
LDFLAGS="-L$h/pe/keyfinder/lib -L$h/pe/keyfinder/lib64 -L/opt/local/lib" \
|
||||||
PKG_CONFIG_PATH=/c/msys64/mingw64/lib/pkgconfig \
|
PKG_CONFIG_PATH=/c/msys64/mingw64/lib/pkgconfig \
|
||||||
$pybin -m pip install --user keyfinder
|
$pybin -m pip install --user keyfinder
|
||||||
@@ -208,6 +224,22 @@ install_vamp() {
|
|||||||
|
|
||||||
$pybin -m pip install --user vamp
|
$pybin -m pip install --user vamp
|
||||||
|
|
||||||
|
cd "$td"
|
||||||
|
echo '#include <vamp-sdk/Plugin.h>' | g++ -x c++ -c -o /dev/null - || [ -e ~/pe/vamp-sdk ] || {
|
||||||
|
printf '\033[33mcould not find the vamp-sdk, building from source\033[0m\n'
|
||||||
|
(dl_files yolo https://code.soundsoftware.ac.uk/attachments/download/2588/vamp-plugin-sdk-2.9.0.tar.gz)
|
||||||
|
sha512sum -c <(
|
||||||
|
echo "7ef7f837d19a08048b059e0da408373a7964ced452b290fae40b85d6d70ca9000bcfb3302cd0b4dc76cf2a848528456f78c1ce1ee0c402228d812bd347b6983b -"
|
||||||
|
) <vamp-plugin-sdk-2.9.0.tar.gz
|
||||||
|
tar -xf vamp-plugin-sdk-2.9.0.tar.gz
|
||||||
|
rm -- *.tar.gz
|
||||||
|
ls -al
|
||||||
|
cd vamp-plugin-sdk-*
|
||||||
|
./configure --prefix=$HOME/pe/vamp-sdk
|
||||||
|
make -j1 install
|
||||||
|
}
|
||||||
|
|
||||||
|
cd "$td"
|
||||||
have_beatroot || {
|
have_beatroot || {
|
||||||
printf '\033[33mcould not find the vamp beatroot plugin, building from source\033[0m\n'
|
printf '\033[33mcould not find the vamp beatroot plugin, building from source\033[0m\n'
|
||||||
(dl_files yolo https://code.soundsoftware.ac.uk/attachments/download/885/beatroot-vamp-v1.0.tar.gz)
|
(dl_files yolo https://code.soundsoftware.ac.uk/attachments/download/885/beatroot-vamp-v1.0.tar.gz)
|
||||||
@@ -215,8 +247,11 @@ install_vamp() {
|
|||||||
echo "1f444d1d58ccf565c0adfe99f1a1aa62789e19f5071e46857e2adfbc9d453037bc1c4dcb039b02c16240e9b97f444aaff3afb625c86aa2470233e711f55b6874 -"
|
echo "1f444d1d58ccf565c0adfe99f1a1aa62789e19f5071e46857e2adfbc9d453037bc1c4dcb039b02c16240e9b97f444aaff3afb625c86aa2470233e711f55b6874 -"
|
||||||
) <beatroot-vamp-v1.0.tar.gz
|
) <beatroot-vamp-v1.0.tar.gz
|
||||||
tar -xf beatroot-vamp-v1.0.tar.gz
|
tar -xf beatroot-vamp-v1.0.tar.gz
|
||||||
|
rm -- *.tar.gz
|
||||||
cd beatroot-vamp-v1.0
|
cd beatroot-vamp-v1.0
|
||||||
make -f Makefile.linux -j4
|
[ -e ~/pe/vamp-sdk ] &&
|
||||||
|
sed -ri 's`^(CFLAGS :=.*)`\1 -I'$HOME'/pe/vamp-sdk/include`' Makefile.linux
|
||||||
|
make -f Makefile.linux -j4 LDFLAGS=-L$HOME/pe/vamp-sdk/lib
|
||||||
# /home/ed/vamp /home/ed/.vamp /usr/local/lib/vamp
|
# /home/ed/vamp /home/ed/.vamp /usr/local/lib/vamp
|
||||||
mkdir ~/vamp
|
mkdir ~/vamp
|
||||||
cp -pv beatroot-vamp.* ~/vamp/
|
cp -pv beatroot-vamp.* ~/vamp/
|
||||||
@@ -230,6 +265,7 @@ install_vamp() {
|
|||||||
|
|
||||||
# not in use because it kinda segfaults, also no windows support
|
# not in use because it kinda segfaults, also no windows support
|
||||||
install_soundtouch() {
|
install_soundtouch() {
|
||||||
|
cd "$td"
|
||||||
gitlab_tarball https://gitlab.com/api/v4/projects/soundtouch%2Fsoundtouch/releases
|
gitlab_tarball https://gitlab.com/api/v4/projects/soundtouch%2Fsoundtouch/releases
|
||||||
|
|
||||||
tar -xvf soundtouch-*
|
tar -xvf soundtouch-*
|
||||||
|
|||||||
38
bin/mtag/mousepad.py
Normal file
38
bin/mtag/mousepad.py
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import subprocess as sp
|
||||||
|
|
||||||
|
|
||||||
|
"""
|
||||||
|
mtp test -- opens a texteditor
|
||||||
|
|
||||||
|
usage:
|
||||||
|
-vsrv/v1:v1:r:c,mte=+x1:c,mtp=x1=ad,p,bin/mtag/mousepad.py
|
||||||
|
|
||||||
|
explained:
|
||||||
|
c,mte: list of tags to index in this volume
|
||||||
|
c,mtp: add new tag provider
|
||||||
|
x1: dummy tag to provide
|
||||||
|
ad: dontcare if audio or not
|
||||||
|
p: priority 1 (run after initial tag-scan with ffprobe or mutagen)
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
env = os.environ.copy()
|
||||||
|
env["DISPLAY"] = ":0.0"
|
||||||
|
|
||||||
|
if False:
|
||||||
|
# open the uploaded file
|
||||||
|
fp = sys.argv[-1]
|
||||||
|
else:
|
||||||
|
# display stdin contents (`oth_tags`)
|
||||||
|
fp = "/dev/stdin"
|
||||||
|
|
||||||
|
p = sp.Popen(["/usr/bin/mousepad", fp])
|
||||||
|
p.communicate()
|
||||||
|
|
||||||
|
|
||||||
|
main()
|
||||||
76
bin/mtag/rclone-upload.py
Normal file
76
bin/mtag/rclone-upload.py
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
import subprocess as sp
|
||||||
|
import sys
|
||||||
|
import time
|
||||||
|
|
||||||
|
try:
|
||||||
|
from copyparty.util import fsenc
|
||||||
|
except:
|
||||||
|
|
||||||
|
def fsenc(p):
|
||||||
|
return p.encode("utf-8")
|
||||||
|
|
||||||
|
|
||||||
|
_ = r"""
|
||||||
|
first checks the tag "vidchk" which must be "ok" to continue,
|
||||||
|
then uploads all files to some cloud storage (RCLONE_REMOTE)
|
||||||
|
and DELETES THE ORIGINAL FILES if rclone returns 0 ("success")
|
||||||
|
|
||||||
|
deps:
|
||||||
|
rclone
|
||||||
|
|
||||||
|
usage:
|
||||||
|
-mtp x2=t43200,ay,p2,bin/mtag/rclone-upload.py
|
||||||
|
|
||||||
|
explained:
|
||||||
|
t43200: timeout 12h
|
||||||
|
ay: only process files which contain audio (including video with audio)
|
||||||
|
p2: set priority 2 (after vidchk's suggested priority of 1),
|
||||||
|
so the output of vidchk will be passed in here
|
||||||
|
|
||||||
|
complete usage example as vflags along with vidchk:
|
||||||
|
-vsrv/vidchk:vidchk:r:rw,ed:c,e2dsa,e2ts,mtp=vidchk=t600,p,bin/mtag/vidchk.py:c,mtp=rupload=t43200,ay,p2,bin/mtag/rclone-upload.py:c,mte=+vidchk,rupload
|
||||||
|
|
||||||
|
setup: see https://rclone.org/drive/
|
||||||
|
|
||||||
|
if you wanna use this script standalone / separately from copyparty,
|
||||||
|
either set CONDITIONAL_UPLOAD False or provide the following stdin:
|
||||||
|
{"vidchk":"ok"}
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
RCLONE_REMOTE = "notmybox"
|
||||||
|
CONDITIONAL_UPLOAD = True
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
fp = sys.argv[1]
|
||||||
|
if CONDITIONAL_UPLOAD:
|
||||||
|
zb = sys.stdin.buffer.read()
|
||||||
|
zs = zb.decode("utf-8", "replace")
|
||||||
|
md = json.loads(zs)
|
||||||
|
|
||||||
|
chk = md.get("vidchk", None)
|
||||||
|
if chk != "ok":
|
||||||
|
print(f"vidchk={chk}", file=sys.stderr)
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
dst = f"{RCLONE_REMOTE}:".encode("utf-8")
|
||||||
|
cmd = [b"rclone", b"copy", b"--", fsenc(fp), dst]
|
||||||
|
|
||||||
|
t0 = time.time()
|
||||||
|
try:
|
||||||
|
sp.check_call(cmd)
|
||||||
|
except:
|
||||||
|
print("rclone failed", file=sys.stderr)
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
print(f"{time.time() - t0:.1f} sec")
|
||||||
|
os.unlink(fsenc(fp))
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
21
bin/mtag/res/twitter-unmute.user.js
Normal file
21
bin/mtag/res/twitter-unmute.user.js
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
// ==UserScript==
|
||||||
|
// @name twitter-unmute
|
||||||
|
// @namespace http://ocv.me/
|
||||||
|
// @version 0.1
|
||||||
|
// @description memes
|
||||||
|
// @author ed <irc.rizon.net>
|
||||||
|
// @match https://twitter.com/*
|
||||||
|
// @icon https://www.google.com/s2/favicons?domain=twitter.com
|
||||||
|
// @grant GM_addStyle
|
||||||
|
// ==/UserScript==
|
||||||
|
|
||||||
|
function grunnur() {
|
||||||
|
setInterval(function () {
|
||||||
|
//document.querySelector('div[aria-label="Unmute"]').click();
|
||||||
|
document.querySelector('video').muted = false;
|
||||||
|
}, 200);
|
||||||
|
}
|
||||||
|
|
||||||
|
var scr = document.createElement('script');
|
||||||
|
scr.textContent = '(' + grunnur.toString() + ')();';
|
||||||
|
(document.head || document.getElementsByTagName('head')[0]).appendChild(scr);
|
||||||
139
bin/mtag/very-bad-idea.py
Executable file
139
bin/mtag/very-bad-idea.py
Executable file
@@ -0,0 +1,139 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
"""
|
||||||
|
use copyparty as a chromecast replacement:
|
||||||
|
* post a URL and it will open in the default browser
|
||||||
|
* upload a file and it will open in the default application
|
||||||
|
* the `key` command simulates keyboard input
|
||||||
|
* the `x` command executes other xdotool commands
|
||||||
|
* the `c` command executes arbitrary unix commands
|
||||||
|
|
||||||
|
the android app makes it a breeze to post pics and links:
|
||||||
|
https://github.com/9001/party-up/releases
|
||||||
|
(iOS devices have to rely on the web-UI)
|
||||||
|
|
||||||
|
goes without saying, but this is HELLA DANGEROUS,
|
||||||
|
GIVES RCE TO ANYONE WHO HAVE UPLOAD PERMISSIONS
|
||||||
|
|
||||||
|
example copyparty config to use this:
|
||||||
|
--urlform save,get -v.::w:c,e2d,e2t,mte=+a1:c,mtp=a1=ad,kn,c0,bin/mtag/very-bad-idea.py
|
||||||
|
|
||||||
|
recommended deps:
|
||||||
|
apt install xdotool libnotify-bin
|
||||||
|
https://github.com/9001/copyparty/blob/hovudstraum/contrib/plugins/meadup.js
|
||||||
|
|
||||||
|
and you probably want `twitter-unmute.user.js` from the res folder
|
||||||
|
|
||||||
|
|
||||||
|
-----------------------------------------------------------------------
|
||||||
|
-- startup script:
|
||||||
|
-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
|
||||||
|
# create qr code
|
||||||
|
ip=$(ip r | awk '/^default/{print$(NF-2)}'); echo http://$ip:3923/ | qrencode -o - -s 4 >/dev/shm/cpp-qr.png
|
||||||
|
/usr/bin/feh -x /dev/shm/cpp-qr.png &
|
||||||
|
|
||||||
|
# reposition and make topmost (with janky raspbian support)
|
||||||
|
( sleep 0.5
|
||||||
|
xdotool search --name cpp-qr.png windowactivate --sync windowmove 1780 0
|
||||||
|
wmctrl -r :ACTIVE: -b toggle,above || true
|
||||||
|
|
||||||
|
ps aux | grep -E 'sleep[ ]7\.27' ||
|
||||||
|
while true; do
|
||||||
|
w=$(xdotool getactivewindow)
|
||||||
|
xdotool search --name cpp-qr.png windowactivate windowraise windowfocus
|
||||||
|
xdotool windowactivate $w
|
||||||
|
xdotool windowfocus $w
|
||||||
|
sleep 7.27 || break
|
||||||
|
done &
|
||||||
|
xeyes # distraction window to prevent ^w from closing the qr-code
|
||||||
|
) &
|
||||||
|
|
||||||
|
# bail if copyparty is already running
|
||||||
|
ps aux | grep -E '[3] copy[p]arty' && exit 0
|
||||||
|
|
||||||
|
# dumb chrome wrapper to allow autoplay
|
||||||
|
cat >/usr/local/bin/chromium-browser <<'EOF'
|
||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
/usr/bin/chromium-browser --autoplay-policy=no-user-gesture-required "$@"
|
||||||
|
EOF
|
||||||
|
chmod 755 /usr/local/bin/chromium-browser
|
||||||
|
|
||||||
|
# start the server (note: replace `-v.::rw:` with `-v.::w:` to disallow retrieving uploaded stuff)
|
||||||
|
cd ~/Downloads; python3 copyparty-sfx.py --urlform save,get -v.::rw:c,e2d,e2t,mte=+a1:c,mtp=a1=ad,kn,very-bad-idea.py
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import time
|
||||||
|
import subprocess as sp
|
||||||
|
from urllib.parse import unquote_to_bytes as unquote
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
fp = os.path.abspath(sys.argv[1])
|
||||||
|
with open(fp, "rb") as f:
|
||||||
|
txt = f.read(4096)
|
||||||
|
|
||||||
|
if txt.startswith(b"msg="):
|
||||||
|
open_post(txt)
|
||||||
|
else:
|
||||||
|
open_url(fp)
|
||||||
|
|
||||||
|
|
||||||
|
def open_post(txt):
|
||||||
|
txt = unquote(txt.replace(b"+", b" ")).decode("utf-8")[4:]
|
||||||
|
try:
|
||||||
|
k, v = txt.split(" ", 1)
|
||||||
|
except:
|
||||||
|
open_url(txt)
|
||||||
|
|
||||||
|
if k == "key":
|
||||||
|
sp.call(["xdotool", "key"] + v.split(" "))
|
||||||
|
elif k == "x":
|
||||||
|
sp.call(["xdotool"] + v.split(" "))
|
||||||
|
elif k == "c":
|
||||||
|
env = os.environ.copy()
|
||||||
|
while " " in v:
|
||||||
|
v1, v2 = v.split(" ", 1)
|
||||||
|
if "=" not in v1:
|
||||||
|
break
|
||||||
|
|
||||||
|
ek, ev = v1.split("=", 1)
|
||||||
|
env[ek] = ev
|
||||||
|
v = v2
|
||||||
|
|
||||||
|
sp.call(v.split(" "), env=env)
|
||||||
|
else:
|
||||||
|
open_url(txt)
|
||||||
|
|
||||||
|
|
||||||
|
def open_url(txt):
|
||||||
|
ext = txt.rsplit(".")[-1].lower()
|
||||||
|
sp.call(["notify-send", "--", txt])
|
||||||
|
if ext not in ["jpg", "jpeg", "png", "gif", "webp"]:
|
||||||
|
# sp.call(["wmctrl", "-c", ":ACTIVE:"]) # closes the active window correctly
|
||||||
|
sp.call(["killall", "vlc"])
|
||||||
|
sp.call(["killall", "mpv"])
|
||||||
|
sp.call(["killall", "feh"])
|
||||||
|
time.sleep(0.5)
|
||||||
|
for _ in range(20):
|
||||||
|
sp.call(["xdotool", "key", "ctrl+w"]) # closes the open tab correctly
|
||||||
|
# else:
|
||||||
|
# sp.call(["xdotool", "getactivewindow", "windowminimize"]) # minimizes the focused windo
|
||||||
|
|
||||||
|
# close any error messages:
|
||||||
|
sp.call(["xdotool", "search", "--name", "Error", "windowclose"])
|
||||||
|
# sp.call(["xdotool", "key", "ctrl+alt+d"]) # doesnt work at all
|
||||||
|
# sp.call(["xdotool", "keydown", "--delay", "100", "ctrl+alt+d"])
|
||||||
|
# sp.call(["xdotool", "keyup", "ctrl+alt+d"])
|
||||||
|
sp.call(["xdg-open", txt])
|
||||||
|
|
||||||
|
|
||||||
|
main()
|
||||||
131
bin/mtag/vidchk.py
Executable file
131
bin/mtag/vidchk.py
Executable file
@@ -0,0 +1,131 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import json
|
||||||
|
import re
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import subprocess as sp
|
||||||
|
|
||||||
|
try:
|
||||||
|
from copyparty.util import fsenc
|
||||||
|
except:
|
||||||
|
|
||||||
|
def fsenc(p):
|
||||||
|
return p.encode("utf-8")
|
||||||
|
|
||||||
|
|
||||||
|
_ = r"""
|
||||||
|
inspects video files for errors and such
|
||||||
|
plus stores a bunch of metadata to filename.ff.json
|
||||||
|
|
||||||
|
usage:
|
||||||
|
-mtp vidchk=t600,ay,p,bin/mtag/vidchk.py
|
||||||
|
|
||||||
|
explained:
|
||||||
|
t600: timeout 10min
|
||||||
|
ay: only process files which contain audio (including video with audio)
|
||||||
|
p: set priority 1 (lowest priority after initial ffprobe/mutagen for base tags),
|
||||||
|
makes copyparty feed base tags into this script as json
|
||||||
|
|
||||||
|
if you wanna use this script standalone / separately from copyparty,
|
||||||
|
provide the video resolution on stdin as json: {"res":"1920x1080"}
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
FAST = True # parse entire file at container level
|
||||||
|
# FAST = False # fully decode audio and video streams
|
||||||
|
|
||||||
|
|
||||||
|
# warnings to ignore
|
||||||
|
harmless = re.compile(
|
||||||
|
r"Unsupported codec with id |Could not find codec parameters.*Attachment:|analyzeduration"
|
||||||
|
+ r"|timescale not set"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def wfilter(lines):
|
||||||
|
return [x for x in lines if x.strip() and not harmless.search(x)]
|
||||||
|
|
||||||
|
|
||||||
|
def errchk(so, se, rc, dbg):
|
||||||
|
if dbg:
|
||||||
|
with open(dbg, "wb") as f:
|
||||||
|
f.write(b"so:\n" + so + b"\nse:\n" + se + b"\n")
|
||||||
|
|
||||||
|
if rc:
|
||||||
|
err = (so + se).decode("utf-8", "replace").split("\n", 1)
|
||||||
|
err = wfilter(err) or err
|
||||||
|
return f"ERROR {rc}: {err[0]}"
|
||||||
|
|
||||||
|
if se:
|
||||||
|
err = se.decode("utf-8", "replace").split("\n", 1)
|
||||||
|
err = wfilter(err)
|
||||||
|
if err:
|
||||||
|
return f"Warning: {err[0]}"
|
||||||
|
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
fp = sys.argv[1]
|
||||||
|
zb = sys.stdin.buffer.read()
|
||||||
|
zs = zb.decode("utf-8", "replace")
|
||||||
|
md = json.loads(zs)
|
||||||
|
|
||||||
|
fdir = os.path.dirname(os.path.realpath(fp))
|
||||||
|
flag = os.path.join(fdir, ".processed")
|
||||||
|
if os.path.exists(flag):
|
||||||
|
return "already processed"
|
||||||
|
|
||||||
|
try:
|
||||||
|
w, h = [int(x) for x in md["res"].split("x")]
|
||||||
|
if not w + h:
|
||||||
|
raise Exception()
|
||||||
|
except:
|
||||||
|
return "could not determine resolution"
|
||||||
|
|
||||||
|
# grab streams/format metadata + 2 seconds of frames at the start and end
|
||||||
|
zs = "ffprobe -hide_banner -v warning -of json -show_streams -show_format -show_packets -show_data_hash crc32 -read_intervals %+2,999999%+2"
|
||||||
|
cmd = zs.encode("ascii").split(b" ") + [fsenc(fp)]
|
||||||
|
p = sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE)
|
||||||
|
so, se = p.communicate()
|
||||||
|
|
||||||
|
# spaces to tabs, drops filesize from 69k to 48k
|
||||||
|
so = b"\n".join(
|
||||||
|
[
|
||||||
|
b"\t" * int((len(x) - len(x.lstrip())) / 4) + x.lstrip()
|
||||||
|
for x in (so or b"").split(b"\n")
|
||||||
|
]
|
||||||
|
)
|
||||||
|
with open(fsenc(f"{fp}.ff.json"), "wb") as f:
|
||||||
|
f.write(so)
|
||||||
|
|
||||||
|
err = errchk(so, se, p.returncode, f"{fp}.vidchk")
|
||||||
|
if err:
|
||||||
|
return err
|
||||||
|
|
||||||
|
if max(w, h) < 1280 and min(w, h) < 720:
|
||||||
|
return "resolution too small"
|
||||||
|
|
||||||
|
zs = (
|
||||||
|
"ffmpeg -y -hide_banner -nostdin -v warning"
|
||||||
|
+ " -err_detect +crccheck+bitstream+buffer+careful+compliant+aggressive+explode"
|
||||||
|
+ " -xerror -i"
|
||||||
|
)
|
||||||
|
|
||||||
|
cmd = zs.encode("ascii").split(b" ") + [fsenc(fp)]
|
||||||
|
|
||||||
|
if FAST:
|
||||||
|
zs = "-c copy -f null -"
|
||||||
|
else:
|
||||||
|
zs = "-vcodec rawvideo -acodec pcm_s16le -f null -"
|
||||||
|
|
||||||
|
cmd += zs.encode("ascii").split(b" ")
|
||||||
|
|
||||||
|
p = sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE)
|
||||||
|
so, se = p.communicate()
|
||||||
|
return errchk(so, se, p.returncode, f"{fp}.vidchk")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
print(main() or "ok")
|
||||||
@@ -1,6 +1,11 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
DEPRECATED -- replaced by event hooks;
|
||||||
|
https://github.com/9001/copyparty/blob/hovudstraum/bin/hooks/wget.py
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
use copyparty as a file downloader by POSTing URLs as
|
use copyparty as a file downloader by POSTing URLs as
|
||||||
application/x-www-form-urlencoded (for example using the
|
application/x-www-form-urlencoded (for example using the
|
||||||
message/pager function on the website)
|
message/pager function on the website)
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
"""copyparty-fuse-streaming: remote copyparty as a local filesystem"""
|
"""partyfuse-streaming: remote copyparty as a local filesystem"""
|
||||||
__author__ = "ed <copyparty@ocv.me>"
|
__author__ = "ed <copyparty@ocv.me>"
|
||||||
__copyright__ = 2020
|
__copyright__ = 2020
|
||||||
__license__ = "MIT"
|
__license__ = "MIT"
|
||||||
@@ -12,7 +12,7 @@ __url__ = "https://github.com/9001/copyparty/"
|
|||||||
mount a copyparty server (local or remote) as a filesystem
|
mount a copyparty server (local or remote) as a filesystem
|
||||||
|
|
||||||
usage:
|
usage:
|
||||||
python copyparty-fuse-streaming.py http://192.168.1.69:3923/ ./music
|
python partyfuse-streaming.py http://192.168.1.69:3923/ ./music
|
||||||
|
|
||||||
dependencies:
|
dependencies:
|
||||||
python3 -m pip install --user fusepy
|
python3 -m pip install --user fusepy
|
||||||
@@ -21,7 +21,7 @@ dependencies:
|
|||||||
+ on Windows: https://github.com/billziss-gh/winfsp/releases/latest
|
+ on Windows: https://github.com/billziss-gh/winfsp/releases/latest
|
||||||
|
|
||||||
this was a mistake:
|
this was a mistake:
|
||||||
fork of copyparty-fuse.py with a streaming cache rather than readahead,
|
fork of partyfuse.py with a streaming cache rather than readahead,
|
||||||
thought this was gonna be way faster (and it kind of is)
|
thought this was gonna be way faster (and it kind of is)
|
||||||
except the overhead of reopening connections on trunc totally kills it
|
except the overhead of reopening connections on trunc totally kills it
|
||||||
"""
|
"""
|
||||||
@@ -42,6 +42,7 @@ import threading
|
|||||||
import traceback
|
import traceback
|
||||||
import http.client # py2: httplib
|
import http.client # py2: httplib
|
||||||
import urllib.parse
|
import urllib.parse
|
||||||
|
import calendar
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
from urllib.parse import quote_from_bytes as quote
|
from urllib.parse import quote_from_bytes as quote
|
||||||
from urllib.parse import unquote_to_bytes as unquote
|
from urllib.parse import unquote_to_bytes as unquote
|
||||||
@@ -61,12 +62,12 @@ except:
|
|||||||
else:
|
else:
|
||||||
libfuse = "apt install libfuse\n modprobe fuse"
|
libfuse = "apt install libfuse\n modprobe fuse"
|
||||||
|
|
||||||
print(
|
m = """\033[33m
|
||||||
"\n could not import fuse; these may help:"
|
could not import fuse; these may help:
|
||||||
+ "\n python3 -m pip install --user fusepy\n "
|
{} -m pip install --user fusepy
|
||||||
+ libfuse
|
{}
|
||||||
+ "\n"
|
\033[0m"""
|
||||||
)
|
print(m.format(sys.executable, libfuse))
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
|
||||||
@@ -153,7 +154,7 @@ def dewin(txt):
|
|||||||
class RecentLog(object):
|
class RecentLog(object):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.mtx = threading.Lock()
|
self.mtx = threading.Lock()
|
||||||
self.f = None # open("copyparty-fuse.log", "wb")
|
self.f = None # open("partyfuse.log", "wb")
|
||||||
self.q = []
|
self.q = []
|
||||||
|
|
||||||
thr = threading.Thread(target=self.printer)
|
thr = threading.Thread(target=self.printer)
|
||||||
@@ -184,9 +185,9 @@ class RecentLog(object):
|
|||||||
print("".join(q), end="")
|
print("".join(q), end="")
|
||||||
|
|
||||||
|
|
||||||
# [windows/cmd/cpy3] python dev\copyparty\bin\copyparty-fuse.py q: http://192.168.1.159:1234/
|
# [windows/cmd/cpy3] python dev\copyparty\bin\partyfuse.py q: http://192.168.1.159:1234/
|
||||||
# [windows/cmd/msys2] C:\msys64\mingw64\bin\python3 dev\copyparty\bin\copyparty-fuse.py q: http://192.168.1.159:1234/
|
# [windows/cmd/msys2] C:\msys64\mingw64\bin\python3 dev\copyparty\bin\partyfuse.py q: http://192.168.1.159:1234/
|
||||||
# [windows/mty/msys2] /mingw64/bin/python3 /c/Users/ed/dev/copyparty/bin/copyparty-fuse.py q: http://192.168.1.159:1234/
|
# [windows/mty/msys2] /mingw64/bin/python3 /c/Users/ed/dev/copyparty/bin/partyfuse.py q: http://192.168.1.159:1234/
|
||||||
#
|
#
|
||||||
# [windows] find /q/music/albums/Phant*24bit -printf '%s %p\n' | sort -n | tail -n 8 | sed -r 's/^[0-9]+ //' | while IFS= read -r x; do dd if="$x" of=/dev/null bs=4k count=8192 & done
|
# [windows] find /q/music/albums/Phant*24bit -printf '%s %p\n' | sort -n | tail -n 8 | sed -r 's/^[0-9]+ //' | while IFS= read -r x; do dd if="$x" of=/dev/null bs=4k count=8192 & done
|
||||||
# [alpine] ll t; for x in t/2020_0724_16{2,3}*; do dd if="$x" of=/dev/null bs=4k count=10240 & done
|
# [alpine] ll t; for x in t/2020_0724_16{2,3}*; do dd if="$x" of=/dev/null bs=4k count=10240 & done
|
||||||
@@ -495,7 +496,7 @@ class Gateway(object):
|
|||||||
ts = 60 * 60 * 24 * 2
|
ts = 60 * 60 * 24 * 2
|
||||||
try:
|
try:
|
||||||
sz = int(fsize)
|
sz = int(fsize)
|
||||||
ts = datetime.strptime(fdate, "%Y-%m-%d %H:%M:%S").timestamp()
|
ts = calendar.timegm(time.strptime(fdate, "%Y-%m-%d %H:%M:%S"))
|
||||||
except:
|
except:
|
||||||
info("bad HTML or OS [{}] [{}]".format(fdate, fsize))
|
info("bad HTML or OS [{}] [{}]".format(fdate, fsize))
|
||||||
# python cannot strptime(1959-01-01) on windows
|
# python cannot strptime(1959-01-01) on windows
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
"""copyparty-fuse: remote copyparty as a local filesystem"""
|
"""partyfuse: remote copyparty as a local filesystem"""
|
||||||
__author__ = "ed <copyparty@ocv.me>"
|
__author__ = "ed <copyparty@ocv.me>"
|
||||||
__copyright__ = 2019
|
__copyright__ = 2019
|
||||||
__license__ = "MIT"
|
__license__ = "MIT"
|
||||||
@@ -12,7 +12,7 @@ __url__ = "https://github.com/9001/copyparty/"
|
|||||||
mount a copyparty server (local or remote) as a filesystem
|
mount a copyparty server (local or remote) as a filesystem
|
||||||
|
|
||||||
usage:
|
usage:
|
||||||
python copyparty-fuse.py http://192.168.1.69:3923/ ./music
|
python partyfuse.py http://192.168.1.69:3923/ ./music
|
||||||
|
|
||||||
dependencies:
|
dependencies:
|
||||||
python3 -m pip install --user fusepy
|
python3 -m pip install --user fusepy
|
||||||
@@ -45,6 +45,7 @@ import threading
|
|||||||
import traceback
|
import traceback
|
||||||
import http.client # py2: httplib
|
import http.client # py2: httplib
|
||||||
import urllib.parse
|
import urllib.parse
|
||||||
|
import calendar
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
from urllib.parse import quote_from_bytes as quote
|
from urllib.parse import quote_from_bytes as quote
|
||||||
from urllib.parse import unquote_to_bytes as unquote
|
from urllib.parse import unquote_to_bytes as unquote
|
||||||
@@ -73,12 +74,12 @@ except:
|
|||||||
else:
|
else:
|
||||||
libfuse = "apt install libfuse3-3\n modprobe fuse"
|
libfuse = "apt install libfuse3-3\n modprobe fuse"
|
||||||
|
|
||||||
print(
|
m = """\033[33m
|
||||||
"\n could not import fuse; these may help:"
|
could not import fuse; these may help:
|
||||||
+ "\n python3 -m pip install --user fusepy\n "
|
{} -m pip install --user fusepy
|
||||||
+ libfuse
|
{}
|
||||||
+ "\n"
|
\033[0m"""
|
||||||
)
|
print(m.format(sys.executable, libfuse))
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
|
||||||
@@ -165,7 +166,7 @@ def dewin(txt):
|
|||||||
class RecentLog(object):
|
class RecentLog(object):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.mtx = threading.Lock()
|
self.mtx = threading.Lock()
|
||||||
self.f = None # open("copyparty-fuse.log", "wb")
|
self.f = None # open("partyfuse.log", "wb")
|
||||||
self.q = []
|
self.q = []
|
||||||
|
|
||||||
thr = threading.Thread(target=self.printer)
|
thr = threading.Thread(target=self.printer)
|
||||||
@@ -196,9 +197,9 @@ class RecentLog(object):
|
|||||||
print("".join(q), end="")
|
print("".join(q), end="")
|
||||||
|
|
||||||
|
|
||||||
# [windows/cmd/cpy3] python dev\copyparty\bin\copyparty-fuse.py q: http://192.168.1.159:1234/
|
# [windows/cmd/cpy3] python dev\copyparty\bin\partyfuse.py q: http://192.168.1.159:1234/
|
||||||
# [windows/cmd/msys2] C:\msys64\mingw64\bin\python3 dev\copyparty\bin\copyparty-fuse.py q: http://192.168.1.159:1234/
|
# [windows/cmd/msys2] C:\msys64\mingw64\bin\python3 dev\copyparty\bin\partyfuse.py q: http://192.168.1.159:1234/
|
||||||
# [windows/mty/msys2] /mingw64/bin/python3 /c/Users/ed/dev/copyparty/bin/copyparty-fuse.py q: http://192.168.1.159:1234/
|
# [windows/mty/msys2] /mingw64/bin/python3 /c/Users/ed/dev/copyparty/bin/partyfuse.py q: http://192.168.1.159:1234/
|
||||||
#
|
#
|
||||||
# [windows] find /q/music/albums/Phant*24bit -printf '%s %p\n' | sort -n | tail -n 8 | sed -r 's/^[0-9]+ //' | while IFS= read -r x; do dd if="$x" of=/dev/null bs=4k count=8192 & done
|
# [windows] find /q/music/albums/Phant*24bit -printf '%s %p\n' | sort -n | tail -n 8 | sed -r 's/^[0-9]+ //' | while IFS= read -r x; do dd if="$x" of=/dev/null bs=4k count=8192 & done
|
||||||
# [alpine] ll t; for x in t/2020_0724_16{2,3}*; do dd if="$x" of=/dev/null bs=4k count=10240 & done
|
# [alpine] ll t; for x in t/2020_0724_16{2,3}*; do dd if="$x" of=/dev/null bs=4k count=10240 & done
|
||||||
@@ -443,7 +444,7 @@ class Gateway(object):
|
|||||||
ts = 60 * 60 * 24 * 2
|
ts = 60 * 60 * 24 * 2
|
||||||
try:
|
try:
|
||||||
sz = int(fsize)
|
sz = int(fsize)
|
||||||
ts = datetime.strptime(fdate, "%Y-%m-%d %H:%M:%S").timestamp()
|
ts = calendar.timegm(time.strptime(fdate, "%Y-%m-%d %H:%M:%S"))
|
||||||
except:
|
except:
|
||||||
info("bad HTML or OS [{}] [{}]".format(fdate, fsize))
|
info("bad HTML or OS [{}] [{}]".format(fdate, fsize))
|
||||||
# python cannot strptime(1959-01-01) on windows
|
# python cannot strptime(1959-01-01) on windows
|
||||||
@@ -996,7 +997,7 @@ def main():
|
|||||||
ap.add_argument(
|
ap.add_argument(
|
||||||
"-cf", metavar="NUM_BLOCKS", type=int, default=nf, help="file cache"
|
"-cf", metavar="NUM_BLOCKS", type=int, default=nf, help="file cache"
|
||||||
)
|
)
|
||||||
ap.add_argument("-a", metavar="PASSWORD", help="password")
|
ap.add_argument("-a", metavar="PASSWORD", help="password or $filepath")
|
||||||
ap.add_argument("-d", action="store_true", help="enable debug")
|
ap.add_argument("-d", action="store_true", help="enable debug")
|
||||||
ap.add_argument("-te", metavar="PEM_FILE", help="certificate to expect/verify")
|
ap.add_argument("-te", metavar="PEM_FILE", help="certificate to expect/verify")
|
||||||
ap.add_argument("-td", action="store_true", help="disable certificate check")
|
ap.add_argument("-td", action="store_true", help="disable certificate check")
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
"""copyparty-fuseb: remote copyparty as a local filesystem"""
|
"""partyfuse2: remote copyparty as a local filesystem"""
|
||||||
__author__ = "ed <copyparty@ocv.me>"
|
__author__ = "ed <copyparty@ocv.me>"
|
||||||
__copyright__ = 2020
|
__copyright__ = 2020
|
||||||
__license__ = "MIT"
|
__license__ = "MIT"
|
||||||
@@ -11,14 +11,18 @@ import re
|
|||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
import time
|
import time
|
||||||
|
import json
|
||||||
import stat
|
import stat
|
||||||
import errno
|
import errno
|
||||||
import struct
|
import struct
|
||||||
|
import codecs
|
||||||
|
import platform
|
||||||
import threading
|
import threading
|
||||||
import http.client # py2: httplib
|
import http.client # py2: httplib
|
||||||
import urllib.parse
|
import urllib.parse
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
from urllib.parse import quote_from_bytes as quote
|
from urllib.parse import quote_from_bytes as quote
|
||||||
|
from urllib.parse import unquote_to_bytes as unquote
|
||||||
|
|
||||||
try:
|
try:
|
||||||
import fuse
|
import fuse
|
||||||
@@ -28,9 +32,19 @@ try:
|
|||||||
if not hasattr(fuse, "__version__"):
|
if not hasattr(fuse, "__version__"):
|
||||||
raise Exception("your fuse-python is way old")
|
raise Exception("your fuse-python is way old")
|
||||||
except:
|
except:
|
||||||
print(
|
if WINDOWS:
|
||||||
"\n could not import fuse; these may help:\n python3 -m pip install --user fuse-python\n apt install libfuse\n modprobe fuse\n"
|
libfuse = "install https://github.com/billziss-gh/winfsp/releases/latest"
|
||||||
)
|
elif MACOS:
|
||||||
|
libfuse = "install https://osxfuse.github.io/"
|
||||||
|
else:
|
||||||
|
libfuse = "apt install libfuse\n modprobe fuse"
|
||||||
|
|
||||||
|
m = """\033[33m
|
||||||
|
could not import fuse; these may help:
|
||||||
|
{} -m pip install --user fuse-python
|
||||||
|
{}
|
||||||
|
\033[0m"""
|
||||||
|
print(m.format(sys.executable, libfuse))
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
|
||||||
@@ -38,18 +52,22 @@ except:
|
|||||||
mount a copyparty server (local or remote) as a filesystem
|
mount a copyparty server (local or remote) as a filesystem
|
||||||
|
|
||||||
usage:
|
usage:
|
||||||
python ./copyparty-fuseb.py -f -o allow_other,auto_unmount,nonempty,url=http://192.168.1.69:3923 /mnt/nas
|
python ./partyfuse2.py -f -o allow_other,auto_unmount,nonempty,pw=wark,url=http://192.168.1.69:3923 /mnt/nas
|
||||||
|
|
||||||
dependencies:
|
dependencies:
|
||||||
sudo apk add fuse-dev python3-dev
|
sudo apk add fuse-dev python3-dev
|
||||||
python3 -m pip install --user fuse-python
|
python3 -m pip install --user fuse-python
|
||||||
|
|
||||||
fork of copyparty-fuse.py based on fuse-python which
|
fork of partyfuse.py based on fuse-python which
|
||||||
appears to be more compliant than fusepy? since this works with samba
|
appears to be more compliant than fusepy? since this works with samba
|
||||||
(probably just my garbage code tbh)
|
(probably just my garbage code tbh)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
WINDOWS = sys.platform == "win32"
|
||||||
|
MACOS = platform.system() == "Darwin"
|
||||||
|
|
||||||
|
|
||||||
def threadless_log(msg):
|
def threadless_log(msg):
|
||||||
print(msg + "\n", end="")
|
print(msg + "\n", end="")
|
||||||
|
|
||||||
@@ -93,6 +111,41 @@ def html_dec(txt):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def register_wtf8():
|
||||||
|
def wtf8_enc(text):
|
||||||
|
return str(text).encode("utf-8", "surrogateescape"), len(text)
|
||||||
|
|
||||||
|
def wtf8_dec(binary):
|
||||||
|
return bytes(binary).decode("utf-8", "surrogateescape"), len(binary)
|
||||||
|
|
||||||
|
def wtf8_search(encoding_name):
|
||||||
|
return codecs.CodecInfo(wtf8_enc, wtf8_dec, name="wtf-8")
|
||||||
|
|
||||||
|
codecs.register(wtf8_search)
|
||||||
|
|
||||||
|
|
||||||
|
bad_good = {}
|
||||||
|
good_bad = {}
|
||||||
|
|
||||||
|
|
||||||
|
def enwin(txt):
|
||||||
|
return "".join([bad_good.get(x, x) for x in txt])
|
||||||
|
|
||||||
|
for bad, good in bad_good.items():
|
||||||
|
txt = txt.replace(bad, good)
|
||||||
|
|
||||||
|
return txt
|
||||||
|
|
||||||
|
|
||||||
|
def dewin(txt):
|
||||||
|
return "".join([good_bad.get(x, x) for x in txt])
|
||||||
|
|
||||||
|
for bad, good in bad_good.items():
|
||||||
|
txt = txt.replace(good, bad)
|
||||||
|
|
||||||
|
return txt
|
||||||
|
|
||||||
|
|
||||||
class CacheNode(object):
|
class CacheNode(object):
|
||||||
def __init__(self, tag, data):
|
def __init__(self, tag, data):
|
||||||
self.tag = tag
|
self.tag = tag
|
||||||
@@ -115,8 +168,9 @@ class Stat(fuse.Stat):
|
|||||||
|
|
||||||
|
|
||||||
class Gateway(object):
|
class Gateway(object):
|
||||||
def __init__(self, base_url):
|
def __init__(self, base_url, pw):
|
||||||
self.base_url = base_url
|
self.base_url = base_url
|
||||||
|
self.pw = pw
|
||||||
|
|
||||||
ui = urllib.parse.urlparse(base_url)
|
ui = urllib.parse.urlparse(base_url)
|
||||||
self.web_root = ui.path.strip("/")
|
self.web_root = ui.path.strip("/")
|
||||||
@@ -135,8 +189,7 @@ class Gateway(object):
|
|||||||
self.conns = {}
|
self.conns = {}
|
||||||
|
|
||||||
def quotep(self, path):
|
def quotep(self, path):
|
||||||
# TODO: mojibake support
|
path = path.encode("wtf-8")
|
||||||
path = path.encode("utf-8", "ignore")
|
|
||||||
return quote(path, safe="/")
|
return quote(path, safe="/")
|
||||||
|
|
||||||
def getconn(self, tid=None):
|
def getconn(self, tid=None):
|
||||||
@@ -159,20 +212,29 @@ class Gateway(object):
|
|||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def sendreq(self, *args, **kwargs):
|
def sendreq(self, *args, **ka):
|
||||||
tid = get_tid()
|
tid = get_tid()
|
||||||
|
if self.pw:
|
||||||
|
ck = "cppwd=" + self.pw
|
||||||
|
try:
|
||||||
|
ka["headers"]["Cookie"] = ck
|
||||||
|
except:
|
||||||
|
ka["headers"] = {"Cookie": ck}
|
||||||
try:
|
try:
|
||||||
c = self.getconn(tid)
|
c = self.getconn(tid)
|
||||||
c.request(*list(args), **kwargs)
|
c.request(*list(args), **ka)
|
||||||
return c.getresponse()
|
return c.getresponse()
|
||||||
except:
|
except:
|
||||||
self.closeconn(tid)
|
self.closeconn(tid)
|
||||||
c = self.getconn(tid)
|
c = self.getconn(tid)
|
||||||
c.request(*list(args), **kwargs)
|
c.request(*list(args), **ka)
|
||||||
return c.getresponse()
|
return c.getresponse()
|
||||||
|
|
||||||
def listdir(self, path):
|
def listdir(self, path):
|
||||||
web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots"
|
if bad_good:
|
||||||
|
path = dewin(path)
|
||||||
|
|
||||||
|
web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots&ls"
|
||||||
r = self.sendreq("GET", web_path)
|
r = self.sendreq("GET", web_path)
|
||||||
if r.status != 200:
|
if r.status != 200:
|
||||||
self.closeconn()
|
self.closeconn()
|
||||||
@@ -182,9 +244,12 @@ class Gateway(object):
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
return self.parse_html(r)
|
return self.parse_jls(r)
|
||||||
|
|
||||||
def download_file_range(self, path, ofs1, ofs2):
|
def download_file_range(self, path, ofs1, ofs2):
|
||||||
|
if bad_good:
|
||||||
|
path = dewin(path)
|
||||||
|
|
||||||
web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?raw"
|
web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?raw"
|
||||||
hdr_range = "bytes={}-{}".format(ofs1, ofs2 - 1)
|
hdr_range = "bytes={}-{}".format(ofs1, ofs2 - 1)
|
||||||
log("downloading {}".format(hdr_range))
|
log("downloading {}".format(hdr_range))
|
||||||
@@ -200,40 +265,27 @@ class Gateway(object):
|
|||||||
|
|
||||||
return r.read()
|
return r.read()
|
||||||
|
|
||||||
def parse_html(self, datasrc):
|
def parse_jls(self, datasrc):
|
||||||
ret = []
|
rsp = b""
|
||||||
remainder = b""
|
|
||||||
ptn = re.compile(
|
|
||||||
r"^<tr><td>(-|DIR)</td><td><a [^>]+>([^<]+)</a></td><td>([^<]+)</td><td>([^<]+)</td></tr>$"
|
|
||||||
)
|
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
buf = remainder + datasrc.read(4096)
|
buf = datasrc.read(1024 * 32)
|
||||||
# print('[{}]'.format(buf.decode('utf-8')))
|
|
||||||
if not buf:
|
if not buf:
|
||||||
break
|
break
|
||||||
|
|
||||||
remainder = b""
|
rsp += buf
|
||||||
endpos = buf.rfind(b"\n")
|
|
||||||
if endpos >= 0:
|
|
||||||
remainder = buf[endpos + 1 :]
|
|
||||||
buf = buf[:endpos]
|
|
||||||
|
|
||||||
lines = buf.decode("utf-8").split("\n")
|
rsp = json.loads(rsp.decode("utf-8"))
|
||||||
for line in lines:
|
ret = []
|
||||||
m = ptn.match(line)
|
for statfun, nodes in [
|
||||||
if not m:
|
[self.stat_dir, rsp["dirs"]],
|
||||||
# print(line)
|
[self.stat_file, rsp["files"]],
|
||||||
continue
|
]:
|
||||||
|
for n in nodes:
|
||||||
|
fname = unquote(n["href"].split("?")[0]).rstrip(b"/").decode("wtf-8")
|
||||||
|
if bad_good:
|
||||||
|
fname = enwin(fname)
|
||||||
|
|
||||||
ftype, fname, fsize, fdate = m.groups()
|
ret.append([fname, statfun(n["ts"], n["sz"]), 0])
|
||||||
fname = html_dec(fname)
|
|
||||||
ts = datetime.strptime(fdate, "%Y-%m-%d %H:%M:%S").timestamp()
|
|
||||||
sz = int(fsize)
|
|
||||||
if ftype == "-":
|
|
||||||
ret.append([fname, self.stat_file(ts, sz), 0])
|
|
||||||
else:
|
|
||||||
ret.append([fname, self.stat_dir(ts, sz), 0])
|
|
||||||
|
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
@@ -262,6 +314,7 @@ class CPPF(Fuse):
|
|||||||
Fuse.__init__(self, *args, **kwargs)
|
Fuse.__init__(self, *args, **kwargs)
|
||||||
|
|
||||||
self.url = None
|
self.url = None
|
||||||
|
self.pw = None
|
||||||
|
|
||||||
self.dircache = []
|
self.dircache = []
|
||||||
self.dircache_mtx = threading.Lock()
|
self.dircache_mtx = threading.Lock()
|
||||||
@@ -271,7 +324,7 @@ class CPPF(Fuse):
|
|||||||
|
|
||||||
def init2(self):
|
def init2(self):
|
||||||
# TODO figure out how python-fuse wanted this to go
|
# TODO figure out how python-fuse wanted this to go
|
||||||
self.gw = Gateway(self.url) # .decode('utf-8'))
|
self.gw = Gateway(self.url, self.pw) # .decode('utf-8'))
|
||||||
info("up")
|
info("up")
|
||||||
|
|
||||||
def clean_dircache(self):
|
def clean_dircache(self):
|
||||||
@@ -536,6 +589,8 @@ class CPPF(Fuse):
|
|||||||
|
|
||||||
def getattr(self, path):
|
def getattr(self, path):
|
||||||
log("getattr [{}]".format(path))
|
log("getattr [{}]".format(path))
|
||||||
|
if WINDOWS:
|
||||||
|
path = enwin(path) # windows occasionally decodes f0xx to xx
|
||||||
|
|
||||||
path = path.strip("/")
|
path = path.strip("/")
|
||||||
try:
|
try:
|
||||||
@@ -568,9 +623,25 @@ class CPPF(Fuse):
|
|||||||
|
|
||||||
def main():
|
def main():
|
||||||
time.strptime("19970815", "%Y%m%d") # python#7980
|
time.strptime("19970815", "%Y%m%d") # python#7980
|
||||||
|
register_wtf8()
|
||||||
|
if WINDOWS:
|
||||||
|
os.system("rem")
|
||||||
|
|
||||||
|
for ch in '<>:"\\|?*':
|
||||||
|
# microsoft maps illegal characters to f0xx
|
||||||
|
# (e000 to f8ff is basic-plane private-use)
|
||||||
|
bad_good[ch] = chr(ord(ch) + 0xF000)
|
||||||
|
|
||||||
|
for n in range(0, 0x100):
|
||||||
|
# map surrogateescape to another private-use area
|
||||||
|
bad_good[chr(n + 0xDC00)] = chr(n + 0xF100)
|
||||||
|
|
||||||
|
for k, v in bad_good.items():
|
||||||
|
good_bad[v] = k
|
||||||
|
|
||||||
server = CPPF()
|
server = CPPF()
|
||||||
server.parser.add_option(mountopt="url", metavar="BASE_URL", default=None)
|
server.parser.add_option(mountopt="url", metavar="BASE_URL", default=None)
|
||||||
|
server.parser.add_option(mountopt="pw", metavar="PASSWORD", default=None)
|
||||||
server.parse(values=server, errex=1)
|
server.parse(values=server, errex=1)
|
||||||
if not server.url or not str(server.url).startswith("http"):
|
if not server.url or not str(server.url).startswith("http"):
|
||||||
print("\nerror:")
|
print("\nerror:")
|
||||||
@@ -578,7 +649,7 @@ def main():
|
|||||||
print(" need argument: mount-path")
|
print(" need argument: mount-path")
|
||||||
print("example:")
|
print("example:")
|
||||||
print(
|
print(
|
||||||
" ./copyparty-fuseb.py -f -o allow_other,auto_unmount,nonempty,url=http://192.168.1.69:3923 /mnt/nas"
|
" ./partyfuse2.py -f -o allow_other,auto_unmount,nonempty,pw=wark,url=http://192.168.1.69:3923 /mnt/nas"
|
||||||
)
|
)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
177
bin/partyjournal.py
Executable file
177
bin/partyjournal.py
Executable file
@@ -0,0 +1,177 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
"""
|
||||||
|
partyjournal.py: chronological history of uploads
|
||||||
|
2021-12-31, v0.1, ed <irc.rizon.net>, MIT-Licensed
|
||||||
|
https://github.com/9001/copyparty/blob/hovudstraum/bin/partyjournal.py
|
||||||
|
|
||||||
|
produces a chronological list of all uploads,
|
||||||
|
by collecting info from up2k databases and the filesystem
|
||||||
|
|
||||||
|
specify subnet `192.168.1.*` with argument `.=192.168.1.`,
|
||||||
|
affecting all successive mappings
|
||||||
|
|
||||||
|
usage:
|
||||||
|
./partyjournal.py > partyjournal.html .=192.168.1. cart=125 steen=114 steen=131 sleepy=121 fscarlet=144 ed=101 ed=123
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import base64
|
||||||
|
import sqlite3
|
||||||
|
import argparse
|
||||||
|
from datetime import datetime
|
||||||
|
from urllib.parse import quote_from_bytes as quote
|
||||||
|
from urllib.parse import unquote_to_bytes as unquote
|
||||||
|
|
||||||
|
|
||||||
|
FS_ENCODING = sys.getfilesystemencoding()
|
||||||
|
|
||||||
|
|
||||||
|
class APF(argparse.ArgumentDefaultsHelpFormatter, argparse.RawDescriptionHelpFormatter):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
##
|
||||||
|
## snibbed from copyparty
|
||||||
|
|
||||||
|
|
||||||
|
def s3dec(v):
|
||||||
|
if not v.startswith("//"):
|
||||||
|
return v
|
||||||
|
|
||||||
|
v = base64.urlsafe_b64decode(v.encode("ascii")[2:])
|
||||||
|
return v.decode(FS_ENCODING, "replace")
|
||||||
|
|
||||||
|
|
||||||
|
def quotep(txt):
|
||||||
|
btxt = txt.encode("utf-8", "replace")
|
||||||
|
quot1 = quote(btxt, safe=b"/")
|
||||||
|
quot1 = quot1.encode("ascii")
|
||||||
|
quot2 = quot1.replace(b" ", b"+")
|
||||||
|
return quot2.decode("utf-8", "replace")
|
||||||
|
|
||||||
|
|
||||||
|
def html_escape(s, quote=False, crlf=False):
|
||||||
|
"""html.escape but also newlines"""
|
||||||
|
s = s.replace("&", "&").replace("<", "<").replace(">", ">")
|
||||||
|
if quote:
|
||||||
|
s = s.replace('"', """).replace("'", "'")
|
||||||
|
if crlf:
|
||||||
|
s = s.replace("\r", " ").replace("\n", " ")
|
||||||
|
|
||||||
|
return s
|
||||||
|
|
||||||
|
|
||||||
|
## end snibs
|
||||||
|
##
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
ap = argparse.ArgumentParser(formatter_class=APF)
|
||||||
|
ap.add_argument("who", nargs="*")
|
||||||
|
ar = ap.parse_args()
|
||||||
|
|
||||||
|
imap = {}
|
||||||
|
subnet = ""
|
||||||
|
for v in ar.who:
|
||||||
|
if "=" not in v:
|
||||||
|
raise Exception("bad who: " + v)
|
||||||
|
|
||||||
|
k, v = v.split("=")
|
||||||
|
if k == ".":
|
||||||
|
subnet = v
|
||||||
|
continue
|
||||||
|
|
||||||
|
imap["{}{}".format(subnet, v)] = k
|
||||||
|
|
||||||
|
print(repr(imap), file=sys.stderr)
|
||||||
|
|
||||||
|
print(
|
||||||
|
"""\
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head><meta charset="utf-8"><style>
|
||||||
|
|
||||||
|
html, body {
|
||||||
|
color: #ccc;
|
||||||
|
background: #222;
|
||||||
|
font-family: sans-serif;
|
||||||
|
}
|
||||||
|
a {
|
||||||
|
color: #fc5;
|
||||||
|
}
|
||||||
|
td, th {
|
||||||
|
padding: .2em .5em;
|
||||||
|
border: 1px solid #999;
|
||||||
|
border-width: 0 1px 1px 0;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
td:nth-child(1),
|
||||||
|
td:nth-child(2),
|
||||||
|
td:nth-child(3) {
|
||||||
|
font-family: monospace, monospace;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
tr:first-child {
|
||||||
|
position: sticky;
|
||||||
|
top: -1px;
|
||||||
|
}
|
||||||
|
th {
|
||||||
|
background: #222;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
</style></head><body><table><tr>
|
||||||
|
<th>wark</th>
|
||||||
|
<th>time</th>
|
||||||
|
<th>size</th>
|
||||||
|
<th>who</th>
|
||||||
|
<th>link</th>
|
||||||
|
</tr>"""
|
||||||
|
)
|
||||||
|
|
||||||
|
db_path = ".hist/up2k.db"
|
||||||
|
conn = sqlite3.connect(db_path)
|
||||||
|
q = r"pragma table_info(up)"
|
||||||
|
inf = conn.execute(q).fetchall()
|
||||||
|
cols = [x[1] for x in inf]
|
||||||
|
print("<!-- " + str(cols) + " -->")
|
||||||
|
# ['w', 'mt', 'sz', 'rd', 'fn', 'ip', 'at']
|
||||||
|
|
||||||
|
q = r"select * from up order by case when at > 0 then at else mt end"
|
||||||
|
for w, mt, sz, rd, fn, ip, at in conn.execute(q):
|
||||||
|
link = "/".join([s3dec(x) for x in [rd, fn] if x])
|
||||||
|
if fn.startswith("put-") and sz < 4096:
|
||||||
|
try:
|
||||||
|
with open(link, "rb") as f:
|
||||||
|
txt = f.read().decode("utf-8", "replace")
|
||||||
|
except:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if txt.startswith("msg="):
|
||||||
|
txt = txt.encode("utf-8", "replace")
|
||||||
|
txt = unquote(txt.replace(b"+", b" "))
|
||||||
|
link = txt.decode("utf-8")[4:]
|
||||||
|
|
||||||
|
sz = "{:,}".format(sz)
|
||||||
|
v = [
|
||||||
|
w[:16],
|
||||||
|
datetime.utcfromtimestamp(at if at > 0 else mt).strftime(
|
||||||
|
"%Y-%m-%d %H:%M:%S"
|
||||||
|
),
|
||||||
|
sz,
|
||||||
|
imap.get(ip, ip),
|
||||||
|
]
|
||||||
|
|
||||||
|
row = "<tr>\n "
|
||||||
|
row += "\n ".join(["<td>{}</th>".format(x) for x in v])
|
||||||
|
row += '\n <td><a href="{}">{}</a></td>'.format(link, html_escape(link))
|
||||||
|
row += "\n</tr>"
|
||||||
|
print(row)
|
||||||
|
|
||||||
|
print("</table></body></html>")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
72
bin/prisonparty.sh
Normal file → Executable file
72
bin/prisonparty.sh
Normal file → Executable file
@@ -4,17 +4,23 @@ set -e
|
|||||||
# runs copyparty (or any other program really) in a chroot
|
# runs copyparty (or any other program really) in a chroot
|
||||||
#
|
#
|
||||||
# assumption: these directories, and everything within, are owned by root
|
# assumption: these directories, and everything within, are owned by root
|
||||||
sysdirs=( /bin /lib /lib32 /lib64 /sbin /usr )
|
sysdirs=( /bin /lib /lib32 /lib64 /sbin /usr /etc/alternatives )
|
||||||
|
|
||||||
|
|
||||||
# error-handler
|
# error-handler
|
||||||
help() { cat <<'EOF'
|
help() { cat <<'EOF'
|
||||||
|
|
||||||
usage:
|
usage:
|
||||||
./prisonparty.sh <ROOTDIR> <UID> <GID> [VOLDIR [VOLDIR...]] -- copyparty-sfx.py [...]"
|
./prisonparty.sh <ROOTDIR> <UID> <GID> [VOLDIR [VOLDIR...]] -- python3 copyparty-sfx.py [...]
|
||||||
|
|
||||||
example:
|
example:
|
||||||
./prisonparty.sh /var/lib/copyparty-jail 1000 1000 /mnt/nas/music -- copyparty-sfx.py -v /mnt/nas/music::rwmd"
|
./prisonparty.sh /var/lib/copyparty-jail 1000 1000 /mnt/nas/music -- python3 copyparty-sfx.py -v /mnt/nas/music::rwmd
|
||||||
|
|
||||||
|
example for running straight from source (instead of using an sfx):
|
||||||
|
PYTHONPATH=$PWD ./prisonparty.sh /var/lib/copyparty-jail 1000 1000 /mnt/nas/music -- python3 -um copyparty -v /mnt/nas/music::rwmd
|
||||||
|
|
||||||
|
note that if you have python modules installed as --user (such as bpm/key detectors),
|
||||||
|
you should add /home/foo/.local as a VOLDIR
|
||||||
|
|
||||||
EOF
|
EOF
|
||||||
exit 1
|
exit 1
|
||||||
@@ -35,10 +41,20 @@ while true; do
|
|||||||
vols+=( "$(realpath "$v")" )
|
vols+=( "$(realpath "$v")" )
|
||||||
done
|
done
|
||||||
pybin="$1"; shift
|
pybin="$1"; shift
|
||||||
pybin="$(realpath "$pybin")"
|
pybin="$(command -v "$pybin")"
|
||||||
|
pyarg=
|
||||||
|
while true; do
|
||||||
|
v="$1"
|
||||||
|
[ "${v:0:1}" = - ] || break
|
||||||
|
pyarg="$pyarg $v"
|
||||||
|
shift
|
||||||
|
done
|
||||||
cpp="$1"; shift
|
cpp="$1"; shift
|
||||||
cpp="$(realpath "$cpp")"
|
[ -d "$cpp" ] && cppdir="$PWD" || {
|
||||||
cppdir="$(dirname "$cpp")"
|
# sfx, not module
|
||||||
|
cpp="$(realpath "$cpp")"
|
||||||
|
cppdir="$(dirname "$cpp")"
|
||||||
|
}
|
||||||
trap - EXIT
|
trap - EXIT
|
||||||
|
|
||||||
|
|
||||||
@@ -60,11 +76,10 @@ echo
|
|||||||
|
|
||||||
# remove any trailing slashes
|
# remove any trailing slashes
|
||||||
jail="${jail%/}"
|
jail="${jail%/}"
|
||||||
cppdir="${cppdir%/}"
|
|
||||||
|
|
||||||
|
|
||||||
# bind-mount system directories and volumes
|
# bind-mount system directories and volumes
|
||||||
printf '%s\n' "${sysdirs[@]}" "${vols[@]}" | LC_ALL=C sort |
|
printf '%s\n' "${sysdirs[@]}" "${vols[@]}" | sed -r 's`/$``' | LC_ALL=C sort | uniq |
|
||||||
while IFS= read -r v; do
|
while IFS= read -r v; do
|
||||||
[ -e "$v" ] || {
|
[ -e "$v" ] || {
|
||||||
# printf '\033[1;31mfolder does not exist:\033[0m %s\n' "/$v"
|
# printf '\033[1;31mfolder does not exist:\033[0m %s\n' "/$v"
|
||||||
@@ -72,6 +87,7 @@ while IFS= read -r v; do
|
|||||||
}
|
}
|
||||||
i1=$(stat -c%D.%i "$v" 2>/dev/null || echo a)
|
i1=$(stat -c%D.%i "$v" 2>/dev/null || echo a)
|
||||||
i2=$(stat -c%D.%i "$jail$v" 2>/dev/null || echo b)
|
i2=$(stat -c%D.%i "$jail$v" 2>/dev/null || echo b)
|
||||||
|
# echo "v [$v] i1 [$i1] i2 [$i2]"
|
||||||
[ $i1 = $i2 ] && continue
|
[ $i1 = $i2 ] && continue
|
||||||
|
|
||||||
mkdir -p "$jail$v"
|
mkdir -p "$jail$v"
|
||||||
@@ -79,21 +95,37 @@ while IFS= read -r v; do
|
|||||||
done
|
done
|
||||||
|
|
||||||
|
|
||||||
|
cln() {
|
||||||
|
rv=$?
|
||||||
|
wait -f -p rv $p || true
|
||||||
|
cd /
|
||||||
|
echo "stopping chroot..."
|
||||||
|
lsof "$jail" | grep -F "$jail" &&
|
||||||
|
echo "chroot is in use; will not unmount" ||
|
||||||
|
{
|
||||||
|
mount | grep -F " on $jail" |
|
||||||
|
awk '{sub(/ type .*/,"");sub(/.* on /,"");print}' |
|
||||||
|
LC_ALL=C sort -r | tee /dev/stderr | tr '\n' '\0' | xargs -r0 umount
|
||||||
|
}
|
||||||
|
exit $rv
|
||||||
|
}
|
||||||
|
trap cln EXIT
|
||||||
|
|
||||||
|
|
||||||
# create a tmp
|
# create a tmp
|
||||||
mkdir -p "$jail/tmp"
|
mkdir -p "$jail/tmp"
|
||||||
chmod 777 "$jail/tmp"
|
chmod 777 "$jail/tmp"
|
||||||
|
|
||||||
|
|
||||||
# run copyparty
|
# run copyparty
|
||||||
/sbin/chroot --userspec=$uid:$gid "$jail" "$pybin" "$cpp" "$@" && rv=0 || rv=$?
|
export HOME=$(getent passwd $uid | cut -d: -f6)
|
||||||
|
export USER=$(getent passwd $uid | cut -d: -f1)
|
||||||
|
export LOGNAME="$USER"
|
||||||
# cleanup if not in use
|
#echo "pybin [$pybin]"
|
||||||
lsof "$jail" | grep -qF "$jail" &&
|
#echo "pyarg [$pyarg]"
|
||||||
echo "chroot is in use, will not cleanup" ||
|
#echo "cpp [$cpp]"
|
||||||
{
|
chroot --userspec=$uid:$gid "$jail" "$pybin" $pyarg "$cpp" "$@" &
|
||||||
mount | grep -qF " on $jail" |
|
p=$!
|
||||||
awk '{sub(/ type .*/,"");sub(/.* on /,"");print}' |
|
trap 'kill -USR1 $p' USR1
|
||||||
LC_ALL=C sort -r | tee /dev/stderr | tr '\n' '\0' | xargs -r0 umount
|
trap 'kill $p' INT TERM
|
||||||
}
|
wait
|
||||||
exit $rv
|
|
||||||
|
|||||||
99
bin/unforget.py
Executable file
99
bin/unforget.py
Executable file
@@ -0,0 +1,99 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
"""
|
||||||
|
unforget.py: rebuild db from logfiles
|
||||||
|
2022-09-07, v0.1, ed <irc.rizon.net>, MIT-Licensed
|
||||||
|
https://github.com/9001/copyparty/blob/hovudstraum/bin/unforget.py
|
||||||
|
|
||||||
|
only makes sense if running copyparty with --no-forget
|
||||||
|
(e.g. immediately shifting uploads to other storage)
|
||||||
|
|
||||||
|
usage:
|
||||||
|
xz -d < log | ./unforget.py .hist/up2k.db
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
import re
|
||||||
|
import sys
|
||||||
|
import json
|
||||||
|
import base64
|
||||||
|
import sqlite3
|
||||||
|
import argparse
|
||||||
|
|
||||||
|
|
||||||
|
FS_ENCODING = sys.getfilesystemencoding()
|
||||||
|
|
||||||
|
|
||||||
|
class APF(argparse.ArgumentDefaultsHelpFormatter, argparse.RawDescriptionHelpFormatter):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
mem_cur = sqlite3.connect(":memory:").cursor()
|
||||||
|
mem_cur.execute(r"create table a (b text)")
|
||||||
|
|
||||||
|
|
||||||
|
def s3enc(rd: str, fn: str) -> tuple[str, str]:
|
||||||
|
ret: list[str] = []
|
||||||
|
for v in [rd, fn]:
|
||||||
|
try:
|
||||||
|
mem_cur.execute("select * from a where b = ?", (v,))
|
||||||
|
ret.append(v)
|
||||||
|
except:
|
||||||
|
wtf8 = v.encode(FS_ENCODING, "surrogateescape")
|
||||||
|
ret.append("//" + base64.urlsafe_b64encode(wtf8).decode("ascii"))
|
||||||
|
|
||||||
|
return ret[0], ret[1]
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
ap = argparse.ArgumentParser()
|
||||||
|
ap.add_argument("db")
|
||||||
|
ar = ap.parse_args()
|
||||||
|
|
||||||
|
db = sqlite3.connect(ar.db).cursor()
|
||||||
|
ptn_times = re.compile(r"no more chunks, setting times \(([0-9]+)")
|
||||||
|
at = 0
|
||||||
|
ctr = 0
|
||||||
|
|
||||||
|
for ln in [x.decode("utf-8", "replace").rstrip() for x in sys.stdin.buffer]:
|
||||||
|
if "no more chunks, setting times (" in ln:
|
||||||
|
m = ptn_times.search(ln)
|
||||||
|
if m:
|
||||||
|
at = int(m.group(1))
|
||||||
|
|
||||||
|
if '"hash": []' in ln:
|
||||||
|
try:
|
||||||
|
ofs = ln.find("{")
|
||||||
|
j = json.loads(ln[ofs:])
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
w = j["wark"]
|
||||||
|
if db.execute("select w from up where w = ?", (w,)).fetchone():
|
||||||
|
continue
|
||||||
|
|
||||||
|
# PYTHONPATH=/home/ed/dev/copyparty/ python3 -m copyparty -e2dsa -v foo:foo:rwmd,ed -aed:wark --no-forget
|
||||||
|
# 05:34:43.845 127.0.0.1 42496 no more chunks, setting times (1662528883, 1658001882)
|
||||||
|
# 05:34:43.863 127.0.0.1 42496 {"name": "f\"2", "purl": "/foo/bar/baz/", "size": 1674, "lmod": 1658001882, "sprs": true, "hash": [], "wark": "LKIWpp2jEAh9dH3fu-DobuURFGEKlODXDGTpZ1otMhUg"}
|
||||||
|
# | w | mt | sz | rd | fn | ip | at |
|
||||||
|
# | LKIWpp2jEAh9dH3fu-DobuURFGEKlODXDGTpZ1otMhUg | 1658001882 | 1674 | bar/baz | f"2 | 127.0.0.1 | 1662528883 |
|
||||||
|
|
||||||
|
rd, fn = s3enc(j["purl"].strip("/"), j["name"])
|
||||||
|
ip = ln.split(" ")[1].split("m")[-1]
|
||||||
|
|
||||||
|
q = "insert into up values (?,?,?,?,?,?,?)"
|
||||||
|
v = (w, int(j["lmod"]), int(j["size"]), rd, fn, ip, at)
|
||||||
|
db.execute(q, v)
|
||||||
|
ctr += 1
|
||||||
|
if ctr % 1024 == 1023:
|
||||||
|
print(f"{ctr} commit...")
|
||||||
|
db.connection.commit()
|
||||||
|
|
||||||
|
if ctr:
|
||||||
|
db.connection.commit()
|
||||||
|
|
||||||
|
print(f"unforgot {ctr} files")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
576
bin/up2k.py
576
bin/up2k.py
@@ -3,14 +3,12 @@ from __future__ import print_function, unicode_literals
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
up2k.py: upload to copyparty
|
up2k.py: upload to copyparty
|
||||||
2021-11-28, v0.13, ed <irc.rizon.net>, MIT-Licensed
|
2023-01-13, v1.2, ed <irc.rizon.net>, MIT-Licensed
|
||||||
https://github.com/9001/copyparty/blob/hovudstraum/bin/up2k.py
|
https://github.com/9001/copyparty/blob/hovudstraum/bin/up2k.py
|
||||||
|
|
||||||
- dependencies: requests
|
- dependencies: requests
|
||||||
- supports python 2.6, 2.7, and 3.3 through 3.10
|
- supports python 2.6, 2.7, and 3.3 through 3.12
|
||||||
|
- if something breaks just try again and it'll autoresume
|
||||||
- almost zero error-handling
|
|
||||||
- but if something breaks just try again and it'll autoresume
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
import os
|
import os
|
||||||
@@ -22,19 +20,37 @@ import atexit
|
|||||||
import signal
|
import signal
|
||||||
import base64
|
import base64
|
||||||
import hashlib
|
import hashlib
|
||||||
import argparse
|
|
||||||
import platform
|
import platform
|
||||||
import threading
|
import threading
|
||||||
import requests
|
|
||||||
import datetime
|
import datetime
|
||||||
|
|
||||||
|
try:
|
||||||
|
import argparse
|
||||||
|
except:
|
||||||
|
m = "\n ERROR: need 'argparse'; download it here:\n https://github.com/ThomasWaldmann/argparse/raw/master/argparse.py\n"
|
||||||
|
print(m)
|
||||||
|
raise
|
||||||
|
|
||||||
|
try:
|
||||||
|
import requests
|
||||||
|
except ImportError:
|
||||||
|
if sys.version_info > (2, 7):
|
||||||
|
m = "\nERROR: need 'requests'; please run this command:\n {0} -m pip install --user requests\n"
|
||||||
|
else:
|
||||||
|
m = "requests/2.18.4 urllib3/1.23 chardet/3.0.4 certifi/2020.4.5.1 idna/2.7"
|
||||||
|
m = [" https://pypi.org/project/" + x + "/#files" for x in m.split()]
|
||||||
|
m = "\n ERROR: need these:\n" + "\n".join(m) + "\n"
|
||||||
|
m += "\n for f in *.whl; do unzip $f; done; rm -r *.dist-info\n"
|
||||||
|
|
||||||
|
print(m.format(sys.executable))
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
|
||||||
# from copyparty/__init__.py
|
# from copyparty/__init__.py
|
||||||
PY2 = sys.version_info[0] == 2
|
PY2 = sys.version_info < (3,)
|
||||||
if PY2:
|
if PY2:
|
||||||
from Queue import Queue
|
from Queue import Queue
|
||||||
from urllib import unquote
|
from urllib import quote, unquote
|
||||||
from urllib import quote
|
|
||||||
|
|
||||||
sys.dont_write_bytecode = True
|
sys.dont_write_bytecode = True
|
||||||
bytes = str
|
bytes = str
|
||||||
@@ -51,6 +67,14 @@ VT100 = platform.system() != "Windows"
|
|||||||
req_ses = requests.Session()
|
req_ses = requests.Session()
|
||||||
|
|
||||||
|
|
||||||
|
class Daemon(threading.Thread):
|
||||||
|
def __init__(self, target, name=None, a=None):
|
||||||
|
# type: (Any, Any, Any) -> None
|
||||||
|
threading.Thread.__init__(self, target=target, args=a or (), name=name)
|
||||||
|
self.daemon = True
|
||||||
|
self.start()
|
||||||
|
|
||||||
|
|
||||||
class File(object):
|
class File(object):
|
||||||
"""an up2k upload task; represents a single file"""
|
"""an up2k upload task; represents a single file"""
|
||||||
|
|
||||||
@@ -68,6 +92,7 @@ class File(object):
|
|||||||
self.kchunks = {} # type: dict[str, tuple[int, int]] # hash: [ ofs, sz ]
|
self.kchunks = {} # type: dict[str, tuple[int, int]] # hash: [ ofs, sz ]
|
||||||
|
|
||||||
# set by handshake
|
# set by handshake
|
||||||
|
self.recheck = False # duplicate; redo handshake after all files done
|
||||||
self.ucids = [] # type: list[str] # chunks which need to be uploaded
|
self.ucids = [] # type: list[str] # chunks which need to be uploaded
|
||||||
self.wark = None # type: str
|
self.wark = None # type: str
|
||||||
self.url = None # type: str
|
self.url = None # type: str
|
||||||
@@ -76,15 +101,15 @@ class File(object):
|
|||||||
self.up_b = 0 # type: int
|
self.up_b = 0 # type: int
|
||||||
self.up_c = 0 # type: int
|
self.up_c = 0 # type: int
|
||||||
|
|
||||||
# m = "size({}) lmod({}) top({}) rel({}) abs({}) name({})\n"
|
# t = "size({}) lmod({}) top({}) rel({}) abs({}) name({})\n"
|
||||||
# eprint(m.format(self.size, self.lmod, self.top, self.rel, self.abs, self.name))
|
# eprint(t.format(self.size, self.lmod, self.top, self.rel, self.abs, self.name))
|
||||||
|
|
||||||
|
|
||||||
class FileSlice(object):
|
class FileSlice(object):
|
||||||
"""file-like object providing a fixed window into a file"""
|
"""file-like object providing a fixed window into a file"""
|
||||||
|
|
||||||
def __init__(self, file, cid):
|
def __init__(self, file, cid):
|
||||||
# type: (File, str) -> FileSlice
|
# type: (File, str) -> None
|
||||||
|
|
||||||
self.car, self.len = file.kchunks[cid]
|
self.car, self.len = file.kchunks[cid]
|
||||||
self.cdr = self.car + self.len
|
self.cdr = self.car + self.len
|
||||||
@@ -125,6 +150,86 @@ class FileSlice(object):
|
|||||||
return ret
|
return ret
|
||||||
|
|
||||||
|
|
||||||
|
class MTHash(object):
|
||||||
|
def __init__(self, cores):
|
||||||
|
self.f = None
|
||||||
|
self.sz = 0
|
||||||
|
self.csz = 0
|
||||||
|
self.omutex = threading.Lock()
|
||||||
|
self.imutex = threading.Lock()
|
||||||
|
self.work_q = Queue()
|
||||||
|
self.done_q = Queue()
|
||||||
|
self.thrs = []
|
||||||
|
for _ in range(cores):
|
||||||
|
self.thrs.append(Daemon(self.worker))
|
||||||
|
|
||||||
|
def hash(self, f, fsz, chunksz, pcb=None, pcb_opaque=None):
|
||||||
|
with self.omutex:
|
||||||
|
self.f = f
|
||||||
|
self.sz = fsz
|
||||||
|
self.csz = chunksz
|
||||||
|
|
||||||
|
chunks = {}
|
||||||
|
nchunks = int(math.ceil(fsz / chunksz))
|
||||||
|
for nch in range(nchunks):
|
||||||
|
self.work_q.put(nch)
|
||||||
|
|
||||||
|
ex = ""
|
||||||
|
for nch in range(nchunks):
|
||||||
|
qe = self.done_q.get()
|
||||||
|
try:
|
||||||
|
nch, dig, ofs, csz = qe
|
||||||
|
chunks[nch] = [dig, ofs, csz]
|
||||||
|
except:
|
||||||
|
ex = ex or qe
|
||||||
|
|
||||||
|
if pcb:
|
||||||
|
pcb(pcb_opaque, chunksz * nch)
|
||||||
|
|
||||||
|
if ex:
|
||||||
|
raise Exception(ex)
|
||||||
|
|
||||||
|
ret = []
|
||||||
|
for n in range(nchunks):
|
||||||
|
ret.append(chunks[n])
|
||||||
|
|
||||||
|
self.f = None
|
||||||
|
self.csz = 0
|
||||||
|
self.sz = 0
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def worker(self):
|
||||||
|
while True:
|
||||||
|
ofs = self.work_q.get()
|
||||||
|
try:
|
||||||
|
v = self.hash_at(ofs)
|
||||||
|
except Exception as ex:
|
||||||
|
v = str(ex)
|
||||||
|
|
||||||
|
self.done_q.put(v)
|
||||||
|
|
||||||
|
def hash_at(self, nch):
|
||||||
|
f = self.f
|
||||||
|
ofs = ofs0 = nch * self.csz
|
||||||
|
hashobj = hashlib.sha512()
|
||||||
|
chunk_sz = chunk_rem = min(self.csz, self.sz - ofs)
|
||||||
|
while chunk_rem > 0:
|
||||||
|
with self.imutex:
|
||||||
|
f.seek(ofs)
|
||||||
|
buf = f.read(min(chunk_rem, 1024 * 1024 * 12))
|
||||||
|
|
||||||
|
if not buf:
|
||||||
|
raise Exception("EOF at " + str(ofs))
|
||||||
|
|
||||||
|
hashobj.update(buf)
|
||||||
|
chunk_rem -= len(buf)
|
||||||
|
ofs += len(buf)
|
||||||
|
|
||||||
|
digest = hashobj.digest()[:33]
|
||||||
|
digest = base64.urlsafe_b64encode(digest).decode("utf-8")
|
||||||
|
return nch, digest, ofs0, chunk_sz
|
||||||
|
|
||||||
|
|
||||||
_print = print
|
_print = print
|
||||||
|
|
||||||
|
|
||||||
@@ -150,18 +255,16 @@ if not VT100:
|
|||||||
|
|
||||||
|
|
||||||
def termsize():
|
def termsize():
|
||||||
import os
|
|
||||||
|
|
||||||
env = os.environ
|
env = os.environ
|
||||||
|
|
||||||
def ioctl_GWINSZ(fd):
|
def ioctl_GWINSZ(fd):
|
||||||
try:
|
try:
|
||||||
import fcntl, termios, struct, os
|
import fcntl, termios, struct
|
||||||
|
|
||||||
cr = struct.unpack("hh", fcntl.ioctl(fd, termios.TIOCGWINSZ, "1234"))
|
r = struct.unpack(b"hh", fcntl.ioctl(fd, termios.TIOCGWINSZ, b"AAAA"))
|
||||||
|
return r[::-1]
|
||||||
except:
|
except:
|
||||||
return
|
return None
|
||||||
return cr
|
|
||||||
|
|
||||||
cr = ioctl_GWINSZ(0) or ioctl_GWINSZ(1) or ioctl_GWINSZ(2)
|
cr = ioctl_GWINSZ(0) or ioctl_GWINSZ(1) or ioctl_GWINSZ(2)
|
||||||
if not cr:
|
if not cr:
|
||||||
@@ -171,12 +274,11 @@ def termsize():
|
|||||||
os.close(fd)
|
os.close(fd)
|
||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
if not cr:
|
|
||||||
try:
|
try:
|
||||||
cr = (env["LINES"], env["COLUMNS"])
|
return cr or (int(env["COLUMNS"]), int(env["LINES"]))
|
||||||
except:
|
except:
|
||||||
cr = (25, 80)
|
return 80, 25
|
||||||
return int(cr[1]), int(cr[0])
|
|
||||||
|
|
||||||
|
|
||||||
class CTermsize(object):
|
class CTermsize(object):
|
||||||
@@ -191,9 +293,7 @@ class CTermsize(object):
|
|||||||
except:
|
except:
|
||||||
return
|
return
|
||||||
|
|
||||||
thr = threading.Thread(target=self.worker)
|
Daemon(self.worker)
|
||||||
thr.daemon = True
|
|
||||||
thr.start()
|
|
||||||
|
|
||||||
def worker(self):
|
def worker(self):
|
||||||
while True:
|
while True:
|
||||||
@@ -217,8 +317,8 @@ class CTermsize(object):
|
|||||||
eprint("\033[s\033[r\033[u")
|
eprint("\033[s\033[r\033[u")
|
||||||
else:
|
else:
|
||||||
self.g = 1 + self.h - margin
|
self.g = 1 + self.h - margin
|
||||||
m = "{0}\033[{1}A".format("\n" * margin, margin)
|
t = "{0}\033[{1}A".format("\n" * margin, margin)
|
||||||
eprint("{0}\033[s\033[1;{1}r\033[u".format(m, self.g - 1))
|
eprint("{0}\033[s\033[1;{1}r\033[u".format(t, self.g - 1))
|
||||||
|
|
||||||
|
|
||||||
ss = CTermsize()
|
ss = CTermsize()
|
||||||
@@ -231,8 +331,8 @@ def _scd(err, top):
|
|||||||
abspath = os.path.join(top, fh.name)
|
abspath = os.path.join(top, fh.name)
|
||||||
try:
|
try:
|
||||||
yield [abspath, fh.stat()]
|
yield [abspath, fh.stat()]
|
||||||
except:
|
except Exception as ex:
|
||||||
err.append(abspath)
|
err.append((abspath, str(ex)))
|
||||||
|
|
||||||
|
|
||||||
def _lsd(err, top):
|
def _lsd(err, top):
|
||||||
@@ -241,40 +341,49 @@ def _lsd(err, top):
|
|||||||
abspath = os.path.join(top, name)
|
abspath = os.path.join(top, name)
|
||||||
try:
|
try:
|
||||||
yield [abspath, os.stat(abspath)]
|
yield [abspath, os.stat(abspath)]
|
||||||
except:
|
except Exception as ex:
|
||||||
err.append(abspath)
|
err.append((abspath, str(ex)))
|
||||||
|
|
||||||
|
|
||||||
if hasattr(os, "scandir"):
|
if hasattr(os, "scandir") and sys.version_info > (3, 6):
|
||||||
statdir = _scd
|
statdir = _scd
|
||||||
else:
|
else:
|
||||||
statdir = _lsd
|
statdir = _lsd
|
||||||
|
|
||||||
|
|
||||||
def walkdir(err, top):
|
def walkdir(err, top, seen):
|
||||||
"""recursive statdir"""
|
"""recursive statdir"""
|
||||||
|
atop = os.path.abspath(os.path.realpath(top))
|
||||||
|
if atop in seen:
|
||||||
|
err.append((top, "recursive-symlink"))
|
||||||
|
return
|
||||||
|
|
||||||
|
seen = seen[:] + [atop]
|
||||||
for ap, inf in sorted(statdir(err, top)):
|
for ap, inf in sorted(statdir(err, top)):
|
||||||
|
yield ap, inf
|
||||||
if stat.S_ISDIR(inf.st_mode):
|
if stat.S_ISDIR(inf.st_mode):
|
||||||
try:
|
try:
|
||||||
for x in walkdir(err, ap):
|
for x in walkdir(err, ap, seen):
|
||||||
yield x
|
yield x
|
||||||
except:
|
except Exception as ex:
|
||||||
err.append(ap)
|
err.append((ap, str(ex)))
|
||||||
else:
|
|
||||||
yield ap, inf
|
|
||||||
|
|
||||||
|
|
||||||
def walkdirs(err, tops):
|
def walkdirs(err, tops):
|
||||||
"""recursive statdir for a list of tops, yields [top, relpath, stat]"""
|
"""recursive statdir for a list of tops, yields [top, relpath, stat]"""
|
||||||
sep = "{0}".format(os.sep).encode("ascii")
|
sep = "{0}".format(os.sep).encode("ascii")
|
||||||
for top in tops:
|
for top in tops:
|
||||||
|
isdir = os.path.isdir(top)
|
||||||
if top[-1:] == sep:
|
if top[-1:] == sep:
|
||||||
stop = top.rstrip(sep)
|
stop = top.rstrip(sep)
|
||||||
|
yield stop, b"", os.stat(stop)
|
||||||
else:
|
else:
|
||||||
stop = os.path.dirname(top)
|
stop, dn = os.path.split(top)
|
||||||
|
if isdir:
|
||||||
|
yield stop, dn, os.stat(stop)
|
||||||
|
|
||||||
if os.path.isdir(top):
|
if isdir:
|
||||||
for ap, inf in walkdir(err, top):
|
for ap, inf in walkdir(err, top, []):
|
||||||
yield stop, ap[len(stop) :].lstrip(sep), inf
|
yield stop, ap[len(stop) :].lstrip(sep), inf
|
||||||
else:
|
else:
|
||||||
d, n = top.rsplit(sep, 1)
|
d, n = top.rsplit(sep, 1)
|
||||||
@@ -315,7 +424,7 @@ def up2k_chunksize(filesize):
|
|||||||
while True:
|
while True:
|
||||||
for mul in [1, 2]:
|
for mul in [1, 2]:
|
||||||
nchunks = math.ceil(filesize * 1.0 / chunksize)
|
nchunks = math.ceil(filesize * 1.0 / chunksize)
|
||||||
if nchunks <= 256 or chunksize >= 32 * 1024 * 1024:
|
if nchunks <= 256 or (chunksize >= 32 * 1024 * 1024 and nchunks < 4096):
|
||||||
return chunksize
|
return chunksize
|
||||||
|
|
||||||
chunksize += stepsize
|
chunksize += stepsize
|
||||||
@@ -323,8 +432,8 @@ def up2k_chunksize(filesize):
|
|||||||
|
|
||||||
|
|
||||||
# mostly from copyparty/up2k.py
|
# mostly from copyparty/up2k.py
|
||||||
def get_hashlist(file, pcb):
|
def get_hashlist(file, pcb, mth):
|
||||||
# type: (File, any) -> None
|
# type: (File, any, any) -> None
|
||||||
"""generates the up2k hashlist from file contents, inserts it into `file`"""
|
"""generates the up2k hashlist from file contents, inserts it into `file`"""
|
||||||
|
|
||||||
chunk_sz = up2k_chunksize(file.size)
|
chunk_sz = up2k_chunksize(file.size)
|
||||||
@@ -332,7 +441,12 @@ def get_hashlist(file, pcb):
|
|||||||
file_ofs = 0
|
file_ofs = 0
|
||||||
ret = []
|
ret = []
|
||||||
with open(file.abs, "rb", 512 * 1024) as f:
|
with open(file.abs, "rb", 512 * 1024) as f:
|
||||||
|
if mth and file.size >= 1024 * 512:
|
||||||
|
ret = mth.hash(f, file.size, chunk_sz, pcb, file)
|
||||||
|
file_rem = 0
|
||||||
|
|
||||||
while file_rem > 0:
|
while file_rem > 0:
|
||||||
|
# same as `hash_at` except for `imutex` / bufsz
|
||||||
hashobj = hashlib.sha512()
|
hashobj = hashlib.sha512()
|
||||||
chunk_sz = chunk_rem = min(chunk_sz, file_rem)
|
chunk_sz = chunk_rem = min(chunk_sz, file_rem)
|
||||||
while chunk_rem > 0:
|
while chunk_rem > 0:
|
||||||
@@ -359,14 +473,17 @@ def get_hashlist(file, pcb):
|
|||||||
file.kchunks[k] = [v1, v2]
|
file.kchunks[k] = [v1, v2]
|
||||||
|
|
||||||
|
|
||||||
def handshake(req_ses, url, file, pw, search):
|
def handshake(ar, file, search):
|
||||||
# type: (requests.Session, str, File, any, bool) -> List[str]
|
# type: (argparse.Namespace, File, bool) -> tuple[list[str], bool]
|
||||||
"""
|
"""
|
||||||
performs a handshake with the server; reply is:
|
performs a handshake with the server; reply is:
|
||||||
if search, a list of search results
|
if search, a list of search results
|
||||||
otherwise, a list of chunks to upload
|
otherwise, a list of chunks to upload
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
url = ar.url
|
||||||
|
pw = ar.a
|
||||||
|
|
||||||
req = {
|
req = {
|
||||||
"hash": [x[0] for x in file.cids],
|
"hash": [x[0] for x in file.cids],
|
||||||
"name": file.name,
|
"name": file.name,
|
||||||
@@ -375,22 +492,43 @@ def handshake(req_ses, url, file, pw, search):
|
|||||||
}
|
}
|
||||||
if search:
|
if search:
|
||||||
req["srch"] = 1
|
req["srch"] = 1
|
||||||
|
elif ar.dr:
|
||||||
|
req["replace"] = True
|
||||||
|
|
||||||
headers = {"Content-Type": "text/plain"} # wtf ed
|
headers = {"Content-Type": "text/plain"} # <=1.5.1 compat
|
||||||
if pw:
|
if pw:
|
||||||
headers["Cookie"] = "=".join(["cppwd", pw])
|
headers["Cookie"] = "=".join(["cppwd", pw])
|
||||||
|
|
||||||
|
file.recheck = False
|
||||||
if file.url:
|
if file.url:
|
||||||
url = file.url
|
url = file.url
|
||||||
elif b"/" in file.rel:
|
elif b"/" in file.rel:
|
||||||
url += quotep(file.rel.rsplit(b"/", 1)[0]).decode("utf-8", "replace")
|
url += quotep(file.rel.rsplit(b"/", 1)[0]).decode("utf-8", "replace")
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
|
sc = 600
|
||||||
|
txt = ""
|
||||||
try:
|
try:
|
||||||
r = req_ses.post(url, headers=headers, json=req)
|
r = req_ses.post(url, headers=headers, json=req)
|
||||||
break
|
sc = r.status_code
|
||||||
except:
|
txt = r.text
|
||||||
eprint("handshake failed, retrying: {0}\n".format(file.name))
|
if sc < 400:
|
||||||
|
break
|
||||||
|
|
||||||
|
raise Exception("http {0}: {1}".format(sc, txt))
|
||||||
|
|
||||||
|
except Exception as ex:
|
||||||
|
em = str(ex).split("SSLError(")[-1].split("\nURL: ")[0].strip()
|
||||||
|
|
||||||
|
if sc == 422 or "<pre>partial upload exists at a different" in txt:
|
||||||
|
file.recheck = True
|
||||||
|
return [], False
|
||||||
|
elif sc == 409 or "<pre>upload rejected, file already exists" in txt:
|
||||||
|
return [], False
|
||||||
|
elif "<pre>you don't have " in txt:
|
||||||
|
raise
|
||||||
|
|
||||||
|
eprint("handshake failed, retrying: {0}\n {1}\n\n".format(file.name, em))
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@@ -399,7 +537,7 @@ def handshake(req_ses, url, file, pw, search):
|
|||||||
raise Exception(r.text)
|
raise Exception(r.text)
|
||||||
|
|
||||||
if search:
|
if search:
|
||||||
return r["hits"]
|
return r["hits"], False
|
||||||
|
|
||||||
try:
|
try:
|
||||||
pre, url = url.split("://")
|
pre, url = url.split("://")
|
||||||
@@ -411,11 +549,11 @@ def handshake(req_ses, url, file, pw, search):
|
|||||||
file.name = r["name"]
|
file.name = r["name"]
|
||||||
file.wark = r["wark"]
|
file.wark = r["wark"]
|
||||||
|
|
||||||
return r["hash"]
|
return r["hash"], r["sprs"]
|
||||||
|
|
||||||
|
|
||||||
def upload(req_ses, file, cid, pw):
|
def upload(file, cid, pw):
|
||||||
# type: (requests.Session, File, str, any) -> None
|
# type: (File, str, str) -> None
|
||||||
"""upload one specific chunk, `cid` (a chunk-hash)"""
|
"""upload one specific chunk, `cid` (a chunk-hash)"""
|
||||||
|
|
||||||
headers = {
|
headers = {
|
||||||
@@ -437,51 +575,52 @@ def upload(req_ses, file, cid, pw):
|
|||||||
f.f.close()
|
f.f.close()
|
||||||
|
|
||||||
|
|
||||||
class Daemon(threading.Thread):
|
|
||||||
def __init__(self, *a, **ka):
|
|
||||||
threading.Thread.__init__(self, *a, **ka)
|
|
||||||
self.daemon = True
|
|
||||||
|
|
||||||
|
|
||||||
class Ctl(object):
|
class Ctl(object):
|
||||||
"""
|
"""
|
||||||
this will be the coordinator which runs everything in parallel
|
the coordinator which runs everything in parallel
|
||||||
(hashing, handshakes, uploads) but right now it's p dumb
|
(hashing, handshakes, uploads)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, ar):
|
def _scan(self):
|
||||||
self.ar = ar
|
ar = self.ar
|
||||||
ar.files = [
|
|
||||||
os.path.abspath(os.path.realpath(x.encode("utf-8")))
|
|
||||||
+ (x[-1:] if x[-1:] == os.sep else "").encode("utf-8")
|
|
||||||
for x in ar.files
|
|
||||||
]
|
|
||||||
ar.url = ar.url.rstrip("/") + "/"
|
|
||||||
if "://" not in ar.url:
|
|
||||||
ar.url = "http://" + ar.url
|
|
||||||
|
|
||||||
eprint("\nscanning {0} locations\n".format(len(ar.files)))
|
eprint("\nscanning {0} locations\n".format(len(ar.files)))
|
||||||
|
|
||||||
nfiles = 0
|
nfiles = 0
|
||||||
nbytes = 0
|
nbytes = 0
|
||||||
err = []
|
err = []
|
||||||
for _, _, inf in walkdirs(err, ar.files):
|
for _, _, inf in walkdirs(err, ar.files):
|
||||||
|
if stat.S_ISDIR(inf.st_mode):
|
||||||
|
continue
|
||||||
|
|
||||||
nfiles += 1
|
nfiles += 1
|
||||||
nbytes += inf.st_size
|
nbytes += inf.st_size
|
||||||
|
|
||||||
if err:
|
if err:
|
||||||
eprint("\n# failed to access {0} paths:\n".format(len(err)))
|
eprint("\n# failed to access {0} paths:\n".format(len(err)))
|
||||||
for x in err:
|
for ap, msg in err:
|
||||||
eprint(x.decode("utf-8", "replace") + "\n")
|
if ar.v:
|
||||||
|
eprint("{0}\n `-{1}\n\n".format(ap.decode("utf-8", "replace"), msg))
|
||||||
|
else:
|
||||||
|
eprint(ap.decode("utf-8", "replace") + "\n")
|
||||||
|
|
||||||
eprint("^ failed to access those {0} paths ^\n\n".format(len(err)))
|
eprint("^ failed to access those {0} paths ^\n\n".format(len(err)))
|
||||||
|
|
||||||
|
if not ar.v:
|
||||||
|
eprint("hint: set -v for detailed error messages\n")
|
||||||
|
|
||||||
if not ar.ok:
|
if not ar.ok:
|
||||||
eprint("aborting because --ok is not set\n")
|
eprint("hint: aborting because --ok is not set\n")
|
||||||
return
|
return
|
||||||
|
|
||||||
eprint("found {0} files, {1}\n\n".format(nfiles, humansize(nbytes)))
|
eprint("found {0} files, {1}\n\n".format(nfiles, humansize(nbytes)))
|
||||||
self.nfiles = nfiles
|
return nfiles, nbytes
|
||||||
self.nbytes = nbytes
|
|
||||||
|
def __init__(self, ar, stats=None):
|
||||||
|
self.ar = ar
|
||||||
|
self.stats = stats or self._scan()
|
||||||
|
if not self.stats:
|
||||||
|
return
|
||||||
|
|
||||||
|
self.nfiles, self.nbytes = self.stats
|
||||||
|
|
||||||
if ar.td:
|
if ar.td:
|
||||||
requests.packages.urllib3.disable_warnings()
|
requests.packages.urllib3.disable_warnings()
|
||||||
@@ -491,24 +630,53 @@ class Ctl(object):
|
|||||||
|
|
||||||
self.filegen = walkdirs([], ar.files)
|
self.filegen = walkdirs([], ar.files)
|
||||||
if ar.safe:
|
if ar.safe:
|
||||||
self.safe()
|
self._safe()
|
||||||
else:
|
else:
|
||||||
self.fancy()
|
self.hash_f = 0
|
||||||
|
self.hash_c = 0
|
||||||
|
self.hash_b = 0
|
||||||
|
self.up_f = 0
|
||||||
|
self.up_c = 0
|
||||||
|
self.up_b = 0
|
||||||
|
self.up_br = 0
|
||||||
|
self.hasher_busy = 1
|
||||||
|
self.handshaker_busy = 0
|
||||||
|
self.uploader_busy = 0
|
||||||
|
self.serialized = False
|
||||||
|
|
||||||
def safe(self):
|
self.t0 = time.time()
|
||||||
|
self.t0_up = None
|
||||||
|
self.spd = None
|
||||||
|
|
||||||
|
self.mutex = threading.Lock()
|
||||||
|
self.q_handshake = Queue() # type: Queue[File]
|
||||||
|
self.q_upload = Queue() # type: Queue[tuple[File, str]]
|
||||||
|
self.recheck = [] # type: list[File]
|
||||||
|
|
||||||
|
self.st_hash = [None, "(idle, starting...)"] # type: tuple[File, int]
|
||||||
|
self.st_up = [None, "(idle, starting...)"] # type: tuple[File, int]
|
||||||
|
|
||||||
|
self.mth = MTHash(ar.J) if ar.J > 1 else None
|
||||||
|
|
||||||
|
self._fancy()
|
||||||
|
|
||||||
|
def _safe(self):
|
||||||
"""minimal basic slow boring fallback codepath"""
|
"""minimal basic slow boring fallback codepath"""
|
||||||
search = self.ar.s
|
search = self.ar.s
|
||||||
for nf, (top, rel, inf) in enumerate(self.filegen):
|
for nf, (top, rel, inf) in enumerate(self.filegen):
|
||||||
|
if stat.S_ISDIR(inf.st_mode) or not rel:
|
||||||
|
continue
|
||||||
|
|
||||||
file = File(top, rel, inf.st_size, inf.st_mtime)
|
file = File(top, rel, inf.st_size, inf.st_mtime)
|
||||||
upath = file.abs.decode("utf-8", "replace")
|
upath = file.abs.decode("utf-8", "replace")
|
||||||
|
|
||||||
print("{0} {1}\n hash...".format(self.nfiles - nf, upath))
|
print("{0} {1}\n hash...".format(self.nfiles - nf, upath))
|
||||||
get_hashlist(file, None)
|
get_hashlist(file, None, None)
|
||||||
|
|
||||||
burl = self.ar.url[:12] + self.ar.url[8:].split("/")[0] + "/"
|
burl = self.ar.url[:12] + self.ar.url[8:].split("/")[0] + "/"
|
||||||
while True:
|
while True:
|
||||||
print(" hs...")
|
print(" hs...")
|
||||||
hs = handshake(req_ses, self.ar.url, file, self.ar.a, search)
|
hs, _ = handshake(self.ar, file, search)
|
||||||
if search:
|
if search:
|
||||||
if hs:
|
if hs:
|
||||||
for hit in hs:
|
for hit in hs:
|
||||||
@@ -525,41 +693,28 @@ class Ctl(object):
|
|||||||
ncs = len(hs)
|
ncs = len(hs)
|
||||||
for nc, cid in enumerate(hs):
|
for nc, cid in enumerate(hs):
|
||||||
print(" {0} up {1}".format(ncs - nc, cid))
|
print(" {0} up {1}".format(ncs - nc, cid))
|
||||||
upload(req_ses, file, cid, self.ar.a)
|
upload(file, cid, self.ar.a)
|
||||||
|
|
||||||
print(" ok!")
|
print(" ok!")
|
||||||
|
if file.recheck:
|
||||||
|
self.recheck.append(file)
|
||||||
|
|
||||||
def fancy(self):
|
if not self.recheck:
|
||||||
self.hash_f = 0
|
return
|
||||||
self.hash_c = 0
|
|
||||||
self.hash_b = 0
|
|
||||||
self.up_f = 0
|
|
||||||
self.up_c = 0
|
|
||||||
self.up_b = 0
|
|
||||||
self.up_br = 0
|
|
||||||
self.hasher_busy = 1
|
|
||||||
self.handshaker_busy = 0
|
|
||||||
self.uploader_busy = 0
|
|
||||||
|
|
||||||
self.t0 = time.time()
|
eprint("finalizing {0} duplicate files".format(len(self.recheck)))
|
||||||
self.t0_up = None
|
for file in self.recheck:
|
||||||
self.spd = None
|
handshake(self.ar, file, search)
|
||||||
|
|
||||||
self.mutex = threading.Lock()
|
def _fancy(self):
|
||||||
self.q_handshake = Queue() # type: Queue[File]
|
if VT100 and not self.ar.ns:
|
||||||
self.q_recheck = Queue() # type: Queue[File] # partial upload exists [...]
|
|
||||||
self.q_upload = Queue() # type: Queue[tuple[File, str]]
|
|
||||||
|
|
||||||
self.st_hash = [None, "(idle, starting...)"] # type: tuple[File, int]
|
|
||||||
self.st_up = [None, "(idle, starting...)"] # type: tuple[File, int]
|
|
||||||
if VT100:
|
|
||||||
atexit.register(self.cleanup_vt100)
|
atexit.register(self.cleanup_vt100)
|
||||||
ss.scroll_region(3)
|
ss.scroll_region(3)
|
||||||
|
|
||||||
Daemon(target=self.hasher).start()
|
Daemon(self.hasher)
|
||||||
for _ in range(self.ar.j):
|
for _ in range(self.ar.j):
|
||||||
Daemon(target=self.handshaker).start()
|
Daemon(self.handshaker)
|
||||||
Daemon(target=self.uploader).start()
|
Daemon(self.uploader)
|
||||||
|
|
||||||
idles = 0
|
idles = 0
|
||||||
while idles < 3:
|
while idles < 3:
|
||||||
@@ -576,7 +731,7 @@ class Ctl(object):
|
|||||||
else:
|
else:
|
||||||
idles = 0
|
idles = 0
|
||||||
|
|
||||||
if VT100:
|
if VT100 and not self.ar.ns:
|
||||||
maxlen = ss.w - len(str(self.nfiles)) - 14
|
maxlen = ss.w - len(str(self.nfiles)) - 14
|
||||||
txt = "\033[s\033[{0}H".format(ss.g)
|
txt = "\033[s\033[{0}H".format(ss.g)
|
||||||
for y, k, st, f in [
|
for y, k, st, f in [
|
||||||
@@ -597,8 +752,8 @@ class Ctl(object):
|
|||||||
if "/" in name:
|
if "/" in name:
|
||||||
name = "\033[36m{0}\033[0m/{1}".format(*name.rsplit("/", 1))
|
name = "\033[36m{0}\033[0m/{1}".format(*name.rsplit("/", 1))
|
||||||
|
|
||||||
m = "{0:6.1f}% {1} {2}\033[K"
|
t = "{0:6.1f}% {1} {2}\033[K"
|
||||||
txt += m.format(p, self.nfiles - f, name)
|
txt += t.format(p, self.nfiles - f, name)
|
||||||
|
|
||||||
txt += "\033[{0}H ".format(ss.g + 2)
|
txt += "\033[{0}H ".format(ss.g + 2)
|
||||||
else:
|
else:
|
||||||
@@ -614,11 +769,19 @@ class Ctl(object):
|
|||||||
|
|
||||||
spd = humansize(spd)
|
spd = humansize(spd)
|
||||||
eta = str(datetime.timedelta(seconds=int(eta)))
|
eta = str(datetime.timedelta(seconds=int(eta)))
|
||||||
left = humansize(self.nbytes - self.up_b)
|
sleft = humansize(self.nbytes - self.up_b)
|
||||||
tail = "\033[K\033[u" if VT100 else "\r"
|
nleft = self.nfiles - self.up_f
|
||||||
|
tail = "\033[K\033[u" if VT100 and not self.ar.ns else "\r"
|
||||||
|
|
||||||
m = "eta: {0} @ {1}/s, {2} left".format(eta, spd, left)
|
t = "{0} eta @ {1}/s, {2}, {3}# left".format(eta, spd, sleft, nleft)
|
||||||
eprint(txt + "\033]0;{0}\033\\\r{1}{2}".format(m, m, tail))
|
eprint(txt + "\033]0;{0}\033\\\r{0}{1}".format(t, tail))
|
||||||
|
|
||||||
|
if not self.recheck:
|
||||||
|
return
|
||||||
|
|
||||||
|
eprint("finalizing {0} duplicate files".format(len(self.recheck)))
|
||||||
|
for file in self.recheck:
|
||||||
|
handshake(self.ar, file, False)
|
||||||
|
|
||||||
def cleanup_vt100(self):
|
def cleanup_vt100(self):
|
||||||
ss.scroll_region(None)
|
ss.scroll_region(None)
|
||||||
@@ -631,8 +794,10 @@ class Ctl(object):
|
|||||||
prd = None
|
prd = None
|
||||||
ls = {}
|
ls = {}
|
||||||
for top, rel, inf in self.filegen:
|
for top, rel, inf in self.filegen:
|
||||||
if self.ar.z:
|
isdir = stat.S_ISDIR(inf.st_mode)
|
||||||
rd = os.path.dirname(rel)
|
if self.ar.z or self.ar.drd:
|
||||||
|
rd = rel if isdir else os.path.dirname(rel)
|
||||||
|
srd = rd.decode("utf-8", "replace").replace("\\", "/")
|
||||||
if prd != rd:
|
if prd != rd:
|
||||||
prd = rd
|
prd = rd
|
||||||
headers = {}
|
headers = {}
|
||||||
@@ -641,19 +806,37 @@ class Ctl(object):
|
|||||||
|
|
||||||
ls = {}
|
ls = {}
|
||||||
try:
|
try:
|
||||||
print(" ls ~{0}".format(rd.decode("utf-8", "replace")))
|
print(" ls ~{0}".format(srd))
|
||||||
r = req_ses.get(
|
zb = self.ar.url.encode("utf-8")
|
||||||
self.ar.url.encode("utf-8") + quotep(rd) + b"?ls",
|
zb += quotep(rd.replace(b"\\", b"/"))
|
||||||
headers=headers,
|
r = req_ses.get(zb + b"?ls&dots", headers=headers)
|
||||||
)
|
if not r:
|
||||||
for f in r.json()["files"]:
|
raise Exception("HTTP {}".format(r.status_code))
|
||||||
rfn = f["href"].split("?")[0].encode("utf-8", "replace")
|
|
||||||
ls[unquote(rfn)] = f
|
|
||||||
except:
|
|
||||||
print(" mkdir ~{0}".format(rd.decode("utf-8", "replace")))
|
|
||||||
|
|
||||||
|
j = r.json()
|
||||||
|
for f in j["dirs"] + j["files"]:
|
||||||
|
rfn = f["href"].split("?")[0].rstrip("/")
|
||||||
|
ls[unquote(rfn.encode("utf-8", "replace"))] = f
|
||||||
|
except Exception as ex:
|
||||||
|
print(" mkdir ~{0} ({1})".format(srd, ex))
|
||||||
|
|
||||||
|
if self.ar.drd:
|
||||||
|
dp = os.path.join(top, rd)
|
||||||
|
lnodes = set(os.listdir(dp))
|
||||||
|
bnames = [x for x in ls if x not in lnodes]
|
||||||
|
if bnames:
|
||||||
|
vpath = self.ar.url.split("://")[-1].split("/", 1)[-1]
|
||||||
|
names = [x.decode("utf-8", "replace") for x in bnames]
|
||||||
|
locs = [vpath + srd + "/" + x for x in names]
|
||||||
|
print("DELETING ~{0}/#{1}".format(srd, len(names)))
|
||||||
|
req_ses.post(self.ar.url + "?delete", json=locs)
|
||||||
|
|
||||||
|
if isdir:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if self.ar.z:
|
||||||
rf = ls.get(os.path.basename(rel), None)
|
rf = ls.get(os.path.basename(rel), None)
|
||||||
if rf and rf["sz"] == inf.st_size and abs(rf["ts"] - inf.st_mtime) <= 1:
|
if rf and rf["sz"] == inf.st_size and abs(rf["ts"] - inf.st_mtime) <= 2:
|
||||||
self.nfiles -= 1
|
self.nfiles -= 1
|
||||||
self.nbytes -= inf.st_size
|
self.nbytes -= inf.st_size
|
||||||
continue
|
continue
|
||||||
@@ -662,22 +845,24 @@ class Ctl(object):
|
|||||||
while True:
|
while True:
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
if (
|
if (
|
||||||
self.hash_b - self.up_b < 1024 * 1024 * 128
|
self.hash_f - self.up_f == 1
|
||||||
and self.hash_c - self.up_c < 64
|
or (
|
||||||
and (
|
self.hash_b - self.up_b < 1024 * 1024 * 1024
|
||||||
not self.ar.nh
|
and self.hash_c - self.up_c < 512
|
||||||
or (
|
)
|
||||||
self.q_upload.empty()
|
) and (
|
||||||
and self.q_handshake.empty()
|
not self.ar.nh
|
||||||
and not self.uploader_busy
|
or (
|
||||||
)
|
self.q_upload.empty()
|
||||||
|
and self.q_handshake.empty()
|
||||||
|
and not self.uploader_busy
|
||||||
)
|
)
|
||||||
):
|
):
|
||||||
break
|
break
|
||||||
|
|
||||||
time.sleep(0.05)
|
time.sleep(0.05)
|
||||||
|
|
||||||
get_hashlist(file, self.cb_hasher)
|
get_hashlist(file, self.cb_hasher, self.mth)
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
self.hash_f += 1
|
self.hash_f += 1
|
||||||
self.hash_c += len(file.cids)
|
self.hash_c += len(file.cids)
|
||||||
@@ -690,16 +875,10 @@ class Ctl(object):
|
|||||||
|
|
||||||
def handshaker(self):
|
def handshaker(self):
|
||||||
search = self.ar.s
|
search = self.ar.s
|
||||||
q = self.q_handshake
|
|
||||||
burl = self.ar.url[:8] + self.ar.url[8:].split("/")[0] + "/"
|
burl = self.ar.url[:8] + self.ar.url[8:].split("/")[0] + "/"
|
||||||
while True:
|
while True:
|
||||||
file = q.get()
|
file = self.q_handshake.get()
|
||||||
if not file:
|
if not file:
|
||||||
if q == self.q_handshake:
|
|
||||||
q = self.q_recheck
|
|
||||||
q.put(None)
|
|
||||||
continue
|
|
||||||
|
|
||||||
self.q_upload.put(None)
|
self.q_upload.put(None)
|
||||||
break
|
break
|
||||||
|
|
||||||
@@ -707,21 +886,12 @@ class Ctl(object):
|
|||||||
self.handshaker_busy += 1
|
self.handshaker_busy += 1
|
||||||
|
|
||||||
upath = file.abs.decode("utf-8", "replace")
|
upath = file.abs.decode("utf-8", "replace")
|
||||||
|
hs, sprs = handshake(self.ar, file, search)
|
||||||
try:
|
|
||||||
hs = handshake(req_ses, self.ar.url, file, self.ar.a, search)
|
|
||||||
except Exception as ex:
|
|
||||||
if q == self.q_handshake and "<pre>partial upload exists" in str(ex):
|
|
||||||
self.q_recheck.put(file)
|
|
||||||
hs = []
|
|
||||||
else:
|
|
||||||
raise
|
|
||||||
|
|
||||||
if search:
|
if search:
|
||||||
if hs:
|
if hs:
|
||||||
for hit in hs:
|
for hit in hs:
|
||||||
m = "found: {0}\n {1}{2}\n"
|
t = "found: {0}\n {1}{2}\n"
|
||||||
print(m.format(upath, burl, hit["rp"]), end="")
|
print(t.format(upath, burl, hit["rp"]), end="")
|
||||||
else:
|
else:
|
||||||
print("NOT found: {0}\n".format(upath), end="")
|
print("NOT found: {0}\n".format(upath), end="")
|
||||||
|
|
||||||
@@ -733,13 +903,25 @@ class Ctl(object):
|
|||||||
|
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
if file.recheck:
|
||||||
|
self.recheck.append(file)
|
||||||
|
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
|
if hs and not sprs and not self.serialized:
|
||||||
|
t = "server filesystem does not support sparse files; serializing uploads\n"
|
||||||
|
eprint(t)
|
||||||
|
self.serialized = True
|
||||||
|
for _ in range(self.ar.j - 1):
|
||||||
|
self.q_upload.put(None)
|
||||||
if not hs:
|
if not hs:
|
||||||
# all chunks done
|
# all chunks done
|
||||||
self.up_f += 1
|
self.up_f += 1
|
||||||
self.up_c += len(file.cids) - file.up_c
|
self.up_c += len(file.cids) - file.up_c
|
||||||
self.up_b += file.size - file.up_b
|
self.up_b += file.size - file.up_b
|
||||||
|
|
||||||
|
if not file.recheck:
|
||||||
|
self.up_done(file)
|
||||||
|
|
||||||
if hs and file.up_c:
|
if hs and file.up_c:
|
||||||
# some chunks failed
|
# some chunks failed
|
||||||
self.up_c -= len(hs)
|
self.up_c -= len(hs)
|
||||||
@@ -771,10 +953,10 @@ class Ctl(object):
|
|||||||
|
|
||||||
file, cid = task
|
file, cid = task
|
||||||
try:
|
try:
|
||||||
upload(req_ses, file, cid, self.ar.a)
|
upload(file, cid, self.ar.a)
|
||||||
except:
|
except:
|
||||||
eprint("upload failed, retrying: {0} #{1}\n".format(file.name, cid[:8]))
|
eprint("upload failed, retrying: {0} #{1}\n".format(file.name, cid[:8]))
|
||||||
pass # handshake will fix it
|
# handshake will fix it
|
||||||
|
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
sz = file.kchunks[cid][1]
|
sz = file.kchunks[cid][1]
|
||||||
@@ -790,6 +972,10 @@ class Ctl(object):
|
|||||||
self.up_c += 1
|
self.up_c += 1
|
||||||
self.uploader_busy -= 1
|
self.uploader_busy -= 1
|
||||||
|
|
||||||
|
def up_done(self, file):
|
||||||
|
if self.ar.dl:
|
||||||
|
os.unlink(file.abs)
|
||||||
|
|
||||||
|
|
||||||
class APF(argparse.ArgumentDefaultsHelpFormatter, argparse.RawDescriptionHelpFormatter):
|
class APF(argparse.ArgumentDefaultsHelpFormatter, argparse.RawDescriptionHelpFormatter):
|
||||||
pass
|
pass
|
||||||
@@ -800,6 +986,9 @@ def main():
|
|||||||
if not VT100:
|
if not VT100:
|
||||||
os.system("rem") # enables colors
|
os.system("rem") # enables colors
|
||||||
|
|
||||||
|
cores = (os.cpu_count() if hasattr(os, "cpu_count") else 0) or 2
|
||||||
|
hcores = min(cores, 3) # 4% faster than 4+ on py3.9 @ r5-4500U
|
||||||
|
|
||||||
# fmt: off
|
# fmt: off
|
||||||
ap = app = argparse.ArgumentParser(formatter_class=APF, epilog="""
|
ap = app = argparse.ArgumentParser(formatter_class=APF, epilog="""
|
||||||
NOTE:
|
NOTE:
|
||||||
@@ -810,20 +999,75 @@ source file/folder selection uses rsync syntax, meaning that:
|
|||||||
|
|
||||||
ap.add_argument("url", type=unicode, help="server url, including destination folder")
|
ap.add_argument("url", type=unicode, help="server url, including destination folder")
|
||||||
ap.add_argument("files", type=unicode, nargs="+", help="files and/or folders to process")
|
ap.add_argument("files", type=unicode, nargs="+", help="files and/or folders to process")
|
||||||
ap.add_argument("-a", metavar="PASSWORD", help="password")
|
ap.add_argument("-v", action="store_true", help="verbose")
|
||||||
|
ap.add_argument("-a", metavar="PASSWORD", help="password or $filepath")
|
||||||
ap.add_argument("-s", action="store_true", help="file-search (disables upload)")
|
ap.add_argument("-s", action="store_true", help="file-search (disables upload)")
|
||||||
ap.add_argument("--ok", action="store_true", help="continue even if some local files are inaccessible")
|
ap.add_argument("--ok", action="store_true", help="continue even if some local files are inaccessible")
|
||||||
|
|
||||||
|
ap = app.add_argument_group("compatibility")
|
||||||
|
ap.add_argument("--cls", action="store_true", help="clear screen before start")
|
||||||
|
ap.add_argument("--ws", action="store_true", help="copyparty is running on windows; wait before deleting files after uploading")
|
||||||
|
|
||||||
|
ap = app.add_argument_group("folder sync")
|
||||||
|
ap.add_argument("--dl", action="store_true", help="delete local files after uploading")
|
||||||
|
ap.add_argument("--dr", action="store_true", help="delete remote files which don't exist locally")
|
||||||
|
ap.add_argument("--drd", action="store_true", help="delete remote files during upload instead of afterwards; reduces peak disk space usage, but will reupload instead of detecting renames")
|
||||||
|
|
||||||
ap = app.add_argument_group("performance tweaks")
|
ap = app.add_argument_group("performance tweaks")
|
||||||
ap.add_argument("-j", type=int, metavar="THREADS", default=4, help="parallel connections")
|
ap.add_argument("-j", type=int, metavar="THREADS", default=4, help="parallel connections")
|
||||||
|
ap.add_argument("-J", type=int, metavar="THREADS", default=hcores, help="num cpu-cores to use for hashing; set 0 or 1 for single-core hashing")
|
||||||
ap.add_argument("-nh", action="store_true", help="disable hashing while uploading")
|
ap.add_argument("-nh", action="store_true", help="disable hashing while uploading")
|
||||||
|
ap.add_argument("-ns", action="store_true", help="no status panel (for slow consoles)")
|
||||||
ap.add_argument("--safe", action="store_true", help="use simple fallback approach")
|
ap.add_argument("--safe", action="store_true", help="use simple fallback approach")
|
||||||
ap.add_argument("-z", action="store_true", help="ZOOMIN' (skip uploading files if they exist at the destination with the ~same last-modified timestamp, so same as yolo / turbo with date-chk but even faster)")
|
ap.add_argument("-z", action="store_true", help="ZOOMIN' (skip uploading files if they exist at the destination with the ~same last-modified timestamp, so same as yolo / turbo with date-chk but even faster)")
|
||||||
|
|
||||||
ap = app.add_argument_group("tls")
|
ap = app.add_argument_group("tls")
|
||||||
ap.add_argument("-te", metavar="PEM_FILE", help="certificate to expect/verify")
|
ap.add_argument("-te", metavar="PEM_FILE", help="certificate to expect/verify")
|
||||||
ap.add_argument("-td", action="store_true", help="disable certificate check")
|
ap.add_argument("-td", action="store_true", help="disable certificate check")
|
||||||
# fmt: on
|
# fmt: on
|
||||||
|
|
||||||
Ctl(app.parse_args())
|
ar = app.parse_args()
|
||||||
|
if ar.drd:
|
||||||
|
ar.dr = True
|
||||||
|
|
||||||
|
for k in "dl dr drd".split():
|
||||||
|
errs = []
|
||||||
|
if ar.safe and getattr(ar, k):
|
||||||
|
errs.append(k)
|
||||||
|
|
||||||
|
if errs:
|
||||||
|
raise Exception("--safe is incompatible with " + str(errs))
|
||||||
|
|
||||||
|
ar.files = [
|
||||||
|
os.path.abspath(os.path.realpath(x.encode("utf-8")))
|
||||||
|
+ (x[-1:] if x[-1:] == os.sep else "").encode("utf-8")
|
||||||
|
for x in ar.files
|
||||||
|
]
|
||||||
|
|
||||||
|
ar.url = ar.url.rstrip("/") + "/"
|
||||||
|
if "://" not in ar.url:
|
||||||
|
ar.url = "http://" + ar.url
|
||||||
|
|
||||||
|
if ar.a and ar.a.startswith("$"):
|
||||||
|
fn = ar.a[1:]
|
||||||
|
print("reading password from file [{}]".format(fn))
|
||||||
|
with open(fn, "rb") as f:
|
||||||
|
ar.a = f.read().decode("utf-8").strip()
|
||||||
|
|
||||||
|
if ar.cls:
|
||||||
|
print("\x1b\x5b\x48\x1b\x5b\x32\x4a\x1b\x5b\x33\x4a", end="")
|
||||||
|
|
||||||
|
ctl = Ctl(ar)
|
||||||
|
|
||||||
|
if ar.dr and not ar.drd:
|
||||||
|
print("\npass 2/2: delete")
|
||||||
|
if getattr(ctl, "up_br") and ar.ws:
|
||||||
|
# wait for up2k to mtime if there was uploads
|
||||||
|
time.sleep(4)
|
||||||
|
|
||||||
|
ar.drd = True
|
||||||
|
ar.z = True
|
||||||
|
Ctl(ar, ctl.stats)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|||||||
@@ -1,3 +1,6 @@
|
|||||||
|
### [`plugins/`](plugins/)
|
||||||
|
* example extensions
|
||||||
|
|
||||||
### [`copyparty.bat`](copyparty.bat)
|
### [`copyparty.bat`](copyparty.bat)
|
||||||
* launches copyparty with no arguments (anon read+write within same folder)
|
* launches copyparty with no arguments (anon read+write within same folder)
|
||||||
* intended for windows machines with no python.exe in PATH
|
* intended for windows machines with no python.exe in PATH
|
||||||
@@ -19,13 +22,23 @@ however if your copyparty is behind a reverse-proxy, you may want to use [`share
|
|||||||
* `URL`: full URL to the root folder (with trailing slash) followed by `$regex:1|1$`
|
* `URL`: full URL to the root folder (with trailing slash) followed by `$regex:1|1$`
|
||||||
* `pw`: password (remove `Parameters` if anon-write)
|
* `pw`: password (remove `Parameters` if anon-write)
|
||||||
|
|
||||||
|
### [`media-osd-bgone.ps1`](media-osd-bgone.ps1)
|
||||||
|
* disables the [windows OSD popup](https://user-images.githubusercontent.com/241032/122821375-0e08df80-d2dd-11eb-9fd9-184e8aacf1d0.png) (the thing on the left) which appears every time you hit media hotkeys to adjust volume or change song while playing music with the copyparty web-ui, or most other audio players really
|
||||||
|
|
||||||
### [`explorer-nothumbs-nofoldertypes.reg`](explorer-nothumbs-nofoldertypes.reg)
|
### [`explorer-nothumbs-nofoldertypes.reg`](explorer-nothumbs-nofoldertypes.reg)
|
||||||
* disables thumbnails and folder-type detection in windows explorer
|
* disables thumbnails and folder-type detection in windows explorer
|
||||||
* makes it way faster (especially for slow/networked locations (such as copyparty-fuse))
|
* makes it way faster (especially for slow/networked locations (such as partyfuse))
|
||||||
|
|
||||||
|
### [`webdav-cfg.reg`](webdav-cfg.bat)
|
||||||
|
* improves the native webdav support in windows;
|
||||||
|
* removes the 47.6 MiB filesize limit when downloading from webdav
|
||||||
|
* optionally enables webdav basic-auth over plaintext http
|
||||||
|
* optionally helps disable wpad, removing the 10sec latency
|
||||||
|
|
||||||
### [`cfssl.sh`](cfssl.sh)
|
### [`cfssl.sh`](cfssl.sh)
|
||||||
* creates CA and server certificates using cfssl
|
* creates CA and server certificates using cfssl
|
||||||
* give a 3rd argument to install it to your copyparty config
|
* give a 3rd argument to install it to your copyparty config
|
||||||
|
* systemd service at [`systemd/cfssl.service`](systemd/cfssl.service)
|
||||||
|
|
||||||
# OS integration
|
# OS integration
|
||||||
init-scripts to start copyparty as a service
|
init-scripts to start copyparty as a service
|
||||||
|
|||||||
15
contrib/apache/copyparty.conf
Normal file
15
contrib/apache/copyparty.conf
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
# when running copyparty behind a reverse proxy,
|
||||||
|
# the following arguments are recommended:
|
||||||
|
#
|
||||||
|
# --http-only lower latency on initial connection
|
||||||
|
# -i 127.0.0.1 only accept connections from nginx
|
||||||
|
#
|
||||||
|
# if you are doing location-based proxying (such as `/stuff` below)
|
||||||
|
# you must run copyparty with --rp-loc=stuff
|
||||||
|
#
|
||||||
|
# on fedora/rhel, remember to setsebool -P httpd_can_network_connect 1
|
||||||
|
|
||||||
|
LoadModule proxy_module modules/mod_proxy.so
|
||||||
|
ProxyPass "/stuff" "http://127.0.0.1:3923/stuff"
|
||||||
|
# do not specify ProxyPassReverse
|
||||||
|
RequestHeader set "X-Forwarded-Proto" expr=%{REQUEST_SCHEME}
|
||||||
@@ -7,7 +7,7 @@ srv_fqdn="$2"
|
|||||||
|
|
||||||
[ -z "$srv_fqdn" ] && {
|
[ -z "$srv_fqdn" ] && {
|
||||||
echo "need arg 1: ca name"
|
echo "need arg 1: ca name"
|
||||||
echo "need arg 2: server fqdn"
|
echo "need arg 2: server fqdn and/or IPs, comma-separated"
|
||||||
echo "optional arg 3: if set, write cert into copyparty cfg"
|
echo "optional arg 3: if set, write cert into copyparty cfg"
|
||||||
exit 1
|
exit 1
|
||||||
}
|
}
|
||||||
|
|||||||
104
contrib/media-osd-bgone.ps1
Normal file
104
contrib/media-osd-bgone.ps1
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
# media-osd-bgone.ps1: disable media-control OSD on win10do
|
||||||
|
# v1.1, 2021-06-25, ed <irc.rizon.net>, MIT-licensed
|
||||||
|
# https://github.com/9001/copyparty/blob/hovudstraum/contrib/media-osd-bgone.ps1
|
||||||
|
#
|
||||||
|
# locates the first window that looks like the media OSD and minimizes it;
|
||||||
|
# doing this once after each reboot should do the trick
|
||||||
|
# (adjust the width/height filter if it doesn't work)
|
||||||
|
#
|
||||||
|
# ---------------------------------------------------------------------
|
||||||
|
#
|
||||||
|
# tip: save the following as "media-osd-bgone.bat" next to this script:
|
||||||
|
# start cmd /c "powershell -command ""set-executionpolicy -scope process bypass; .\media-osd-bgone.ps1"" & ping -n 2 127.1 >nul"
|
||||||
|
#
|
||||||
|
# then create a shortcut to that bat-file and move the shortcut here:
|
||||||
|
# %appdata%\Microsoft\Windows\Start Menu\Programs\Startup
|
||||||
|
#
|
||||||
|
# and now this will autorun on bootup
|
||||||
|
|
||||||
|
|
||||||
|
Add-Type -TypeDefinition @"
|
||||||
|
using System;
|
||||||
|
using System.IO;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
using System.Windows.Forms;
|
||||||
|
|
||||||
|
namespace A {
|
||||||
|
public class B : Control {
|
||||||
|
|
||||||
|
[DllImport("user32.dll")]
|
||||||
|
static extern void keybd_event(byte bVk, byte bScan, uint dwFlags, int dwExtraInfo);
|
||||||
|
|
||||||
|
[DllImport("user32.dll", SetLastError = true)]
|
||||||
|
static extern IntPtr FindWindowEx(IntPtr hwndParent, IntPtr hwndChildAfter, string lpszClass, string lpszWindow);
|
||||||
|
|
||||||
|
[DllImport("user32.dll", SetLastError=true)]
|
||||||
|
static extern bool GetWindowRect(IntPtr hwnd, out RECT lpRect);
|
||||||
|
|
||||||
|
[DllImport("user32.dll")]
|
||||||
|
static extern bool ShowWindow(IntPtr hWnd, int nCmdShow);
|
||||||
|
|
||||||
|
[StructLayout(LayoutKind.Sequential)]
|
||||||
|
public struct RECT {
|
||||||
|
public int x;
|
||||||
|
public int y;
|
||||||
|
public int x2;
|
||||||
|
public int y2;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool fa() {
|
||||||
|
RECT r;
|
||||||
|
IntPtr it = IntPtr.Zero;
|
||||||
|
while ((it = FindWindowEx(IntPtr.Zero, it, "NativeHWNDHost", "")) != IntPtr.Zero) {
|
||||||
|
if (FindWindowEx(it, IntPtr.Zero, "DirectUIHWND", "") == IntPtr.Zero)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (!GetWindowRect(it, out r))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
int w = r.x2 - r.x + 1;
|
||||||
|
int h = r.y2 - r.y + 1;
|
||||||
|
|
||||||
|
Console.WriteLine("[*] hwnd {0:x} @ {1}x{2} sz {3}x{4}", it, r.x, r.y, w, h);
|
||||||
|
if (h != 141)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
ShowWindow(it, 6);
|
||||||
|
Console.WriteLine("[+] poof");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void fb() {
|
||||||
|
keybd_event((byte)Keys.VolumeMute, 0, 0, 0);
|
||||||
|
keybd_event((byte)Keys.VolumeMute, 0, 2, 0);
|
||||||
|
Thread.Sleep(500);
|
||||||
|
keybd_event((byte)Keys.VolumeMute, 0, 0, 0);
|
||||||
|
keybd_event((byte)Keys.VolumeMute, 0, 2, 0);
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if (fa()) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
Console.WriteLine("[!] not found");
|
||||||
|
Thread.Sleep(1000);
|
||||||
|
}
|
||||||
|
this.Invoke((MethodInvoker)delegate {
|
||||||
|
Application.Exit();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Run() {
|
||||||
|
Console.WriteLine("[+] hi");
|
||||||
|
new Thread(new ThreadStart(fb)).Start();
|
||||||
|
Application.Run();
|
||||||
|
Console.WriteLine("[+] bye");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"@ -ReferencedAssemblies System.Windows.Forms
|
||||||
|
|
||||||
|
(New-Object -TypeName A.B).Run()
|
||||||
@@ -1,15 +1,17 @@
|
|||||||
# when running copyparty behind a reverse proxy,
|
# when running copyparty behind a reverse proxy,
|
||||||
# the following arguments are recommended:
|
# the following arguments are recommended:
|
||||||
#
|
#
|
||||||
# -nc 512 important, see next paragraph
|
|
||||||
# --http-only lower latency on initial connection
|
# --http-only lower latency on initial connection
|
||||||
# -i 127.0.0.1 only accept connections from nginx
|
# -i 127.0.0.1 only accept connections from nginx
|
||||||
#
|
#
|
||||||
# -nc must match or exceed the webserver's max number of concurrent clients;
|
# -nc must match or exceed the webserver's max number of concurrent clients;
|
||||||
|
# copyparty default is 1024 if OS permits it (see "max clients:" on startup),
|
||||||
# nginx default is 512 (worker_processes 1, worker_connections 512)
|
# nginx default is 512 (worker_processes 1, worker_connections 512)
|
||||||
#
|
#
|
||||||
# you may also consider adding -j0 for CPU-intensive configurations
|
# you may also consider adding -j0 for CPU-intensive configurations
|
||||||
# (not that i can really think of any good examples)
|
# (not that i can really think of any good examples)
|
||||||
|
#
|
||||||
|
# on fedora/rhel, remember to setsebool -P httpd_can_network_connect 1
|
||||||
|
|
||||||
upstream cpp {
|
upstream cpp {
|
||||||
server 127.0.0.1:3923;
|
server 127.0.0.1:3923;
|
||||||
|
|||||||
@@ -14,5 +14,5 @@ name="$SVCNAME"
|
|||||||
command_background=true
|
command_background=true
|
||||||
pidfile="/var/run/$SVCNAME.pid"
|
pidfile="/var/run/$SVCNAME.pid"
|
||||||
|
|
||||||
command="/usr/bin/python /usr/local/bin/copyparty-sfx.py"
|
command="/usr/bin/python3 /usr/local/bin/copyparty-sfx.py"
|
||||||
command_args="-q -v /mnt::rw"
|
command_args="-q -v /mnt::rw"
|
||||||
|
|||||||
57
contrib/package/arch/PKGBUILD
Normal file
57
contrib/package/arch/PKGBUILD
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
# Maintainer: icxes <dev.null@need.moe>
|
||||||
|
pkgname=copyparty
|
||||||
|
pkgver="1.6.5"
|
||||||
|
pkgrel=1
|
||||||
|
pkgdesc="Portable file sharing hub"
|
||||||
|
arch=("any")
|
||||||
|
url="https://github.com/9001/${pkgname}"
|
||||||
|
license=('MIT')
|
||||||
|
depends=("python" "lsof")
|
||||||
|
optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tags"
|
||||||
|
"python-jinja: faster html generator"
|
||||||
|
"python-mutagen: music tags (alternative)"
|
||||||
|
"python-pillow: thumbnails for images"
|
||||||
|
"python-pyvips: thumbnails for images (higher quality, faster, uses more ram)"
|
||||||
|
"libkeyfinder-git: detection of musical keys"
|
||||||
|
"qm-vamp-plugins: BPM detection"
|
||||||
|
"python-pyopenssl: ftps functionality"
|
||||||
|
"python-impacket-git: smb support (bad idea)"
|
||||||
|
)
|
||||||
|
source=("${url}/releases/download/v${pkgver}/${pkgname}-sfx.py"
|
||||||
|
"${pkgname}.conf"
|
||||||
|
"${pkgname}.service"
|
||||||
|
"prisonparty.service"
|
||||||
|
"index.md"
|
||||||
|
"https://raw.githubusercontent.com/9001/${pkgname}/v${pkgver}/bin/prisonparty.sh"
|
||||||
|
"https://raw.githubusercontent.com/9001/${pkgname}/v${pkgver}/LICENSE"
|
||||||
|
)
|
||||||
|
backup=("etc/${pkgname}.d/init" )
|
||||||
|
sha256sums=("947d3f191f96f6a9e451bbcb35c5582ba210d81cfdc92dfa9ab0390dbecf26ee"
|
||||||
|
"b8565eba5e64dedba1cf6c7aac7e31c5a731ed7153d6810288a28f00a36c28b2"
|
||||||
|
"f65c207e0670f9d78ad2e399bda18d5502ff30d2ac79e0e7fc48e7fbdc39afdc"
|
||||||
|
"c4f396b083c9ec02ad50b52412c84d2a82be7f079b2d016e1c9fad22d68285ff"
|
||||||
|
"dba701de9fd584405917e923ea1e59dbb249b96ef23bad479cf4e42740b774c8"
|
||||||
|
"746971e95817c54445ce7f9c8406822dffc814cd5eb8113abd36dd472fd677d7"
|
||||||
|
"cb2ce3d6277bf2f5a82ecf336cc44963bc6490bcf496ffbd75fc9e21abaa75f3"
|
||||||
|
)
|
||||||
|
|
||||||
|
package() {
|
||||||
|
cd "${srcdir}/"
|
||||||
|
|
||||||
|
install -dm755 "${pkgdir}/etc/${pkgname}.d"
|
||||||
|
install -Dm755 "${pkgname}-sfx.py" "${pkgdir}/usr/bin/${pkgname}"
|
||||||
|
install -Dm755 "prisonparty.sh" "${pkgdir}/usr/bin/prisonparty"
|
||||||
|
install -Dm644 "${pkgname}.conf" "${pkgdir}/etc/${pkgname}.d/init"
|
||||||
|
install -Dm644 "${pkgname}.service" "${pkgdir}/usr/lib/systemd/system/${pkgname}.service"
|
||||||
|
install -Dm644 "prisonparty.service" "${pkgdir}/usr/lib/systemd/system/prisonparty.service"
|
||||||
|
install -Dm644 "index.md" "${pkgdir}/var/lib/${pkgname}-jail/README.md"
|
||||||
|
install -Dm644 "LICENSE" "${pkgdir}/usr/share/licenses/${pkgname}/LICENSE"
|
||||||
|
|
||||||
|
find /etc/${pkgname}.d -iname '*.conf' 2>/dev/null | grep -qE . && return
|
||||||
|
echo "┏━━━━━━━━━━━━━━━──-"
|
||||||
|
echo "┃ Configure ${pkgname} by adding .conf files into /etc/${pkgname}.d/"
|
||||||
|
echo "┃ and maybe copy+edit one of the following to /etc/systemd/system/:"
|
||||||
|
echo "┣━♦ /usr/lib/systemd/system/${pkgname}.service (standard)"
|
||||||
|
echo "┣━♦ /usr/lib/systemd/system/prisonparty.service (chroot)"
|
||||||
|
echo "┗━━━━━━━━━━━━━━━──-"
|
||||||
|
}
|
||||||
7
contrib/package/arch/copyparty.conf
Normal file
7
contrib/package/arch/copyparty.conf
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
## import all *.conf files from the current folder (/etc/copyparty.d)
|
||||||
|
% ./
|
||||||
|
|
||||||
|
# add additional .conf files to this folder;
|
||||||
|
# see example config files for reference:
|
||||||
|
# https://github.com/9001/copyparty/blob/hovudstraum/docs/example.conf
|
||||||
|
# https://github.com/9001/copyparty/tree/hovudstraum/docs/copyparty.d
|
||||||
32
contrib/package/arch/copyparty.service
Normal file
32
contrib/package/arch/copyparty.service
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
# this will start `/usr/bin/copyparty-sfx.py`
|
||||||
|
# and read config from `/etc/copyparty.d/*.conf`
|
||||||
|
#
|
||||||
|
# you probably want to:
|
||||||
|
# change "User=cpp" and "/home/cpp/" to another user
|
||||||
|
#
|
||||||
|
# unless you add -q to disable logging, you may want to remove the
|
||||||
|
# following line to allow buffering (slightly better performance):
|
||||||
|
# Environment=PYTHONUNBUFFERED=x
|
||||||
|
|
||||||
|
[Unit]
|
||||||
|
Description=copyparty file server
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=notify
|
||||||
|
SyslogIdentifier=copyparty
|
||||||
|
Environment=PYTHONUNBUFFERED=x
|
||||||
|
WorkingDirectory=/var/lib/copyparty-jail
|
||||||
|
ExecReload=/bin/kill -s USR1 $MAINPID
|
||||||
|
|
||||||
|
# user to run as + where the TLS certificate is (if any)
|
||||||
|
User=cpp
|
||||||
|
Environment=XDG_CONFIG_HOME=/home/cpp/.config
|
||||||
|
|
||||||
|
# stop systemd-tmpfiles-clean.timer from deleting copyparty while it's running
|
||||||
|
ExecStartPre=+/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf'
|
||||||
|
|
||||||
|
# run copyparty
|
||||||
|
ExecStart=/usr/bin/python3 /usr/bin/copyparty -c /etc/copyparty.d/init
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
3
contrib/package/arch/index.md
Normal file
3
contrib/package/arch/index.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
this is `/var/lib/copyparty-jail`, the fallback webroot when copyparty has not yet been configured
|
||||||
|
|
||||||
|
please add some `*.conf` files to `/etc/copyparty.d/`
|
||||||
31
contrib/package/arch/prisonparty.service
Normal file
31
contrib/package/arch/prisonparty.service
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
# this will start `/usr/bin/copyparty-sfx.py`
|
||||||
|
# in a chroot, preventing accidental access elsewhere
|
||||||
|
# and read config from `/etc/copyparty.d/*.conf`
|
||||||
|
#
|
||||||
|
# expose additional filesystem locations to copyparty
|
||||||
|
# by listing them between the last `1000` and `--`
|
||||||
|
#
|
||||||
|
# `1000 1000` = what user to run copyparty as
|
||||||
|
#
|
||||||
|
# unless you add -q to disable logging, you may want to remove the
|
||||||
|
# following line to allow buffering (slightly better performance):
|
||||||
|
# Environment=PYTHONUNBUFFERED=x
|
||||||
|
|
||||||
|
[Unit]
|
||||||
|
Description=copyparty file server
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
SyslogIdentifier=prisonparty
|
||||||
|
Environment=PYTHONUNBUFFERED=x
|
||||||
|
WorkingDirectory=/var/lib/copyparty-jail
|
||||||
|
ExecReload=/bin/kill -s USR1 $MAINPID
|
||||||
|
|
||||||
|
# stop systemd-tmpfiles-clean.timer from deleting copyparty while it's running
|
||||||
|
ExecStartPre=+/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf'
|
||||||
|
|
||||||
|
# run copyparty
|
||||||
|
ExecStart=/bin/bash /usr/bin/prisonparty /var/lib/copyparty-jail 1000 1000 /etc/copyparty.d -- \
|
||||||
|
/usr/bin/python3 /usr/bin/copyparty -c /etc/copyparty.d/init
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
33
contrib/plugins/README.md
Normal file
33
contrib/plugins/README.md
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
# example resource files
|
||||||
|
|
||||||
|
can be provided to copyparty to tweak things
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## example `.epilogue.html`
|
||||||
|
save one of these as `.epilogue.html` inside a folder to customize it:
|
||||||
|
|
||||||
|
* [`minimal-up2k.html`](minimal-up2k.html) will [simplify the upload ui](https://user-images.githubusercontent.com/241032/118311195-dd6ca380-b4ef-11eb-86f3-75a3ff2e1332.png)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## example browser-js
|
||||||
|
point `--js-browser` to one of these by URL:
|
||||||
|
|
||||||
|
* [`minimal-up2k.js`](minimal-up2k.js) is similar to the above `minimal-up2k.html` except it applies globally to all write-only folders
|
||||||
|
* [`up2k-hooks.js`](up2k-hooks.js) lets you specify a ruleset for files to skip uploading
|
||||||
|
* [`up2k-hook-ytid.js`](up2k-hook-ytid.js) is a more specific example checking youtube-IDs against some API
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## example browser-css
|
||||||
|
point `--css-browser` to one of these by URL:
|
||||||
|
|
||||||
|
* [`browser-icons.css`](browser-icons.css) adds filetype icons
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## meadup.js
|
||||||
|
|
||||||
|
* turns copyparty into chromecast just more flexible (and probably way more buggy)
|
||||||
|
* usage: put the js somewhere in the webroot and `--js-browser /memes/meadup.js`
|
||||||
506
contrib/plugins/meadup.js
Normal file
506
contrib/plugins/meadup.js
Normal file
@@ -0,0 +1,506 @@
|
|||||||
|
// USAGE:
|
||||||
|
// place this file somewhere in the webroot and then
|
||||||
|
// python3 -m copyparty --js-browser /memes/meadup.js
|
||||||
|
//
|
||||||
|
// FEATURES:
|
||||||
|
// * adds an onscreen keyboard for operating a media center remotely,
|
||||||
|
// relies on https://github.com/9001/copyparty/blob/hovudstraum/bin/mtag/very-bad-idea.py
|
||||||
|
// * adds an interactive anime girl (if you can find the dependencies)
|
||||||
|
|
||||||
|
var hambagas = [
|
||||||
|
"https://www.youtube.com/watch?v=pFA3KGp4GuU"
|
||||||
|
];
|
||||||
|
|
||||||
|
// keybaord,
|
||||||
|
// onscreen keyboard by @steinuil
|
||||||
|
function initKeybaord(BASE_URL, HAMBAGA, consoleLog, consoleError) {
|
||||||
|
document.querySelector('.keybaord-container').innerHTML = `
|
||||||
|
<div class="keybaord-body">
|
||||||
|
<div class="keybaord-row keybaord-row-1">
|
||||||
|
<div class="keybaord-key" data-keybaord-key="Escape">
|
||||||
|
esc
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F1">
|
||||||
|
F1
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F2">
|
||||||
|
F2
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F3">
|
||||||
|
F3
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F4">
|
||||||
|
F4
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F5">
|
||||||
|
F5
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F6">
|
||||||
|
F6
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F7">
|
||||||
|
F7
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F8">
|
||||||
|
F8
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F9">
|
||||||
|
F9
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F10">
|
||||||
|
F10
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F11">
|
||||||
|
F11
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="F12">
|
||||||
|
F12
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="Insert">
|
||||||
|
ins
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="Delete">
|
||||||
|
del
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-row keybaord-row-2">
|
||||||
|
<div class="keybaord-key" data-keybaord-key="\`">
|
||||||
|
\`
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="1">
|
||||||
|
1
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="2">
|
||||||
|
2
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="3">
|
||||||
|
3
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="4">
|
||||||
|
4
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="5">
|
||||||
|
5
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="6">
|
||||||
|
6
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="7">
|
||||||
|
7
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="8">
|
||||||
|
8
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="9">
|
||||||
|
9
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="0">
|
||||||
|
0
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="-">
|
||||||
|
-
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="=">
|
||||||
|
=
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key keybaord-backspace" data-keybaord-key="BackSpace">
|
||||||
|
backspace
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-row keybaord-row-3">
|
||||||
|
<div class="keybaord-key keybaord-tab" data-keybaord-key="Tab">
|
||||||
|
tab
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="q">
|
||||||
|
q
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="w">
|
||||||
|
w
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="e">
|
||||||
|
e
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="r">
|
||||||
|
r
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="t">
|
||||||
|
t
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="y">
|
||||||
|
y
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="u">
|
||||||
|
u
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="i">
|
||||||
|
i
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="o">
|
||||||
|
o
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="p">
|
||||||
|
p
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="[">
|
||||||
|
[
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="]">
|
||||||
|
]
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key keybaord-enter" data-keybaord-key="Return">
|
||||||
|
enter
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-row keybaord-row-4">
|
||||||
|
<div class="keybaord-key keybaord-capslock" data-keybaord-key="HAMBAGA">
|
||||||
|
🍔
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="a">
|
||||||
|
a
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="s">
|
||||||
|
s
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="d">
|
||||||
|
d
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="f">
|
||||||
|
f
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="g">
|
||||||
|
g
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="h">
|
||||||
|
h
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="j">
|
||||||
|
j
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="k">
|
||||||
|
k
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="l">
|
||||||
|
l
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key=";">
|
||||||
|
;
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="'">
|
||||||
|
'
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key keybaord-backslash" data-keybaord-key="\\">
|
||||||
|
\\
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-row keybaord-row-5">
|
||||||
|
<div class="keybaord-key keybaord-lshift" data-keybaord-key="Shift_L">
|
||||||
|
shift
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="\\">
|
||||||
|
\\
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="z">
|
||||||
|
z
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="x">
|
||||||
|
x
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="c">
|
||||||
|
c
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="v">
|
||||||
|
v
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="b">
|
||||||
|
b
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="n">
|
||||||
|
n
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="m">
|
||||||
|
m
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key=",">
|
||||||
|
,
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key=".">
|
||||||
|
.
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="/">
|
||||||
|
/
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key keybaord-rshift" data-keybaord-key="Shift_R">
|
||||||
|
shift
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-row keybaord-row-6">
|
||||||
|
<div class="keybaord-key keybaord-lctrl" data-keybaord-key="Control_L">
|
||||||
|
ctrl
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key keybaord-super" data-keybaord-key="Meta_L">
|
||||||
|
win
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key keybaord-alt" data-keybaord-key="Alt_L">
|
||||||
|
alt
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key keybaord-spacebar" data-keybaord-key="space">
|
||||||
|
space
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key keybaord-altgr" data-keybaord-key="Alt_R">
|
||||||
|
altgr
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key keybaord-what" data-keybaord-key="Menu">
|
||||||
|
menu
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key keybaord-rctrl" data-keybaord-key="Control_R">
|
||||||
|
ctrl
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-row">
|
||||||
|
<div class="keybaord-key" data-keybaord-key="XF86AudioLowerVolume">
|
||||||
|
🔉
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="XF86AudioRaiseVolume">
|
||||||
|
🔊
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="Left">
|
||||||
|
⬅️
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="Down">
|
||||||
|
⬇️
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="Up">
|
||||||
|
⬆️
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="Right">
|
||||||
|
➡️
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="Page_Up">
|
||||||
|
PgUp
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="Page_Down">
|
||||||
|
PgDn
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="Home">
|
||||||
|
🏠
|
||||||
|
</div>
|
||||||
|
<div class="keybaord-key" data-keybaord-key="End">
|
||||||
|
End
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
`;
|
||||||
|
|
||||||
|
function arraySample(array) {
|
||||||
|
return array[Math.floor(Math.random() * array.length)];
|
||||||
|
}
|
||||||
|
|
||||||
|
function sendMessage(msg) {
|
||||||
|
return fetch(BASE_URL, {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/x-www-form-urlencoded;charset=UTF-8",
|
||||||
|
},
|
||||||
|
body: "msg=" + encodeURIComponent(msg),
|
||||||
|
}).then(
|
||||||
|
(r) => r.text(), // so the response body shows up in network tab
|
||||||
|
(err) => consoleError(err)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
const MODIFIER_ON_CLASS = "keybaord-modifier-on";
|
||||||
|
const KEY_DATASET = "data-keybaord-key";
|
||||||
|
const KEY_CLASS = "keybaord-key";
|
||||||
|
|
||||||
|
const modifiers = new Set()
|
||||||
|
|
||||||
|
function toggleModifier(button, key) {
|
||||||
|
button.classList.toggle(MODIFIER_ON_CLASS);
|
||||||
|
if (modifiers.has(key)) {
|
||||||
|
modifiers.delete(key);
|
||||||
|
} else {
|
||||||
|
modifiers.add(key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function popModifiers() {
|
||||||
|
let modifierString = "";
|
||||||
|
|
||||||
|
modifiers.forEach((mod) => {
|
||||||
|
document.querySelector("[" + KEY_DATASET + "='" + mod + "']")
|
||||||
|
.classList.remove(MODIFIER_ON_CLASS);
|
||||||
|
|
||||||
|
modifierString += mod + "+";
|
||||||
|
});
|
||||||
|
|
||||||
|
modifiers.clear();
|
||||||
|
|
||||||
|
return modifierString;
|
||||||
|
}
|
||||||
|
|
||||||
|
Array.from(document.querySelectorAll("." + KEY_CLASS)).forEach((button) => {
|
||||||
|
const key = button.dataset.keybaordKey;
|
||||||
|
|
||||||
|
button.addEventListener("click", (ev) => {
|
||||||
|
switch (key) {
|
||||||
|
case "HAMBAGA":
|
||||||
|
sendMessage(arraySample(HAMBAGA));
|
||||||
|
break;
|
||||||
|
|
||||||
|
case "Shift_L":
|
||||||
|
case "Shift_R":
|
||||||
|
|
||||||
|
case "Control_L":
|
||||||
|
case "Control_R":
|
||||||
|
|
||||||
|
case "Meta_L":
|
||||||
|
|
||||||
|
case "Alt_L":
|
||||||
|
case "Alt_R":
|
||||||
|
toggleModifier(button, key);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default: {
|
||||||
|
const keyWithModifiers = popModifiers() + key;
|
||||||
|
|
||||||
|
consoleLog(keyWithModifiers);
|
||||||
|
|
||||||
|
sendMessage("key " + keyWithModifiers)
|
||||||
|
.then(() => consoleLog(keyWithModifiers + " OK"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// keybaord integration
|
||||||
|
(function () {
|
||||||
|
var o = mknod('div');
|
||||||
|
clmod(o, 'keybaord-container', 1);
|
||||||
|
ebi('op_msg').appendChild(o);
|
||||||
|
|
||||||
|
o = mknod('style');
|
||||||
|
o.innerHTML = `
|
||||||
|
.keybaord-body {
|
||||||
|
display: flex;
|
||||||
|
flex-flow: column nowrap;
|
||||||
|
margin: .6em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-row {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key {
|
||||||
|
border: 1px solid rgba(128,128,128,0.2);
|
||||||
|
width: 41px;
|
||||||
|
height: 40px;
|
||||||
|
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key:active {
|
||||||
|
background-color: lightgrey;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-modifier-on {
|
||||||
|
background-color: lightblue;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-backspace {
|
||||||
|
width: 82px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-tab {
|
||||||
|
width: 55px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-enter {
|
||||||
|
width: 69px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-capslock {
|
||||||
|
width: 80px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-backslash {
|
||||||
|
width: 88px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-lshift {
|
||||||
|
width: 65px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-rshift {
|
||||||
|
width: 103px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-lctrl {
|
||||||
|
width: 55px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-super {
|
||||||
|
width: 55px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-alt {
|
||||||
|
width: 55px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-altgr {
|
||||||
|
width: 55px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-what {
|
||||||
|
width: 55px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-rctrl {
|
||||||
|
width: 55px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.keybaord-key.keybaord-spacebar {
|
||||||
|
width: 302px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
document.head.appendChild(o);
|
||||||
|
|
||||||
|
initKeybaord('/', hambagas,
|
||||||
|
(msg) => { toast.inf(2, msg.toString()) },
|
||||||
|
(msg) => { toast.err(30, msg.toString()) });
|
||||||
|
})();
|
||||||
|
|
||||||
|
|
||||||
|
// live2d (dumb pointless meme)
|
||||||
|
// dependencies for this part are not tracked in git
|
||||||
|
// so delete this section if you wanna use this file
|
||||||
|
// (or supply your own l2d model and js)
|
||||||
|
(function () {
|
||||||
|
var o = mknod('link');
|
||||||
|
o.setAttribute('rel', 'stylesheet');
|
||||||
|
o.setAttribute('href', "/bad-memes/pio.css");
|
||||||
|
document.head.appendChild(o);
|
||||||
|
|
||||||
|
o = mknod('style');
|
||||||
|
o.innerHTML = '.pio-container{text-shadow:none;z-index:1}';
|
||||||
|
document.head.appendChild(o);
|
||||||
|
|
||||||
|
o = mknod('div');
|
||||||
|
clmod(o, 'pio-container', 1);
|
||||||
|
o.innerHTML = '<div class="pio-action"></div><canvas id="pio" width="280" height="500"></canvas>';
|
||||||
|
document.body.appendChild(o);
|
||||||
|
|
||||||
|
var remaining = 3;
|
||||||
|
for (var a of ['pio', 'l2d', 'fireworks']) {
|
||||||
|
import_js(`/bad-memes/${a}.js`, function () {
|
||||||
|
if (remaining --> 1)
|
||||||
|
return;
|
||||||
|
|
||||||
|
o = mknod('script');
|
||||||
|
o.innerHTML = 'var pio = new Paul_Pio({"selector":[],"mode":"fixed","hidden":false,"content":{"close":"ok bye"},"model":["/bad-memes/sagiri/model.json"]});';
|
||||||
|
document.body.appendChild(o);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})();
|
||||||
@@ -1,13 +1,22 @@
|
|||||||
<!--
|
<!--
|
||||||
|
NOTE: DEPRECATED; please use the javascript version instead:
|
||||||
|
https://github.com/9001/copyparty/blob/hovudstraum/contrib/plugins/minimal-up2k.js
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
save this as .epilogue.html inside a write-only folder to declutter the UI, makes it look like
|
save this as .epilogue.html inside a write-only folder to declutter the UI, makes it look like
|
||||||
https://user-images.githubusercontent.com/241032/118311195-dd6ca380-b4ef-11eb-86f3-75a3ff2e1332.png
|
https://user-images.githubusercontent.com/241032/118311195-dd6ca380-b4ef-11eb-86f3-75a3ff2e1332.png
|
||||||
|
|
||||||
|
only works if you disable the prologue/epilogue sandbox with --no-sb-lg
|
||||||
|
which should probably be combined with --no-dot-ren to prevent damage
|
||||||
|
(`no_sb_lg` can also be set per-volume with volflags)
|
||||||
-->
|
-->
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
|
|
||||||
/* make the up2k ui REALLY minimal by hiding a bunch of stuff: */
|
/* make the up2k ui REALLY minimal by hiding a bunch of stuff: */
|
||||||
|
|
||||||
#ops, #tree, #path, #wrap>h2:last-child, /* main tabs and navigators (tree/breadcrumbs) */
|
#ops, #tree, #path, #wfp, /* main tabs and navigators (tree/breadcrumbs) */
|
||||||
|
|
||||||
#u2conf tr:first-child>td[rowspan]:not(#u2btn_cw), /* most of the config options */
|
#u2conf tr:first-child>td[rowspan]:not(#u2btn_cw), /* most of the config options */
|
||||||
|
|
||||||
59
contrib/plugins/minimal-up2k.js
Normal file
59
contrib/plugins/minimal-up2k.js
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
makes the up2k ui REALLY minimal by hiding a bunch of stuff
|
||||||
|
|
||||||
|
almost the same as minimal-up2k.html except this one...:
|
||||||
|
|
||||||
|
-- applies to every write-only folder when used with --js-browser
|
||||||
|
|
||||||
|
-- only applies if javascript is enabled
|
||||||
|
|
||||||
|
-- doesn't hide the total upload ETA display
|
||||||
|
|
||||||
|
-- looks slightly better
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
var u2min = `
|
||||||
|
<style>
|
||||||
|
|
||||||
|
#ops, #path, #tree, #files, #wfp,
|
||||||
|
#u2conf td.c+.c, #u2cards, #srch_dz, #srch_zd {
|
||||||
|
display: none !important;
|
||||||
|
}
|
||||||
|
#u2conf {margin:5em auto 0 auto !important}
|
||||||
|
#u2conf.ww {width:70em}
|
||||||
|
#u2conf.w {width:50em}
|
||||||
|
#u2conf.w .c,
|
||||||
|
#u2conf.w #u2btn_cw {text-align:left}
|
||||||
|
#u2conf.w #u2btn_cw {width:70%}
|
||||||
|
#u2etaw {margin:3em auto}
|
||||||
|
#u2etaw.w {
|
||||||
|
text-align: center;
|
||||||
|
margin: -3.5em auto 5em auto;
|
||||||
|
}
|
||||||
|
#u2etaw.w #u2etas {margin-right:-37em}
|
||||||
|
#u2etaw.w #u2etas.o {margin-top:-2.2em}
|
||||||
|
#u2etaw.ww {margin:-1em auto}
|
||||||
|
#u2etaw.ww #u2etas {padding-left:4em}
|
||||||
|
#u2etas {
|
||||||
|
background: none !important;
|
||||||
|
border: none !important;
|
||||||
|
}
|
||||||
|
#wrap {margin-left:2em !important}
|
||||||
|
.logue {
|
||||||
|
border: none !important;
|
||||||
|
margin: 2em auto !important;
|
||||||
|
}
|
||||||
|
.logue:before {content:'' !important}
|
||||||
|
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<a href="#" onclick="this.parentNode.innerHTML='';">show advanced options</a>
|
||||||
|
`;
|
||||||
|
|
||||||
|
if (!has(perms, 'read')) {
|
||||||
|
var e2 = mknod('div');
|
||||||
|
e2.innerHTML = u2min;
|
||||||
|
ebi('wrap').insertBefore(e2, QS('#wfp'));
|
||||||
|
}
|
||||||
297
contrib/plugins/up2k-hook-ytid.js
Normal file
297
contrib/plugins/up2k-hook-ytid.js
Normal file
@@ -0,0 +1,297 @@
|
|||||||
|
// way more specific example --
|
||||||
|
// assumes all files dropped into the uploader have a youtube-id somewhere in the filename,
|
||||||
|
// locates the youtube-ids and passes them to an API which returns a list of IDs which should be uploaded
|
||||||
|
//
|
||||||
|
// also tries to find the youtube-id in the embedded metadata
|
||||||
|
//
|
||||||
|
// assumes copyparty is behind nginx as /ytq is a standalone service which must be rproxied in place
|
||||||
|
|
||||||
|
function up2k_namefilter(good_files, nil_files, bad_files, hooks) {
|
||||||
|
var passthru = up2k.uc.fsearch;
|
||||||
|
if (passthru)
|
||||||
|
return hooks[0](good_files, nil_files, bad_files, hooks.slice(1));
|
||||||
|
|
||||||
|
a_up2k_namefilter(good_files, nil_files, bad_files, hooks).then(() => { });
|
||||||
|
}
|
||||||
|
|
||||||
|
// ebi('op_up2k').appendChild(mknod('input','unick'));
|
||||||
|
|
||||||
|
function bstrpos(buf, ptn) {
|
||||||
|
var ofs = 0,
|
||||||
|
ch0 = ptn[0],
|
||||||
|
sz = buf.byteLength;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
ofs = buf.indexOf(ch0, ofs);
|
||||||
|
if (ofs < 0 || ofs >= sz)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
for (var a = 1; a < ptn.length; a++)
|
||||||
|
if (buf[ofs + a] !== ptn[a])
|
||||||
|
break;
|
||||||
|
|
||||||
|
if (a === ptn.length)
|
||||||
|
return ofs;
|
||||||
|
|
||||||
|
++ofs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function a_up2k_namefilter(good_files, nil_files, bad_files, hooks) {
|
||||||
|
var t0 = Date.now(),
|
||||||
|
yt_ids = new Set(),
|
||||||
|
textdec = new TextDecoder('latin1'),
|
||||||
|
md_ptn = new TextEncoder().encode('youtube.com/watch?v='),
|
||||||
|
file_ids = [], // all IDs found for each good_files
|
||||||
|
md_only = [], // `${id} ${fn}` where ID was only found in metadata
|
||||||
|
mofs = 0,
|
||||||
|
mnchk = 0,
|
||||||
|
mfile = '',
|
||||||
|
myid = localStorage.getItem('ytid_t0');
|
||||||
|
|
||||||
|
if (!myid)
|
||||||
|
localStorage.setItem('ytid_t0', myid = Date.now());
|
||||||
|
|
||||||
|
for (var a = 0; a < good_files.length; a++) {
|
||||||
|
var [fobj, name] = good_files[a],
|
||||||
|
cname = name, // will clobber
|
||||||
|
sz = fobj.size,
|
||||||
|
ids = [],
|
||||||
|
fn_ids = [],
|
||||||
|
md_ids = [],
|
||||||
|
id_ok = false,
|
||||||
|
m;
|
||||||
|
|
||||||
|
// all IDs found in this file
|
||||||
|
file_ids.push(ids);
|
||||||
|
|
||||||
|
// look for ID in filename; reduce the
|
||||||
|
// metadata-scan intensity if the id looks safe
|
||||||
|
m = /[\[(-]([\w-]{11})[\])]?\.(?:mp4|webm|mkv|flv|opus|ogg|mp3|m4a|aac)$/i.exec(name);
|
||||||
|
id_ok = !!m;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
// fuzzy catch-all;
|
||||||
|
// some ytdl fork did %(title)-%(id).%(ext) ...
|
||||||
|
m = /(?:^|[^\w])([\w-]{11})(?:$|[^\w-])/.exec(cname);
|
||||||
|
if (!m)
|
||||||
|
break;
|
||||||
|
|
||||||
|
cname = cname.replace(m[1], '');
|
||||||
|
yt_ids.add(m[1]);
|
||||||
|
fn_ids.unshift(m[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// look for IDs in video metadata,
|
||||||
|
if (/\.(mp4|webm|mkv|flv|opus|ogg|mp3|m4a|aac)$/i.exec(name)) {
|
||||||
|
toast.show('inf r', 0, `analyzing file ${a + 1} / ${good_files.length} :\n${name}\n\nhave analysed ${++mnchk} files in ${(Date.now() - t0) / 1000} seconds, ${humantime((good_files.length - (a + 1)) * (((Date.now() - t0) / 1000) / mnchk))} remaining,\n\nbiggest offset so far is ${mofs}, in this file:\n\n${mfile}`);
|
||||||
|
|
||||||
|
// check first and last 128 MiB;
|
||||||
|
// pWxOroN5WCo.mkv @ 6edb98 (6.92M)
|
||||||
|
// Nf-nN1wF5Xo.mp4 @ 4a98034 (74.6M)
|
||||||
|
var chunksz = 1024 * 1024 * 2, // byte
|
||||||
|
aspan = id_ok ? 128 : 512; // MiB
|
||||||
|
|
||||||
|
aspan = parseInt(Math.min(sz / 2, aspan * 1024 * 1024) / chunksz) * chunksz;
|
||||||
|
if (!aspan)
|
||||||
|
aspan = Math.min(sz, chunksz);
|
||||||
|
|
||||||
|
for (var side = 0; side < 2; side++) {
|
||||||
|
var ofs = side ? Math.max(0, sz - aspan) : 0,
|
||||||
|
nchunks = aspan / chunksz;
|
||||||
|
|
||||||
|
for (var chunk = 0; chunk < nchunks; chunk++) {
|
||||||
|
var bchunk = await fobj.slice(ofs, ofs + chunksz + 16).arrayBuffer(),
|
||||||
|
uchunk = new Uint8Array(bchunk, 0, bchunk.byteLength),
|
||||||
|
bofs = bstrpos(uchunk, md_ptn),
|
||||||
|
absofs = Math.min(ofs + bofs, (sz - ofs) + bofs),
|
||||||
|
txt = bofs < 0 ? '' : textdec.decode(uchunk.subarray(bofs)),
|
||||||
|
m;
|
||||||
|
|
||||||
|
//console.log(`side ${ side }, chunk ${ chunk }, ofs ${ ofs }, bchunk ${ bchunk.byteLength }, txt ${ txt.length }`);
|
||||||
|
while (true) {
|
||||||
|
// mkv/webm have [a-z] immediately after url
|
||||||
|
m = /(youtube\.com\/watch\?v=[\w-]{11})/.exec(txt);
|
||||||
|
if (!m)
|
||||||
|
break;
|
||||||
|
|
||||||
|
txt = txt.replace(m[1], '');
|
||||||
|
m = m[1].slice(-11);
|
||||||
|
|
||||||
|
console.log(`found ${m} @${bofs}, ${name} `);
|
||||||
|
yt_ids.add(m);
|
||||||
|
if (!has(fn_ids, m) && !has(md_ids, m)) {
|
||||||
|
md_ids.push(m);
|
||||||
|
md_only.push(`${m} ${name}`);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
// id appears several times; make it preferred
|
||||||
|
md_ids.unshift(m);
|
||||||
|
|
||||||
|
// bail after next iteration
|
||||||
|
chunk = nchunks - 1;
|
||||||
|
side = 9;
|
||||||
|
|
||||||
|
if (mofs < absofs) {
|
||||||
|
mofs = absofs;
|
||||||
|
mfile = name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ofs += chunksz;
|
||||||
|
if (ofs >= sz)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var yi of md_ids)
|
||||||
|
ids.push(yi);
|
||||||
|
|
||||||
|
for (var yi of fn_ids)
|
||||||
|
if (!has(ids, yi))
|
||||||
|
ids.push(yi);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (md_only.length)
|
||||||
|
console.log('recovered the following youtube-IDs by inspecting metadata:\n\n' + md_only.join('\n'));
|
||||||
|
else if (yt_ids.size)
|
||||||
|
console.log('did not discover any additional youtube-IDs by inspecting metadata; all the IDs also existed in the filenames');
|
||||||
|
else
|
||||||
|
console.log('failed to find any youtube-IDs at all, sorry');
|
||||||
|
|
||||||
|
if (false) {
|
||||||
|
var msg = `finished analysing ${mnchk} files in ${(Date.now() - t0) / 1000} seconds,\n\nbiggest offset was ${mofs} in this file:\n\n${mfile}`,
|
||||||
|
mfun = function () { toast.ok(0, msg); };
|
||||||
|
|
||||||
|
mfun();
|
||||||
|
setTimeout(mfun, 200);
|
||||||
|
|
||||||
|
return hooks[0]([], [], [], hooks.slice(1));
|
||||||
|
}
|
||||||
|
|
||||||
|
var el = ebi('unick'), unick = el ? el.value : '';
|
||||||
|
if (unick) {
|
||||||
|
console.log(`sending uploader nickname [${unick}]`);
|
||||||
|
fetch(document.location, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/x-www-form-urlencoded;charset=UTF-8' },
|
||||||
|
body: 'msg=' + encodeURIComponent(unick)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
toast.inf(5, `running query for ${yt_ids.size} youtube-IDs...`);
|
||||||
|
|
||||||
|
var xhr = new XHR();
|
||||||
|
xhr.open('POST', '/ytq', true);
|
||||||
|
xhr.setRequestHeader('Content-Type', 'text/plain');
|
||||||
|
xhr.onload = xhr.onerror = function () {
|
||||||
|
if (this.status != 200)
|
||||||
|
return toast.err(0, `sorry, database query failed ;_;\n\nplease let us know so we can look at it, thx!!\n\nerror ${this.status}: ${(this.response && this.response.err) || this.responseText}`);
|
||||||
|
|
||||||
|
process_id_list(this.responseText);
|
||||||
|
};
|
||||||
|
xhr.send(Array.from(yt_ids).join('\n'));
|
||||||
|
|
||||||
|
function process_id_list(txt) {
|
||||||
|
var wanted_ids = new Set(txt.trim().split('\n')),
|
||||||
|
name_id = {},
|
||||||
|
wanted_names = new Set(), // basenames with a wanted ID -- not including relpath
|
||||||
|
wanted_names_scoped = {}, // basenames with a wanted ID -> list of dirs to search under
|
||||||
|
wanted_files = new Set(); // filedrops
|
||||||
|
|
||||||
|
for (var a = 0; a < good_files.length; a++) {
|
||||||
|
var name = good_files[a][1];
|
||||||
|
for (var b = 0; b < file_ids[a].length; b++)
|
||||||
|
if (wanted_ids.has(file_ids[a][b])) {
|
||||||
|
// let the next stage handle this to prevent dupes
|
||||||
|
//wanted_files.add(good_files[a]);
|
||||||
|
|
||||||
|
var m = /(.*)\.(mp4|webm|mkv|flv|opus|ogg|mp3|m4a|aac)$/i.exec(name);
|
||||||
|
if (!m)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
var [rd, fn] = vsplit(m[1]);
|
||||||
|
|
||||||
|
if (fn in wanted_names_scoped)
|
||||||
|
wanted_names_scoped[fn].push(rd);
|
||||||
|
else
|
||||||
|
wanted_names_scoped[fn] = [rd];
|
||||||
|
|
||||||
|
wanted_names.add(fn);
|
||||||
|
name_id[m[1]] = file_ids[a][b];
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// add all files with the same basename as each explicitly wanted file
|
||||||
|
// (infojson/chatlog/etc when ID was discovered from metadata)
|
||||||
|
for (var a = 0; a < good_files.length; a++) {
|
||||||
|
var [rd, name] = vsplit(good_files[a][1]);
|
||||||
|
for (var b = 0; b < 3; b++) {
|
||||||
|
name = name.replace(/\.[^\.]+$/, '');
|
||||||
|
if (!wanted_names.has(name))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
var vid_fp = false;
|
||||||
|
for (var c of wanted_names_scoped[name])
|
||||||
|
if (rd.startsWith(c))
|
||||||
|
vid_fp = c + name;
|
||||||
|
|
||||||
|
if (!vid_fp)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
var subdir = name_id[vid_fp];
|
||||||
|
subdir = `v${subdir.slice(0, 1)}/${subdir}-${myid}`;
|
||||||
|
var newpath = subdir + '/' + good_files[a][1].split(/\//g).pop();
|
||||||
|
|
||||||
|
// check if this file is a dupe
|
||||||
|
for (var c of good_files)
|
||||||
|
if (c[1] == newpath)
|
||||||
|
newpath = null;
|
||||||
|
|
||||||
|
if (!newpath)
|
||||||
|
break;
|
||||||
|
|
||||||
|
good_files[a][1] = newpath;
|
||||||
|
wanted_files.add(good_files[a]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function upload_filtered() {
|
||||||
|
if (!wanted_files.size)
|
||||||
|
return modal.alert('Good news -- turns out we already have all those.\n\nBut thank you for checking in!');
|
||||||
|
|
||||||
|
hooks[0](Array.from(wanted_files), nil_files, bad_files, hooks.slice(1));
|
||||||
|
}
|
||||||
|
|
||||||
|
function upload_all() {
|
||||||
|
hooks[0](good_files, nil_files, bad_files, hooks.slice(1));
|
||||||
|
}
|
||||||
|
|
||||||
|
var n_skip = good_files.length - wanted_files.size,
|
||||||
|
msg = `you added ${good_files.length} files; ${good_files.length == n_skip ? 'all' : n_skip} of them were skipped --\neither because we already have them,\nor because there is no youtube-ID in your filenames.\n\n<code>OK</code> / <code>Enter</code> = continue uploading just the ${wanted_files.size} files we definitely need\n\n<code>Cancel</code> / <code>ESC</code> = override the filter; upload ALL the files you added`;
|
||||||
|
|
||||||
|
if (!n_skip)
|
||||||
|
upload_filtered();
|
||||||
|
else
|
||||||
|
modal.confirm(msg, upload_filtered, upload_all);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
up2k_hooks.push(function () {
|
||||||
|
up2k.gotallfiles.unshift(up2k_namefilter);
|
||||||
|
});
|
||||||
|
|
||||||
|
// persist/restore nickname field if present
|
||||||
|
setInterval(function () {
|
||||||
|
var o = ebi('unick');
|
||||||
|
if (!o || document.activeElement == o)
|
||||||
|
return;
|
||||||
|
|
||||||
|
o.oninput = function () {
|
||||||
|
localStorage.setItem('unick', o.value);
|
||||||
|
};
|
||||||
|
o.value = localStorage.getItem('unick') || '';
|
||||||
|
}, 1000);
|
||||||
45
contrib/plugins/up2k-hooks.js
Normal file
45
contrib/plugins/up2k-hooks.js
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
// hooks into up2k
|
||||||
|
|
||||||
|
function up2k_namefilter(good_files, nil_files, bad_files, hooks) {
|
||||||
|
// is called when stuff is dropped into the browser,
|
||||||
|
// after iterating through the directory tree and discovering all files,
|
||||||
|
// before the upload confirmation dialogue is shown
|
||||||
|
|
||||||
|
// good_files will successfully upload
|
||||||
|
// nil_files are empty files and will show an alert in the final hook
|
||||||
|
// bad_files are unreadable and cannot be uploaded
|
||||||
|
var file_lists = [good_files, nil_files, bad_files];
|
||||||
|
|
||||||
|
// build a list of filenames
|
||||||
|
var filenames = [];
|
||||||
|
for (var lst of file_lists)
|
||||||
|
for (var ent of lst)
|
||||||
|
filenames.push(ent[1]);
|
||||||
|
|
||||||
|
toast.inf(5, "running database query...");
|
||||||
|
|
||||||
|
// simulate delay while passing the list to some api for checking
|
||||||
|
setTimeout(function () {
|
||||||
|
|
||||||
|
// only keep webm files as an example
|
||||||
|
var new_lists = [];
|
||||||
|
for (var lst of file_lists) {
|
||||||
|
var keep = [];
|
||||||
|
new_lists.push(keep);
|
||||||
|
|
||||||
|
for (var ent of lst)
|
||||||
|
if (/\.webm$/.test(ent[1]))
|
||||||
|
keep.push(ent);
|
||||||
|
}
|
||||||
|
|
||||||
|
// finally, call the next hook in the chain
|
||||||
|
[good_files, nil_files, bad_files] = new_lists;
|
||||||
|
hooks[0](good_files, nil_files, bad_files, hooks.slice(1));
|
||||||
|
|
||||||
|
}, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
// register
|
||||||
|
up2k_hooks.push(function () {
|
||||||
|
up2k.gotallfiles.unshift(up2k_namefilter);
|
||||||
|
});
|
||||||
23
contrib/systemd/cfssl.service
Normal file
23
contrib/systemd/cfssl.service
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
# systemd service which generates a new TLS certificate on each boot,
|
||||||
|
# that way the one-year expiry time won't cause any issues --
|
||||||
|
# just have everyone trust the ca.pem once every 10 years
|
||||||
|
#
|
||||||
|
# assumptions/placeholder values:
|
||||||
|
# * this script and copyparty runs as user "cpp"
|
||||||
|
# * copyparty repo is at ~cpp/dev/copyparty
|
||||||
|
# * CA is named partylan
|
||||||
|
# * server IPs = 10.1.2.3 and 192.168.123.1
|
||||||
|
# * server hostname = party.lan
|
||||||
|
|
||||||
|
[Unit]
|
||||||
|
Description=copyparty certificate generator
|
||||||
|
Before=copyparty.service
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
User=cpp
|
||||||
|
Type=oneshot
|
||||||
|
SyslogIdentifier=cpp-cert
|
||||||
|
ExecStart=/bin/bash -c 'cd ~/dev/copyparty/contrib && ./cfssl.sh partylan 10.1.2.3,192.168.123.1,party.lan y'
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
@@ -2,16 +2,22 @@
|
|||||||
# and share '/mnt' with anonymous read+write
|
# and share '/mnt' with anonymous read+write
|
||||||
#
|
#
|
||||||
# installation:
|
# installation:
|
||||||
# cp -pv copyparty.service /etc/systemd/system && systemctl enable --now copyparty
|
# cp -pv copyparty.service /etc/systemd/system
|
||||||
# restorecon -vr /etc/systemd/system/copyparty.service
|
# restorecon -vr /etc/systemd/system/copyparty.service
|
||||||
# firewall-cmd --permanent --add-port={80,443,3923}/tcp
|
# firewall-cmd --permanent --add-port={80,443,3923}/tcp # --zone=libvirt
|
||||||
# firewall-cmd --reload
|
# firewall-cmd --reload
|
||||||
|
# systemctl daemon-reload && systemctl enable --now copyparty
|
||||||
#
|
#
|
||||||
# you may want to:
|
# you may want to:
|
||||||
|
# change "User=cpp" and "/home/cpp/" to another user
|
||||||
|
# remove the nft lines to only listen on port 3923
|
||||||
|
# and in the ExecStart= line:
|
||||||
# change '/usr/bin/python3' to another interpreter
|
# change '/usr/bin/python3' to another interpreter
|
||||||
# change '/mnt::rw' to another location or permission-set
|
# change '/mnt::rw' to another location or permission-set
|
||||||
# remove '-p 80,443,3923' to only listen on port 3923
|
# add '-q' to disable logging on busy servers
|
||||||
# add '-i 127.0.0.1' to only allow local connections
|
# add '-i 127.0.0.1' to only allow local connections
|
||||||
|
# add '-e2dsa' to enable filesystem scanning + indexing
|
||||||
|
# add '-e2ts' to enable metadata indexing
|
||||||
#
|
#
|
||||||
# with `Type=notify`, copyparty will signal systemd when it is ready to
|
# with `Type=notify`, copyparty will signal systemd when it is ready to
|
||||||
# accept connections; correctly delaying units depending on copyparty.
|
# accept connections; correctly delaying units depending on copyparty.
|
||||||
@@ -19,9 +25,11 @@
|
|||||||
# python disabling line-buffering, so messages are out-of-order:
|
# python disabling line-buffering, so messages are out-of-order:
|
||||||
# https://user-images.githubusercontent.com/241032/126040249-cb535cc7-c599-4931-a796-a5d9af691bad.png
|
# https://user-images.githubusercontent.com/241032/126040249-cb535cc7-c599-4931-a796-a5d9af691bad.png
|
||||||
#
|
#
|
||||||
# if you remove -q to enable logging, you may also want to remove the
|
# unless you add -q to disable logging, you may want to remove the
|
||||||
# following line to enable buffering (slightly better performance):
|
# following line to allow buffering (slightly better performance):
|
||||||
# Environment=PYTHONUNBUFFERED=x
|
# Environment=PYTHONUNBUFFERED=x
|
||||||
|
#
|
||||||
|
# keep ExecStartPre before ExecStart, at least on rhel8
|
||||||
|
|
||||||
[Unit]
|
[Unit]
|
||||||
Description=copyparty file server
|
Description=copyparty file server
|
||||||
@@ -31,8 +39,23 @@ Type=notify
|
|||||||
SyslogIdentifier=copyparty
|
SyslogIdentifier=copyparty
|
||||||
Environment=PYTHONUNBUFFERED=x
|
Environment=PYTHONUNBUFFERED=x
|
||||||
ExecReload=/bin/kill -s USR1 $MAINPID
|
ExecReload=/bin/kill -s USR1 $MAINPID
|
||||||
ExecStartPre=/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf'
|
|
||||||
ExecStart=/usr/bin/python3 /usr/local/bin/copyparty-sfx.py -q -p 80,443,3923 -v /mnt::rw
|
# user to run as + where the TLS certificate is (if any)
|
||||||
|
User=cpp
|
||||||
|
Environment=XDG_CONFIG_HOME=/home/cpp/.config
|
||||||
|
|
||||||
|
# setup forwarding from ports 80 and 443 to port 3923
|
||||||
|
ExecStartPre=+/bin/bash -c 'nft -n -a list table nat | awk "/ to :3923 /{print\$NF}" | xargs -rL1 nft delete rule nat prerouting handle; true'
|
||||||
|
ExecStartPre=+nft add table ip nat
|
||||||
|
ExecStartPre=+nft -- add chain ip nat prerouting { type nat hook prerouting priority -100 \; }
|
||||||
|
ExecStartPre=+nft add rule ip nat prerouting tcp dport 80 redirect to :3923
|
||||||
|
ExecStartPre=+nft add rule ip nat prerouting tcp dport 443 redirect to :3923
|
||||||
|
|
||||||
|
# stop systemd-tmpfiles-clean.timer from deleting copyparty while it's running
|
||||||
|
ExecStartPre=+/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf'
|
||||||
|
|
||||||
|
# copyparty settings
|
||||||
|
ExecStart=/usr/bin/python3 /usr/local/bin/copyparty-sfx.py -e2d -v /mnt::rw
|
||||||
|
|
||||||
[Install]
|
[Install]
|
||||||
WantedBy=multi-user.target
|
WantedBy=multi-user.target
|
||||||
|
|||||||
@@ -6,12 +6,17 @@
|
|||||||
# 1) put copyparty-sfx.py and prisonparty.sh in /usr/local/bin
|
# 1) put copyparty-sfx.py and prisonparty.sh in /usr/local/bin
|
||||||
# 2) cp -pv prisonparty.service /etc/systemd/system && systemctl enable --now prisonparty
|
# 2) cp -pv prisonparty.service /etc/systemd/system && systemctl enable --now prisonparty
|
||||||
#
|
#
|
||||||
|
# expose additional filesystem locations to copyparty
|
||||||
|
# by listing them between the last `1000` and `--`
|
||||||
|
#
|
||||||
|
# `1000 1000` = what user to run copyparty as
|
||||||
|
#
|
||||||
# you may want to:
|
# you may want to:
|
||||||
# change '/mnt::rw' to another location or permission-set
|
# change '/mnt::rw' to another location or permission-set
|
||||||
# (remember to change the '/mnt' chroot arg too)
|
# (remember to change the '/mnt' chroot arg too)
|
||||||
#
|
#
|
||||||
# enable line-buffering for realtime logging (slight performance cost):
|
# unless you add -q to disable logging, you may want to remove the
|
||||||
# inside the [Service] block, add the following line:
|
# following line to allow buffering (slightly better performance):
|
||||||
# Environment=PYTHONUNBUFFERED=x
|
# Environment=PYTHONUNBUFFERED=x
|
||||||
|
|
||||||
[Unit]
|
[Unit]
|
||||||
@@ -19,7 +24,14 @@ Description=copyparty file server
|
|||||||
|
|
||||||
[Service]
|
[Service]
|
||||||
SyslogIdentifier=prisonparty
|
SyslogIdentifier=prisonparty
|
||||||
WorkingDirectory=/usr/local/bin
|
Environment=PYTHONUNBUFFERED=x
|
||||||
|
WorkingDirectory=/var/lib/copyparty-jail
|
||||||
|
ExecReload=/bin/kill -s USR1 $MAINPID
|
||||||
|
|
||||||
|
# stop systemd-tmpfiles-clean.timer from deleting copyparty while it's running
|
||||||
|
ExecStartPre=+/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf'
|
||||||
|
|
||||||
|
# run copyparty
|
||||||
ExecStart=/bin/bash /usr/local/bin/prisonparty.sh /var/lib/copyparty-jail 1000 1000 /mnt -- \
|
ExecStart=/bin/bash /usr/local/bin/prisonparty.sh /var/lib/copyparty-jail 1000 1000 /mnt -- \
|
||||||
/usr/bin/python3 /usr/local/bin/copyparty-sfx.py -q -v /mnt::rw
|
/usr/bin/python3 /usr/local/bin/copyparty-sfx.py -q -v /mnt::rw
|
||||||
|
|
||||||
|
|||||||
45
contrib/webdav-cfg.bat
Normal file
45
contrib/webdav-cfg.bat
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
@echo off
|
||||||
|
rem removes the 47.6 MiB filesize limit when downloading from webdav
|
||||||
|
rem + optionally allows/enables password-auth over plaintext http
|
||||||
|
rem + optionally helps disable wpad, removing the 10sec latency
|
||||||
|
|
||||||
|
net session >nul 2>&1
|
||||||
|
if %errorlevel% neq 0 (
|
||||||
|
echo sorry, you must run this as administrator
|
||||||
|
pause
|
||||||
|
exit /b
|
||||||
|
)
|
||||||
|
|
||||||
|
reg add HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\services\WebClient\Parameters /v FileSizeLimitInBytes /t REG_DWORD /d 0xffffffff /f
|
||||||
|
reg add HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\WebClient\Parameters /v FsCtlRequestTimeoutInSec /t REG_DWORD /d 0xffffffff /f
|
||||||
|
|
||||||
|
echo(
|
||||||
|
echo OK;
|
||||||
|
echo allow webdav basic-auth over plaintext http?
|
||||||
|
echo Y: login works, but the password will be visible in wireshark etc
|
||||||
|
echo N: login will NOT work unless you use https and valid certificates
|
||||||
|
choice
|
||||||
|
if %errorlevel% equ 1 (
|
||||||
|
reg add HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\services\WebClient\Parameters /v BasicAuthLevel /t REG_DWORD /d 0x2 /f
|
||||||
|
rem default is 1 (require tls)
|
||||||
|
)
|
||||||
|
|
||||||
|
echo(
|
||||||
|
echo OK;
|
||||||
|
echo do you want to disable wpad?
|
||||||
|
echo can give a HUGE speed boost depending on network settings
|
||||||
|
choice
|
||||||
|
if %errorlevel% equ 1 (
|
||||||
|
echo(
|
||||||
|
echo i'm about to open the [Connections] tab in [Internet Properties] for you;
|
||||||
|
echo please click [LAN settings] and disable [Automatically detect settings]
|
||||||
|
echo(
|
||||||
|
pause
|
||||||
|
control inetcpl.cpl,,4
|
||||||
|
)
|
||||||
|
|
||||||
|
net stop webclient
|
||||||
|
net start webclient
|
||||||
|
echo(
|
||||||
|
echo OK; all done
|
||||||
|
pause
|
||||||
@@ -1,80 +1,53 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import platform
|
|
||||||
import time
|
|
||||||
import sys
|
|
||||||
import os
|
import os
|
||||||
|
import platform
|
||||||
|
import sys
|
||||||
|
import time
|
||||||
|
|
||||||
PY2 = sys.version_info[0] == 2
|
try:
|
||||||
if PY2:
|
from typing import TYPE_CHECKING
|
||||||
sys.dont_write_bytecode = True
|
except:
|
||||||
unicode = unicode
|
TYPE_CHECKING = False
|
||||||
|
|
||||||
|
if True:
|
||||||
|
from typing import Any, Callable
|
||||||
|
|
||||||
|
PY2 = sys.version_info < (3,)
|
||||||
|
if not PY2:
|
||||||
|
unicode: Callable[[Any], str] = str
|
||||||
else:
|
else:
|
||||||
unicode = str
|
sys.dont_write_bytecode = True
|
||||||
|
unicode = unicode # noqa: F821 # pylint: disable=undefined-variable,self-assigning-variable
|
||||||
|
|
||||||
WINDOWS = False
|
WINDOWS: Any = (
|
||||||
if platform.system() == "Windows":
|
[int(x) for x in platform.version().split(".")]
|
||||||
WINDOWS = [int(x) for x in platform.version().split(".")]
|
if platform.system() == "Windows"
|
||||||
|
else False
|
||||||
|
)
|
||||||
|
|
||||||
VT100 = not WINDOWS or WINDOWS >= [10, 0, 14393]
|
VT100 = not WINDOWS or WINDOWS >= [10, 0, 14393]
|
||||||
# introduced in anniversary update
|
# introduced in anniversary update
|
||||||
|
|
||||||
ANYWIN = WINDOWS or sys.platform in ["msys"]
|
ANYWIN = WINDOWS or sys.platform in ["msys", "cygwin"]
|
||||||
|
|
||||||
MACOS = platform.system() == "Darwin"
|
MACOS = platform.system() == "Darwin"
|
||||||
|
|
||||||
|
EXE = bool(getattr(sys, "frozen", False))
|
||||||
|
|
||||||
def get_unixdir():
|
try:
|
||||||
paths = [
|
CORES = len(os.sched_getaffinity(0))
|
||||||
(os.environ.get, "XDG_CONFIG_HOME"),
|
except:
|
||||||
(os.path.expanduser, "~/.config"),
|
CORES = (os.cpu_count() if hasattr(os, "cpu_count") else 0) or 2
|
||||||
(os.environ.get, "TMPDIR"),
|
|
||||||
(os.environ.get, "TEMP"),
|
|
||||||
(os.environ.get, "TMP"),
|
|
||||||
(unicode, "/tmp"),
|
|
||||||
]
|
|
||||||
for chk in [os.listdir, os.mkdir]:
|
|
||||||
for pf, pa in paths:
|
|
||||||
try:
|
|
||||||
p = pf(pa)
|
|
||||||
# print(chk.__name__, p, pa)
|
|
||||||
if not p or p.startswith("~"):
|
|
||||||
continue
|
|
||||||
|
|
||||||
p = os.path.normpath(p)
|
|
||||||
chk(p)
|
|
||||||
p = os.path.join(p, "copyparty")
|
|
||||||
if not os.path.isdir(p):
|
|
||||||
os.mkdir(p)
|
|
||||||
|
|
||||||
return p
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
|
|
||||||
raise Exception("could not find a writable path for config")
|
|
||||||
|
|
||||||
|
|
||||||
class EnvParams(object):
|
class EnvParams(object):
|
||||||
def __init__(self):
|
def __init__(self) -> None:
|
||||||
self.t0 = time.time()
|
self.t0 = time.time()
|
||||||
self.mod = os.path.dirname(os.path.realpath(__file__))
|
self.mod = ""
|
||||||
if self.mod.endswith("__init__"):
|
self.cfg = ""
|
||||||
self.mod = os.path.dirname(self.mod)
|
self.ox = getattr(sys, "oxidized", None)
|
||||||
|
|
||||||
if sys.platform == "win32":
|
|
||||||
self.cfg = os.path.normpath(os.environ["APPDATA"] + "/copyparty")
|
|
||||||
elif sys.platform == "darwin":
|
|
||||||
self.cfg = os.path.expanduser("~/Library/Preferences/copyparty")
|
|
||||||
else:
|
|
||||||
self.cfg = get_unixdir()
|
|
||||||
|
|
||||||
self.cfg = self.cfg.replace("\\", "/")
|
|
||||||
try:
|
|
||||||
os.makedirs(self.cfg)
|
|
||||||
except:
|
|
||||||
if not os.path.isdir(self.cfg):
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
E = EnvParams()
|
E = EnvParams()
|
||||||
|
|||||||
996
copyparty/__main__.py
Normal file → Executable file
996
copyparty/__main__.py
Normal file → Executable file
File diff suppressed because it is too large
Load Diff
@@ -1,8 +1,8 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
|
|
||||||
VERSION = (1, 1, 6)
|
VERSION = (1, 6, 6)
|
||||||
CODENAME = "opus"
|
CODENAME = "cors k"
|
||||||
BUILD_DT = (2021, 12, 7)
|
BUILD_DT = (2023, 2, 26)
|
||||||
|
|
||||||
S_VERSION = ".".join(map(str, VERSION))
|
S_VERSION = ".".join(map(str, VERSION))
|
||||||
S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT)
|
S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT)
|
||||||
|
|||||||
1470
copyparty/authsrv.py
1470
copyparty/authsrv.py
File diff suppressed because it is too large
Load Diff
@@ -2,61 +2,80 @@
|
|||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import os
|
import os
|
||||||
from ..util import fsenc, fsdec, SYMTIME
|
|
||||||
from . import path
|
|
||||||
|
|
||||||
|
from ..util import SYMTIME, fsdec, fsenc
|
||||||
|
from . import path as path
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any, Optional
|
||||||
|
|
||||||
|
_ = (path,)
|
||||||
|
__all__ = ["path"]
|
||||||
|
|
||||||
# grep -hRiE '(^|[^a-zA-Z_\.-])os\.' . | gsed -r 's/ /\n/g;s/\(/(\n/g' | grep -hRiE '(^|[^a-zA-Z_\.-])os\.' | sort | uniq -c
|
# grep -hRiE '(^|[^a-zA-Z_\.-])os\.' . | gsed -r 's/ /\n/g;s/\(/(\n/g' | grep -hRiE '(^|[^a-zA-Z_\.-])os\.' | sort | uniq -c
|
||||||
# printf 'os\.(%s)' "$(grep ^def bos/__init__.py | gsed -r 's/^def //;s/\(.*//' | tr '\n' '|' | gsed -r 's/.$//')"
|
# printf 'os\.(%s)' "$(grep ^def bos/__init__.py | gsed -r 's/^def //;s/\(.*//' | tr '\n' '|' | gsed -r 's/.$//')"
|
||||||
|
|
||||||
|
|
||||||
def chmod(p, mode):
|
def chmod(p: str, mode: int) -> None:
|
||||||
return os.chmod(fsenc(p), mode)
|
return os.chmod(fsenc(p), mode)
|
||||||
|
|
||||||
|
|
||||||
def listdir(p="."):
|
def listdir(p: str = ".") -> list[str]:
|
||||||
return [fsdec(x) for x in os.listdir(fsenc(p))]
|
return [fsdec(x) for x in os.listdir(fsenc(p))]
|
||||||
|
|
||||||
|
|
||||||
def lstat(p):
|
def makedirs(name: str, mode: int = 0o755, exist_ok: bool = True) -> bool:
|
||||||
return os.lstat(fsenc(p))
|
|
||||||
|
|
||||||
|
|
||||||
def makedirs(name, mode=0o755, exist_ok=True):
|
|
||||||
bname = fsenc(name)
|
bname = fsenc(name)
|
||||||
try:
|
try:
|
||||||
os.makedirs(bname, mode)
|
os.makedirs(bname, mode)
|
||||||
|
return True
|
||||||
except:
|
except:
|
||||||
if not exist_ok or not os.path.isdir(bname):
|
if not exist_ok or not os.path.isdir(bname):
|
||||||
raise
|
raise
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
def mkdir(p, mode=0o755):
|
def mkdir(p: str, mode: int = 0o755) -> None:
|
||||||
return os.mkdir(fsenc(p), mode)
|
return os.mkdir(fsenc(p), mode)
|
||||||
|
|
||||||
|
|
||||||
def rename(src, dst):
|
def open(p: str, *a, **ka) -> int:
|
||||||
|
return os.open(fsenc(p), *a, **ka)
|
||||||
|
|
||||||
|
|
||||||
|
def rename(src: str, dst: str) -> None:
|
||||||
return os.rename(fsenc(src), fsenc(dst))
|
return os.rename(fsenc(src), fsenc(dst))
|
||||||
|
|
||||||
|
|
||||||
def replace(src, dst):
|
def replace(src: str, dst: str) -> None:
|
||||||
return os.replace(fsenc(src), fsenc(dst))
|
return os.replace(fsenc(src), fsenc(dst))
|
||||||
|
|
||||||
|
|
||||||
def rmdir(p):
|
def rmdir(p: str) -> None:
|
||||||
return os.rmdir(fsenc(p))
|
return os.rmdir(fsenc(p))
|
||||||
|
|
||||||
|
|
||||||
def stat(p):
|
def stat(p: str) -> os.stat_result:
|
||||||
return os.stat(fsenc(p))
|
return os.stat(fsenc(p))
|
||||||
|
|
||||||
|
|
||||||
def unlink(p):
|
def unlink(p: str) -> None:
|
||||||
return os.unlink(fsenc(p))
|
return os.unlink(fsenc(p))
|
||||||
|
|
||||||
|
|
||||||
def utime(p, times=None, follow_symlinks=True):
|
def utime(
|
||||||
|
p: str, times: Optional[tuple[float, float]] = None, follow_symlinks: bool = True
|
||||||
|
) -> None:
|
||||||
if SYMTIME:
|
if SYMTIME:
|
||||||
return os.utime(fsenc(p), times, follow_symlinks=follow_symlinks)
|
return os.utime(fsenc(p), times, follow_symlinks=follow_symlinks)
|
||||||
else:
|
else:
|
||||||
return os.utime(fsenc(p), times)
|
return os.utime(fsenc(p), times)
|
||||||
|
|
||||||
|
|
||||||
|
if hasattr(os, "lstat"):
|
||||||
|
|
||||||
|
def lstat(p: str) -> os.stat_result:
|
||||||
|
return os.lstat(fsenc(p))
|
||||||
|
|
||||||
|
else:
|
||||||
|
lstat = stat
|
||||||
|
|||||||
@@ -2,39 +2,44 @@
|
|||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import os
|
import os
|
||||||
from ..util import fsenc, fsdec, SYMTIME
|
|
||||||
|
from ..util import SYMTIME, fsdec, fsenc
|
||||||
|
|
||||||
|
|
||||||
def abspath(p):
|
def abspath(p: str) -> str:
|
||||||
return fsdec(os.path.abspath(fsenc(p)))
|
return fsdec(os.path.abspath(fsenc(p)))
|
||||||
|
|
||||||
|
|
||||||
def exists(p):
|
def exists(p: str) -> bool:
|
||||||
return os.path.exists(fsenc(p))
|
return os.path.exists(fsenc(p))
|
||||||
|
|
||||||
|
|
||||||
def getmtime(p, follow_symlinks=True):
|
def getmtime(p: str, follow_symlinks: bool = True) -> float:
|
||||||
if not follow_symlinks and SYMTIME:
|
if not follow_symlinks and SYMTIME:
|
||||||
return os.lstat(fsenc(p)).st_mtime
|
return os.lstat(fsenc(p)).st_mtime
|
||||||
else:
|
else:
|
||||||
return os.path.getmtime(fsenc(p))
|
return os.path.getmtime(fsenc(p))
|
||||||
|
|
||||||
|
|
||||||
def getsize(p):
|
def getsize(p: str) -> int:
|
||||||
return os.path.getsize(fsenc(p))
|
return os.path.getsize(fsenc(p))
|
||||||
|
|
||||||
|
|
||||||
def isfile(p):
|
def isfile(p: str) -> bool:
|
||||||
return os.path.isfile(fsenc(p))
|
return os.path.isfile(fsenc(p))
|
||||||
|
|
||||||
|
|
||||||
def isdir(p):
|
def isdir(p: str) -> bool:
|
||||||
return os.path.isdir(fsenc(p))
|
return os.path.isdir(fsenc(p))
|
||||||
|
|
||||||
|
|
||||||
def islink(p):
|
def islink(p: str) -> bool:
|
||||||
return os.path.islink(fsenc(p))
|
return os.path.islink(fsenc(p))
|
||||||
|
|
||||||
|
|
||||||
def realpath(p):
|
def lexists(p: str) -> bool:
|
||||||
|
return os.path.lexists(fsenc(p))
|
||||||
|
|
||||||
|
|
||||||
|
def realpath(p: str) -> str:
|
||||||
return fsdec(os.path.realpath(fsenc(p)))
|
return fsdec(os.path.realpath(fsenc(p)))
|
||||||
|
|||||||
@@ -1,52 +1,64 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import time
|
|
||||||
import threading
|
import threading
|
||||||
|
import time
|
||||||
|
import traceback
|
||||||
|
|
||||||
from .broker_util import try_exec
|
import queue
|
||||||
|
|
||||||
|
from .__init__ import CORES, TYPE_CHECKING
|
||||||
from .broker_mpw import MpWorker
|
from .broker_mpw import MpWorker
|
||||||
from .util import mp
|
from .broker_util import try_exec
|
||||||
|
from .util import Daemon, mp
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from .svchub import SvcHub
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
|
||||||
|
class MProcess(mp.Process):
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
q_pend: queue.Queue[tuple[int, str, list[Any]]],
|
||||||
|
q_yield: queue.Queue[tuple[int, str, list[Any]]],
|
||||||
|
target: Any,
|
||||||
|
args: Any,
|
||||||
|
) -> None:
|
||||||
|
super(MProcess, self).__init__(target=target, args=args)
|
||||||
|
self.q_pend = q_pend
|
||||||
|
self.q_yield = q_yield
|
||||||
|
|
||||||
|
|
||||||
class BrokerMp(object):
|
class BrokerMp(object):
|
||||||
"""external api; manages MpWorkers"""
|
"""external api; manages MpWorkers"""
|
||||||
|
|
||||||
def __init__(self, hub):
|
def __init__(self, hub: "SvcHub") -> None:
|
||||||
self.hub = hub
|
self.hub = hub
|
||||||
self.log = hub.log
|
self.log = hub.log
|
||||||
self.args = hub.args
|
self.args = hub.args
|
||||||
|
|
||||||
self.procs = []
|
self.procs = []
|
||||||
self.retpend = {}
|
|
||||||
self.retpend_mutex = threading.Lock()
|
|
||||||
self.mutex = threading.Lock()
|
self.mutex = threading.Lock()
|
||||||
|
|
||||||
self.num_workers = self.args.j or mp.cpu_count()
|
self.num_workers = self.args.j or CORES
|
||||||
self.log("broker", "booting {} subprocesses".format(self.num_workers))
|
self.log("broker", "booting {} subprocesses".format(self.num_workers))
|
||||||
for n in range(1, self.num_workers + 1):
|
for n in range(1, self.num_workers + 1):
|
||||||
q_pend = mp.Queue(1)
|
q_pend: queue.Queue[tuple[int, str, list[Any]]] = mp.Queue(1)
|
||||||
q_yield = mp.Queue(64)
|
q_yield: queue.Queue[tuple[int, str, list[Any]]] = mp.Queue(64)
|
||||||
|
|
||||||
proc = mp.Process(target=MpWorker, args=(q_pend, q_yield, self.args, n))
|
|
||||||
proc.q_pend = q_pend
|
|
||||||
proc.q_yield = q_yield
|
|
||||||
proc.clients = {}
|
|
||||||
|
|
||||||
thr = threading.Thread(
|
|
||||||
target=self.collector, args=(proc,), name="mp-sink-{}".format(n)
|
|
||||||
)
|
|
||||||
thr.daemon = True
|
|
||||||
thr.start()
|
|
||||||
|
|
||||||
|
proc = MProcess(q_pend, q_yield, MpWorker, (q_pend, q_yield, self.args, n))
|
||||||
|
Daemon(self.collector, "mp-sink-{}".format(n), (proc,))
|
||||||
self.procs.append(proc)
|
self.procs.append(proc)
|
||||||
proc.start()
|
proc.start()
|
||||||
|
|
||||||
def shutdown(self):
|
def shutdown(self) -> None:
|
||||||
self.log("broker", "shutting down")
|
self.log("broker", "shutting down")
|
||||||
for n, proc in enumerate(self.procs):
|
for n, proc in enumerate(self.procs):
|
||||||
thr = threading.Thread(
|
thr = threading.Thread(
|
||||||
target=proc.q_pend.put([0, "shutdown", []]),
|
target=proc.q_pend.put((0, "shutdown", [])),
|
||||||
name="mp-shutdown-{}-{}".format(n, len(self.procs)),
|
name="mp-shutdown-{}-{}".format(n, len(self.procs)),
|
||||||
)
|
)
|
||||||
thr.start()
|
thr.start()
|
||||||
@@ -62,12 +74,12 @@ class BrokerMp(object):
|
|||||||
|
|
||||||
procs.pop()
|
procs.pop()
|
||||||
|
|
||||||
def reload(self):
|
def reload(self) -> None:
|
||||||
self.log("broker", "reloading")
|
self.log("broker", "reloading")
|
||||||
for _, proc in enumerate(self.procs):
|
for _, proc in enumerate(self.procs):
|
||||||
proc.q_pend.put([0, "reload", []])
|
proc.q_pend.put((0, "reload", []))
|
||||||
|
|
||||||
def collector(self, proc):
|
def collector(self, proc: MProcess) -> None:
|
||||||
"""receive message from hub in other process"""
|
"""receive message from hub in other process"""
|
||||||
while True:
|
while True:
|
||||||
msg = proc.q_yield.get()
|
msg = proc.q_yield.get()
|
||||||
@@ -78,24 +90,24 @@ class BrokerMp(object):
|
|||||||
|
|
||||||
elif dest == "retq":
|
elif dest == "retq":
|
||||||
# response from previous ipc call
|
# response from previous ipc call
|
||||||
with self.retpend_mutex:
|
raise Exception("invalid broker_mp usage")
|
||||||
retq = self.retpend.pop(retq_id)
|
|
||||||
|
|
||||||
retq.put(args)
|
|
||||||
|
|
||||||
else:
|
else:
|
||||||
# new ipc invoking managed service in hub
|
# new ipc invoking managed service in hub
|
||||||
obj = self.hub
|
try:
|
||||||
for node in dest.split("."):
|
obj = self.hub
|
||||||
obj = getattr(obj, node)
|
for node in dest.split("."):
|
||||||
|
obj = getattr(obj, node)
|
||||||
|
|
||||||
# TODO will deadlock if dest performs another ipc
|
# TODO will deadlock if dest performs another ipc
|
||||||
rv = try_exec(retq_id, obj, *args)
|
rv = try_exec(retq_id, obj, *args)
|
||||||
|
except:
|
||||||
|
rv = ["exception", "stack", traceback.format_exc()]
|
||||||
|
|
||||||
if retq_id:
|
if retq_id:
|
||||||
proc.q_pend.put([retq_id, "retq", rv])
|
proc.q_pend.put((retq_id, "retq", rv))
|
||||||
|
|
||||||
def put(self, want_retval, dest, *args):
|
def say(self, dest: str, *args: Any) -> None:
|
||||||
"""
|
"""
|
||||||
send message to non-hub component in other process,
|
send message to non-hub component in other process,
|
||||||
returns a Queue object which eventually contains the response if want_retval
|
returns a Queue object which eventually contains the response if want_retval
|
||||||
@@ -103,7 +115,11 @@ class BrokerMp(object):
|
|||||||
"""
|
"""
|
||||||
if dest == "listen":
|
if dest == "listen":
|
||||||
for p in self.procs:
|
for p in self.procs:
|
||||||
p.q_pend.put([0, dest, [args[0], len(self.procs)]])
|
p.q_pend.put((0, dest, [args[0], len(self.procs)]))
|
||||||
|
|
||||||
|
elif dest == "set_netdevs":
|
||||||
|
for p in self.procs:
|
||||||
|
p.q_pend.put((0, dest, list(args)))
|
||||||
|
|
||||||
elif dest == "cb_httpsrv_up":
|
elif dest == "cb_httpsrv_up":
|
||||||
self.hub.cb_httpsrv_up()
|
self.hub.cb_httpsrv_up()
|
||||||
|
|||||||
@@ -1,20 +1,38 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import sys
|
import argparse
|
||||||
|
import os
|
||||||
import signal
|
import signal
|
||||||
|
import sys
|
||||||
import threading
|
import threading
|
||||||
|
|
||||||
from .broker_util import ExceptionalQueue
|
import queue
|
||||||
|
|
||||||
|
from .__init__ import ANYWIN
|
||||||
|
from .authsrv import AuthSrv
|
||||||
|
from .broker_util import BrokerCli, ExceptionalQueue
|
||||||
from .httpsrv import HttpSrv
|
from .httpsrv import HttpSrv
|
||||||
from .util import FAKE_MP
|
from .util import FAKE_MP, Daemon, HMaccas
|
||||||
from copyparty.authsrv import AuthSrv
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from types import FrameType
|
||||||
|
|
||||||
|
from typing import Any, Optional, Union
|
||||||
|
|
||||||
|
|
||||||
class MpWorker(object):
|
class MpWorker(BrokerCli):
|
||||||
"""one single mp instance"""
|
"""one single mp instance"""
|
||||||
|
|
||||||
def __init__(self, q_pend, q_yield, args, n):
|
def __init__(
|
||||||
|
self,
|
||||||
|
q_pend: queue.Queue[tuple[int, str, list[Any]]],
|
||||||
|
q_yield: queue.Queue[tuple[int, str, list[Any]]],
|
||||||
|
args: argparse.Namespace,
|
||||||
|
n: int,
|
||||||
|
) -> None:
|
||||||
|
super(MpWorker, self).__init__()
|
||||||
|
|
||||||
self.q_pend = q_pend
|
self.q_pend = q_pend
|
||||||
self.q_yield = q_yield
|
self.q_yield = q_yield
|
||||||
self.args = args
|
self.args = args
|
||||||
@@ -22,43 +40,45 @@ class MpWorker(object):
|
|||||||
|
|
||||||
self.log = self._log_disabled if args.q and not args.lo else self._log_enabled
|
self.log = self._log_disabled if args.q and not args.lo else self._log_enabled
|
||||||
|
|
||||||
self.retpend = {}
|
self.retpend: dict[int, Any] = {}
|
||||||
self.retpend_mutex = threading.Lock()
|
self.retpend_mutex = threading.Lock()
|
||||||
self.mutex = threading.Lock()
|
self.mutex = threading.Lock()
|
||||||
|
|
||||||
# we inherited signal_handler from parent,
|
# we inherited signal_handler from parent,
|
||||||
# replace it with something harmless
|
# replace it with something harmless
|
||||||
if not FAKE_MP:
|
if not FAKE_MP:
|
||||||
for sig in [signal.SIGINT, signal.SIGTERM, signal.SIGUSR1]:
|
sigs = [signal.SIGINT, signal.SIGTERM]
|
||||||
|
if not ANYWIN:
|
||||||
|
sigs.append(signal.SIGUSR1)
|
||||||
|
|
||||||
|
for sig in sigs:
|
||||||
signal.signal(sig, self.signal_handler)
|
signal.signal(sig, self.signal_handler)
|
||||||
|
|
||||||
# starting to look like a good idea
|
# starting to look like a good idea
|
||||||
self.asrv = AuthSrv(args, None, False)
|
self.asrv = AuthSrv(args, None, False)
|
||||||
|
|
||||||
# instantiate all services here (TODO: inheritance?)
|
# instantiate all services here (TODO: inheritance?)
|
||||||
|
self.iphash = HMaccas(os.path.join(self.args.E.cfg, "iphash"), 8)
|
||||||
self.httpsrv = HttpSrv(self, n)
|
self.httpsrv = HttpSrv(self, n)
|
||||||
|
|
||||||
# on winxp and some other platforms,
|
# on winxp and some other platforms,
|
||||||
# use thr.join() to block all signals
|
# use thr.join() to block all signals
|
||||||
thr = threading.Thread(target=self.main, name="mpw-main")
|
Daemon(self.main, "mpw-main").join()
|
||||||
thr.daemon = True
|
|
||||||
thr.start()
|
|
||||||
thr.join()
|
|
||||||
|
|
||||||
def signal_handler(self, sig, frame):
|
def signal_handler(self, sig: Optional[int], frame: Optional[FrameType]) -> None:
|
||||||
# print('k')
|
# print('k')
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def _log_enabled(self, src, msg, c=0):
|
def _log_enabled(self, src: str, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
self.q_yield.put([0, "log", [src, msg, c]])
|
self.q_yield.put((0, "log", [src, msg, c]))
|
||||||
|
|
||||||
def _log_disabled(self, src, msg, c=0):
|
def _log_disabled(self, src: str, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def logw(self, msg, c=0):
|
def logw(self, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
self.log("mp{}".format(self.n), msg, c)
|
self.log("mp{}".format(self.n), msg, c)
|
||||||
|
|
||||||
def main(self):
|
def main(self) -> None:
|
||||||
while True:
|
while True:
|
||||||
retq_id, dest, args = self.q_pend.get()
|
retq_id, dest, args = self.q_pend.get()
|
||||||
|
|
||||||
@@ -77,6 +97,9 @@ class MpWorker(object):
|
|||||||
elif dest == "listen":
|
elif dest == "listen":
|
||||||
self.httpsrv.listen(args[0], args[1])
|
self.httpsrv.listen(args[0], args[1])
|
||||||
|
|
||||||
|
elif dest == "set_netdevs":
|
||||||
|
self.httpsrv.set_netdevs(args[0])
|
||||||
|
|
||||||
elif dest == "retq":
|
elif dest == "retq":
|
||||||
# response from previous ipc call
|
# response from previous ipc call
|
||||||
with self.retpend_mutex:
|
with self.retpend_mutex:
|
||||||
@@ -87,15 +110,14 @@ class MpWorker(object):
|
|||||||
else:
|
else:
|
||||||
raise Exception("what is " + str(dest))
|
raise Exception("what is " + str(dest))
|
||||||
|
|
||||||
def put(self, want_retval, dest, *args):
|
def ask(self, dest: str, *args: Any) -> ExceptionalQueue:
|
||||||
if want_retval:
|
retq = ExceptionalQueue(1)
|
||||||
retq = ExceptionalQueue(1)
|
retq_id = id(retq)
|
||||||
retq_id = id(retq)
|
with self.retpend_mutex:
|
||||||
with self.retpend_mutex:
|
self.retpend[retq_id] = retq
|
||||||
self.retpend[retq_id] = retq
|
|
||||||
else:
|
|
||||||
retq = None
|
|
||||||
retq_id = 0
|
|
||||||
|
|
||||||
self.q_yield.put([retq_id, dest, args])
|
self.q_yield.put((retq_id, dest, list(args)))
|
||||||
return retq
|
return retq
|
||||||
|
|
||||||
|
def say(self, dest: str, *args: Any) -> None:
|
||||||
|
self.q_yield.put((0, dest, list(args)))
|
||||||
|
|||||||
@@ -1,16 +1,27 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import os
|
||||||
import threading
|
import threading
|
||||||
|
|
||||||
|
from .__init__ import TYPE_CHECKING
|
||||||
|
from .broker_util import BrokerCli, ExceptionalQueue, try_exec
|
||||||
from .httpsrv import HttpSrv
|
from .httpsrv import HttpSrv
|
||||||
from .broker_util import ExceptionalQueue, try_exec
|
from .util import HMaccas
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from .svchub import SvcHub
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
|
||||||
class BrokerThr(object):
|
class BrokerThr(BrokerCli):
|
||||||
"""external api; behaves like BrokerMP but using plain threads"""
|
"""external api; behaves like BrokerMP but using plain threads"""
|
||||||
|
|
||||||
def __init__(self, hub):
|
def __init__(self, hub: "SvcHub") -> None:
|
||||||
|
super(BrokerThr, self).__init__()
|
||||||
|
|
||||||
self.hub = hub
|
self.hub = hub
|
||||||
self.log = hub.log
|
self.log = hub.log
|
||||||
self.args = hub.args
|
self.args = hub.args
|
||||||
@@ -20,32 +31,43 @@ class BrokerThr(object):
|
|||||||
self.num_workers = 1
|
self.num_workers = 1
|
||||||
|
|
||||||
# instantiate all services here (TODO: inheritance?)
|
# instantiate all services here (TODO: inheritance?)
|
||||||
|
self.iphash = HMaccas(os.path.join(self.args.E.cfg, "iphash"), 8)
|
||||||
self.httpsrv = HttpSrv(self, None)
|
self.httpsrv = HttpSrv(self, None)
|
||||||
self.reload = self.noop
|
self.reload = self.noop
|
||||||
|
|
||||||
def shutdown(self):
|
def shutdown(self) -> None:
|
||||||
# self.log("broker", "shutting down")
|
# self.log("broker", "shutting down")
|
||||||
self.httpsrv.shutdown()
|
self.httpsrv.shutdown()
|
||||||
|
|
||||||
def noop(self):
|
def noop(self) -> None:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def put(self, want_retval, dest, *args):
|
def ask(self, dest: str, *args: Any) -> ExceptionalQueue:
|
||||||
|
|
||||||
|
# new ipc invoking managed service in hub
|
||||||
|
obj = self.hub
|
||||||
|
for node in dest.split("."):
|
||||||
|
obj = getattr(obj, node)
|
||||||
|
|
||||||
|
rv = try_exec(True, obj, *args)
|
||||||
|
|
||||||
|
# pretend we're broker_mp
|
||||||
|
retq = ExceptionalQueue(1)
|
||||||
|
retq.put(rv)
|
||||||
|
return retq
|
||||||
|
|
||||||
|
def say(self, dest: str, *args: Any) -> None:
|
||||||
if dest == "listen":
|
if dest == "listen":
|
||||||
self.httpsrv.listen(args[0], 1)
|
self.httpsrv.listen(args[0], 1)
|
||||||
|
return
|
||||||
|
|
||||||
else:
|
if dest == "set_netdevs":
|
||||||
# new ipc invoking managed service in hub
|
self.httpsrv.set_netdevs(args[0])
|
||||||
obj = self.hub
|
return
|
||||||
for node in dest.split("."):
|
|
||||||
obj = getattr(obj, node)
|
|
||||||
|
|
||||||
# TODO will deadlock if dest performs another ipc
|
# new ipc invoking managed service in hub
|
||||||
rv = try_exec(want_retval, obj, *args)
|
obj = self.hub
|
||||||
if not want_retval:
|
for node in dest.split("."):
|
||||||
return
|
obj = getattr(obj, node)
|
||||||
|
|
||||||
# pretend we're broker_mp
|
try_exec(False, obj, *args)
|
||||||
retq = ExceptionalQueue(1)
|
|
||||||
retq.put(rv)
|
|
||||||
return retq
|
|
||||||
|
|||||||
@@ -1,17 +1,28 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import argparse
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from .util import Pebkac, Queue
|
from queue import Queue
|
||||||
|
|
||||||
|
from .__init__ import TYPE_CHECKING
|
||||||
|
from .authsrv import AuthSrv
|
||||||
|
from .util import HMaccas, Pebkac
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any, Optional, Union
|
||||||
|
|
||||||
|
from .util import RootLogger
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from .httpsrv import HttpSrv
|
||||||
|
|
||||||
|
|
||||||
class ExceptionalQueue(Queue, object):
|
class ExceptionalQueue(Queue, object):
|
||||||
def get(self, block=True, timeout=None):
|
def get(self, block: bool = True, timeout: Optional[float] = None) -> Any:
|
||||||
rv = super(ExceptionalQueue, self).get(block, timeout)
|
rv = super(ExceptionalQueue, self).get(block, timeout)
|
||||||
|
|
||||||
# TODO: how expensive is this?
|
|
||||||
if isinstance(rv, list):
|
if isinstance(rv, list):
|
||||||
if rv[0] == "exception":
|
if rv[0] == "exception":
|
||||||
if rv[1] == "pebkac":
|
if rv[1] == "pebkac":
|
||||||
@@ -22,7 +33,29 @@ class ExceptionalQueue(Queue, object):
|
|||||||
return rv
|
return rv
|
||||||
|
|
||||||
|
|
||||||
def try_exec(want_retval, func, *args):
|
class BrokerCli(object):
|
||||||
|
"""
|
||||||
|
helps mypy understand httpsrv.broker but still fails a few levels deeper,
|
||||||
|
for example resolving httpconn.* in httpcli -- see lines tagged #mypy404
|
||||||
|
"""
|
||||||
|
|
||||||
|
log: "RootLogger"
|
||||||
|
args: argparse.Namespace
|
||||||
|
asrv: AuthSrv
|
||||||
|
httpsrv: "HttpSrv"
|
||||||
|
iphash: HMaccas
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def ask(self, dest: str, *args: Any) -> ExceptionalQueue:
|
||||||
|
return ExceptionalQueue(1)
|
||||||
|
|
||||||
|
def say(self, dest: str, *args: Any) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def try_exec(want_retval: Union[bool, int], func: Any, *args: list[Any]) -> Any:
|
||||||
try:
|
try:
|
||||||
return func(*args)
|
return func(*args)
|
||||||
|
|
||||||
|
|||||||
150
copyparty/cfg.py
Normal file
150
copyparty/cfg.py
Normal file
@@ -0,0 +1,150 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
# awk -F\" '/add_argument\("-[^-]/{print(substr($2,2))}' copyparty/__main__.py | sort | tr '\n' ' '
|
||||||
|
zs = "a c e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vp e2vu ed emp i j lo mcr mte mth mtm mtp nb nc nid nih nw p q s ss sss v z zv"
|
||||||
|
onedash = set(zs.split())
|
||||||
|
|
||||||
|
|
||||||
|
def vf_bmap() -> dict[str, str]:
|
||||||
|
"""argv-to-volflag: simple bools"""
|
||||||
|
ret = {
|
||||||
|
"never_symlink": "neversymlink",
|
||||||
|
"no_dedup": "copydupes",
|
||||||
|
"no_dupe": "nodupe",
|
||||||
|
"no_forget": "noforget",
|
||||||
|
}
|
||||||
|
for k in (
|
||||||
|
"dotsrch",
|
||||||
|
"e2t",
|
||||||
|
"e2ts",
|
||||||
|
"e2tsr",
|
||||||
|
"e2v",
|
||||||
|
"e2vu",
|
||||||
|
"e2vp",
|
||||||
|
"hardlink",
|
||||||
|
"magic",
|
||||||
|
"no_sb_md",
|
||||||
|
"no_sb_lg",
|
||||||
|
"rand",
|
||||||
|
"xdev",
|
||||||
|
"xlink",
|
||||||
|
"xvol",
|
||||||
|
):
|
||||||
|
ret[k] = k
|
||||||
|
return ret
|
||||||
|
|
||||||
|
|
||||||
|
def vf_vmap() -> dict[str, str]:
|
||||||
|
"""argv-to-volflag: simple values"""
|
||||||
|
ret = {}
|
||||||
|
for k in ("lg_sbf", "md_sbf"):
|
||||||
|
ret[k] = k
|
||||||
|
return ret
|
||||||
|
|
||||||
|
|
||||||
|
def vf_cmap() -> dict[str, str]:
|
||||||
|
"""argv-to-volflag: complex/lists"""
|
||||||
|
ret = {}
|
||||||
|
for k in ("dbd", "html_head", "mte", "mth", "nrand"):
|
||||||
|
ret[k] = k
|
||||||
|
return ret
|
||||||
|
|
||||||
|
|
||||||
|
permdescs = {
|
||||||
|
"r": "read; list folder contents, download files",
|
||||||
|
"w": 'write; upload files; need "r" to see the uploads',
|
||||||
|
"m": 'move; move files and folders; need "w" at destination',
|
||||||
|
"d": "delete; permanently delete files and folders",
|
||||||
|
"g": "get; download files, but cannot see folder contents",
|
||||||
|
"G": 'upget; same as "g" but can see filekeys of their own uploads',
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
flagcats = {
|
||||||
|
"uploads, general": {
|
||||||
|
"nodupe": "rejects existing files (instead of symlinking them)",
|
||||||
|
"hardlink": "does dedup with hardlinks instead of symlinks",
|
||||||
|
"neversymlink": "disables symlink fallback; full copy instead",
|
||||||
|
"copydupes": "disables dedup, always saves full copies of dupes",
|
||||||
|
"daw": "enable full WebDAV write support (dangerous);\nPUT-operations will now \033[1;31mOVERWRITE\033[0;35m existing files",
|
||||||
|
"nosub": "forces all uploads into the top folder of the vfs",
|
||||||
|
"magic": "enables filetype detection for nameless uploads",
|
||||||
|
"gz": "allows server-side gzip of uploads with ?gz (also c,xz)",
|
||||||
|
"pk": "forces server-side compression, optional arg: xz,9",
|
||||||
|
},
|
||||||
|
"upload rules": {
|
||||||
|
"maxn=250,600": "max 250 uploads over 15min",
|
||||||
|
"maxb=1g,300": "max 1 GiB over 5min (suffixes: b, k, m, g)",
|
||||||
|
"rand": "force randomized filenames, 9 chars long by default",
|
||||||
|
"nrand=N": "randomized filenames are N chars long",
|
||||||
|
"sz=1k-3m": "allow filesizes between 1 KiB and 3MiB",
|
||||||
|
"df=1g": "ensure 1 GiB free disk space",
|
||||||
|
},
|
||||||
|
"upload rotation\n(moves all uploads into the specified folder structure)": {
|
||||||
|
"rotn=100,3": "3 levels of subfolders with 100 entries in each",
|
||||||
|
"rotf=%Y-%m/%d-%H": "date-formatted organizing",
|
||||||
|
"lifetime=3600": "uploads are deleted after 1 hour",
|
||||||
|
},
|
||||||
|
"database, general": {
|
||||||
|
"e2d": "enable database; makes files searchable + enables upload dedup",
|
||||||
|
"e2ds": "scan writable folders for new files on startup; also sets -e2d",
|
||||||
|
"e2dsa": "scans all folders for new files on startup; also sets -e2d",
|
||||||
|
"e2t": "enable multimedia indexing; makes it possible to search for tags",
|
||||||
|
"e2ts": "scan existing files for tags on startup; also sets -e2t",
|
||||||
|
"e2tsa": "delete all metadata from DB (full rescan); also sets -e2ts",
|
||||||
|
"d2ts": "disables metadata collection for existing files",
|
||||||
|
"d2ds": "disables onboot indexing, overrides -e2ds*",
|
||||||
|
"d2t": "disables metadata collection, overrides -e2t*",
|
||||||
|
"d2v": "disables file verification, overrides -e2v*",
|
||||||
|
"d2d": "disables all database stuff, overrides -e2*",
|
||||||
|
"hist=/tmp/cdb": "puts thumbnails and indexes at that location",
|
||||||
|
"scan=60": "scan for new files every 60sec, same as --re-maxage",
|
||||||
|
"nohash=\\.iso$": "skips hashing file contents if path matches *.iso",
|
||||||
|
"noidx=\\.iso$": "fully ignores the contents at paths matching *.iso",
|
||||||
|
"noforget": "don't forget files when deleted from disk",
|
||||||
|
"dbd=[acid|swal|wal|yolo]": "database speed-durability tradeoff",
|
||||||
|
"xlink": "cross-volume dupe detection / linking",
|
||||||
|
"xdev": "do not descend into other filesystems",
|
||||||
|
"xvol": "skip symlinks leaving the volume root",
|
||||||
|
"dotsrch": "show dotfiles in search results",
|
||||||
|
"nodotsrch": "hide dotfiles in search results (default)",
|
||||||
|
},
|
||||||
|
'database, audio tags\n"mte", "mth", "mtp", "mtm" all work the same as -mte, -mth, ...': {
|
||||||
|
"mtp=.bpm=f,audio-bpm.py": 'uses the "audio-bpm.py" program to\ngenerate ".bpm" tags from uploads (f = overwrite tags)',
|
||||||
|
"mtp=ahash,vhash=media-hash.py": "collects two tags at once",
|
||||||
|
},
|
||||||
|
"thumbnails": {
|
||||||
|
"dthumb": "disables all thumbnails",
|
||||||
|
"dvthumb": "disables video thumbnails",
|
||||||
|
"dathumb": "disables audio thumbnails (spectrograms)",
|
||||||
|
"dithumb": "disables image thumbnails",
|
||||||
|
},
|
||||||
|
"event hooks\n(better explained in --help-hooks)": {
|
||||||
|
"xbu=CMD": "execute CMD before a file upload starts",
|
||||||
|
"xau=CMD": "execute CMD after a file upload finishes",
|
||||||
|
"xiu=CMD": "execute CMD after all uploads finish and volume is idle",
|
||||||
|
"xbr=CMD": "execute CMD before a file rename/move",
|
||||||
|
"xar=CMD": "execute CMD after a file rename/move",
|
||||||
|
"xbd=CMD": "execute CMD before a file delete",
|
||||||
|
"xad=CMD": "execute CMD after a file delete",
|
||||||
|
"xm=CMD": "execute CMD on message",
|
||||||
|
},
|
||||||
|
"client and ux": {
|
||||||
|
"html_head=TXT": "includes TXT in the <head>",
|
||||||
|
"robots": "allows indexing by search engines (default)",
|
||||||
|
"norobots": "kindly asks search engines to leave",
|
||||||
|
"no_sb_md": "disable js sandbox for markdown files",
|
||||||
|
"no_sb_lg": "disable js sandbox for prologue/epilogue",
|
||||||
|
"sb_md": "enable js sandbox for markdown files (default)",
|
||||||
|
"sb_lg": "enable js sandbox for prologue/epilogue (default)",
|
||||||
|
"md_sbf": "list of markdown-sandbox safeguards to disable",
|
||||||
|
"lg_sbf": "list of *logue-sandbox safeguards to disable",
|
||||||
|
},
|
||||||
|
"others": {
|
||||||
|
"fk=8": 'generates per-file accesskeys,\nwhich will then be required at the "g" permission'
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
flagdescs = {k.split("=")[0]: v for tab in flagcats.values() for k, v in tab.items()}
|
||||||
72
copyparty/dxml.py
Normal file
72
copyparty/dxml.py
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
import importlib
|
||||||
|
import sys
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
|
||||||
|
from .__init__ import PY2
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any, Optional
|
||||||
|
|
||||||
|
|
||||||
|
def get_ET() -> ET.XMLParser:
|
||||||
|
pn = "xml.etree.ElementTree"
|
||||||
|
cn = "_elementtree"
|
||||||
|
|
||||||
|
cmod = sys.modules.pop(cn, None)
|
||||||
|
if not cmod:
|
||||||
|
return ET.XMLParser # type: ignore
|
||||||
|
|
||||||
|
pmod = sys.modules.pop(pn)
|
||||||
|
sys.modules[cn] = None # type: ignore
|
||||||
|
|
||||||
|
ret = importlib.import_module(pn)
|
||||||
|
for name, mod in ((pn, pmod), (cn, cmod)):
|
||||||
|
if mod:
|
||||||
|
sys.modules[name] = mod
|
||||||
|
else:
|
||||||
|
sys.modules.pop(name, None)
|
||||||
|
|
||||||
|
sys.modules["xml.etree"].ElementTree = pmod # type: ignore
|
||||||
|
ret.ParseError = ET.ParseError # type: ignore
|
||||||
|
return ret.XMLParser # type: ignore
|
||||||
|
|
||||||
|
|
||||||
|
XMLParser: ET.XMLParser = get_ET()
|
||||||
|
|
||||||
|
|
||||||
|
class DXMLParser(XMLParser): # type: ignore
|
||||||
|
def __init__(self) -> None:
|
||||||
|
tb = ET.TreeBuilder()
|
||||||
|
super(DXMLParser, self).__init__(target=tb)
|
||||||
|
|
||||||
|
p = self._parser if PY2 else self.parser
|
||||||
|
p.StartDoctypeDeclHandler = self.nope
|
||||||
|
p.EntityDeclHandler = self.nope
|
||||||
|
p.UnparsedEntityDeclHandler = self.nope
|
||||||
|
p.ExternalEntityRefHandler = self.nope
|
||||||
|
|
||||||
|
def nope(self, *a: Any, **ka: Any) -> None:
|
||||||
|
raise BadXML("{}, {}".format(a, ka))
|
||||||
|
|
||||||
|
|
||||||
|
class BadXML(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def parse_xml(txt: str) -> ET.Element:
|
||||||
|
parser = DXMLParser()
|
||||||
|
parser.feed(txt)
|
||||||
|
return parser.close() # type: ignore
|
||||||
|
|
||||||
|
|
||||||
|
def mktnod(name: str, text: str) -> ET.Element:
|
||||||
|
el = ET.Element(name)
|
||||||
|
el.text = text
|
||||||
|
return el
|
||||||
|
|
||||||
|
|
||||||
|
def mkenod(name: str, sub_el: Optional[ET.Element] = None) -> ET.Element:
|
||||||
|
el = ET.Element(name)
|
||||||
|
if sub_el is not None:
|
||||||
|
el.append(sub_el)
|
||||||
|
return el
|
||||||
152
copyparty/fsutil.py
Normal file
152
copyparty/fsutil.py
Normal file
@@ -0,0 +1,152 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import time
|
||||||
|
|
||||||
|
from .__init__ import ANYWIN, MACOS
|
||||||
|
from .authsrv import AXS, VFS
|
||||||
|
from .bos import bos
|
||||||
|
from .util import chkcmd, min_ex
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Optional, Union
|
||||||
|
|
||||||
|
from .util import RootLogger
|
||||||
|
|
||||||
|
|
||||||
|
class Fstab(object):
|
||||||
|
def __init__(self, log: "RootLogger"):
|
||||||
|
self.log_func = log
|
||||||
|
|
||||||
|
self.trusted = False
|
||||||
|
self.tab: Optional[VFS] = None
|
||||||
|
self.cache: dict[str, str] = {}
|
||||||
|
self.age = 0.0
|
||||||
|
|
||||||
|
def log(self, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
|
self.log_func("fstab", msg, c)
|
||||||
|
|
||||||
|
def get(self, path: str) -> str:
|
||||||
|
if len(self.cache) > 9000:
|
||||||
|
self.age = time.time()
|
||||||
|
self.tab = None
|
||||||
|
self.cache = {}
|
||||||
|
|
||||||
|
fs = "ext4"
|
||||||
|
msg = "failed to determine filesystem at [{}]; assuming {}\n{}"
|
||||||
|
|
||||||
|
if ANYWIN:
|
||||||
|
fs = "vfat"
|
||||||
|
try:
|
||||||
|
path = self._winpath(path)
|
||||||
|
except:
|
||||||
|
self.log(msg.format(path, fs, min_ex()), 3)
|
||||||
|
return fs
|
||||||
|
|
||||||
|
path = path.lstrip("/")
|
||||||
|
try:
|
||||||
|
return self.cache[path]
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
fs = self.get_w32(path) if ANYWIN else self.get_unix(path)
|
||||||
|
except:
|
||||||
|
self.log(msg.format(path, fs, min_ex()), 3)
|
||||||
|
|
||||||
|
fs = fs.lower()
|
||||||
|
self.cache[path] = fs
|
||||||
|
self.log("found {} at {}".format(fs, path))
|
||||||
|
return fs
|
||||||
|
|
||||||
|
def _winpath(self, path: str) -> str:
|
||||||
|
# try to combine volume-label + st_dev (vsn)
|
||||||
|
path = path.replace("/", "\\")
|
||||||
|
vid = path.split(":", 1)[0].strip("\\").split("\\", 1)[0]
|
||||||
|
try:
|
||||||
|
return "{}*{}".format(vid, bos.stat(path).st_dev)
|
||||||
|
except:
|
||||||
|
return vid
|
||||||
|
|
||||||
|
def build_fallback(self) -> None:
|
||||||
|
self.tab = VFS(self.log_func, "idk", "/", AXS(), {})
|
||||||
|
self.trusted = False
|
||||||
|
|
||||||
|
def build_tab(self) -> None:
|
||||||
|
self.log("building tab")
|
||||||
|
|
||||||
|
sptn = r"^.*? on (.*) type ([^ ]+) \(.*"
|
||||||
|
if MACOS:
|
||||||
|
sptn = r"^.*? on (.*) \(([^ ]+), .*"
|
||||||
|
|
||||||
|
ptn = re.compile(sptn)
|
||||||
|
so, _ = chkcmd(["mount"])
|
||||||
|
tab1: list[tuple[str, str]] = []
|
||||||
|
for ln in so.split("\n"):
|
||||||
|
m = ptn.match(ln)
|
||||||
|
if not m:
|
||||||
|
continue
|
||||||
|
|
||||||
|
zs1, zs2 = m.groups()
|
||||||
|
tab1.append((str(zs1), str(zs2)))
|
||||||
|
|
||||||
|
tab1.sort(key=lambda x: (len(x[0]), x[0]))
|
||||||
|
path1, fs1 = tab1[0]
|
||||||
|
tab = VFS(self.log_func, fs1, path1, AXS(), {})
|
||||||
|
for path, fs in tab1[1:]:
|
||||||
|
tab.add(fs, path.lstrip("/"))
|
||||||
|
|
||||||
|
self.tab = tab
|
||||||
|
|
||||||
|
def relabel(self, path: str, nval: str) -> None:
|
||||||
|
assert self.tab
|
||||||
|
self.cache = {}
|
||||||
|
if ANYWIN:
|
||||||
|
path = self._winpath(path)
|
||||||
|
|
||||||
|
path = path.lstrip("/")
|
||||||
|
ptn = re.compile(r"^[^\\/]*")
|
||||||
|
vn, rem = self.tab._find(path)
|
||||||
|
if not self.trusted:
|
||||||
|
# no mtab access; have to build as we go
|
||||||
|
if "/" in rem:
|
||||||
|
self.tab.add("idk", os.path.join(vn.vpath, rem.split("/")[0]))
|
||||||
|
if rem:
|
||||||
|
self.tab.add(nval, path)
|
||||||
|
else:
|
||||||
|
vn.realpath = nval
|
||||||
|
|
||||||
|
return
|
||||||
|
|
||||||
|
visit = [vn]
|
||||||
|
while visit:
|
||||||
|
vn = visit.pop()
|
||||||
|
vn.realpath = ptn.sub(nval, vn.realpath)
|
||||||
|
visit.extend(list(vn.nodes.values()))
|
||||||
|
|
||||||
|
def get_unix(self, path: str) -> str:
|
||||||
|
if not self.tab:
|
||||||
|
try:
|
||||||
|
self.build_tab()
|
||||||
|
self.trusted = True
|
||||||
|
except:
|
||||||
|
# prisonparty or other restrictive environment
|
||||||
|
self.log("failed to build tab:\n{}".format(min_ex()), 3)
|
||||||
|
self.build_fallback()
|
||||||
|
|
||||||
|
assert self.tab
|
||||||
|
ret = self.tab._find(path)[0]
|
||||||
|
if self.trusted or path == ret.vpath:
|
||||||
|
return ret.realpath.split("/")[0]
|
||||||
|
else:
|
||||||
|
return "idk"
|
||||||
|
|
||||||
|
def get_w32(self, path: str) -> str:
|
||||||
|
if not self.tab:
|
||||||
|
self.build_fallback()
|
||||||
|
|
||||||
|
assert self.tab
|
||||||
|
ret = self.tab._find(path)[0]
|
||||||
|
return ret.realpath
|
||||||
502
copyparty/ftpd.py
Normal file
502
copyparty/ftpd.py
Normal file
@@ -0,0 +1,502 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import argparse
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import stat
|
||||||
|
import sys
|
||||||
|
import time
|
||||||
|
|
||||||
|
from pyftpdlib.authorizers import AuthenticationFailed, DummyAuthorizer
|
||||||
|
from pyftpdlib.filesystems import AbstractedFS, FilesystemError
|
||||||
|
from pyftpdlib.handlers import FTPHandler
|
||||||
|
from pyftpdlib.servers import FTPServer
|
||||||
|
|
||||||
|
from .__init__ import ANYWIN, PY2, TYPE_CHECKING, E
|
||||||
|
from .bos import bos
|
||||||
|
from .authsrv import VFS
|
||||||
|
from .util import (
|
||||||
|
Daemon,
|
||||||
|
Pebkac,
|
||||||
|
exclude_dotfiles,
|
||||||
|
fsenc,
|
||||||
|
ipnorm,
|
||||||
|
pybin,
|
||||||
|
relchk,
|
||||||
|
runhook,
|
||||||
|
sanitize_fn,
|
||||||
|
vjoin,
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
from pyftpdlib.ioloop import IOLoop
|
||||||
|
except ImportError:
|
||||||
|
p = os.path.join(E.mod, "vend")
|
||||||
|
print("loading asynchat from " + p)
|
||||||
|
sys.path.append(p)
|
||||||
|
from pyftpdlib.ioloop import IOLoop
|
||||||
|
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from .svchub import SvcHub
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
import typing
|
||||||
|
from typing import Any, Optional
|
||||||
|
|
||||||
|
|
||||||
|
class FtpAuth(DummyAuthorizer):
|
||||||
|
def __init__(self, hub: "SvcHub") -> None:
|
||||||
|
super(FtpAuth, self).__init__()
|
||||||
|
self.hub = hub
|
||||||
|
|
||||||
|
def validate_authentication(
|
||||||
|
self, username: str, password: str, handler: Any
|
||||||
|
) -> None:
|
||||||
|
handler.username = "{}:{}".format(username, password)
|
||||||
|
|
||||||
|
ip = handler.addr[0]
|
||||||
|
if ip.startswith("::ffff:"):
|
||||||
|
ip = ip[7:]
|
||||||
|
|
||||||
|
ip = ipnorm(ip)
|
||||||
|
bans = self.hub.bans
|
||||||
|
if ip in bans:
|
||||||
|
rt = bans[ip] - time.time()
|
||||||
|
if rt < 0:
|
||||||
|
logging.info("client unbanned")
|
||||||
|
del bans[ip]
|
||||||
|
else:
|
||||||
|
raise AuthenticationFailed("banned")
|
||||||
|
|
||||||
|
asrv = self.hub.asrv
|
||||||
|
if username == "anonymous":
|
||||||
|
uname = "*"
|
||||||
|
else:
|
||||||
|
uname = asrv.iacct.get(password, "") or asrv.iacct.get(username, "") or "*"
|
||||||
|
|
||||||
|
if not uname or not (asrv.vfs.aread.get(uname) or asrv.vfs.awrite.get(uname)):
|
||||||
|
g = self.hub.gpwd
|
||||||
|
if g.lim:
|
||||||
|
bonk, ip = g.bonk(ip, handler.username)
|
||||||
|
if bonk:
|
||||||
|
logging.warning("client banned: invalid passwords")
|
||||||
|
bans[ip] = bonk
|
||||||
|
|
||||||
|
raise AuthenticationFailed("Authentication failed.")
|
||||||
|
|
||||||
|
handler.username = uname
|
||||||
|
|
||||||
|
def get_home_dir(self, username: str) -> str:
|
||||||
|
return "/"
|
||||||
|
|
||||||
|
def has_user(self, username: str) -> bool:
|
||||||
|
asrv = self.hub.asrv
|
||||||
|
return username in asrv.acct
|
||||||
|
|
||||||
|
def has_perm(self, username: str, perm: int, path: Optional[str] = None) -> bool:
|
||||||
|
return True # handled at filesystem layer
|
||||||
|
|
||||||
|
def get_perms(self, username: str) -> str:
|
||||||
|
return "elradfmwMT"
|
||||||
|
|
||||||
|
def get_msg_login(self, username: str) -> str:
|
||||||
|
return "sup {}".format(username)
|
||||||
|
|
||||||
|
def get_msg_quit(self, username: str) -> str:
|
||||||
|
return "cya"
|
||||||
|
|
||||||
|
|
||||||
|
class FtpFs(AbstractedFS):
|
||||||
|
def __init__(
|
||||||
|
self, root: str, cmd_channel: Any
|
||||||
|
) -> None: # pylint: disable=super-init-not-called
|
||||||
|
self.h = self.cmd_channel = cmd_channel # type: FTPHandler
|
||||||
|
self.hub: "SvcHub" = cmd_channel.hub
|
||||||
|
self.args = cmd_channel.args
|
||||||
|
|
||||||
|
self.uname = self.hub.asrv.iacct.get(cmd_channel.password, "*")
|
||||||
|
|
||||||
|
self.cwd = "/" # pyftpdlib convention of leading slash
|
||||||
|
self.root = "/var/lib/empty"
|
||||||
|
|
||||||
|
self.can_read = self.can_write = self.can_move = False
|
||||||
|
self.can_delete = self.can_get = self.can_upget = False
|
||||||
|
|
||||||
|
self.listdirinfo = self.listdir
|
||||||
|
self.chdir(".")
|
||||||
|
|
||||||
|
def die(self, msg):
|
||||||
|
self.h.die(msg)
|
||||||
|
|
||||||
|
def v2a(
|
||||||
|
self,
|
||||||
|
vpath: str,
|
||||||
|
r: bool = False,
|
||||||
|
w: bool = False,
|
||||||
|
m: bool = False,
|
||||||
|
d: bool = False,
|
||||||
|
) -> tuple[str, VFS, str]:
|
||||||
|
try:
|
||||||
|
vpath = vpath.replace("\\", "/").lstrip("/")
|
||||||
|
rd, fn = os.path.split(vpath)
|
||||||
|
if ANYWIN and relchk(rd):
|
||||||
|
logging.warning("malicious vpath: %s", vpath)
|
||||||
|
self.die("Unsupported characters in filepath")
|
||||||
|
|
||||||
|
fn = sanitize_fn(fn or "", "", [".prologue.html", ".epilogue.html"])
|
||||||
|
vpath = vjoin(rd, fn)
|
||||||
|
vfs, rem = self.hub.asrv.vfs.get(vpath, self.uname, r, w, m, d)
|
||||||
|
if not vfs.realpath:
|
||||||
|
self.die("No filesystem mounted at this path")
|
||||||
|
|
||||||
|
return os.path.join(vfs.realpath, rem), vfs, rem
|
||||||
|
except Pebkac as ex:
|
||||||
|
self.die(str(ex))
|
||||||
|
|
||||||
|
def rv2a(
|
||||||
|
self,
|
||||||
|
vpath: str,
|
||||||
|
r: bool = False,
|
||||||
|
w: bool = False,
|
||||||
|
m: bool = False,
|
||||||
|
d: bool = False,
|
||||||
|
) -> tuple[str, VFS, str]:
|
||||||
|
return self.v2a(os.path.join(self.cwd, vpath), r, w, m, d)
|
||||||
|
|
||||||
|
def ftp2fs(self, ftppath: str) -> str:
|
||||||
|
# return self.v2a(ftppath)
|
||||||
|
return ftppath # self.cwd must be vpath
|
||||||
|
|
||||||
|
def fs2ftp(self, fspath: str) -> str:
|
||||||
|
# raise NotImplementedError()
|
||||||
|
return fspath
|
||||||
|
|
||||||
|
def validpath(self, path: str) -> bool:
|
||||||
|
if "/.hist/" in path:
|
||||||
|
if "/up2k." in path or path.endswith("/dir.txt"):
|
||||||
|
self.die("Access to this file is forbidden")
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def open(self, filename: str, mode: str) -> typing.IO[Any]:
|
||||||
|
r = "r" in mode
|
||||||
|
w = "w" in mode or "a" in mode or "+" in mode
|
||||||
|
|
||||||
|
ap = self.rv2a(filename, r, w)[0]
|
||||||
|
if w:
|
||||||
|
try:
|
||||||
|
st = bos.stat(ap)
|
||||||
|
td = time.time() - st.st_mtime
|
||||||
|
except:
|
||||||
|
td = 0
|
||||||
|
|
||||||
|
if td < -1 or td > self.args.ftp_wt:
|
||||||
|
self.die("Cannot open existing file for writing")
|
||||||
|
|
||||||
|
self.validpath(ap)
|
||||||
|
return open(fsenc(ap), mode)
|
||||||
|
|
||||||
|
def chdir(self, path: str) -> None:
|
||||||
|
nwd = join(self.cwd, path)
|
||||||
|
vfs, rem = self.hub.asrv.vfs.get(nwd, self.uname, False, False)
|
||||||
|
ap = vfs.canonical(rem)
|
||||||
|
if not bos.path.isdir(ap):
|
||||||
|
# returning 550 is library-default and suitable
|
||||||
|
self.die("Failed to change directory")
|
||||||
|
|
||||||
|
self.cwd = nwd
|
||||||
|
(
|
||||||
|
self.can_read,
|
||||||
|
self.can_write,
|
||||||
|
self.can_move,
|
||||||
|
self.can_delete,
|
||||||
|
self.can_get,
|
||||||
|
self.can_upget,
|
||||||
|
) = self.hub.asrv.vfs.can_access(self.cwd.lstrip("/"), self.h.username)
|
||||||
|
|
||||||
|
def mkdir(self, path: str) -> None:
|
||||||
|
ap = self.rv2a(path, w=True)[0]
|
||||||
|
bos.mkdir(ap)
|
||||||
|
|
||||||
|
def listdir(self, path: str) -> list[str]:
|
||||||
|
vpath = join(self.cwd, path).lstrip("/")
|
||||||
|
try:
|
||||||
|
vfs, rem = self.hub.asrv.vfs.get(vpath, self.uname, True, False)
|
||||||
|
|
||||||
|
fsroot, vfs_ls1, vfs_virt = vfs.ls(
|
||||||
|
rem,
|
||||||
|
self.uname,
|
||||||
|
not self.args.no_scandir,
|
||||||
|
[[True, False], [False, True]],
|
||||||
|
)
|
||||||
|
vfs_ls = [x[0] for x in vfs_ls1]
|
||||||
|
vfs_ls.extend(vfs_virt.keys())
|
||||||
|
|
||||||
|
if not self.args.ed:
|
||||||
|
vfs_ls = exclude_dotfiles(vfs_ls)
|
||||||
|
|
||||||
|
vfs_ls.sort()
|
||||||
|
return vfs_ls
|
||||||
|
except:
|
||||||
|
if vpath:
|
||||||
|
# display write-only folders as empty
|
||||||
|
return []
|
||||||
|
|
||||||
|
# return list of volumes
|
||||||
|
r = {x.split("/")[0]: 1 for x in self.hub.asrv.vfs.all_vols.keys()}
|
||||||
|
return list(sorted(list(r.keys())))
|
||||||
|
|
||||||
|
def rmdir(self, path: str) -> None:
|
||||||
|
ap = self.rv2a(path, d=True)[0]
|
||||||
|
bos.rmdir(ap)
|
||||||
|
|
||||||
|
def remove(self, path: str) -> None:
|
||||||
|
if self.args.no_del:
|
||||||
|
self.die("The delete feature is disabled in server config")
|
||||||
|
|
||||||
|
vp = join(self.cwd, path).lstrip("/")
|
||||||
|
try:
|
||||||
|
self.hub.up2k.handle_rm(self.uname, self.h.cli_ip, [vp], [])
|
||||||
|
except Exception as ex:
|
||||||
|
self.die(str(ex))
|
||||||
|
|
||||||
|
def rename(self, src: str, dst: str) -> None:
|
||||||
|
if not self.can_move:
|
||||||
|
self.die("Not allowed for user " + self.h.username)
|
||||||
|
|
||||||
|
if self.args.no_mv:
|
||||||
|
self.die("The rename/move feature is disabled in server config")
|
||||||
|
|
||||||
|
svp = join(self.cwd, src).lstrip("/")
|
||||||
|
dvp = join(self.cwd, dst).lstrip("/")
|
||||||
|
try:
|
||||||
|
self.hub.up2k.handle_mv(self.uname, svp, dvp)
|
||||||
|
except Exception as ex:
|
||||||
|
self.die(str(ex))
|
||||||
|
|
||||||
|
def chmod(self, path: str, mode: str) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def stat(self, path: str) -> os.stat_result:
|
||||||
|
try:
|
||||||
|
ap = self.rv2a(path, r=True)[0]
|
||||||
|
return bos.stat(ap)
|
||||||
|
except:
|
||||||
|
ap = self.rv2a(path)[0]
|
||||||
|
st = bos.stat(ap)
|
||||||
|
if not stat.S_ISDIR(st.st_mode):
|
||||||
|
raise
|
||||||
|
|
||||||
|
return st
|
||||||
|
|
||||||
|
def utime(self, path: str, timeval: float) -> None:
|
||||||
|
ap = self.rv2a(path, w=True)[0]
|
||||||
|
return bos.utime(ap, (timeval, timeval))
|
||||||
|
|
||||||
|
def lstat(self, path: str) -> os.stat_result:
|
||||||
|
ap = self.rv2a(path)[0]
|
||||||
|
return bos.stat(ap)
|
||||||
|
|
||||||
|
def isfile(self, path: str) -> bool:
|
||||||
|
try:
|
||||||
|
st = self.stat(path)
|
||||||
|
return stat.S_ISREG(st.st_mode)
|
||||||
|
except:
|
||||||
|
return False # expected for mojibake in ftp_SIZE()
|
||||||
|
|
||||||
|
def islink(self, path: str) -> bool:
|
||||||
|
ap = self.rv2a(path)[0]
|
||||||
|
return bos.path.islink(ap)
|
||||||
|
|
||||||
|
def isdir(self, path: str) -> bool:
|
||||||
|
try:
|
||||||
|
st = self.stat(path)
|
||||||
|
return stat.S_ISDIR(st.st_mode)
|
||||||
|
except:
|
||||||
|
return True
|
||||||
|
|
||||||
|
def getsize(self, path: str) -> int:
|
||||||
|
ap = self.rv2a(path)[0]
|
||||||
|
return bos.path.getsize(ap)
|
||||||
|
|
||||||
|
def getmtime(self, path: str) -> float:
|
||||||
|
ap = self.rv2a(path)[0]
|
||||||
|
return bos.path.getmtime(ap)
|
||||||
|
|
||||||
|
def realpath(self, path: str) -> str:
|
||||||
|
return path
|
||||||
|
|
||||||
|
def lexists(self, path: str) -> bool:
|
||||||
|
ap = self.rv2a(path)[0]
|
||||||
|
return bos.path.lexists(ap)
|
||||||
|
|
||||||
|
def get_user_by_uid(self, uid: int) -> str:
|
||||||
|
return "root"
|
||||||
|
|
||||||
|
def get_group_by_uid(self, gid: int) -> str:
|
||||||
|
return "root"
|
||||||
|
|
||||||
|
|
||||||
|
class FtpHandler(FTPHandler):
|
||||||
|
abstracted_fs = FtpFs
|
||||||
|
hub: "SvcHub"
|
||||||
|
args: argparse.Namespace
|
||||||
|
|
||||||
|
def __init__(self, conn: Any, server: Any, ioloop: Any = None) -> None:
|
||||||
|
self.hub: "SvcHub" = FtpHandler.hub
|
||||||
|
self.args: argparse.Namespace = FtpHandler.args
|
||||||
|
|
||||||
|
if PY2:
|
||||||
|
FTPHandler.__init__(self, conn, server, ioloop)
|
||||||
|
else:
|
||||||
|
super(FtpHandler, self).__init__(conn, server, ioloop)
|
||||||
|
|
||||||
|
cip = self.remote_ip
|
||||||
|
self.cli_ip = cip[7:] if cip.startswith("::ffff:") else cip
|
||||||
|
|
||||||
|
# abspath->vpath mapping to resolve log_transfer paths
|
||||||
|
self.vfs_map: dict[str, str] = {}
|
||||||
|
|
||||||
|
# reduce non-debug logging
|
||||||
|
self.log_cmds_list = [x for x in self.log_cmds_list if x not in ("CWD", "XCWD")]
|
||||||
|
|
||||||
|
def die(self, msg):
|
||||||
|
self.respond("550 {}".format(msg))
|
||||||
|
raise FilesystemError(msg)
|
||||||
|
|
||||||
|
def ftp_STOR(self, file: str, mode: str = "w") -> Any:
|
||||||
|
# Optional[str]
|
||||||
|
vp = join(self.fs.cwd, file).lstrip("/")
|
||||||
|
ap, vfs, rem = self.fs.v2a(vp)
|
||||||
|
self.vfs_map[ap] = vp
|
||||||
|
xbu = vfs.flags.get("xbu")
|
||||||
|
if xbu and not runhook(
|
||||||
|
None,
|
||||||
|
xbu,
|
||||||
|
ap,
|
||||||
|
vfs.canonical(rem),
|
||||||
|
"",
|
||||||
|
self.username,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
self.cli_ip,
|
||||||
|
0,
|
||||||
|
"",
|
||||||
|
):
|
||||||
|
self.die("Upload blocked by xbu server config")
|
||||||
|
|
||||||
|
# print("ftp_STOR: {} {} => {}".format(vp, mode, ap))
|
||||||
|
ret = FTPHandler.ftp_STOR(self, file, mode)
|
||||||
|
# print("ftp_STOR: {} {} OK".format(vp, mode))
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def log_transfer(
|
||||||
|
self,
|
||||||
|
cmd: str,
|
||||||
|
filename: bytes,
|
||||||
|
receive: bool,
|
||||||
|
completed: bool,
|
||||||
|
elapsed: float,
|
||||||
|
bytes: int,
|
||||||
|
) -> Any:
|
||||||
|
# None
|
||||||
|
ap = filename.decode("utf-8", "replace")
|
||||||
|
vp = self.vfs_map.pop(ap, None)
|
||||||
|
# print("xfer_end: {} => {}".format(ap, vp))
|
||||||
|
if vp:
|
||||||
|
vp, fn = os.path.split(vp)
|
||||||
|
vfs, rem = self.hub.asrv.vfs.get(vp, self.username, False, True)
|
||||||
|
vfs, rem = vfs.get_dbv(rem)
|
||||||
|
self.hub.up2k.hash_file(
|
||||||
|
vfs.realpath,
|
||||||
|
vfs.vpath,
|
||||||
|
vfs.flags,
|
||||||
|
rem,
|
||||||
|
fn,
|
||||||
|
self.cli_ip,
|
||||||
|
time.time(),
|
||||||
|
self.username,
|
||||||
|
)
|
||||||
|
|
||||||
|
return FTPHandler.log_transfer(
|
||||||
|
self, cmd, filename, receive, completed, elapsed, bytes
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
try:
|
||||||
|
from pyftpdlib.handlers import TLS_FTPHandler
|
||||||
|
|
||||||
|
class SftpHandler(FtpHandler, TLS_FTPHandler):
|
||||||
|
pass
|
||||||
|
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class Ftpd(object):
|
||||||
|
def __init__(self, hub: "SvcHub") -> None:
|
||||||
|
self.hub = hub
|
||||||
|
self.args = hub.args
|
||||||
|
|
||||||
|
hs = []
|
||||||
|
if self.args.ftp:
|
||||||
|
hs.append([FtpHandler, self.args.ftp])
|
||||||
|
if self.args.ftps:
|
||||||
|
try:
|
||||||
|
h1 = SftpHandler
|
||||||
|
except:
|
||||||
|
t = "\nftps requires pyopenssl;\nplease run the following:\n\n {} -m pip install --user pyopenssl\n"
|
||||||
|
print(t.format(pybin))
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
h1.certfile = os.path.join(self.args.E.cfg, "cert.pem")
|
||||||
|
h1.tls_control_required = True
|
||||||
|
h1.tls_data_required = True
|
||||||
|
|
||||||
|
hs.append([h1, self.args.ftps])
|
||||||
|
|
||||||
|
for h_lp in hs:
|
||||||
|
h2, lp = h_lp
|
||||||
|
h2.hub = hub
|
||||||
|
h2.args = hub.args
|
||||||
|
h2.authorizer = FtpAuth(hub)
|
||||||
|
|
||||||
|
if self.args.ftp_pr:
|
||||||
|
p1, p2 = [int(x) for x in self.args.ftp_pr.split("-")]
|
||||||
|
if self.args.ftp and self.args.ftps:
|
||||||
|
# divide port range in half
|
||||||
|
d = int((p2 - p1) / 2)
|
||||||
|
if lp == self.args.ftp:
|
||||||
|
p2 = p1 + d
|
||||||
|
else:
|
||||||
|
p1 += d + 1
|
||||||
|
|
||||||
|
h2.passive_ports = list(range(p1, p2 + 1))
|
||||||
|
|
||||||
|
if self.args.ftp_nat:
|
||||||
|
h2.masquerade_address = self.args.ftp_nat
|
||||||
|
|
||||||
|
lgr = logging.getLogger("pyftpdlib")
|
||||||
|
lgr.setLevel(logging.DEBUG if self.args.ftpv else logging.INFO)
|
||||||
|
|
||||||
|
ips = self.args.i
|
||||||
|
if "::" in ips:
|
||||||
|
ips.append("0.0.0.0")
|
||||||
|
|
||||||
|
ioloop = IOLoop()
|
||||||
|
for ip in ips:
|
||||||
|
for h, lp in hs:
|
||||||
|
try:
|
||||||
|
FTPServer((ip, int(lp)), h, ioloop)
|
||||||
|
except:
|
||||||
|
if ip != "0.0.0.0" or "::" not in ips:
|
||||||
|
raise
|
||||||
|
|
||||||
|
Daemon(ioloop.loop, "ftp")
|
||||||
|
|
||||||
|
|
||||||
|
def join(p1: str, p2: str) -> str:
|
||||||
|
w = os.path.join(p1, p2.replace("\\", "/"))
|
||||||
|
return os.path.normpath(w).replace("\\", "/")
|
||||||
2387
copyparty/httpcli.py
2387
copyparty/httpcli.py
File diff suppressed because it is too large
Load Diff
@@ -1,24 +1,38 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import re
|
import argparse # typechk
|
||||||
import os
|
import os
|
||||||
import time
|
import re
|
||||||
import socket
|
import socket
|
||||||
|
import threading # typechk
|
||||||
|
import time
|
||||||
|
|
||||||
HAVE_SSL = True
|
|
||||||
try:
|
try:
|
||||||
|
HAVE_SSL = True
|
||||||
import ssl
|
import ssl
|
||||||
except:
|
except:
|
||||||
HAVE_SSL = False
|
HAVE_SSL = False
|
||||||
|
|
||||||
from .__init__ import E
|
from . import util as Util
|
||||||
from .util import Unrecv
|
from .__init__ import TYPE_CHECKING, EnvParams
|
||||||
|
from .authsrv import AuthSrv # typechk
|
||||||
from .httpcli import HttpCli
|
from .httpcli import HttpCli
|
||||||
from .u2idx import U2idx
|
|
||||||
from .th_cli import ThumbCli
|
|
||||||
from .th_srv import HAVE_PIL
|
|
||||||
from .ico import Ico
|
from .ico import Ico
|
||||||
|
from .mtag import HAVE_FFMPEG
|
||||||
|
from .th_cli import ThumbCli
|
||||||
|
from .th_srv import HAVE_PIL, HAVE_VIPS
|
||||||
|
from .u2idx import U2idx
|
||||||
|
from .util import HMaccas, shut_socket
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Optional, Pattern, Union
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from .httpsrv import HttpSrv
|
||||||
|
|
||||||
|
|
||||||
|
PTN_HTTP = re.compile(br"[A-Z]{3}[A-Z ]")
|
||||||
|
|
||||||
|
|
||||||
class HttpConn(object):
|
class HttpConn(object):
|
||||||
@@ -27,39 +41,50 @@ class HttpConn(object):
|
|||||||
creates an HttpCli for each request (Connection: Keep-Alive)
|
creates an HttpCli for each request (Connection: Keep-Alive)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, sck, addr, hsrv):
|
def __init__(
|
||||||
|
self, sck: socket.socket, addr: tuple[str, int], hsrv: "HttpSrv"
|
||||||
|
) -> None:
|
||||||
self.s = sck
|
self.s = sck
|
||||||
|
self.sr: Optional[Util._Unrecv] = None
|
||||||
|
self.cli: Optional[HttpCli] = None
|
||||||
self.addr = addr
|
self.addr = addr
|
||||||
self.hsrv = hsrv
|
self.hsrv = hsrv
|
||||||
|
|
||||||
self.mutex = hsrv.mutex
|
self.mutex: threading.Lock = hsrv.mutex # mypy404
|
||||||
self.args = hsrv.args
|
self.args: argparse.Namespace = hsrv.args # mypy404
|
||||||
self.asrv = hsrv.asrv
|
self.E: EnvParams = self.args.E
|
||||||
|
self.asrv: AuthSrv = hsrv.asrv # mypy404
|
||||||
self.cert_path = hsrv.cert_path
|
self.cert_path = hsrv.cert_path
|
||||||
self.u2fh = hsrv.u2fh
|
self.u2fh: Util.FHC = hsrv.u2fh # mypy404
|
||||||
|
self.iphash: HMaccas = hsrv.broker.iphash
|
||||||
|
self.bans: dict[str, int] = hsrv.bans
|
||||||
|
self.aclose: dict[str, int] = hsrv.aclose
|
||||||
|
|
||||||
enth = HAVE_PIL and not self.args.no_thumb
|
enth = (HAVE_PIL or HAVE_VIPS or HAVE_FFMPEG) and not self.args.no_thumb
|
||||||
self.thumbcli = ThumbCli(hsrv) if enth else None
|
self.thumbcli: Optional[ThumbCli] = ThumbCli(hsrv) if enth else None # mypy404
|
||||||
self.ico = Ico(self.args)
|
self.ico: Ico = Ico(self.args) # mypy404
|
||||||
|
|
||||||
self.t0 = time.time()
|
self.t0: float = time.time() # mypy404
|
||||||
|
self.freshen_pwd: float = 0.0
|
||||||
self.stopping = False
|
self.stopping = False
|
||||||
self.nreq = 0
|
self.nreq: int = -1 # mypy404
|
||||||
self.nbyte = 0
|
self.nbyte: int = 0 # mypy404
|
||||||
self.u2idx = None
|
self.u2idx: Optional[U2idx] = None
|
||||||
self.log_func = hsrv.log
|
self.log_func: "Util.RootLogger" = hsrv.log # mypy404
|
||||||
self.lf_url = re.compile(self.args.lf_url) if self.args.lf_url else None
|
self.log_src: str = "httpconn" # mypy404
|
||||||
|
self.lf_url: Optional[Pattern[str]] = (
|
||||||
|
re.compile(self.args.lf_url) if self.args.lf_url else None
|
||||||
|
) # mypy404
|
||||||
self.set_rproxy()
|
self.set_rproxy()
|
||||||
|
|
||||||
def shutdown(self):
|
def shutdown(self) -> None:
|
||||||
self.stopping = True
|
self.stopping = True
|
||||||
try:
|
try:
|
||||||
self.s.shutdown(socket.SHUT_RDWR)
|
shut_socket(self.log, self.s, 1)
|
||||||
self.s.close()
|
|
||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def set_rproxy(self, ip=None):
|
def set_rproxy(self, ip: Optional[str] = None) -> str:
|
||||||
if ip is None:
|
if ip is None:
|
||||||
color = 36
|
color = 36
|
||||||
ip = self.addr[0]
|
ip = self.addr[0]
|
||||||
@@ -72,35 +97,37 @@ class HttpConn(object):
|
|||||||
self.log_src = "{} \033[{}m{}".format(ip, color, self.addr[1]).ljust(26)
|
self.log_src = "{} \033[{}m{}".format(ip, color, self.addr[1]).ljust(26)
|
||||||
return self.log_src
|
return self.log_src
|
||||||
|
|
||||||
def respath(self, res_name):
|
def respath(self, res_name: str) -> str:
|
||||||
return os.path.join(E.mod, "web", res_name)
|
return os.path.join(self.E.mod, "web", res_name)
|
||||||
|
|
||||||
def log(self, msg, c=0):
|
def log(self, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
self.log_func(self.log_src, msg, c)
|
self.log_func(self.log_src, msg, c)
|
||||||
|
|
||||||
def get_u2idx(self):
|
def get_u2idx(self) -> U2idx:
|
||||||
|
# one u2idx per tcp connection;
|
||||||
|
# sqlite3 fully parallelizes under python threads
|
||||||
if not self.u2idx:
|
if not self.u2idx:
|
||||||
self.u2idx = U2idx(self)
|
self.u2idx = U2idx(self)
|
||||||
|
|
||||||
return self.u2idx
|
return self.u2idx
|
||||||
|
|
||||||
def _detect_https(self):
|
def _detect_https(self) -> bool:
|
||||||
method = None
|
method = None
|
||||||
if self.cert_path:
|
if self.cert_path:
|
||||||
try:
|
try:
|
||||||
method = self.s.recv(4, socket.MSG_PEEK)
|
method = self.s.recv(4, socket.MSG_PEEK)
|
||||||
except socket.timeout:
|
except socket.timeout:
|
||||||
return
|
return False
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
# jython does not support msg_peek; forget about https
|
# jython does not support msg_peek; forget about https
|
||||||
method = self.s.recv(4)
|
method = self.s.recv(4)
|
||||||
self.sr = Unrecv(self.s)
|
self.sr = Util.Unrecv(self.s, self.log)
|
||||||
self.sr.buf = method
|
self.sr.buf = method
|
||||||
|
|
||||||
# jython used to do this, they stopped since it's broken
|
# jython used to do this, they stopped since it's broken
|
||||||
# but reimplementing sendall is out of scope for now
|
# but reimplementing sendall is out of scope for now
|
||||||
if not getattr(self.s, "sendall", None):
|
if not getattr(self.s, "sendall", None):
|
||||||
self.s.sendall = self.s.send
|
self.s.sendall = self.s.send # type: ignore
|
||||||
|
|
||||||
if len(method) != 4:
|
if len(method) != 4:
|
||||||
err = "need at least 4 bytes in the first packet; got {}".format(
|
err = "need at least 4 bytes in the first packet; got {}".format(
|
||||||
@@ -110,17 +137,20 @@ class HttpConn(object):
|
|||||||
self.log(err)
|
self.log(err)
|
||||||
|
|
||||||
self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8"))
|
self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8"))
|
||||||
return
|
return False
|
||||||
|
|
||||||
return method not in [None, b"GET ", b"HEAD", b"POST", b"PUT ", b"OPTI"]
|
return not method or not bool(PTN_HTTP.match(method))
|
||||||
|
|
||||||
|
def run(self) -> None:
|
||||||
|
self.s.settimeout(10)
|
||||||
|
|
||||||
def run(self):
|
|
||||||
self.sr = None
|
self.sr = None
|
||||||
if self.args.https_only:
|
if self.args.https_only:
|
||||||
is_https = True
|
is_https = True
|
||||||
elif self.args.http_only or not HAVE_SSL:
|
elif self.args.http_only or not HAVE_SSL:
|
||||||
is_https = False
|
is_https = False
|
||||||
else:
|
else:
|
||||||
|
# raise Exception("asdf")
|
||||||
is_https = self._detect_https()
|
is_https = self._detect_https()
|
||||||
|
|
||||||
if is_https:
|
if is_https:
|
||||||
@@ -149,14 +179,15 @@ class HttpConn(object):
|
|||||||
self.s = ctx.wrap_socket(self.s, server_side=True)
|
self.s = ctx.wrap_socket(self.s, server_side=True)
|
||||||
msg = [
|
msg = [
|
||||||
"\033[1;3{:d}m{}".format(c, s)
|
"\033[1;3{:d}m{}".format(c, s)
|
||||||
for c, s in zip([0, 5, 0], self.s.cipher())
|
for c, s in zip([0, 5, 0], self.s.cipher()) # type: ignore
|
||||||
]
|
]
|
||||||
self.log(" ".join(msg) + "\033[0m")
|
self.log(" ".join(msg) + "\033[0m")
|
||||||
|
|
||||||
if self.args.ssl_dbg and hasattr(self.s, "shared_ciphers"):
|
if self.args.ssl_dbg and hasattr(self.s, "shared_ciphers"):
|
||||||
overlap = [y[::-1] for y in self.s.shared_ciphers()]
|
ciphers = self.s.shared_ciphers()
|
||||||
lines = [str(x) for x in (["TLS cipher overlap:"] + overlap)]
|
assert ciphers
|
||||||
self.log("\n".join(lines))
|
overlap = [str(y[::-1]) for y in ciphers]
|
||||||
|
self.log("TLS cipher overlap:" + "\n".join(overlap))
|
||||||
for k, v in [
|
for k, v in [
|
||||||
["compression", self.s.compression()],
|
["compression", self.s.compression()],
|
||||||
["ALPN proto", self.s.selected_alpn_protocol()],
|
["ALPN proto", self.s.selected_alpn_protocol()],
|
||||||
@@ -167,11 +198,7 @@ class HttpConn(object):
|
|||||||
except Exception as ex:
|
except Exception as ex:
|
||||||
em = str(ex)
|
em = str(ex)
|
||||||
|
|
||||||
if "ALERT_BAD_CERTIFICATE" in em:
|
if "ALERT_CERTIFICATE_UNKNOWN" in em:
|
||||||
# firefox-linux if there is no exception yet
|
|
||||||
self.log("client rejected our certificate (nice)")
|
|
||||||
|
|
||||||
elif "ALERT_CERTIFICATE_UNKNOWN" in em:
|
|
||||||
# android-chrome keeps doing this
|
# android-chrome keeps doing this
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@@ -181,10 +208,10 @@ class HttpConn(object):
|
|||||||
return
|
return
|
||||||
|
|
||||||
if not self.sr:
|
if not self.sr:
|
||||||
self.sr = Unrecv(self.s)
|
self.sr = Util.Unrecv(self.s, self.log)
|
||||||
|
|
||||||
while not self.stopping:
|
while not self.stopping:
|
||||||
self.nreq += 1
|
self.nreq += 1
|
||||||
cli = HttpCli(self)
|
self.cli = HttpCli(self)
|
||||||
if not cli.run():
|
if not self.cli.run():
|
||||||
return
|
return
|
||||||
|
|||||||
@@ -1,17 +1,29 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
import time
|
|
||||||
import math
|
|
||||||
import base64
|
import base64
|
||||||
|
import math
|
||||||
|
import os
|
||||||
import socket
|
import socket
|
||||||
|
import sys
|
||||||
import threading
|
import threading
|
||||||
|
import time
|
||||||
|
|
||||||
|
import queue
|
||||||
|
|
||||||
|
from .__init__ import ANYWIN, EXE, MACOS, TYPE_CHECKING, EnvParams
|
||||||
|
|
||||||
|
try:
|
||||||
|
MNFE = ModuleNotFoundError
|
||||||
|
except:
|
||||||
|
MNFE = ImportError
|
||||||
|
|
||||||
try:
|
try:
|
||||||
import jinja2
|
import jinja2
|
||||||
except ImportError:
|
except MNFE:
|
||||||
|
if EXE:
|
||||||
|
raise
|
||||||
|
|
||||||
print(
|
print(
|
||||||
"""\033[1;31m
|
"""\033[1;31m
|
||||||
you do not have jinja2 installed,\033[33m
|
you do not have jinja2 installed,\033[33m
|
||||||
@@ -26,15 +38,30 @@ except ImportError:
|
|||||||
)
|
)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
from .__init__ import E, PY2, MACOS
|
|
||||||
from .util import FHC, spack, min_ex, start_stackmon, start_log_thrs
|
|
||||||
from .bos import bos
|
from .bos import bos
|
||||||
from .httpconn import HttpConn
|
from .httpconn import HttpConn
|
||||||
|
from .util import (
|
||||||
|
E_SCK,
|
||||||
|
FHC,
|
||||||
|
Daemon,
|
||||||
|
Garda,
|
||||||
|
Magician,
|
||||||
|
Netdev,
|
||||||
|
NetMap,
|
||||||
|
ipnorm,
|
||||||
|
min_ex,
|
||||||
|
shut_socket,
|
||||||
|
spack,
|
||||||
|
start_log_thrs,
|
||||||
|
start_stackmon,
|
||||||
|
)
|
||||||
|
|
||||||
if PY2:
|
if TYPE_CHECKING:
|
||||||
import Queue as queue
|
from .broker_util import BrokerCli
|
||||||
else:
|
from .ssdp import SSDPr
|
||||||
import queue
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any, Optional
|
||||||
|
|
||||||
|
|
||||||
class HttpSrv(object):
|
class HttpSrv(object):
|
||||||
@@ -43,46 +70,69 @@ class HttpSrv(object):
|
|||||||
relying on MpSrv for performance (HttpSrv is just plain threads)
|
relying on MpSrv for performance (HttpSrv is just plain threads)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, broker, nid):
|
def __init__(self, broker: "BrokerCli", nid: Optional[int]) -> None:
|
||||||
self.broker = broker
|
self.broker = broker
|
||||||
self.nid = nid
|
self.nid = nid
|
||||||
self.args = broker.args
|
self.args = broker.args
|
||||||
|
self.E: EnvParams = self.args.E
|
||||||
self.log = broker.log
|
self.log = broker.log
|
||||||
self.asrv = broker.asrv
|
self.asrv = broker.asrv
|
||||||
|
|
||||||
nsuf = "-n{}-i{:x}".format(nid, os.getpid()) if nid else ""
|
# redefine in case of multiprocessing
|
||||||
|
socket.setdefaulttimeout(120)
|
||||||
|
|
||||||
|
nsuf = "-n{}-i{:x}".format(nid, os.getpid()) if nid else ""
|
||||||
|
self.magician = Magician()
|
||||||
|
self.nm = NetMap([], {})
|
||||||
|
self.ssdp: Optional["SSDPr"] = None
|
||||||
|
self.gpwd = Garda(self.args.ban_pw)
|
||||||
|
self.g404 = Garda(self.args.ban_404)
|
||||||
|
self.bans: dict[str, int] = {}
|
||||||
|
self.aclose: dict[str, int] = {}
|
||||||
|
|
||||||
|
self.bound: set[tuple[str, int]] = set()
|
||||||
self.name = "hsrv" + nsuf
|
self.name = "hsrv" + nsuf
|
||||||
self.mutex = threading.Lock()
|
self.mutex = threading.Lock()
|
||||||
self.stopping = False
|
self.stopping = False
|
||||||
|
|
||||||
self.tp_nthr = 0 # actual
|
self.tp_nthr = 0 # actual
|
||||||
self.tp_ncli = 0 # fading
|
self.tp_ncli = 0 # fading
|
||||||
self.tp_time = None # latest worker collect
|
self.tp_time = 0.0 # latest worker collect
|
||||||
self.tp_q = None if self.args.no_htp else queue.LifoQueue()
|
self.tp_q: Optional[queue.LifoQueue[Any]] = (
|
||||||
self.t_periodic = None
|
None if self.args.no_htp else queue.LifoQueue()
|
||||||
|
)
|
||||||
|
self.t_periodic: Optional[threading.Thread] = None
|
||||||
|
|
||||||
self.u2fh = FHC()
|
self.u2fh = FHC()
|
||||||
self.srvs = []
|
self.srvs: list[socket.socket] = []
|
||||||
self.ncli = 0 # exact
|
self.ncli = 0 # exact
|
||||||
self.clients = {} # laggy
|
self.clients: set[HttpConn] = set() # laggy
|
||||||
self.nclimax = 0
|
self.nclimax = 0
|
||||||
self.cb_ts = 0
|
self.cb_ts = 0.0
|
||||||
self.cb_v = 0
|
self.cb_v = ""
|
||||||
|
|
||||||
env = jinja2.Environment()
|
env = jinja2.Environment()
|
||||||
env.loader = jinja2.FileSystemLoader(os.path.join(E.mod, "web"))
|
env.loader = jinja2.FileSystemLoader(os.path.join(self.E.mod, "web"))
|
||||||
self.j2 = {
|
jn = ["splash", "svcs", "browser", "browser2", "msg", "md", "mde", "cf"]
|
||||||
x: env.get_template(x + ".html")
|
self.j2 = {x: env.get_template(x + ".html") for x in jn}
|
||||||
for x in ["splash", "browser", "browser2", "msg", "md", "mde"]
|
zs = os.path.join(self.E.mod, "web", "deps", "prism.js.gz")
|
||||||
}
|
self.prism = os.path.exists(zs)
|
||||||
self.prism = os.path.exists(os.path.join(E.mod, "web", "deps", "prism.js.gz"))
|
|
||||||
|
|
||||||
cert_path = os.path.join(E.cfg, "cert.pem")
|
self.mallow = "GET HEAD POST PUT DELETE OPTIONS".split()
|
||||||
|
if not self.args.no_dav:
|
||||||
|
zs = "PROPFIND PROPPATCH LOCK UNLOCK MKCOL COPY MOVE"
|
||||||
|
self.mallow += zs.split()
|
||||||
|
|
||||||
|
if self.args.zs:
|
||||||
|
from .ssdp import SSDPr
|
||||||
|
|
||||||
|
self.ssdp = SSDPr(broker)
|
||||||
|
|
||||||
|
cert_path = os.path.join(self.E.cfg, "cert.pem")
|
||||||
if bos.path.exists(cert_path):
|
if bos.path.exists(cert_path):
|
||||||
self.cert_path = cert_path
|
self.cert_path = cert_path
|
||||||
else:
|
else:
|
||||||
self.cert_path = None
|
self.cert_path = ""
|
||||||
|
|
||||||
if self.tp_q:
|
if self.tp_q:
|
||||||
self.start_threads(4)
|
self.start_threads(4)
|
||||||
@@ -94,28 +144,41 @@ class HttpSrv(object):
|
|||||||
if self.args.log_thrs:
|
if self.args.log_thrs:
|
||||||
start_log_thrs(self.log, self.args.log_thrs, nid)
|
start_log_thrs(self.log, self.args.log_thrs, nid)
|
||||||
|
|
||||||
def start_threads(self, n):
|
self.th_cfg: dict[str, Any] = {}
|
||||||
|
Daemon(self.post_init, "hsrv-init2")
|
||||||
|
|
||||||
|
def post_init(self) -> None:
|
||||||
|
try:
|
||||||
|
x = self.broker.ask("thumbsrv.getcfg")
|
||||||
|
self.th_cfg = x.get()
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def set_netdevs(self, netdevs: dict[str, Netdev]) -> None:
|
||||||
|
ips = set()
|
||||||
|
for ip, _ in self.bound:
|
||||||
|
ips.add(ip)
|
||||||
|
|
||||||
|
self.nm = NetMap(list(ips), netdevs)
|
||||||
|
|
||||||
|
def start_threads(self, n: int) -> None:
|
||||||
self.tp_nthr += n
|
self.tp_nthr += n
|
||||||
if self.args.log_htp:
|
if self.args.log_htp:
|
||||||
self.log(self.name, "workers += {} = {}".format(n, self.tp_nthr), 6)
|
self.log(self.name, "workers += {} = {}".format(n, self.tp_nthr), 6)
|
||||||
|
|
||||||
for _ in range(n):
|
for _ in range(n):
|
||||||
thr = threading.Thread(
|
Daemon(self.thr_poolw, self.name + "-poolw")
|
||||||
target=self.thr_poolw,
|
|
||||||
name=self.name + "-poolw",
|
|
||||||
)
|
|
||||||
thr.daemon = True
|
|
||||||
thr.start()
|
|
||||||
|
|
||||||
def stop_threads(self, n):
|
def stop_threads(self, n: int) -> None:
|
||||||
self.tp_nthr -= n
|
self.tp_nthr -= n
|
||||||
if self.args.log_htp:
|
if self.args.log_htp:
|
||||||
self.log(self.name, "workers -= {} = {}".format(n, self.tp_nthr), 6)
|
self.log(self.name, "workers -= {} = {}".format(n, self.tp_nthr), 6)
|
||||||
|
|
||||||
|
assert self.tp_q
|
||||||
for _ in range(n):
|
for _ in range(n):
|
||||||
self.tp_q.put(None)
|
self.tp_q.put(None)
|
||||||
|
|
||||||
def periodic(self):
|
def periodic(self) -> None:
|
||||||
while True:
|
while True:
|
||||||
time.sleep(2 if self.tp_ncli or self.ncli else 10)
|
time.sleep(2 if self.tp_ncli or self.ncli else 10)
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
@@ -129,65 +192,134 @@ class HttpSrv(object):
|
|||||||
self.t_periodic = None
|
self.t_periodic = None
|
||||||
return
|
return
|
||||||
|
|
||||||
def listen(self, sck, nlisteners):
|
def listen(self, sck: socket.socket, nlisteners: int) -> None:
|
||||||
ip, port = sck.getsockname()
|
if self.args.j != 1:
|
||||||
self.srvs.append(sck)
|
# lost in the pickle; redefine
|
||||||
self.nclimax = math.ceil(self.args.nc * 1.0 / nlisteners)
|
if not ANYWIN or self.args.reuseaddr:
|
||||||
t = threading.Thread(
|
sck.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
|
||||||
target=self.thr_listen,
|
|
||||||
args=(sck,),
|
|
||||||
name="httpsrv-n{}-listen-{}-{}".format(self.nid or "0", ip, port),
|
|
||||||
)
|
|
||||||
t.daemon = True
|
|
||||||
t.start()
|
|
||||||
|
|
||||||
def thr_listen(self, srv_sck):
|
sck.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1)
|
||||||
|
sck.settimeout(None) # < does not inherit, ^ opts above do
|
||||||
|
|
||||||
|
ip, port = sck.getsockname()[:2]
|
||||||
|
self.srvs.append(sck)
|
||||||
|
self.bound.add((ip, port))
|
||||||
|
self.nclimax = math.ceil(self.args.nc * 1.0 / nlisteners)
|
||||||
|
Daemon(
|
||||||
|
self.thr_listen,
|
||||||
|
"httpsrv-n{}-listen-{}-{}".format(self.nid or "0", ip, port),
|
||||||
|
(sck,),
|
||||||
|
)
|
||||||
|
|
||||||
|
def thr_listen(self, srv_sck: socket.socket) -> None:
|
||||||
"""listens on a shared tcp server"""
|
"""listens on a shared tcp server"""
|
||||||
ip, port = srv_sck.getsockname()
|
ip, port = srv_sck.getsockname()[:2]
|
||||||
fno = srv_sck.fileno()
|
fno = srv_sck.fileno()
|
||||||
msg = "subscribed @ {}:{} f{}".format(ip, port, fno)
|
hip = "[{}]".format(ip) if ":" in ip else ip
|
||||||
|
msg = "subscribed @ {}:{} f{} p{}".format(hip, port, fno, os.getpid())
|
||||||
self.log(self.name, msg)
|
self.log(self.name, msg)
|
||||||
|
|
||||||
def fun():
|
def fun() -> None:
|
||||||
self.broker.put(False, "cb_httpsrv_up")
|
self.broker.say("cb_httpsrv_up")
|
||||||
|
|
||||||
threading.Thread(target=fun).start()
|
threading.Thread(target=fun, name="sig-hsrv-up1").start()
|
||||||
|
|
||||||
while not self.stopping:
|
while not self.stopping:
|
||||||
if self.args.log_conn:
|
if self.args.log_conn:
|
||||||
self.log(self.name, "|%sC-ncli" % ("-" * 1,), c="1;30")
|
self.log(self.name, "|%sC-ncli" % ("-" * 1,), c="90")
|
||||||
|
|
||||||
if self.ncli >= self.nclimax:
|
spins = 0
|
||||||
self.log(self.name, "at connection limit; waiting", 3)
|
while self.ncli >= self.nclimax:
|
||||||
while self.ncli >= self.nclimax:
|
if not spins:
|
||||||
time.sleep(0.1)
|
self.log(self.name, "at connection limit; waiting", 3)
|
||||||
|
|
||||||
|
spins += 1
|
||||||
|
time.sleep(0.1)
|
||||||
|
if spins != 50 or not self.args.aclose:
|
||||||
|
continue
|
||||||
|
|
||||||
|
ipfreq: dict[str, int] = {}
|
||||||
|
with self.mutex:
|
||||||
|
for c in self.clients:
|
||||||
|
ip = ipnorm(c.ip)
|
||||||
|
try:
|
||||||
|
ipfreq[ip] += 1
|
||||||
|
except:
|
||||||
|
ipfreq[ip] = 1
|
||||||
|
|
||||||
|
ip, n = sorted(ipfreq.items(), key=lambda x: x[1], reverse=True)[0]
|
||||||
|
if n < self.nclimax / 2:
|
||||||
|
continue
|
||||||
|
|
||||||
|
self.aclose[ip] = int(time.time() + self.args.aclose * 60)
|
||||||
|
nclose = 0
|
||||||
|
nloris = 0
|
||||||
|
nconn = 0
|
||||||
|
with self.mutex:
|
||||||
|
for c in self.clients:
|
||||||
|
cip = ipnorm(c.ip)
|
||||||
|
if ip != cip:
|
||||||
|
continue
|
||||||
|
|
||||||
|
nconn += 1
|
||||||
|
try:
|
||||||
|
if (
|
||||||
|
c.nreq >= 1
|
||||||
|
or not c.cli
|
||||||
|
or c.cli.in_hdr_recv
|
||||||
|
or c.cli.keepalive
|
||||||
|
):
|
||||||
|
Daemon(c.shutdown)
|
||||||
|
nclose += 1
|
||||||
|
if c.nreq <= 0 and (not c.cli or c.cli.in_hdr_recv):
|
||||||
|
nloris += 1
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
t = "{} downgraded to connection:close for {} min; dropped {}/{} connections"
|
||||||
|
self.log(self.name, t.format(ip, self.args.aclose, nclose, nconn), 1)
|
||||||
|
|
||||||
|
if nloris < nconn / 2:
|
||||||
|
continue
|
||||||
|
|
||||||
|
t = "slowloris (idle-conn): {} banned for {} min"
|
||||||
|
self.log(self.name, t.format(ip, self.args.loris, nclose), 1)
|
||||||
|
self.bans[ip] = int(time.time() + self.args.loris * 60)
|
||||||
|
|
||||||
if self.args.log_conn:
|
if self.args.log_conn:
|
||||||
self.log(self.name, "|%sC-acc1" % ("-" * 2,), c="1;30")
|
self.log(self.name, "|%sC-acc1" % ("-" * 2,), c="90")
|
||||||
|
|
||||||
try:
|
try:
|
||||||
sck, addr = srv_sck.accept()
|
sck, saddr = srv_sck.accept()
|
||||||
|
cip, cport = saddr[:2]
|
||||||
|
if cip.startswith("::ffff:"):
|
||||||
|
cip = cip[7:]
|
||||||
|
|
||||||
|
addr = (cip, cport)
|
||||||
except (OSError, socket.error) as ex:
|
except (OSError, socket.error) as ex:
|
||||||
|
if self.stopping:
|
||||||
|
break
|
||||||
|
|
||||||
self.log(self.name, "accept({}): {}".format(fno, ex), c=6)
|
self.log(self.name, "accept({}): {}".format(fno, ex), c=6)
|
||||||
time.sleep(0.02)
|
time.sleep(0.02)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if self.args.log_conn:
|
if self.args.log_conn:
|
||||||
m = "|{}C-acc2 \033[0;36m{} \033[3{}m{}".format(
|
t = "|{}C-acc2 \033[0;36m{} \033[3{}m{}".format(
|
||||||
"-" * 3, ip, port % 8, port
|
"-" * 3, ip, port % 8, port
|
||||||
)
|
)
|
||||||
self.log("%s %s" % addr, m, c="1;30")
|
self.log("%s %s" % addr, t, c="90")
|
||||||
|
|
||||||
self.accept(sck, addr)
|
self.accept(sck, addr)
|
||||||
|
|
||||||
def accept(self, sck, addr):
|
def accept(self, sck: socket.socket, addr: tuple[str, int]) -> None:
|
||||||
"""takes an incoming tcp connection and creates a thread to handle it"""
|
"""takes an incoming tcp connection and creates a thread to handle it"""
|
||||||
now = time.time()
|
now = time.time()
|
||||||
|
|
||||||
if now - (self.tp_time or now) > 300:
|
if now - (self.tp_time or now) > 300:
|
||||||
m = "httpserver threadpool died: tpt {:.2f}, now {:.2f}, nthr {}, ncli {}"
|
t = "httpserver threadpool died: tpt {:.2f}, now {:.2f}, nthr {}, ncli {}"
|
||||||
self.log(self.name, m.format(self.tp_time, now, self.tp_nthr, self.ncli), 1)
|
self.log(self.name, t.format(self.tp_time, now, self.tp_nthr, self.ncli), 1)
|
||||||
self.tp_time = None
|
self.tp_time = 0
|
||||||
self.tp_q = None
|
self.tp_q = None
|
||||||
|
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
@@ -197,10 +329,7 @@ class HttpSrv(object):
|
|||||||
if self.nid:
|
if self.nid:
|
||||||
name += "-{}".format(self.nid)
|
name += "-{}".format(self.nid)
|
||||||
|
|
||||||
t = threading.Thread(target=self.periodic, name=name)
|
self.t_periodic = Daemon(self.periodic, name)
|
||||||
self.t_periodic = t
|
|
||||||
t.daemon = True
|
|
||||||
t.start()
|
|
||||||
|
|
||||||
if self.tp_q:
|
if self.tp_q:
|
||||||
self.tp_time = self.tp_time or now
|
self.tp_time = self.tp_time or now
|
||||||
@@ -212,25 +341,24 @@ class HttpSrv(object):
|
|||||||
return
|
return
|
||||||
|
|
||||||
if not self.args.no_htp:
|
if not self.args.no_htp:
|
||||||
m = "looks like the httpserver threadpool died; please make an issue on github and tell me the story of how you pulled that off, thanks and dog bless\n"
|
t = "looks like the httpserver threadpool died; please make an issue on github and tell me the story of how you pulled that off, thanks and dog bless\n"
|
||||||
self.log(self.name, m, 1)
|
self.log(self.name, t, 1)
|
||||||
|
|
||||||
thr = threading.Thread(
|
Daemon(
|
||||||
target=self.thr_client,
|
self.thr_client,
|
||||||
args=(sck, addr),
|
"httpconn-{}-{}".format(addr[0].split(".", 2)[-1][-6:], addr[1]),
|
||||||
name="httpconn-{}-{}".format(addr[0].split(".", 2)[-1][-6:], addr[1]),
|
(sck, addr),
|
||||||
)
|
)
|
||||||
thr.daemon = True
|
|
||||||
thr.start()
|
|
||||||
|
|
||||||
def thr_poolw(self):
|
def thr_poolw(self) -> None:
|
||||||
|
assert self.tp_q
|
||||||
while True:
|
while True:
|
||||||
task = self.tp_q.get()
|
task = self.tp_q.get()
|
||||||
if not task:
|
if not task:
|
||||||
break
|
break
|
||||||
|
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
self.tp_time = None
|
self.tp_time = 0
|
||||||
|
|
||||||
try:
|
try:
|
||||||
sck, addr = task
|
sck, addr = task
|
||||||
@@ -240,10 +368,13 @@ class HttpSrv(object):
|
|||||||
)
|
)
|
||||||
self.thr_client(sck, addr)
|
self.thr_client(sck, addr)
|
||||||
me.name = self.name + "-poolw"
|
me.name = self.name + "-poolw"
|
||||||
except:
|
except Exception as ex:
|
||||||
self.log(self.name, "thr_client: " + min_ex(), 3)
|
if str(ex).startswith("client d/c "):
|
||||||
|
self.log(self.name, "thr_client: " + str(ex), 6)
|
||||||
|
else:
|
||||||
|
self.log(self.name, "thr_client: " + min_ex(), 3)
|
||||||
|
|
||||||
def shutdown(self):
|
def shutdown(self) -> None:
|
||||||
self.stopping = True
|
self.stopping = True
|
||||||
for srv in self.srvs:
|
for srv in self.srvs:
|
||||||
try:
|
try:
|
||||||
@@ -251,12 +382,12 @@ class HttpSrv(object):
|
|||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
clients = list(self.clients.keys())
|
thrs = []
|
||||||
|
clients = list(self.clients)
|
||||||
for cli in clients:
|
for cli in clients:
|
||||||
try:
|
t = threading.Thread(target=cli.shutdown)
|
||||||
cli.shutdown()
|
thrs.append(t)
|
||||||
except:
|
t.start()
|
||||||
pass
|
|
||||||
|
|
||||||
if self.tp_q:
|
if self.tp_q:
|
||||||
self.stop_threads(self.tp_nthr)
|
self.stop_threads(self.tp_nthr)
|
||||||
@@ -265,25 +396,27 @@ class HttpSrv(object):
|
|||||||
if self.tp_q.empty():
|
if self.tp_q.empty():
|
||||||
break
|
break
|
||||||
|
|
||||||
|
for t in thrs:
|
||||||
|
t.join()
|
||||||
|
|
||||||
self.log(self.name, "ok bye")
|
self.log(self.name, "ok bye")
|
||||||
|
|
||||||
def thr_client(self, sck, addr):
|
def thr_client(self, sck: socket.socket, addr: tuple[str, int]) -> None:
|
||||||
"""thread managing one tcp client"""
|
"""thread managing one tcp client"""
|
||||||
sck.settimeout(120)
|
|
||||||
|
|
||||||
cli = HttpConn(sck, addr, self)
|
cli = HttpConn(sck, addr, self)
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
self.clients[cli] = 0
|
self.clients.add(cli)
|
||||||
|
|
||||||
|
# print("{}\n".format(len(self.clients)), end="")
|
||||||
fno = sck.fileno()
|
fno = sck.fileno()
|
||||||
try:
|
try:
|
||||||
if self.args.log_conn:
|
if self.args.log_conn:
|
||||||
self.log("%s %s" % addr, "|%sC-crun" % ("-" * 4,), c="1;30")
|
self.log("%s %s" % addr, "|%sC-crun" % ("-" * 4,), c="90")
|
||||||
|
|
||||||
cli.run()
|
cli.run()
|
||||||
|
|
||||||
except (OSError, socket.error) as ex:
|
except (OSError, socket.error) as ex:
|
||||||
if ex.errno not in [10038, 10054, 107, 57, 49, 9]:
|
if ex.errno not in E_SCK:
|
||||||
self.log(
|
self.log(
|
||||||
"%s %s" % addr,
|
"%s %s" % addr,
|
||||||
"run({}): {}".format(fno, ex),
|
"run({}): {}".format(fno, ex),
|
||||||
@@ -293,33 +426,26 @@ class HttpSrv(object):
|
|||||||
finally:
|
finally:
|
||||||
sck = cli.s
|
sck = cli.s
|
||||||
if self.args.log_conn:
|
if self.args.log_conn:
|
||||||
self.log("%s %s" % addr, "|%sC-cdone" % ("-" * 5,), c="1;30")
|
self.log("%s %s" % addr, "|%sC-cdone" % ("-" * 5,), c="90")
|
||||||
|
|
||||||
try:
|
try:
|
||||||
fno = sck.fileno()
|
fno = sck.fileno()
|
||||||
sck.shutdown(socket.SHUT_RDWR)
|
shut_socket(cli.log, sck)
|
||||||
sck.close()
|
|
||||||
except (OSError, socket.error) as ex:
|
except (OSError, socket.error) as ex:
|
||||||
if not MACOS:
|
if not MACOS:
|
||||||
self.log(
|
self.log(
|
||||||
"%s %s" % addr,
|
"%s %s" % addr,
|
||||||
"shut({}): {}".format(fno, ex),
|
"shut({}): {}".format(fno, ex),
|
||||||
c="1;30",
|
c="90",
|
||||||
)
|
)
|
||||||
if ex.errno not in [10038, 10054, 107, 57, 49, 9]:
|
if ex.errno not in E_SCK:
|
||||||
# 10038 No longer considered a socket
|
|
||||||
# 10054 Foribly closed by remote
|
|
||||||
# 107 Transport endpoint not connected
|
|
||||||
# 57 Socket is not connected
|
|
||||||
# 49 Can't assign requested address (wifi down)
|
|
||||||
# 9 Bad file descriptor
|
|
||||||
raise
|
raise
|
||||||
finally:
|
finally:
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
del self.clients[cli]
|
self.clients.remove(cli)
|
||||||
self.ncli -= 1
|
self.ncli -= 1
|
||||||
|
|
||||||
def cachebuster(self):
|
def cachebuster(self) -> str:
|
||||||
if time.time() - self.cb_ts < 1:
|
if time.time() - self.cb_ts < 1:
|
||||||
return self.cb_v
|
return self.cb_v
|
||||||
|
|
||||||
@@ -327,9 +453,9 @@ class HttpSrv(object):
|
|||||||
if time.time() - self.cb_ts < 1:
|
if time.time() - self.cb_ts < 1:
|
||||||
return self.cb_v
|
return self.cb_v
|
||||||
|
|
||||||
v = E.t0
|
v = self.E.t0
|
||||||
try:
|
try:
|
||||||
with os.scandir(os.path.join(E.mod, "web")) as dh:
|
with os.scandir(os.path.join(self.E.mod, "web")) as dh:
|
||||||
for fh in dh:
|
for fh in dh:
|
||||||
inf = fh.stat()
|
inf = fh.stat()
|
||||||
v = max(v, inf.st_mtime)
|
v = max(v, inf.st_mtime)
|
||||||
|
|||||||
@@ -1,33 +1,69 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import hashlib
|
import argparse # typechk
|
||||||
import colorsys
|
import colorsys
|
||||||
|
import hashlib
|
||||||
|
|
||||||
from .__init__ import PY2
|
from .__init__ import PY2
|
||||||
|
from .th_srv import HAVE_PIL
|
||||||
|
from .util import BytesIO
|
||||||
|
|
||||||
|
|
||||||
class Ico(object):
|
class Ico(object):
|
||||||
def __init__(self, args):
|
def __init__(self, args: argparse.Namespace) -> None:
|
||||||
self.args = args
|
self.args = args
|
||||||
|
|
||||||
def get(self, ext, as_thumb):
|
def get(self, ext: str, as_thumb: bool, chrome: bool) -> tuple[str, bytes]:
|
||||||
"""placeholder to make thumbnails not break"""
|
"""placeholder to make thumbnails not break"""
|
||||||
|
|
||||||
h = hashlib.md5(ext.encode("utf-8")).digest()[:2]
|
zb = hashlib.sha1(ext.encode("utf-8")).digest()[2:4]
|
||||||
if PY2:
|
if PY2:
|
||||||
h = [ord(x) for x in h]
|
zb = [ord(x) for x in zb]
|
||||||
|
|
||||||
c1 = colorsys.hsv_to_rgb(h[0] / 256.0, 1, 0.3)
|
c1 = colorsys.hsv_to_rgb(zb[0] / 256.0, 1, 0.3)
|
||||||
c2 = colorsys.hsv_to_rgb(h[0] / 256.0, 1, 1)
|
c2 = colorsys.hsv_to_rgb(zb[0] / 256.0, 1, 1)
|
||||||
c = list(c1) + list(c2)
|
ci = [int(x * 255) for x in list(c1) + list(c2)]
|
||||||
c = [int(x * 255) for x in c]
|
c = "".join(["{:02x}".format(x) for x in ci])
|
||||||
c = "".join(["{:02x}".format(x) for x in c])
|
|
||||||
|
|
||||||
|
w = 100
|
||||||
h = 30
|
h = 30
|
||||||
if not self.args.th_no_crop and as_thumb:
|
if not self.args.th_no_crop and as_thumb:
|
||||||
w, h = self.args.th_size.split("x")
|
sw, sh = self.args.th_size.split("x")
|
||||||
h = int(100 / (float(w) / float(h)))
|
h = int(100 / (float(sw) / float(sh)))
|
||||||
|
w = 100
|
||||||
|
|
||||||
|
if chrome and as_thumb:
|
||||||
|
# cannot handle more than ~2000 unique SVGs
|
||||||
|
if HAVE_PIL:
|
||||||
|
# svg: 3s, cache: 6s, this: 8s
|
||||||
|
from PIL import Image, ImageDraw
|
||||||
|
|
||||||
|
h = int(64 * h / w)
|
||||||
|
w = 64
|
||||||
|
img = Image.new("RGB", (w, h), "#" + c[:6])
|
||||||
|
pb = ImageDraw.Draw(img)
|
||||||
|
tw, th = pb.textsize(ext)
|
||||||
|
pb.text(((w - tw) // 2, (h - th) // 2), ext, fill="#" + c[6:])
|
||||||
|
img = img.resize((w * 3, h * 3), Image.NEAREST)
|
||||||
|
|
||||||
|
buf = BytesIO()
|
||||||
|
img.save(buf, format="PNG", compress_level=1)
|
||||||
|
return "image/png", buf.getvalue()
|
||||||
|
|
||||||
|
elif False:
|
||||||
|
# 48s, too slow
|
||||||
|
import pyvips
|
||||||
|
|
||||||
|
h = int(192 * h / w)
|
||||||
|
w = 192
|
||||||
|
img = pyvips.Image.text(
|
||||||
|
ext, width=w, height=h, dpi=192, align=pyvips.Align.CENTRE
|
||||||
|
)
|
||||||
|
img = img.ifthenelse(ci[3:], ci[:3], blend=True)
|
||||||
|
# i = i.resize(3, kernel=pyvips.Kernel.NEAREST)
|
||||||
|
buf = img.write_to_buffer(".png[compression=1]")
|
||||||
|
return "image/png", buf
|
||||||
|
|
||||||
svg = """\
|
svg = """\
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
@@ -37,6 +73,6 @@ class Ico(object):
|
|||||||
fill="#{}" font-family="monospace" font-size="14px" style="letter-spacing:.5px">{}</text>
|
fill="#{}" font-family="monospace" font-size="14px" style="letter-spacing:.5px">{}</text>
|
||||||
</g></svg>
|
</g></svg>
|
||||||
"""
|
"""
|
||||||
svg = svg.format(h, c[:6], c[6:], ext).encode("utf-8")
|
svg = svg.format(h, c[:6], c[6:], ext)
|
||||||
|
|
||||||
return ["image/svg+xml", svg]
|
return "image/svg+xml", svg.encode("utf-8")
|
||||||
|
|||||||
538
copyparty/mdns.py
Normal file
538
copyparty/mdns.py
Normal file
@@ -0,0 +1,538 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import random
|
||||||
|
import select
|
||||||
|
import socket
|
||||||
|
import time
|
||||||
|
|
||||||
|
from ipaddress import IPv4Network, IPv6Network
|
||||||
|
|
||||||
|
from .__init__ import TYPE_CHECKING
|
||||||
|
from .__init__ import unicode as U
|
||||||
|
from .multicast import MC_Sck, MCast
|
||||||
|
from .stolen.dnslib import AAAA
|
||||||
|
from .stolen.dnslib import CLASS as DC
|
||||||
|
from .stolen.dnslib import (
|
||||||
|
NSEC,
|
||||||
|
PTR,
|
||||||
|
QTYPE,
|
||||||
|
RR,
|
||||||
|
SRV,
|
||||||
|
TXT,
|
||||||
|
A,
|
||||||
|
DNSHeader,
|
||||||
|
DNSQuestion,
|
||||||
|
DNSRecord,
|
||||||
|
)
|
||||||
|
from .util import CachedSet, Daemon, Netdev, list_ips, min_ex
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from .svchub import SvcHub
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any, Optional, Union
|
||||||
|
|
||||||
|
|
||||||
|
MDNS4 = "224.0.0.251"
|
||||||
|
MDNS6 = "ff02::fb"
|
||||||
|
|
||||||
|
|
||||||
|
class MDNS_Sck(MC_Sck):
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
sck: socket.socket,
|
||||||
|
nd: Netdev,
|
||||||
|
grp: str,
|
||||||
|
ip: str,
|
||||||
|
net: Union[IPv4Network, IPv6Network],
|
||||||
|
):
|
||||||
|
super(MDNS_Sck, self).__init__(sck, nd, grp, ip, net)
|
||||||
|
|
||||||
|
self.bp_probe = b""
|
||||||
|
self.bp_ip = b""
|
||||||
|
self.bp_svc = b""
|
||||||
|
self.bp_bye = b""
|
||||||
|
|
||||||
|
self.last_tx = 0.0
|
||||||
|
self.tx_ex = False
|
||||||
|
|
||||||
|
|
||||||
|
class MDNS(MCast):
|
||||||
|
def __init__(self, hub: "SvcHub", ngen: int) -> None:
|
||||||
|
al = hub.args
|
||||||
|
grp4 = "" if al.zm6 else MDNS4
|
||||||
|
grp6 = "" if al.zm4 else MDNS6
|
||||||
|
super(MDNS, self).__init__(
|
||||||
|
hub, MDNS_Sck, al.zm_on, al.zm_off, grp4, grp6, 5353, hub.args.zmv
|
||||||
|
)
|
||||||
|
self.srv: dict[socket.socket, MDNS_Sck] = {}
|
||||||
|
self.logsrc = "mDNS-{}".format(ngen)
|
||||||
|
self.ngen = ngen
|
||||||
|
self.ttl = 300
|
||||||
|
|
||||||
|
zs = self.args.name + ".local."
|
||||||
|
zs = zs.encode("ascii", "replace").decode("ascii", "replace")
|
||||||
|
self.hn = "-".join(x for x in zs.split("?") if x) or (
|
||||||
|
"vault-{}".format(random.randint(1, 255))
|
||||||
|
)
|
||||||
|
self.lhn = self.hn.lower()
|
||||||
|
|
||||||
|
# requester ip -> (response deadline, srv, body):
|
||||||
|
self.q: dict[str, tuple[float, MDNS_Sck, bytes]] = {}
|
||||||
|
self.rx4 = CachedSet(0.42) # 3 probes @ 250..500..750 => 500ms span
|
||||||
|
self.rx6 = CachedSet(0.42)
|
||||||
|
self.svcs, self.sfqdns = self.build_svcs()
|
||||||
|
self.lsvcs = {k.lower(): v for k, v in self.svcs.items()}
|
||||||
|
self.lsfqdns = set([x.lower() for x in self.sfqdns])
|
||||||
|
|
||||||
|
self.probing = 0.0
|
||||||
|
self.unsolicited: list[float] = [] # scheduled announces on all nics
|
||||||
|
self.defend: dict[MDNS_Sck, float] = {} # server -> deadline
|
||||||
|
|
||||||
|
def log(self, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
|
self.log_func(self.logsrc, msg, c)
|
||||||
|
|
||||||
|
def build_svcs(self) -> tuple[dict[str, dict[str, Any]], set[str]]:
|
||||||
|
zms = self.args.zms
|
||||||
|
http = {"port": 80 if 80 in self.args.p else self.args.p[0]}
|
||||||
|
https = {"port": 443 if 443 in self.args.p else self.args.p[0]}
|
||||||
|
webdav = http.copy()
|
||||||
|
webdavs = https.copy()
|
||||||
|
webdav["u"] = webdavs["u"] = "u" # KDE requires username
|
||||||
|
ftp = {"port": (self.args.ftp if "f" in zms else self.args.ftps)}
|
||||||
|
smb = {"port": self.args.smb_port}
|
||||||
|
|
||||||
|
# some gvfs require path
|
||||||
|
zs = self.args.zm_ld or "/"
|
||||||
|
if zs:
|
||||||
|
webdav["path"] = zs
|
||||||
|
webdavs["path"] = zs
|
||||||
|
|
||||||
|
if self.args.zm_lh:
|
||||||
|
http["path"] = self.args.zm_lh
|
||||||
|
https["path"] = self.args.zm_lh
|
||||||
|
|
||||||
|
if self.args.zm_lf:
|
||||||
|
ftp["path"] = self.args.zm_lf
|
||||||
|
|
||||||
|
if self.args.zm_ls:
|
||||||
|
smb["path"] = self.args.zm_ls
|
||||||
|
|
||||||
|
svcs: dict[str, dict[str, Any]] = {}
|
||||||
|
|
||||||
|
if "d" in zms:
|
||||||
|
svcs["_webdav._tcp.local."] = webdav
|
||||||
|
|
||||||
|
if "D" in zms:
|
||||||
|
svcs["_webdavs._tcp.local."] = webdavs
|
||||||
|
|
||||||
|
if "h" in zms:
|
||||||
|
svcs["_http._tcp.local."] = http
|
||||||
|
|
||||||
|
if "H" in zms:
|
||||||
|
svcs["_https._tcp.local."] = https
|
||||||
|
|
||||||
|
if "f" in zms.lower():
|
||||||
|
svcs["_ftp._tcp.local."] = ftp
|
||||||
|
|
||||||
|
if "s" in zms.lower():
|
||||||
|
svcs["_smb._tcp.local."] = smb
|
||||||
|
|
||||||
|
sfqdns: set[str] = set()
|
||||||
|
for k, v in svcs.items():
|
||||||
|
name = "{}-c-{}".format(self.args.name, k.split(".")[0][1:])
|
||||||
|
v["name"] = name
|
||||||
|
sfqdns.add("{}.{}".format(name, k))
|
||||||
|
|
||||||
|
return svcs, sfqdns
|
||||||
|
|
||||||
|
def build_replies(self) -> None:
|
||||||
|
for srv in self.srv.values():
|
||||||
|
probe = DNSRecord(DNSHeader(0, 0), q=DNSQuestion(self.hn, QTYPE.ANY))
|
||||||
|
areply = DNSRecord(DNSHeader(0, 0x8400))
|
||||||
|
sreply = DNSRecord(DNSHeader(0, 0x8400))
|
||||||
|
bye = DNSRecord(DNSHeader(0, 0x8400))
|
||||||
|
|
||||||
|
have4 = have6 = False
|
||||||
|
for s2 in self.srv.values():
|
||||||
|
if srv.idx != s2.idx:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if s2.v6:
|
||||||
|
have6 = True
|
||||||
|
else:
|
||||||
|
have4 = True
|
||||||
|
|
||||||
|
for ip in srv.ips:
|
||||||
|
if ":" in ip:
|
||||||
|
qt = QTYPE.AAAA
|
||||||
|
ar = {"rclass": DC.F_IN, "rdata": AAAA(ip)}
|
||||||
|
else:
|
||||||
|
qt = QTYPE.A
|
||||||
|
ar = {"rclass": DC.F_IN, "rdata": A(ip)}
|
||||||
|
|
||||||
|
r0 = RR(self.hn, qt, ttl=0, **ar)
|
||||||
|
r120 = RR(self.hn, qt, ttl=120, **ar)
|
||||||
|
# rfc-10:
|
||||||
|
# SHOULD rr ttl 120sec for A/AAAA/SRV
|
||||||
|
# (and recommend 75min for all others)
|
||||||
|
|
||||||
|
probe.add_auth(r120)
|
||||||
|
areply.add_answer(r120)
|
||||||
|
sreply.add_answer(r120)
|
||||||
|
bye.add_answer(r0)
|
||||||
|
|
||||||
|
for sclass, props in self.svcs.items():
|
||||||
|
sname = props["name"]
|
||||||
|
sport = props["port"]
|
||||||
|
sfqdn = sname + "." + sclass
|
||||||
|
|
||||||
|
k = "_services._dns-sd._udp.local."
|
||||||
|
r = RR(k, QTYPE.PTR, DC.IN, 4500, PTR(sclass))
|
||||||
|
sreply.add_answer(r)
|
||||||
|
|
||||||
|
r = RR(sclass, QTYPE.PTR, DC.IN, 4500, PTR(sfqdn))
|
||||||
|
sreply.add_answer(r)
|
||||||
|
|
||||||
|
r = RR(sfqdn, QTYPE.SRV, DC.F_IN, 120, SRV(0, 0, sport, self.hn))
|
||||||
|
sreply.add_answer(r)
|
||||||
|
areply.add_answer(r)
|
||||||
|
|
||||||
|
r = RR(sfqdn, QTYPE.SRV, DC.F_IN, 0, SRV(0, 0, sport, self.hn))
|
||||||
|
bye.add_answer(r)
|
||||||
|
|
||||||
|
txts = []
|
||||||
|
for k in ("u", "path"):
|
||||||
|
if k not in props:
|
||||||
|
continue
|
||||||
|
|
||||||
|
zb = "{}={}".format(k, props[k]).encode("utf-8")
|
||||||
|
if len(zb) > 255:
|
||||||
|
t = "value too long for mdns: [{}]"
|
||||||
|
raise Exception(t.format(props[k]))
|
||||||
|
|
||||||
|
txts.append(zb)
|
||||||
|
|
||||||
|
# gvfs really wants txt even if they're empty
|
||||||
|
r = RR(sfqdn, QTYPE.TXT, DC.F_IN, 4500, TXT(txts))
|
||||||
|
sreply.add_answer(r)
|
||||||
|
|
||||||
|
if not (have4 and have6) and not self.args.zm_noneg:
|
||||||
|
ns = NSEC(self.hn, ["AAAA" if have6 else "A"])
|
||||||
|
r = RR(self.hn, QTYPE.NSEC, DC.F_IN, 120, ns)
|
||||||
|
areply.add_ar(r)
|
||||||
|
if len(sreply.pack()) < 1400:
|
||||||
|
sreply.add_ar(r)
|
||||||
|
|
||||||
|
srv.bp_probe = probe.pack()
|
||||||
|
srv.bp_ip = areply.pack()
|
||||||
|
srv.bp_svc = sreply.pack()
|
||||||
|
srv.bp_bye = bye.pack()
|
||||||
|
|
||||||
|
# since all replies are small enough to fit in one packet,
|
||||||
|
# always send full replies rather than just a/aaaa records
|
||||||
|
srv.bp_ip = srv.bp_svc
|
||||||
|
|
||||||
|
def send_probes(self) -> None:
|
||||||
|
slp = random.random() * 0.25
|
||||||
|
for _ in range(3):
|
||||||
|
time.sleep(slp)
|
||||||
|
slp = 0.25
|
||||||
|
if not self.running:
|
||||||
|
break
|
||||||
|
|
||||||
|
if self.args.zmv:
|
||||||
|
self.log("sending hostname probe...")
|
||||||
|
|
||||||
|
# ipv4: need to probe each ip (each server)
|
||||||
|
# ipv6: only need to probe each set of looped nics
|
||||||
|
probed6: set[str] = set()
|
||||||
|
for srv in self.srv.values():
|
||||||
|
if srv.ip in probed6:
|
||||||
|
continue
|
||||||
|
|
||||||
|
try:
|
||||||
|
srv.sck.sendto(srv.bp_probe, (srv.grp, 5353))
|
||||||
|
if srv.v6:
|
||||||
|
for ip in srv.ips:
|
||||||
|
probed6.add(ip)
|
||||||
|
except Exception as ex:
|
||||||
|
self.log("sendto failed: {} ({})".format(srv.ip, ex), "90")
|
||||||
|
|
||||||
|
def run(self) -> None:
|
||||||
|
try:
|
||||||
|
bound = self.create_servers()
|
||||||
|
except:
|
||||||
|
t = "no server IP matches the mdns config\n{}"
|
||||||
|
self.log(t.format(min_ex()), 1)
|
||||||
|
bound = []
|
||||||
|
|
||||||
|
if not bound:
|
||||||
|
self.log("failed to announce copyparty services on the network", 3)
|
||||||
|
return
|
||||||
|
|
||||||
|
self.build_replies()
|
||||||
|
Daemon(self.send_probes)
|
||||||
|
zf = time.time() + 2
|
||||||
|
self.probing = zf # cant unicast so give everyone an extra sec
|
||||||
|
self.unsolicited = [zf, zf + 1, zf + 3, zf + 7] # rfc-8.3
|
||||||
|
last_hop = time.time()
|
||||||
|
ihop = self.args.mc_hop
|
||||||
|
while self.running:
|
||||||
|
timeout = (
|
||||||
|
0.02 + random.random() * 0.07
|
||||||
|
if self.probing or self.q or self.defend or self.unsolicited
|
||||||
|
else (last_hop + ihop if ihop else 180)
|
||||||
|
)
|
||||||
|
rdy = select.select(self.srv, [], [], timeout)
|
||||||
|
rx: list[socket.socket] = rdy[0] # type: ignore
|
||||||
|
self.rx4.cln()
|
||||||
|
self.rx6.cln()
|
||||||
|
buf = b""
|
||||||
|
addr = ("0", 0)
|
||||||
|
for sck in rx:
|
||||||
|
try:
|
||||||
|
buf, addr = sck.recvfrom(4096)
|
||||||
|
self.eat(buf, addr, sck)
|
||||||
|
except:
|
||||||
|
if not self.running:
|
||||||
|
self.log("stopped", 2)
|
||||||
|
return
|
||||||
|
|
||||||
|
t = "{} {} \033[33m|{}| {}\n{}".format(
|
||||||
|
self.srv[sck].name, addr, len(buf), repr(buf)[2:-1], min_ex()
|
||||||
|
)
|
||||||
|
self.log(t, 6)
|
||||||
|
|
||||||
|
if not self.probing:
|
||||||
|
self.process()
|
||||||
|
continue
|
||||||
|
|
||||||
|
if self.probing < time.time():
|
||||||
|
t = "probe ok; announcing [{}]"
|
||||||
|
self.log(t.format(self.hn[:-1]), 2)
|
||||||
|
self.probing = 0
|
||||||
|
|
||||||
|
self.log("stopped", 2)
|
||||||
|
|
||||||
|
def stop(self, panic=False) -> None:
|
||||||
|
self.running = False
|
||||||
|
for srv in self.srv.values():
|
||||||
|
try:
|
||||||
|
if panic:
|
||||||
|
srv.sck.close()
|
||||||
|
else:
|
||||||
|
srv.sck.sendto(srv.bp_bye, (srv.grp, 5353))
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
self.srv = {}
|
||||||
|
|
||||||
|
def eat(self, buf: bytes, addr: tuple[str, int], sck: socket.socket) -> None:
|
||||||
|
cip = addr[0]
|
||||||
|
v6 = ":" in cip
|
||||||
|
if (cip.startswith("169.254") and not self.ll_ok) or (
|
||||||
|
v6 and not cip.startswith("fe80")
|
||||||
|
):
|
||||||
|
return
|
||||||
|
|
||||||
|
cache = self.rx6 if v6 else self.rx4
|
||||||
|
if buf in cache.c:
|
||||||
|
return
|
||||||
|
|
||||||
|
srv: Optional[MDNS_Sck] = self.srv[sck] if v6 else self.map_client(cip) # type: ignore
|
||||||
|
if not srv:
|
||||||
|
return
|
||||||
|
|
||||||
|
cache.add(buf)
|
||||||
|
now = time.time()
|
||||||
|
|
||||||
|
if self.args.zmv and cip != srv.ip and cip not in srv.ips:
|
||||||
|
t = "{} [{}] \033[36m{} \033[0m|{}|"
|
||||||
|
self.log(t.format(srv.name, srv.ip, cip, len(buf)), "90")
|
||||||
|
|
||||||
|
p = DNSRecord.parse(buf)
|
||||||
|
if self.args.zmvv:
|
||||||
|
self.log(str(p))
|
||||||
|
|
||||||
|
# check for incoming probes for our hostname
|
||||||
|
cips = [U(x.rdata) for x in p.auth if U(x.rname).lower() == self.lhn]
|
||||||
|
if cips and self.sips.isdisjoint(cips):
|
||||||
|
if not [x for x in cips if x not in ("::1", "127.0.0.1")]:
|
||||||
|
# avahi broadcasting 127.0.0.1-only packets
|
||||||
|
return
|
||||||
|
|
||||||
|
self.log("someone trying to steal our hostname: {}".format(cips), 3)
|
||||||
|
# immediately unicast
|
||||||
|
if not self.probing:
|
||||||
|
srv.sck.sendto(srv.bp_ip, (cip, 5353))
|
||||||
|
|
||||||
|
# and schedule multicast
|
||||||
|
self.defend[srv] = self.defend.get(srv, now + 0.1)
|
||||||
|
return
|
||||||
|
|
||||||
|
# check for someone rejecting our probe / hijacking our hostname
|
||||||
|
cips = [
|
||||||
|
U(x.rdata)
|
||||||
|
for x in p.rr
|
||||||
|
if U(x.rname).lower() == self.lhn and x.rclass == DC.F_IN
|
||||||
|
]
|
||||||
|
if cips and self.sips.isdisjoint(cips):
|
||||||
|
if not [x for x in cips if x not in ("::1", "127.0.0.1")]:
|
||||||
|
# avahi broadcasting 127.0.0.1-only packets
|
||||||
|
return
|
||||||
|
|
||||||
|
# check if we've been given additional IPs
|
||||||
|
for ip in list_ips():
|
||||||
|
if ip in cips:
|
||||||
|
self.sips.add(ip)
|
||||||
|
|
||||||
|
if not self.sips.isdisjoint(cips):
|
||||||
|
return
|
||||||
|
|
||||||
|
t = "mdns zeroconf: "
|
||||||
|
if self.probing:
|
||||||
|
t += "Cannot start; hostname '{}' is occupied"
|
||||||
|
else:
|
||||||
|
t += "Emergency stop; hostname '{}' got stolen"
|
||||||
|
|
||||||
|
t += " on {}! Use --name to set another hostname.\n\nName taken by {}\n\nYour IPs: {}\n"
|
||||||
|
self.log(t.format(self.args.name, srv.name, cips, list(self.sips)), 1)
|
||||||
|
self.stop(True)
|
||||||
|
return
|
||||||
|
|
||||||
|
# then rfc-6.7; dns pretending to be mdns (android...)
|
||||||
|
if p.header.id or addr[1] != 5353:
|
||||||
|
rsp: Optional[DNSRecord] = None
|
||||||
|
for r in p.questions:
|
||||||
|
try:
|
||||||
|
lhn = U(r.qname).lower()
|
||||||
|
except:
|
||||||
|
self.log("invalid question: {}".format(r))
|
||||||
|
continue
|
||||||
|
|
||||||
|
if lhn != self.lhn:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if p.header.id and r.qtype in (QTYPE.A, QTYPE.AAAA):
|
||||||
|
rsp = rsp or DNSRecord(DNSHeader(p.header.id, 0x8400))
|
||||||
|
rsp.add_question(r)
|
||||||
|
for ip in srv.ips:
|
||||||
|
qt = r.qtype
|
||||||
|
v6 = ":" in ip
|
||||||
|
if v6 == (qt == QTYPE.AAAA):
|
||||||
|
rd = AAAA(ip) if v6 else A(ip)
|
||||||
|
rr = RR(self.hn, qt, DC.IN, 10, rd)
|
||||||
|
rsp.add_answer(rr)
|
||||||
|
if rsp:
|
||||||
|
srv.sck.sendto(rsp.pack(), addr[:2])
|
||||||
|
# but don't return in case it's a differently broken client
|
||||||
|
|
||||||
|
# then a/aaaa records
|
||||||
|
for r in p.questions:
|
||||||
|
try:
|
||||||
|
lhn = U(r.qname).lower()
|
||||||
|
except:
|
||||||
|
self.log("invalid question: {}".format(r))
|
||||||
|
continue
|
||||||
|
|
||||||
|
if lhn != self.lhn:
|
||||||
|
continue
|
||||||
|
|
||||||
|
# gvfs keeps repeating itself
|
||||||
|
found = False
|
||||||
|
unicast = False
|
||||||
|
for rr in p.rr:
|
||||||
|
try:
|
||||||
|
rname = U(rr.rname).lower()
|
||||||
|
except:
|
||||||
|
self.log("invalid rr: {}".format(rr))
|
||||||
|
continue
|
||||||
|
|
||||||
|
if rname == self.lhn:
|
||||||
|
if rr.ttl > 60:
|
||||||
|
found = True
|
||||||
|
if rr.rclass == DC.F_IN:
|
||||||
|
unicast = True
|
||||||
|
|
||||||
|
if unicast:
|
||||||
|
# spec-compliant mDNS-over-unicast
|
||||||
|
srv.sck.sendto(srv.bp_ip, (cip, 5353))
|
||||||
|
elif addr[1] != 5353:
|
||||||
|
# just in case some clients use (and want us to use) invalid ports
|
||||||
|
srv.sck.sendto(srv.bp_ip, addr[:2])
|
||||||
|
|
||||||
|
if not found:
|
||||||
|
self.q[cip] = (0, srv, srv.bp_ip)
|
||||||
|
return
|
||||||
|
|
||||||
|
deadline = now + (0.5 if p.header.tc else 0.02) # rfc-7.2
|
||||||
|
|
||||||
|
# and service queries
|
||||||
|
for r in p.questions:
|
||||||
|
if not r or not r.qname:
|
||||||
|
continue
|
||||||
|
|
||||||
|
qname = U(r.qname).lower()
|
||||||
|
if qname in self.lsvcs or qname == "_services._dns-sd._udp.local.":
|
||||||
|
self.q[cip] = (deadline, srv, srv.bp_svc)
|
||||||
|
break
|
||||||
|
# heed rfc-7.1 if there was an announce in the past 12sec
|
||||||
|
# (workaround gvfs race-condition where it occasionally
|
||||||
|
# doesn't read/decode the full response...)
|
||||||
|
if now < srv.last_tx + 12:
|
||||||
|
for rr in p.rr:
|
||||||
|
if not rr.rdata:
|
||||||
|
continue
|
||||||
|
|
||||||
|
rdata = U(rr.rdata).lower()
|
||||||
|
if rdata in self.lsfqdns:
|
||||||
|
if rr.ttl > 2250:
|
||||||
|
self.q.pop(cip, None)
|
||||||
|
break
|
||||||
|
|
||||||
|
def process(self) -> None:
|
||||||
|
tx = set()
|
||||||
|
now = time.time()
|
||||||
|
cooldown = 0.9 # rfc-6: 1
|
||||||
|
if self.unsolicited and self.unsolicited[0] < now:
|
||||||
|
self.unsolicited.pop(0)
|
||||||
|
cooldown = 0.1
|
||||||
|
for srv in self.srv.values():
|
||||||
|
tx.add(srv)
|
||||||
|
|
||||||
|
for srv, deadline in list(self.defend.items()):
|
||||||
|
if now < deadline:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if self._tx(srv, srv.bp_ip, 0.02): # rfc-6: 0.25
|
||||||
|
self.defend.pop(srv)
|
||||||
|
|
||||||
|
for cip, (deadline, srv, msg) in list(self.q.items()):
|
||||||
|
if now < deadline:
|
||||||
|
continue
|
||||||
|
|
||||||
|
self.q.pop(cip)
|
||||||
|
self._tx(srv, msg, cooldown)
|
||||||
|
|
||||||
|
for srv in tx:
|
||||||
|
self._tx(srv, srv.bp_svc, cooldown)
|
||||||
|
|
||||||
|
def _tx(self, srv: MDNS_Sck, msg: bytes, cooldown: float) -> bool:
|
||||||
|
now = time.time()
|
||||||
|
if now < srv.last_tx + cooldown:
|
||||||
|
return False
|
||||||
|
|
||||||
|
try:
|
||||||
|
srv.sck.sendto(msg, (srv.grp, 5353))
|
||||||
|
srv.last_tx = now
|
||||||
|
except Exception as ex:
|
||||||
|
if srv.tx_ex:
|
||||||
|
return True
|
||||||
|
|
||||||
|
srv.tx_ex = True
|
||||||
|
t = "tx({},|{}|,{}): {}"
|
||||||
|
self.log(t.format(srv.ip, len(msg), cooldown, ex), 3)
|
||||||
|
|
||||||
|
return True
|
||||||
@@ -1,28 +1,44 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import os
|
import argparse
|
||||||
import sys
|
|
||||||
import json
|
import json
|
||||||
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
import subprocess as sp
|
import subprocess as sp
|
||||||
|
import sys
|
||||||
|
|
||||||
from .__init__ import PY2, WINDOWS, unicode
|
from .__init__ import EXE, PY2, WINDOWS, E, unicode
|
||||||
from .util import fsenc, fsdec, uncyg, runcmd, REKOBO_LKEY
|
|
||||||
from .bos import bos
|
from .bos import bos
|
||||||
|
from .util import (
|
||||||
|
FFMPEG_URL,
|
||||||
|
REKOBO_LKEY,
|
||||||
|
fsenc,
|
||||||
|
min_ex,
|
||||||
|
pybin,
|
||||||
|
retchk,
|
||||||
|
runcmd,
|
||||||
|
sfsenc,
|
||||||
|
uncyg,
|
||||||
|
)
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any, Union
|
||||||
|
|
||||||
|
from .util import RootLogger
|
||||||
|
|
||||||
|
|
||||||
def have_ff(cmd):
|
def have_ff(scmd: str) -> bool:
|
||||||
if PY2:
|
if PY2:
|
||||||
print("# checking {}".format(cmd))
|
print("# checking {}".format(scmd))
|
||||||
cmd = (cmd + " -version").encode("ascii").split(b" ")
|
acmd = (scmd + " -version").encode("ascii").split(b" ")
|
||||||
try:
|
try:
|
||||||
sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE).communicate()
|
sp.Popen(acmd, stdout=sp.PIPE, stderr=sp.PIPE).communicate()
|
||||||
return True
|
return True
|
||||||
except:
|
except:
|
||||||
return False
|
return False
|
||||||
else:
|
else:
|
||||||
return bool(shutil.which(cmd))
|
return bool(shutil.which(scmd))
|
||||||
|
|
||||||
|
|
||||||
HAVE_FFMPEG = have_ff("ffmpeg")
|
HAVE_FFMPEG = have_ff("ffmpeg")
|
||||||
@@ -30,13 +46,16 @@ HAVE_FFPROBE = have_ff("ffprobe")
|
|||||||
|
|
||||||
|
|
||||||
class MParser(object):
|
class MParser(object):
|
||||||
def __init__(self, cmdline):
|
def __init__(self, cmdline: str) -> None:
|
||||||
self.tag, args = cmdline.split("=", 1)
|
self.tag, args = cmdline.split("=", 1)
|
||||||
self.tags = self.tag.split(",")
|
self.tags = self.tag.split(",")
|
||||||
|
|
||||||
self.timeout = 30
|
self.timeout = 60
|
||||||
self.force = False
|
self.force = False
|
||||||
|
self.kill = "t" # tree; all children recursively
|
||||||
|
self.capture = 3 # outputs to consume
|
||||||
self.audio = "y"
|
self.audio = "y"
|
||||||
|
self.pri = 0 # priority; higher = later
|
||||||
self.ext = []
|
self.ext = []
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
@@ -58,6 +77,14 @@ class MParser(object):
|
|||||||
self.audio = arg[1:] # [r]equire [n]ot [d]ontcare
|
self.audio = arg[1:] # [r]equire [n]ot [d]ontcare
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
if arg.startswith("k"):
|
||||||
|
self.kill = arg[1:] # [t]ree [m]ain [n]one
|
||||||
|
continue
|
||||||
|
|
||||||
|
if arg.startswith("c"):
|
||||||
|
self.capture = int(arg[1:]) # 0=none 1=stdout 2=stderr 3=both
|
||||||
|
continue
|
||||||
|
|
||||||
if arg == "f":
|
if arg == "f":
|
||||||
self.force = True
|
self.force = True
|
||||||
continue
|
continue
|
||||||
@@ -70,10 +97,16 @@ class MParser(object):
|
|||||||
self.ext.append(arg[1:])
|
self.ext.append(arg[1:])
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
if arg.startswith("p"):
|
||||||
|
self.pri = int(arg[1:] or "1")
|
||||||
|
continue
|
||||||
|
|
||||||
raise Exception()
|
raise Exception()
|
||||||
|
|
||||||
|
|
||||||
def ffprobe(abspath, timeout=10):
|
def ffprobe(
|
||||||
|
abspath: str, timeout: int = 60
|
||||||
|
) -> tuple[dict[str, tuple[int, Any]], dict[str, list[Any]]]:
|
||||||
cmd = [
|
cmd = [
|
||||||
b"ffprobe",
|
b"ffprobe",
|
||||||
b"-hide_banner",
|
b"-hide_banner",
|
||||||
@@ -82,19 +115,20 @@ def ffprobe(abspath, timeout=10):
|
|||||||
b"--",
|
b"--",
|
||||||
fsenc(abspath),
|
fsenc(abspath),
|
||||||
]
|
]
|
||||||
rc = runcmd(cmd, timeout=timeout)
|
rc, so, se = runcmd(cmd, timeout=timeout)
|
||||||
return parse_ffprobe(rc[1])
|
retchk(rc, cmd, se)
|
||||||
|
return parse_ffprobe(so)
|
||||||
|
|
||||||
|
|
||||||
def parse_ffprobe(txt):
|
def parse_ffprobe(txt: str) -> tuple[dict[str, tuple[int, Any]], dict[str, list[Any]]]:
|
||||||
"""ffprobe -show_format -show_streams"""
|
"""ffprobe -show_format -show_streams"""
|
||||||
streams = []
|
streams = []
|
||||||
fmt = {}
|
fmt = {}
|
||||||
g = None
|
g = {}
|
||||||
for ln in [x.rstrip("\r") for x in txt.split("\n")]:
|
for ln in [x.rstrip("\r") for x in txt.split("\n")]:
|
||||||
try:
|
try:
|
||||||
k, v = ln.split("=", 1)
|
sk, sv = ln.split("=", 1)
|
||||||
g[k] = v
|
g[sk] = sv
|
||||||
continue
|
continue
|
||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
@@ -108,8 +142,8 @@ def parse_ffprobe(txt):
|
|||||||
fmt = g
|
fmt = g
|
||||||
|
|
||||||
streams = [fmt] + streams
|
streams = [fmt] + streams
|
||||||
ret = {} # processed
|
ret: dict[str, Any] = {} # processed
|
||||||
md = {} # raw tags
|
md: dict[str, list[Any]] = {} # raw tags
|
||||||
|
|
||||||
is_audio = fmt.get("format_name") in ["mp3", "ogg", "flac", "wav"]
|
is_audio = fmt.get("format_name") in ["mp3", "ogg", "flac", "wav"]
|
||||||
if fmt.get("filename", "").split(".")[-1].lower() in ["m4a", "aac"]:
|
if fmt.get("filename", "").split(".")[-1].lower() in ["m4a", "aac"]:
|
||||||
@@ -157,52 +191,55 @@ def parse_ffprobe(txt):
|
|||||||
]
|
]
|
||||||
|
|
||||||
if typ == "format":
|
if typ == "format":
|
||||||
kvm = [["duration", ".dur"], ["bit_rate", ".q"]]
|
kvm = [["duration", ".dur"], ["bit_rate", ".q"], ["format_name", "fmt"]]
|
||||||
|
|
||||||
for sk, rk in kvm:
|
for sk, rk in kvm:
|
||||||
v = strm.get(sk)
|
v1 = strm.get(sk)
|
||||||
if v is None:
|
if v1 is None:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if rk.startswith("."):
|
if rk.startswith("."):
|
||||||
try:
|
try:
|
||||||
v = float(v)
|
zf = float(v1)
|
||||||
v2 = ret.get(rk)
|
v2 = ret.get(rk)
|
||||||
if v2 is None or v > v2:
|
if v2 is None or zf > v2:
|
||||||
ret[rk] = v
|
ret[rk] = zf
|
||||||
except:
|
except:
|
||||||
# sqlite doesnt care but the code below does
|
# sqlite doesnt care but the code below does
|
||||||
if v not in ["N/A"]:
|
if v1 not in ["N/A"]:
|
||||||
ret[rk] = v
|
ret[rk] = v1
|
||||||
else:
|
else:
|
||||||
ret[rk] = v
|
ret[rk] = v1
|
||||||
|
|
||||||
if ret.get("vc") == "ansi": # shellscript
|
if ret.get("vc") == "ansi": # shellscript
|
||||||
return {}, {}
|
return {}, {}
|
||||||
|
|
||||||
for strm in streams:
|
for strm in streams:
|
||||||
for k, v in strm.items():
|
for sk, sv in strm.items():
|
||||||
if not k.startswith("TAG:"):
|
if not sk.startswith("TAG:"):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
k = k[4:].strip()
|
sk = sk[4:].strip()
|
||||||
v = v.strip()
|
sv = sv.strip()
|
||||||
if k and v and k not in md:
|
if sk and sv and sk not in md:
|
||||||
md[k] = [v]
|
md[sk] = [sv]
|
||||||
|
|
||||||
for k in [".q", ".vq", ".aq"]:
|
for sk in [".q", ".vq", ".aq"]:
|
||||||
if k in ret:
|
if sk in ret:
|
||||||
ret[k] /= 1000 # bit_rate=320000
|
ret[sk] /= 1000 # bit_rate=320000
|
||||||
|
|
||||||
for k in [".q", ".vq", ".aq", ".resw", ".resh"]:
|
for sk in [".q", ".vq", ".aq", ".resw", ".resh"]:
|
||||||
if k in ret:
|
if sk in ret:
|
||||||
ret[k] = int(ret[k])
|
ret[sk] = int(ret[sk])
|
||||||
|
|
||||||
if ".fps" in ret:
|
if ".fps" in ret:
|
||||||
fps = ret[".fps"]
|
fps = ret[".fps"]
|
||||||
if "/" in fps:
|
if "/" in fps:
|
||||||
fa, fb = fps.split("/")
|
fa, fb = fps.split("/")
|
||||||
fps = int(fa) * 1.0 / int(fb)
|
try:
|
||||||
|
fps = int(fa) * 1.0 / int(fb)
|
||||||
|
except:
|
||||||
|
fps = 9001
|
||||||
|
|
||||||
if fps < 1000 and fmt.get("format_name") not in ["image2", "png_pipe"]:
|
if fps < 1000 and fmt.get("format_name") not in ["image2", "png_pipe"]:
|
||||||
ret[".fps"] = round(fps, 3)
|
ret[".fps"] = round(fps, 3)
|
||||||
@@ -215,33 +252,34 @@ def parse_ffprobe(txt):
|
|||||||
if ".q" in ret:
|
if ".q" in ret:
|
||||||
del ret[".q"]
|
del ret[".q"]
|
||||||
|
|
||||||
|
if "fmt" in ret:
|
||||||
|
ret["fmt"] = ret["fmt"].split(",")[0]
|
||||||
|
|
||||||
if ".resw" in ret and ".resh" in ret:
|
if ".resw" in ret and ".resh" in ret:
|
||||||
ret["res"] = "{}x{}".format(ret[".resw"], ret[".resh"])
|
ret["res"] = "{}x{}".format(ret[".resw"], ret[".resh"])
|
||||||
|
|
||||||
ret = {k: [0, v] for k, v in ret.items()}
|
zd = {k: (0, v) for k, v in ret.items()}
|
||||||
|
|
||||||
return ret, md
|
return zd, md
|
||||||
|
|
||||||
|
|
||||||
class MTag(object):
|
class MTag(object):
|
||||||
def __init__(self, log_func, args):
|
def __init__(self, log_func: "RootLogger", args: argparse.Namespace) -> None:
|
||||||
self.log_func = log_func
|
self.log_func = log_func
|
||||||
self.args = args
|
self.args = args
|
||||||
self.usable = True
|
self.usable = True
|
||||||
self.prefer_mt = not args.no_mtag_ff
|
self.prefer_mt = not args.no_mtag_ff
|
||||||
self.backend = "ffprobe" if args.no_mutagen else "mutagen"
|
self.backend = (
|
||||||
self.can_ffprobe = (
|
"ffprobe" if args.no_mutagen or (HAVE_FFPROBE and EXE) else "mutagen"
|
||||||
HAVE_FFPROBE
|
|
||||||
and not args.no_mtag_ff
|
|
||||||
and (not WINDOWS or sys.version_info >= (3, 8))
|
|
||||||
)
|
)
|
||||||
|
self.can_ffprobe = HAVE_FFPROBE and not args.no_mtag_ff
|
||||||
mappings = args.mtm
|
mappings = args.mtm
|
||||||
or_ffprobe = " or FFprobe"
|
or_ffprobe = " or FFprobe"
|
||||||
|
|
||||||
if self.backend == "mutagen":
|
if self.backend == "mutagen":
|
||||||
self.get = self.get_mutagen
|
self.get = self.get_mutagen
|
||||||
try:
|
try:
|
||||||
import mutagen
|
from mutagen import version # noqa: F401
|
||||||
except:
|
except:
|
||||||
self.log("could not load Mutagen, trying FFprobe instead", c=3)
|
self.log("could not load Mutagen, trying FFprobe instead", c=3)
|
||||||
self.backend = "ffprobe"
|
self.backend = "ffprobe"
|
||||||
@@ -258,15 +296,15 @@ class MTag(object):
|
|||||||
msg = "found FFprobe but it was disabled by --no-mtag-ff"
|
msg = "found FFprobe but it was disabled by --no-mtag-ff"
|
||||||
self.log(msg, c=3)
|
self.log(msg, c=3)
|
||||||
|
|
||||||
elif WINDOWS and sys.version_info < (3, 8):
|
|
||||||
or_ffprobe = " or python >= 3.8"
|
|
||||||
msg = "found FFprobe but your python is too old; need 3.8 or newer"
|
|
||||||
self.log(msg, c=1)
|
|
||||||
|
|
||||||
if not self.usable:
|
if not self.usable:
|
||||||
|
if EXE:
|
||||||
|
t = "copyparty.exe cannot use mutagen; need ffprobe.exe to read media tags: "
|
||||||
|
self.log(t + FFMPEG_URL)
|
||||||
|
return
|
||||||
|
|
||||||
msg = "need Mutagen{} to read media tags so please run this:\n{}{} -m pip install --user mutagen\n"
|
msg = "need Mutagen{} to read media tags so please run this:\n{}{} -m pip install --user mutagen\n"
|
||||||
pybin = os.path.basename(sys.executable)
|
pyname = os.path.basename(pybin)
|
||||||
self.log(msg.format(or_ffprobe, " " * 37, pybin), c=1)
|
self.log(msg.format(or_ffprobe, " " * 37, pyname), c=1)
|
||||||
return
|
return
|
||||||
|
|
||||||
# https://picard-docs.musicbrainz.org/downloads/MusicBrainz_Picard_Tag_Map.html
|
# https://picard-docs.musicbrainz.org/downloads/MusicBrainz_Picard_Tag_Map.html
|
||||||
@@ -338,41 +376,49 @@ class MTag(object):
|
|||||||
}
|
}
|
||||||
# self.get = self.compare
|
# self.get = self.compare
|
||||||
|
|
||||||
def log(self, msg, c=0):
|
def log(self, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
self.log_func("mtag", msg, c)
|
self.log_func("mtag", msg, c)
|
||||||
|
|
||||||
def normalize_tags(self, ret, md):
|
def normalize_tags(
|
||||||
for k, v in dict(md).items():
|
self, parser_output: dict[str, tuple[int, Any]], md: dict[str, list[Any]]
|
||||||
if not v:
|
) -> dict[str, Union[str, float]]:
|
||||||
|
for sk, tv in dict(md).items():
|
||||||
|
if not tv:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
k = k.lower().split("::")[0].strip()
|
sk = sk.lower().split("::")[0].strip()
|
||||||
mk = self.rmap.get(k)
|
key_mapping = self.rmap.get(sk)
|
||||||
if not mk:
|
if not key_mapping:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
pref, mk = mk
|
priority, alias = key_mapping
|
||||||
if mk not in ret or ret[mk][0] > pref:
|
if alias not in parser_output or parser_output[alias][0] > priority:
|
||||||
ret[mk] = [pref, v[0]]
|
parser_output[alias] = (priority, tv[0])
|
||||||
|
|
||||||
# take first value
|
# take first value (lowest priority / most preferred)
|
||||||
ret = {k: unicode(v[1]).strip() for k, v in ret.items()}
|
ret: dict[str, Union[str, float]] = {
|
||||||
|
sk: unicode(tv[1]).strip() for sk, tv in parser_output.items()
|
||||||
|
}
|
||||||
|
|
||||||
# track 3/7 => track 3
|
# track 3/7 => track 3
|
||||||
for k, v in ret.items():
|
for sk, zv in ret.items():
|
||||||
if k[0] == ".":
|
if sk[0] == ".":
|
||||||
v = v.split("/")[0].strip().lstrip("0")
|
sv = str(zv).split("/")[0].strip().lstrip("0")
|
||||||
ret[k] = v or 0
|
ret[sk] = sv or 0
|
||||||
|
|
||||||
# normalize key notation to rkeobo
|
# normalize key notation to rkeobo
|
||||||
okey = ret.get("key")
|
okey = ret.get("key")
|
||||||
if okey:
|
if okey:
|
||||||
key = okey.replace(" ", "").replace("maj", "").replace("min", "m")
|
key = str(okey).replace(" ", "").replace("maj", "").replace("min", "m")
|
||||||
ret["key"] = REKOBO_LKEY.get(key.lower(), okey)
|
ret["key"] = REKOBO_LKEY.get(key.lower(), okey)
|
||||||
|
|
||||||
|
if self.args.mtag_vv:
|
||||||
|
zl = " ".join("\033[36m{} \033[33m{}".format(k, v) for k, v in ret.items())
|
||||||
|
self.log("norm: {}\033[0m".format(zl), "90")
|
||||||
|
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
def compare(self, abspath):
|
def compare(self, abspath: str) -> dict[str, Union[str, float]]:
|
||||||
if abspath.endswith(".au"):
|
if abspath.endswith(".au"):
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
@@ -410,20 +456,34 @@ class MTag(object):
|
|||||||
|
|
||||||
return r1
|
return r1
|
||||||
|
|
||||||
def get_mutagen(self, abspath):
|
def get_mutagen(self, abspath: str) -> dict[str, Union[str, float]]:
|
||||||
|
ret: dict[str, tuple[int, Any]] = {}
|
||||||
|
|
||||||
if not bos.path.isfile(abspath):
|
if not bos.path.isfile(abspath):
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
import mutagen
|
from mutagen import File
|
||||||
|
|
||||||
try:
|
try:
|
||||||
md = mutagen.File(fsenc(abspath), easy=True)
|
md = File(fsenc(abspath), easy=True)
|
||||||
x = md.info.length
|
assert md
|
||||||
|
if self.args.mtag_vv:
|
||||||
|
for zd in (md.info.__dict__, dict(md.tags)):
|
||||||
|
zl = ["\033[36m{} \033[33m{}".format(k, v) for k, v in zd.items()]
|
||||||
|
self.log("mutagen: {}\033[0m".format(" ".join(zl)), "90")
|
||||||
|
if not md.info.length and not md.info.codec:
|
||||||
|
raise Exception()
|
||||||
except Exception as ex:
|
except Exception as ex:
|
||||||
|
if self.args.mtag_v:
|
||||||
|
self.log("mutagen-err [{}] @ [{}]".format(ex, abspath), "90")
|
||||||
|
|
||||||
return self.get_ffprobe(abspath) if self.can_ffprobe else {}
|
return self.get_ffprobe(abspath) if self.can_ffprobe else {}
|
||||||
|
|
||||||
sz = bos.path.getsize(abspath)
|
sz = bos.path.getsize(abspath)
|
||||||
ret = {".q": [0, int((sz / md.info.length) / 128)]}
|
try:
|
||||||
|
ret[".q"] = (0, int((sz / md.info.length) / 128))
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
for attr, k, norm in [
|
for attr, k, norm in [
|
||||||
["codec", "ac", unicode],
|
["codec", "ac", unicode],
|
||||||
@@ -454,54 +514,83 @@ class MTag(object):
|
|||||||
if k == "ac" and v.startswith("mp4a.40."):
|
if k == "ac" and v.startswith("mp4a.40."):
|
||||||
v = "aac"
|
v = "aac"
|
||||||
|
|
||||||
ret[k] = [0, norm(v)]
|
ret[k] = (0, norm(v))
|
||||||
|
|
||||||
return self.normalize_tags(ret, md)
|
return self.normalize_tags(ret, md)
|
||||||
|
|
||||||
def get_ffprobe(self, abspath):
|
def get_ffprobe(self, abspath: str) -> dict[str, Union[str, float]]:
|
||||||
if not bos.path.isfile(abspath):
|
if not bos.path.isfile(abspath):
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
ret, md = ffprobe(abspath)
|
ret, md = ffprobe(abspath, self.args.mtag_to)
|
||||||
|
|
||||||
|
if self.args.mtag_vv:
|
||||||
|
for zd in (ret, dict(md)):
|
||||||
|
zl = ["\033[36m{} \033[33m{}".format(k, v) for k, v in zd.items()]
|
||||||
|
self.log("ffprobe: {}\033[0m".format(" ".join(zl)), "90")
|
||||||
|
|
||||||
return self.normalize_tags(ret, md)
|
return self.normalize_tags(ret, md)
|
||||||
|
|
||||||
def get_bin(self, parsers, abspath):
|
def get_bin(
|
||||||
|
self, parsers: dict[str, MParser], abspath: str, oth_tags: dict[str, Any]
|
||||||
|
) -> dict[str, Any]:
|
||||||
if not bos.path.isfile(abspath):
|
if not bos.path.isfile(abspath):
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
pypath = os.path.abspath(os.path.dirname(os.path.dirname(__file__)))
|
|
||||||
pypath = [str(pypath)] + [str(x) for x in sys.path if x]
|
|
||||||
pypath = str(os.pathsep.join(pypath))
|
|
||||||
env = os.environ.copy()
|
env = os.environ.copy()
|
||||||
env["PYTHONPATH"] = pypath
|
try:
|
||||||
|
if EXE:
|
||||||
|
raise Exception()
|
||||||
|
|
||||||
ret = {}
|
pypath = os.path.abspath(os.path.dirname(os.path.dirname(__file__)))
|
||||||
for tagname, mp in parsers.items():
|
zsl = [str(pypath)] + [str(x) for x in sys.path if x]
|
||||||
|
pypath = str(os.pathsep.join(zsl))
|
||||||
|
env["PYTHONPATH"] = pypath
|
||||||
|
except:
|
||||||
|
if not E.ox and not EXE:
|
||||||
|
raise
|
||||||
|
|
||||||
|
ret: dict[str, Any] = {}
|
||||||
|
for tagname, parser in sorted(parsers.items(), key=lambda x: (x[1].pri, x[0])):
|
||||||
try:
|
try:
|
||||||
cmd = [mp.bin, abspath]
|
cmd = [parser.bin, abspath]
|
||||||
if mp.bin.endswith(".py"):
|
if parser.bin.endswith(".py"):
|
||||||
cmd = [sys.executable] + cmd
|
cmd = [pybin] + cmd
|
||||||
|
|
||||||
args = {"env": env, "timeout": mp.timeout}
|
args = {
|
||||||
|
"env": env,
|
||||||
|
"timeout": parser.timeout,
|
||||||
|
"kill": parser.kill,
|
||||||
|
"capture": parser.capture,
|
||||||
|
}
|
||||||
|
|
||||||
|
if parser.pri:
|
||||||
|
zd = oth_tags.copy()
|
||||||
|
zd.update(ret)
|
||||||
|
args["sin"] = json.dumps(zd).encode("utf-8", "replace")
|
||||||
|
|
||||||
if WINDOWS:
|
if WINDOWS:
|
||||||
args["creationflags"] = 0x4000
|
args["creationflags"] = 0x4000
|
||||||
else:
|
else:
|
||||||
cmd = ["nice"] + cmd
|
cmd = ["nice"] + cmd
|
||||||
|
|
||||||
cmd = [fsenc(x) for x in cmd]
|
bcmd = [sfsenc(x) for x in cmd[:-1]] + [fsenc(cmd[-1])]
|
||||||
v = sp.check_output(cmd, **args).strip()
|
rc, v, err = runcmd(bcmd, **args) # type: ignore
|
||||||
|
retchk(rc, bcmd, err, self.log, 5, self.args.mtag_v)
|
||||||
|
v = v.strip()
|
||||||
if not v:
|
if not v:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if "," not in tagname:
|
if "," not in tagname:
|
||||||
ret[tagname] = v.decode("utf-8")
|
ret[tagname] = v
|
||||||
else:
|
else:
|
||||||
v = json.loads(v)
|
zj = json.loads(v)
|
||||||
for tag in tagname.split(","):
|
for tag in tagname.split(","):
|
||||||
if tag and tag in v:
|
if tag and tag in zj:
|
||||||
ret[tag] = v[tag]
|
ret[tag] = zj[tag]
|
||||||
except:
|
except:
|
||||||
pass
|
if self.args.mtag_v:
|
||||||
|
t = "mtag error: tagname {}, parser {}, file {} => {}"
|
||||||
|
self.log(t.format(tagname, parser.bin, abspath, min_ex()))
|
||||||
|
|
||||||
return ret
|
return ret
|
||||||
|
|||||||
370
copyparty/multicast.py
Normal file
370
copyparty/multicast.py
Normal file
@@ -0,0 +1,370 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import socket
|
||||||
|
import time
|
||||||
|
|
||||||
|
import ipaddress
|
||||||
|
from ipaddress import (
|
||||||
|
IPv4Address,
|
||||||
|
IPv4Network,
|
||||||
|
IPv6Address,
|
||||||
|
IPv6Network,
|
||||||
|
ip_address,
|
||||||
|
ip_network,
|
||||||
|
)
|
||||||
|
|
||||||
|
from .__init__ import MACOS, TYPE_CHECKING
|
||||||
|
from .util import Netdev, find_prefix, min_ex, spack
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from .svchub import SvcHub
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Optional, Union
|
||||||
|
|
||||||
|
if not hasattr(socket, "IPPROTO_IPV6"):
|
||||||
|
setattr(socket, "IPPROTO_IPV6", 41)
|
||||||
|
|
||||||
|
|
||||||
|
class NoIPs(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class MC_Sck(object):
|
||||||
|
"""there is one socket for each server ip"""
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
sck: socket.socket,
|
||||||
|
nd: Netdev,
|
||||||
|
grp: str,
|
||||||
|
ip: str,
|
||||||
|
net: Union[IPv4Network, IPv6Network],
|
||||||
|
):
|
||||||
|
self.sck = sck
|
||||||
|
self.idx = nd.idx
|
||||||
|
self.name = nd.name
|
||||||
|
self.grp = grp
|
||||||
|
self.mreq = b""
|
||||||
|
self.ip = ip
|
||||||
|
self.net = net
|
||||||
|
self.ips = {ip: net}
|
||||||
|
self.v6 = ":" in ip
|
||||||
|
self.have4 = ":" not in ip
|
||||||
|
self.have6 = ":" in ip
|
||||||
|
|
||||||
|
|
||||||
|
class MCast(object):
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
hub: "SvcHub",
|
||||||
|
Srv: type[MC_Sck],
|
||||||
|
on: list[str],
|
||||||
|
off: list[str],
|
||||||
|
mc_grp_4: str,
|
||||||
|
mc_grp_6: str,
|
||||||
|
port: int,
|
||||||
|
vinit: bool,
|
||||||
|
) -> None:
|
||||||
|
"""disable ipv%d by setting mc_grp_%d empty"""
|
||||||
|
self.hub = hub
|
||||||
|
self.Srv = Srv
|
||||||
|
self.args = hub.args
|
||||||
|
self.asrv = hub.asrv
|
||||||
|
self.log_func = hub.log
|
||||||
|
self.on = on
|
||||||
|
self.off = off
|
||||||
|
self.grp4 = mc_grp_4
|
||||||
|
self.grp6 = mc_grp_6
|
||||||
|
self.port = port
|
||||||
|
self.vinit = vinit
|
||||||
|
|
||||||
|
self.srv: dict[socket.socket, MC_Sck] = {} # listening sockets
|
||||||
|
self.sips: set[str] = set() # all listening ips (including failed attempts)
|
||||||
|
self.ll_ok: set[str] = set() # fallback linklocal IPv4 and IPv6 addresses
|
||||||
|
self.b2srv: dict[bytes, MC_Sck] = {} # binary-ip -> server socket
|
||||||
|
self.b4: list[bytes] = [] # sorted list of binary-ips
|
||||||
|
self.b6: list[bytes] = [] # sorted list of binary-ips
|
||||||
|
self.cscache: dict[str, Optional[MC_Sck]] = {} # client ip -> server cache
|
||||||
|
|
||||||
|
self.running = True
|
||||||
|
|
||||||
|
def log(self, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
|
self.log_func("multicast", msg, c)
|
||||||
|
|
||||||
|
def create_servers(self) -> list[str]:
|
||||||
|
bound: list[str] = []
|
||||||
|
netdevs = self.hub.tcpsrv.netdevs
|
||||||
|
ips = [x[0] for x in self.hub.tcpsrv.bound]
|
||||||
|
|
||||||
|
if "::" in ips:
|
||||||
|
ips = [x for x in ips if x != "::"] + list(
|
||||||
|
[x.split("/")[0] for x in netdevs if ":" in x]
|
||||||
|
)
|
||||||
|
ips.append("0.0.0.0")
|
||||||
|
|
||||||
|
if "0.0.0.0" in ips:
|
||||||
|
ips = [x for x in ips if x != "0.0.0.0"] + list(
|
||||||
|
[x.split("/")[0] for x in netdevs if ":" not in x]
|
||||||
|
)
|
||||||
|
|
||||||
|
ips = [x for x in ips if x not in ("::1", "127.0.0.1")]
|
||||||
|
ips = find_prefix(ips, netdevs)
|
||||||
|
|
||||||
|
on = self.on[:]
|
||||||
|
off = self.off[:]
|
||||||
|
for lst in (on, off):
|
||||||
|
for av in list(lst):
|
||||||
|
try:
|
||||||
|
arg_net = ip_network(av, False)
|
||||||
|
except:
|
||||||
|
arg_net = None
|
||||||
|
|
||||||
|
for sk, sv in netdevs.items():
|
||||||
|
if arg_net:
|
||||||
|
net_ip = ip_address(sk.split("/")[0])
|
||||||
|
if net_ip in arg_net and sk not in lst:
|
||||||
|
lst.append(sk)
|
||||||
|
|
||||||
|
if (av == str(sv.idx) or av == sv.name) and sk not in lst:
|
||||||
|
lst.append(sk)
|
||||||
|
|
||||||
|
if on:
|
||||||
|
ips = [x for x in ips if x in on]
|
||||||
|
elif off:
|
||||||
|
ips = [x for x in ips if x not in off]
|
||||||
|
|
||||||
|
if not self.grp4:
|
||||||
|
ips = [x for x in ips if ":" in x]
|
||||||
|
|
||||||
|
if not self.grp6:
|
||||||
|
ips = [x for x in ips if ":" not in x]
|
||||||
|
|
||||||
|
ips = list(set(ips))
|
||||||
|
all_selected = ips[:]
|
||||||
|
|
||||||
|
# discard non-linklocal ipv6
|
||||||
|
ips = [x for x in ips if ":" not in x or x.startswith("fe80")]
|
||||||
|
|
||||||
|
if not ips:
|
||||||
|
raise NoIPs()
|
||||||
|
|
||||||
|
for ip in ips:
|
||||||
|
v6 = ":" in ip
|
||||||
|
netdev = netdevs[ip]
|
||||||
|
if not netdev.idx:
|
||||||
|
t = "using INADDR_ANY for ip [{}], netdev [{}]"
|
||||||
|
if not self.srv and ip not in ["::", "0.0.0.0"]:
|
||||||
|
self.log(t.format(ip, netdev), 3)
|
||||||
|
|
||||||
|
ipv = socket.AF_INET6 if v6 else socket.AF_INET
|
||||||
|
sck = socket.socket(ipv, socket.SOCK_DGRAM, socket.IPPROTO_UDP)
|
||||||
|
sck.settimeout(None)
|
||||||
|
sck.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
|
||||||
|
try:
|
||||||
|
sck.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEPORT, 1)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
# most ipv6 clients expect multicast on linklocal ip only;
|
||||||
|
# add a/aaaa records for the other nic IPs
|
||||||
|
other_ips: set[str] = set()
|
||||||
|
if v6:
|
||||||
|
for nd in netdevs.values():
|
||||||
|
if nd.idx == netdev.idx and nd.ip in all_selected and ":" in nd.ip:
|
||||||
|
other_ips.add(nd.ip)
|
||||||
|
|
||||||
|
net = ipaddress.ip_network(ip, False)
|
||||||
|
ip = ip.split("/")[0]
|
||||||
|
srv = self.Srv(sck, netdev, self.grp6 if ":" in ip else self.grp4, ip, net)
|
||||||
|
for oth_ip in other_ips:
|
||||||
|
srv.ips[oth_ip.split("/")[0]] = ipaddress.ip_network(oth_ip, False)
|
||||||
|
|
||||||
|
# gvfs breaks if a linklocal ip appears in a dns reply
|
||||||
|
ll = {
|
||||||
|
k: v
|
||||||
|
for k, v in srv.ips.items()
|
||||||
|
if k.startswith("169.254") or k.startswith("fe80")
|
||||||
|
}
|
||||||
|
rt = {k: v for k, v in srv.ips.items() if k not in ll}
|
||||||
|
|
||||||
|
if self.args.ll or not rt:
|
||||||
|
self.ll_ok.update(list(ll))
|
||||||
|
|
||||||
|
if not self.args.ll:
|
||||||
|
srv.ips = rt or ll
|
||||||
|
|
||||||
|
if not srv.ips:
|
||||||
|
self.log("no IPs on {}; skipping [{}]".format(netdev, ip), 3)
|
||||||
|
continue
|
||||||
|
|
||||||
|
try:
|
||||||
|
self.setup_socket(srv)
|
||||||
|
self.srv[sck] = srv
|
||||||
|
bound.append(ip)
|
||||||
|
except:
|
||||||
|
t = "announce failed on {} [{}]:\n{}"
|
||||||
|
self.log(t.format(netdev, ip, min_ex()), 3)
|
||||||
|
|
||||||
|
if self.args.zm_msub:
|
||||||
|
for s1 in self.srv.values():
|
||||||
|
for s2 in self.srv.values():
|
||||||
|
if s1.idx != s2.idx:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if s1.ip not in s2.ips:
|
||||||
|
s2.ips[s1.ip] = s1.net
|
||||||
|
|
||||||
|
if self.args.zm_mnic:
|
||||||
|
for s1 in self.srv.values():
|
||||||
|
for s2 in self.srv.values():
|
||||||
|
for ip1, net1 in list(s1.ips.items()):
|
||||||
|
for ip2, net2 in list(s2.ips.items()):
|
||||||
|
if net1 == net2 and ip1 != ip2:
|
||||||
|
s1.ips[ip2] = net2
|
||||||
|
|
||||||
|
self.sips = set([x.split("/")[0] for x in all_selected])
|
||||||
|
for srv in self.srv.values():
|
||||||
|
assert srv.ip in self.sips
|
||||||
|
|
||||||
|
return bound
|
||||||
|
|
||||||
|
def setup_socket(self, srv: MC_Sck) -> None:
|
||||||
|
sck = srv.sck
|
||||||
|
if srv.v6:
|
||||||
|
if self.vinit:
|
||||||
|
zsl = list(srv.ips.keys())
|
||||||
|
self.log("v6({}) idx({}) {}".format(srv.ip, srv.idx, zsl), 6)
|
||||||
|
|
||||||
|
for ip in srv.ips:
|
||||||
|
bip = socket.inet_pton(socket.AF_INET6, ip)
|
||||||
|
self.b2srv[bip] = srv
|
||||||
|
self.b6.append(bip)
|
||||||
|
|
||||||
|
grp = self.grp6 if srv.idx else ""
|
||||||
|
try:
|
||||||
|
if MACOS:
|
||||||
|
raise Exception()
|
||||||
|
|
||||||
|
sck.bind((grp, self.port, 0, srv.idx))
|
||||||
|
except:
|
||||||
|
sck.bind(("", self.port, 0, srv.idx))
|
||||||
|
|
||||||
|
bgrp = socket.inet_pton(socket.AF_INET6, self.grp6)
|
||||||
|
dev = spack(b"@I", srv.idx)
|
||||||
|
srv.mreq = bgrp + dev
|
||||||
|
if srv.idx != socket.INADDR_ANY:
|
||||||
|
sck.setsockopt(socket.IPPROTO_IPV6, socket.IPV6_MULTICAST_IF, dev)
|
||||||
|
|
||||||
|
try:
|
||||||
|
sck.setsockopt(socket.IPPROTO_IPV6, socket.IPV6_MULTICAST_HOPS, 255)
|
||||||
|
sck.setsockopt(socket.IPPROTO_IPV6, socket.IPV6_MULTICAST_LOOP, 1)
|
||||||
|
except:
|
||||||
|
# macos
|
||||||
|
t = "failed to set IPv6 TTL/LOOP; announcements may not survive multiple switches/routers"
|
||||||
|
self.log(t, 3)
|
||||||
|
else:
|
||||||
|
if self.vinit:
|
||||||
|
self.log("v4({}) idx({})".format(srv.ip, srv.idx), 6)
|
||||||
|
|
||||||
|
bip = socket.inet_aton(srv.ip)
|
||||||
|
self.b2srv[bip] = srv
|
||||||
|
self.b4.append(bip)
|
||||||
|
|
||||||
|
grp = self.grp4 if srv.idx else ""
|
||||||
|
try:
|
||||||
|
if MACOS:
|
||||||
|
raise Exception()
|
||||||
|
|
||||||
|
sck.bind((grp, self.port))
|
||||||
|
except:
|
||||||
|
sck.bind(("", self.port))
|
||||||
|
|
||||||
|
bgrp = socket.inet_aton(self.grp4)
|
||||||
|
dev = (
|
||||||
|
spack(b"=I", socket.INADDR_ANY)
|
||||||
|
if srv.idx == socket.INADDR_ANY
|
||||||
|
else socket.inet_aton(srv.ip)
|
||||||
|
)
|
||||||
|
srv.mreq = bgrp + dev
|
||||||
|
if srv.idx != socket.INADDR_ANY:
|
||||||
|
sck.setsockopt(socket.IPPROTO_IP, socket.IP_MULTICAST_IF, dev)
|
||||||
|
|
||||||
|
try:
|
||||||
|
sck.setsockopt(socket.IPPROTO_IP, socket.IP_MULTICAST_TTL, 255)
|
||||||
|
sck.setsockopt(socket.IPPROTO_IP, socket.IP_MULTICAST_LOOP, 1)
|
||||||
|
except:
|
||||||
|
# probably can't happen but dontcare if it does
|
||||||
|
t = "failed to set IPv4 TTL/LOOP; announcements may not survive multiple switches/routers"
|
||||||
|
self.log(t, 3)
|
||||||
|
|
||||||
|
self.hop(srv)
|
||||||
|
self.b4.sort(reverse=True)
|
||||||
|
self.b6.sort(reverse=True)
|
||||||
|
|
||||||
|
def hop(self, srv: MC_Sck) -> None:
|
||||||
|
"""rejoin to keepalive on routers/switches without igmp-snooping"""
|
||||||
|
sck = srv.sck
|
||||||
|
req = srv.mreq
|
||||||
|
if ":" in srv.ip:
|
||||||
|
try:
|
||||||
|
sck.setsockopt(socket.IPPROTO_IPV6, socket.IPV6_LEAVE_GROUP, req)
|
||||||
|
# linux does leaves/joins twice with 0.2~1.05s spacing
|
||||||
|
time.sleep(1.2)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
sck.setsockopt(socket.IPPROTO_IPV6, socket.IPV6_JOIN_GROUP, req)
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
sck.setsockopt(socket.IPPROTO_IP, socket.IP_DROP_MEMBERSHIP, req)
|
||||||
|
time.sleep(1.2)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
# t = "joining {} from ip {} idx {} with mreq {}"
|
||||||
|
# self.log(t.format(srv.grp, srv.ip, srv.idx, repr(srv.mreq)), 6)
|
||||||
|
sck.setsockopt(socket.IPPROTO_IP, socket.IP_ADD_MEMBERSHIP, req)
|
||||||
|
|
||||||
|
def map_client(self, cip: str) -> Optional[MC_Sck]:
|
||||||
|
try:
|
||||||
|
return self.cscache[cip]
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
ret: Optional[MC_Sck] = None
|
||||||
|
v6 = ":" in cip
|
||||||
|
ci = IPv6Address(cip) if v6 else IPv4Address(cip)
|
||||||
|
for x in self.b6 if v6 else self.b4:
|
||||||
|
srv = self.b2srv[x]
|
||||||
|
if any([x for x in srv.ips.values() if ci in x]):
|
||||||
|
ret = srv
|
||||||
|
break
|
||||||
|
|
||||||
|
if not ret and cip in ("127.0.0.1", "::1"):
|
||||||
|
# just give it something
|
||||||
|
ret = list(self.srv.values())[0]
|
||||||
|
|
||||||
|
if not ret and cip.startswith("169.254"):
|
||||||
|
# idk how to map LL IPv4 msgs to nics;
|
||||||
|
# just pick one and hope for the best
|
||||||
|
lls = (
|
||||||
|
x
|
||||||
|
for x in self.srv.values()
|
||||||
|
if next((y for y in x.ips if y in self.ll_ok), None)
|
||||||
|
)
|
||||||
|
ret = next(lls, None)
|
||||||
|
|
||||||
|
if ret:
|
||||||
|
t = "new client on {} ({}): {}"
|
||||||
|
self.log(t.format(ret.name, ret.net, cip), 6)
|
||||||
|
else:
|
||||||
|
t = "could not map client {} to known subnet; maybe forwarded from another network?"
|
||||||
|
self.log(t.format(cip), 3)
|
||||||
|
|
||||||
|
if len(self.cscache) > 9000:
|
||||||
|
self.cscache = {}
|
||||||
|
|
||||||
|
self.cscache[cip] = ret
|
||||||
|
return ret
|
||||||
337
copyparty/smbd.py
Normal file
337
copyparty/smbd.py
Normal file
@@ -0,0 +1,337 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
import inspect
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import random
|
||||||
|
import stat
|
||||||
|
import sys
|
||||||
|
import time
|
||||||
|
from types import SimpleNamespace
|
||||||
|
|
||||||
|
from .__init__ import ANYWIN, EXE, TYPE_CHECKING
|
||||||
|
from .authsrv import LEELOO_DALLAS, VFS
|
||||||
|
from .bos import bos
|
||||||
|
from .util import Daemon, min_ex, pybin, runhook
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any, Union
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from .svchub import SvcHub
|
||||||
|
|
||||||
|
|
||||||
|
lg = logging.getLogger("smb")
|
||||||
|
debug, info, warning, error = (lg.debug, lg.info, lg.warning, lg.error)
|
||||||
|
|
||||||
|
|
||||||
|
class SMB(object):
|
||||||
|
def __init__(self, hub: "SvcHub") -> None:
|
||||||
|
self.hub = hub
|
||||||
|
self.args = hub.args
|
||||||
|
self.asrv = hub.asrv
|
||||||
|
self.log = hub.log
|
||||||
|
self.files: dict[int, tuple[float, str]] = {}
|
||||||
|
|
||||||
|
lg.setLevel(logging.DEBUG if self.args.smbvvv else logging.INFO)
|
||||||
|
for x in ["impacket", "impacket.smbserver"]:
|
||||||
|
lgr = logging.getLogger(x)
|
||||||
|
lgr.setLevel(logging.DEBUG if self.args.smbvv else logging.INFO)
|
||||||
|
|
||||||
|
try:
|
||||||
|
from impacket import smbserver
|
||||||
|
from impacket.ntlm import compute_lmhash, compute_nthash
|
||||||
|
except ImportError:
|
||||||
|
if EXE:
|
||||||
|
print("copyparty.exe cannot do SMB")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
m = "\033[36m\n{}\033[31m\n\nERROR: need 'impacket'; please run this command:\033[33m\n {} -m pip install --user impacket\n\033[0m"
|
||||||
|
print(m.format(min_ex(), pybin))
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
# patch vfs into smbserver.os
|
||||||
|
fos = SimpleNamespace()
|
||||||
|
for k in os.__dict__:
|
||||||
|
try:
|
||||||
|
setattr(fos, k, getattr(os, k))
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
fos.close = self._close
|
||||||
|
fos.listdir = self._listdir
|
||||||
|
fos.mkdir = self._mkdir
|
||||||
|
fos.open = self._open
|
||||||
|
fos.remove = self._unlink
|
||||||
|
fos.rename = self._rename
|
||||||
|
fos.stat = self._stat
|
||||||
|
fos.unlink = self._unlink
|
||||||
|
fos.utime = self._utime
|
||||||
|
smbserver.os = fos
|
||||||
|
|
||||||
|
# ...and smbserver.os.path
|
||||||
|
fop = SimpleNamespace()
|
||||||
|
for k in os.path.__dict__:
|
||||||
|
try:
|
||||||
|
setattr(fop, k, getattr(os.path, k))
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
fop.exists = self._p_exists
|
||||||
|
fop.getsize = self._p_getsize
|
||||||
|
fop.isdir = self._p_isdir
|
||||||
|
smbserver.os.path = fop
|
||||||
|
|
||||||
|
if not self.args.smb_nwa_2:
|
||||||
|
fop.join = self._p_join
|
||||||
|
|
||||||
|
# other patches
|
||||||
|
smbserver.isInFileJail = self._is_in_file_jail
|
||||||
|
self._disarm()
|
||||||
|
|
||||||
|
ip = next((x for x in self.args.i if ":" not in x), None)
|
||||||
|
if not ip:
|
||||||
|
self.log("smb", "IPv6 not supported for SMB; listening on 0.0.0.0", 3)
|
||||||
|
ip = "0.0.0.0"
|
||||||
|
|
||||||
|
port = int(self.args.smb_port)
|
||||||
|
srv = smbserver.SimpleSMBServer(listenAddress=ip, listenPort=port)
|
||||||
|
|
||||||
|
ro = "no" if self.args.smbw else "yes" # (does nothing)
|
||||||
|
srv.addShare("A", "/", readOnly=ro)
|
||||||
|
srv.setSMB2Support(not self.args.smb1)
|
||||||
|
|
||||||
|
for name, pwd in self.asrv.acct.items():
|
||||||
|
for u, p in ((name, pwd), (pwd, "k")):
|
||||||
|
lmhash = compute_lmhash(p)
|
||||||
|
nthash = compute_nthash(p)
|
||||||
|
srv.addCredential(u, 0, lmhash, nthash)
|
||||||
|
|
||||||
|
chi = [random.randint(0, 255) for x in range(8)]
|
||||||
|
cha = "".join(["{:02x}".format(x) for x in chi])
|
||||||
|
srv.setSMBChallenge(cha)
|
||||||
|
|
||||||
|
self.srv = srv
|
||||||
|
self.stop = srv.stop
|
||||||
|
self.log("smb", "listening @ {}:{}".format(ip, port))
|
||||||
|
|
||||||
|
def nlog(self, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
|
self.log("smb", msg, c)
|
||||||
|
|
||||||
|
def start(self) -> None:
|
||||||
|
Daemon(self.srv.start)
|
||||||
|
|
||||||
|
def _v2a(self, caller: str, vpath: str, *a: Any) -> tuple[VFS, str]:
|
||||||
|
vpath = vpath.replace("\\", "/").lstrip("/")
|
||||||
|
# cf = inspect.currentframe().f_back
|
||||||
|
# c1 = cf.f_back.f_code.co_name
|
||||||
|
# c2 = cf.f_code.co_name
|
||||||
|
debug('%s("%s", %s)\033[K\033[0m', caller, vpath, str(a))
|
||||||
|
|
||||||
|
# TODO find a way to grab `identity` in smbComSessionSetupAndX and smb2SessionSetup
|
||||||
|
vfs, rem = self.asrv.vfs.get(vpath, LEELOO_DALLAS, True, True)
|
||||||
|
return vfs, vfs.canonical(rem)
|
||||||
|
|
||||||
|
def _listdir(self, vpath: str, *a: Any, **ka: Any) -> list[str]:
|
||||||
|
vpath = vpath.replace("\\", "/").lstrip("/")
|
||||||
|
# caller = inspect.currentframe().f_back.f_code.co_name
|
||||||
|
debug('listdir("%s", %s)\033[K\033[0m', vpath, str(a))
|
||||||
|
vfs, rem = self.asrv.vfs.get(vpath, LEELOO_DALLAS, False, False)
|
||||||
|
_, vfs_ls, vfs_virt = vfs.ls(
|
||||||
|
rem, LEELOO_DALLAS, not self.args.no_scandir, [[False, False]]
|
||||||
|
)
|
||||||
|
dirs = [x[0] for x in vfs_ls if stat.S_ISDIR(x[1].st_mode)]
|
||||||
|
fils = [x[0] for x in vfs_ls if x[0] not in dirs]
|
||||||
|
ls = list(vfs_virt.keys()) + dirs + fils
|
||||||
|
if self.args.smb_nwa_1:
|
||||||
|
return ls
|
||||||
|
|
||||||
|
# clients crash somewhere around 65760 byte
|
||||||
|
ret = []
|
||||||
|
sz = 112 * 2 # ['.', '..']
|
||||||
|
for n, fn in enumerate(ls):
|
||||||
|
if sz >= 64000:
|
||||||
|
t = "listing only %d of %d files (%d byte); see impacket#1433"
|
||||||
|
warning(t, n, len(ls), sz)
|
||||||
|
break
|
||||||
|
|
||||||
|
nsz = len(fn.encode("utf-16", "replace"))
|
||||||
|
nsz = ((nsz + 7) // 8) * 8
|
||||||
|
sz += 104 + nsz
|
||||||
|
ret.append(fn)
|
||||||
|
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def _open(
|
||||||
|
self, vpath: str, flags: int, *a: Any, chmod: int = 0o777, **ka: Any
|
||||||
|
) -> Any:
|
||||||
|
f_ro = os.O_RDONLY
|
||||||
|
if ANYWIN:
|
||||||
|
f_ro |= os.O_BINARY
|
||||||
|
|
||||||
|
wr = flags != f_ro
|
||||||
|
if wr and not self.args.smbw:
|
||||||
|
yeet("blocked write (no --smbw): " + vpath)
|
||||||
|
|
||||||
|
vfs, ap = self._v2a("open", vpath, *a)
|
||||||
|
if wr:
|
||||||
|
if not vfs.axs.uwrite:
|
||||||
|
yeet("blocked write (no-write-acc): " + vpath)
|
||||||
|
|
||||||
|
xbu = vfs.flags.get("xbu")
|
||||||
|
if xbu and not runhook(
|
||||||
|
self.nlog, xbu, ap, vpath, "", "", 0, 0, "1.7.6.2", 0, ""
|
||||||
|
):
|
||||||
|
yeet("blocked by xbu server config: " + vpath)
|
||||||
|
|
||||||
|
ret = bos.open(ap, flags, *a, mode=chmod, **ka)
|
||||||
|
if wr:
|
||||||
|
now = time.time()
|
||||||
|
nf = len(self.files)
|
||||||
|
if nf > 9000:
|
||||||
|
oldest = min([x[0] for x in self.files.values()])
|
||||||
|
cutoff = oldest + (now - oldest) / 2
|
||||||
|
self.files = {k: v for k, v in self.files.items() if v[0] > cutoff}
|
||||||
|
info("was tracking %d files, now %d", nf, len(self.files))
|
||||||
|
|
||||||
|
vpath = vpath.replace("\\", "/").lstrip("/")
|
||||||
|
self.files[ret] = (now, vpath)
|
||||||
|
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def _close(self, fd: int) -> None:
|
||||||
|
os.close(fd)
|
||||||
|
if fd not in self.files:
|
||||||
|
return
|
||||||
|
|
||||||
|
_, vp = self.files.pop(fd)
|
||||||
|
vp, fn = os.path.split(vp)
|
||||||
|
vfs, rem = self.hub.asrv.vfs.get(vp, LEELOO_DALLAS, False, True)
|
||||||
|
vfs, rem = vfs.get_dbv(rem)
|
||||||
|
self.hub.up2k.hash_file(
|
||||||
|
vfs.realpath,
|
||||||
|
vfs.vpath,
|
||||||
|
vfs.flags,
|
||||||
|
rem,
|
||||||
|
fn,
|
||||||
|
"1.7.6.2",
|
||||||
|
time.time(),
|
||||||
|
"",
|
||||||
|
)
|
||||||
|
|
||||||
|
def _rename(self, vp1: str, vp2: str) -> None:
|
||||||
|
if not self.args.smbw:
|
||||||
|
yeet("blocked rename (no --smbw): " + vp1)
|
||||||
|
|
||||||
|
vp1 = vp1.lstrip("/")
|
||||||
|
vp2 = vp2.lstrip("/")
|
||||||
|
|
||||||
|
vfs2, ap2 = self._v2a("rename", vp2, vp1)
|
||||||
|
if not vfs2.axs.uwrite:
|
||||||
|
yeet("blocked rename (no-write-acc): " + vp2)
|
||||||
|
|
||||||
|
vfs1, _ = self.asrv.vfs.get(vp1, LEELOO_DALLAS, True, True)
|
||||||
|
if not vfs1.axs.umove:
|
||||||
|
yeet("blocked rename (no-move-acc): " + vp1)
|
||||||
|
|
||||||
|
self.hub.up2k.handle_mv(LEELOO_DALLAS, vp1, vp2)
|
||||||
|
try:
|
||||||
|
bos.makedirs(ap2)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def _mkdir(self, vpath: str) -> None:
|
||||||
|
if not self.args.smbw:
|
||||||
|
yeet("blocked mkdir (no --smbw): " + vpath)
|
||||||
|
|
||||||
|
vfs, ap = self._v2a("mkdir", vpath)
|
||||||
|
if not vfs.axs.uwrite:
|
||||||
|
yeet("blocked mkdir (no-write-acc): " + vpath)
|
||||||
|
|
||||||
|
return bos.mkdir(ap)
|
||||||
|
|
||||||
|
def _stat(self, vpath: str, *a: Any, **ka: Any) -> os.stat_result:
|
||||||
|
return bos.stat(self._v2a("stat", vpath, *a)[1], *a, **ka)
|
||||||
|
|
||||||
|
def _unlink(self, vpath: str) -> None:
|
||||||
|
if not self.args.smbw:
|
||||||
|
yeet("blocked delete (no --smbw): " + vpath)
|
||||||
|
|
||||||
|
# return bos.unlink(self._v2a("stat", vpath, *a)[1])
|
||||||
|
vfs, ap = self._v2a("delete", vpath)
|
||||||
|
if not vfs.axs.udel:
|
||||||
|
yeet("blocked delete (no-del-acc): " + vpath)
|
||||||
|
|
||||||
|
vpath = vpath.replace("\\", "/").lstrip("/")
|
||||||
|
self.hub.up2k.handle_rm(LEELOO_DALLAS, "1.7.6.2", [vpath], [])
|
||||||
|
|
||||||
|
def _utime(self, vpath: str, times: tuple[float, float]) -> None:
|
||||||
|
if not self.args.smbw:
|
||||||
|
yeet("blocked utime (no --smbw): " + vpath)
|
||||||
|
|
||||||
|
vfs, ap = self._v2a("utime", vpath)
|
||||||
|
if not vfs.axs.uwrite:
|
||||||
|
yeet("blocked utime (no-write-acc): " + vpath)
|
||||||
|
|
||||||
|
return bos.utime(ap, times)
|
||||||
|
|
||||||
|
def _p_exists(self, vpath: str) -> bool:
|
||||||
|
try:
|
||||||
|
bos.stat(self._v2a("p.exists", vpath)[1])
|
||||||
|
return True
|
||||||
|
except:
|
||||||
|
return False
|
||||||
|
|
||||||
|
def _p_getsize(self, vpath: str) -> int:
|
||||||
|
st = bos.stat(self._v2a("p.getsize", vpath)[1])
|
||||||
|
return st.st_size
|
||||||
|
|
||||||
|
def _p_isdir(self, vpath: str) -> bool:
|
||||||
|
try:
|
||||||
|
st = bos.stat(self._v2a("p.isdir", vpath)[1])
|
||||||
|
return stat.S_ISDIR(st.st_mode)
|
||||||
|
except:
|
||||||
|
return False
|
||||||
|
|
||||||
|
def _p_join(self, *a) -> str:
|
||||||
|
# impacket.smbserver reads globs from queryDirectoryRequest['Buffer']
|
||||||
|
# where somehow `fds.*` becomes `fds"*` so lets fix that
|
||||||
|
ret = os.path.join(*a)
|
||||||
|
return ret.replace('"', ".") # type: ignore
|
||||||
|
|
||||||
|
def _hook(self, *a: Any, **ka: Any) -> None:
|
||||||
|
src = inspect.currentframe().f_back.f_code.co_name
|
||||||
|
error("\033[31m%s:hook(%s)\033[0m", src, a)
|
||||||
|
raise Exception("nope")
|
||||||
|
|
||||||
|
def _disarm(self) -> None:
|
||||||
|
from impacket import smbserver
|
||||||
|
|
||||||
|
smbserver.os.chmod = self._hook
|
||||||
|
smbserver.os.chown = self._hook
|
||||||
|
smbserver.os.ftruncate = self._hook
|
||||||
|
smbserver.os.lchown = self._hook
|
||||||
|
smbserver.os.link = self._hook
|
||||||
|
smbserver.os.lstat = self._hook
|
||||||
|
smbserver.os.replace = self._hook
|
||||||
|
smbserver.os.scandir = self._hook
|
||||||
|
smbserver.os.symlink = self._hook
|
||||||
|
smbserver.os.truncate = self._hook
|
||||||
|
smbserver.os.walk = self._hook
|
||||||
|
|
||||||
|
smbserver.os.path.abspath = self._hook
|
||||||
|
smbserver.os.path.expanduser = self._hook
|
||||||
|
smbserver.os.path.getatime = self._hook
|
||||||
|
smbserver.os.path.getctime = self._hook
|
||||||
|
smbserver.os.path.getmtime = self._hook
|
||||||
|
smbserver.os.path.isabs = self._hook
|
||||||
|
smbserver.os.path.isfile = self._hook
|
||||||
|
smbserver.os.path.islink = self._hook
|
||||||
|
smbserver.os.path.realpath = self._hook
|
||||||
|
|
||||||
|
def _is_in_file_jail(self, *a: Any) -> bool:
|
||||||
|
# handled by vfs
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def yeet(msg: str) -> None:
|
||||||
|
info(msg)
|
||||||
|
raise Exception(msg)
|
||||||
210
copyparty/ssdp.py
Normal file
210
copyparty/ssdp.py
Normal file
@@ -0,0 +1,210 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import re
|
||||||
|
import select
|
||||||
|
import socket
|
||||||
|
from email.utils import formatdate
|
||||||
|
|
||||||
|
from .__init__ import TYPE_CHECKING
|
||||||
|
from .multicast import MC_Sck, MCast
|
||||||
|
from .util import CachedSet, html_escape, min_ex
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from .broker_util import BrokerCli
|
||||||
|
from .httpcli import HttpCli
|
||||||
|
from .svchub import SvcHub
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Optional, Union
|
||||||
|
|
||||||
|
|
||||||
|
GRP = "239.255.255.250"
|
||||||
|
|
||||||
|
|
||||||
|
class SSDP_Sck(MC_Sck):
|
||||||
|
def __init__(self, *a):
|
||||||
|
super(SSDP_Sck, self).__init__(*a)
|
||||||
|
self.hport = 0
|
||||||
|
|
||||||
|
|
||||||
|
class SSDPr(object):
|
||||||
|
"""generates http responses for httpcli"""
|
||||||
|
|
||||||
|
def __init__(self, broker: "BrokerCli") -> None:
|
||||||
|
self.broker = broker
|
||||||
|
self.args = broker.args
|
||||||
|
|
||||||
|
def reply(self, hc: "HttpCli") -> bool:
|
||||||
|
if hc.vpath.endswith("device.xml"):
|
||||||
|
return self.tx_device(hc)
|
||||||
|
|
||||||
|
hc.reply(b"unknown request", 400)
|
||||||
|
return False
|
||||||
|
|
||||||
|
def tx_device(self, hc: "HttpCli") -> bool:
|
||||||
|
zs = """
|
||||||
|
<?xml version="1.0"?>
|
||||||
|
<root xmlns="urn:schemas-upnp-org:device-1-0">
|
||||||
|
<specVersion>
|
||||||
|
<major>1</major>
|
||||||
|
<minor>0</minor>
|
||||||
|
</specVersion>
|
||||||
|
<URLBase>{}</URLBase>
|
||||||
|
<device>
|
||||||
|
<presentationURL>{}</presentationURL>
|
||||||
|
<deviceType>urn:schemas-upnp-org:device:Basic:1</deviceType>
|
||||||
|
<friendlyName>{}</friendlyName>
|
||||||
|
<modelDescription>file server</modelDescription>
|
||||||
|
<manufacturer>ed</manufacturer>
|
||||||
|
<manufacturerURL>https://ocv.me/</manufacturerURL>
|
||||||
|
<modelName>copyparty</modelName>
|
||||||
|
<modelURL>https://github.com/9001/copyparty/</modelURL>
|
||||||
|
<UDN>{}</UDN>
|
||||||
|
<serviceList>
|
||||||
|
<service>
|
||||||
|
<serviceType>urn:schemas-upnp-org:device:Basic:1</serviceType>
|
||||||
|
<serviceId>urn:schemas-upnp-org:device:Basic</serviceId>
|
||||||
|
<controlURL>/.cpr/ssdp/services.xml</controlURL>
|
||||||
|
<eventSubURL>/.cpr/ssdp/services.xml</eventSubURL>
|
||||||
|
<SCPDURL>/.cpr/ssdp/services.xml</SCPDURL>
|
||||||
|
</service>
|
||||||
|
</serviceList>
|
||||||
|
</device>
|
||||||
|
</root>"""
|
||||||
|
|
||||||
|
c = html_escape
|
||||||
|
sip, sport = hc.s.getsockname()[:2]
|
||||||
|
sip = sip.replace("::ffff:", "")
|
||||||
|
proto = "https" if self.args.https_only else "http"
|
||||||
|
ubase = "{}://{}:{}".format(proto, sip, sport)
|
||||||
|
zsl = self.args.zsl
|
||||||
|
url = zsl if "://" in zsl else ubase + "/" + zsl.lstrip("/")
|
||||||
|
name = "{} @ {}".format(self.args.doctitle, self.args.name)
|
||||||
|
zs = zs.strip().format(c(ubase), c(url), c(name), c(self.args.zsid))
|
||||||
|
hc.reply(zs.encode("utf-8", "replace"))
|
||||||
|
return False # close connectino
|
||||||
|
|
||||||
|
|
||||||
|
class SSDPd(MCast):
|
||||||
|
"""communicates with ssdp clients over multicast"""
|
||||||
|
|
||||||
|
def __init__(self, hub: "SvcHub", ngen: int) -> None:
|
||||||
|
al = hub.args
|
||||||
|
vinit = al.zsv and not al.zmv
|
||||||
|
super(SSDPd, self).__init__(
|
||||||
|
hub, SSDP_Sck, al.zs_on, al.zs_off, GRP, "", 1900, vinit
|
||||||
|
)
|
||||||
|
self.srv: dict[socket.socket, SSDP_Sck] = {}
|
||||||
|
self.logsrc = "SSDP-{}".format(ngen)
|
||||||
|
self.ngen = ngen
|
||||||
|
|
||||||
|
self.rxc = CachedSet(0.7)
|
||||||
|
self.txc = CachedSet(5) # win10: every 3 sec
|
||||||
|
self.ptn_st = re.compile(b"\nst: *upnp:rootdevice", re.I)
|
||||||
|
|
||||||
|
def log(self, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
|
self.log_func(self.logsrc, msg, c)
|
||||||
|
|
||||||
|
def run(self) -> None:
|
||||||
|
try:
|
||||||
|
bound = self.create_servers()
|
||||||
|
except:
|
||||||
|
t = "no server IP matches the ssdp config\n{}"
|
||||||
|
self.log(t.format(min_ex()), 1)
|
||||||
|
bound = []
|
||||||
|
|
||||||
|
if not bound:
|
||||||
|
self.log("failed to announce copyparty services on the network", 3)
|
||||||
|
return
|
||||||
|
|
||||||
|
# find http port for this listening ip
|
||||||
|
for srv in self.srv.values():
|
||||||
|
tcps = self.hub.tcpsrv.bound
|
||||||
|
hp = next((x[1] for x in tcps if x[0] in ("0.0.0.0", srv.ip)), 0)
|
||||||
|
hp = hp or next((x[1] for x in tcps if x[0] == "::"), 0)
|
||||||
|
if not hp:
|
||||||
|
hp = tcps[0][1]
|
||||||
|
self.log("assuming port {} for {}".format(hp, srv.ip), 3)
|
||||||
|
srv.hport = hp
|
||||||
|
|
||||||
|
self.log("listening")
|
||||||
|
while self.running:
|
||||||
|
rdy = select.select(self.srv, [], [], self.args.z_chk or 180)
|
||||||
|
rx: list[socket.socket] = rdy[0] # type: ignore
|
||||||
|
self.rxc.cln()
|
||||||
|
buf = b""
|
||||||
|
addr = ("0", 0)
|
||||||
|
for sck in rx:
|
||||||
|
try:
|
||||||
|
buf, addr = sck.recvfrom(4096)
|
||||||
|
self.eat(buf, addr)
|
||||||
|
except:
|
||||||
|
if not self.running:
|
||||||
|
break
|
||||||
|
|
||||||
|
t = "{} {} \033[33m|{}| {}\n{}".format(
|
||||||
|
self.srv[sck].name, addr, len(buf), repr(buf)[2:-1], min_ex()
|
||||||
|
)
|
||||||
|
self.log(t, 6)
|
||||||
|
|
||||||
|
self.log("stopped", 2)
|
||||||
|
|
||||||
|
def stop(self) -> None:
|
||||||
|
self.running = False
|
||||||
|
for srv in self.srv.values():
|
||||||
|
try:
|
||||||
|
srv.sck.close()
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
self.srv = {}
|
||||||
|
|
||||||
|
def eat(self, buf: bytes, addr: tuple[str, int]) -> None:
|
||||||
|
cip = addr[0]
|
||||||
|
if cip.startswith("169.254") and not self.ll_ok:
|
||||||
|
return
|
||||||
|
|
||||||
|
if buf in self.rxc.c:
|
||||||
|
return
|
||||||
|
|
||||||
|
srv: Optional[SSDP_Sck] = self.map_client(cip) # type: ignore
|
||||||
|
if not srv:
|
||||||
|
return
|
||||||
|
|
||||||
|
self.rxc.add(buf)
|
||||||
|
if not buf.startswith(b"M-SEARCH * HTTP/1."):
|
||||||
|
return
|
||||||
|
|
||||||
|
if not self.ptn_st.search(buf):
|
||||||
|
return
|
||||||
|
|
||||||
|
if self.args.zsv:
|
||||||
|
t = "{} [{}] \033[36m{} \033[0m|{}|"
|
||||||
|
self.log(t.format(srv.name, srv.ip, cip, len(buf)), "90")
|
||||||
|
|
||||||
|
zs = """
|
||||||
|
HTTP/1.1 200 OK
|
||||||
|
CACHE-CONTROL: max-age=1800
|
||||||
|
DATE: {0}
|
||||||
|
EXT:
|
||||||
|
LOCATION: http://{1}:{2}/.cpr/ssdp/device.xml
|
||||||
|
OPT: "http://schemas.upnp.org/upnp/1/0/"; ns=01
|
||||||
|
01-NLS: {3}
|
||||||
|
SERVER: UPnP/1.0
|
||||||
|
ST: upnp:rootdevice
|
||||||
|
USN: {3}::upnp:rootdevice
|
||||||
|
BOOTID.UPNP.ORG: 0
|
||||||
|
CONFIGID.UPNP.ORG: 1
|
||||||
|
|
||||||
|
"""
|
||||||
|
v4 = srv.ip.replace("::ffff:", "")
|
||||||
|
zs = zs.format(formatdate(usegmt=True), v4, srv.hport, self.args.zsid)
|
||||||
|
zb = zs[1:].replace("\n", "\r\n").encode("utf-8", "replace")
|
||||||
|
srv.sck.sendto(zb, addr[:2])
|
||||||
|
|
||||||
|
if cip not in self.txc.c:
|
||||||
|
self.log("{} [{}] --> {}".format(srv.name, srv.ip, cip), "6")
|
||||||
|
|
||||||
|
self.txc.add(cip)
|
||||||
|
self.txc.cln()
|
||||||
@@ -1,23 +1,30 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import stat
|
||||||
import tarfile
|
import tarfile
|
||||||
import threading
|
|
||||||
|
|
||||||
from .sutil import errdesc
|
from queue import Queue
|
||||||
from .util import Queue, fsenc
|
|
||||||
from .bos import bos
|
from .bos import bos
|
||||||
|
from .sutil import StreamArc, errdesc
|
||||||
|
from .util import Daemon, fsenc, min_ex
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any, Generator, Optional
|
||||||
|
|
||||||
|
from .util import NamedLogger
|
||||||
|
|
||||||
|
|
||||||
class QFile(object):
|
class QFile(object): # inherit io.StringIO for painful typing
|
||||||
"""file-like object which buffers writes into a queue"""
|
"""file-like object which buffers writes into a queue"""
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self) -> None:
|
||||||
self.q = Queue(64)
|
self.q: Queue[Optional[bytes]] = Queue(64)
|
||||||
self.bq = []
|
self.bq: list[bytes] = []
|
||||||
self.nq = 0
|
self.nq = 0
|
||||||
|
|
||||||
def write(self, buf):
|
def write(self, buf: Optional[bytes]) -> None:
|
||||||
if buf is None or self.nq >= 240 * 1024:
|
if buf is None or self.nq >= 240 * 1024:
|
||||||
self.q.put(b"".join(self.bq))
|
self.q.put(b"".join(self.bq))
|
||||||
self.bq = []
|
self.bq = []
|
||||||
@@ -30,44 +37,52 @@ class QFile(object):
|
|||||||
self.nq += len(buf)
|
self.nq += len(buf)
|
||||||
|
|
||||||
|
|
||||||
class StreamTar(object):
|
class StreamTar(StreamArc):
|
||||||
"""construct in-memory tar file from the given path"""
|
"""construct in-memory tar file from the given path"""
|
||||||
|
|
||||||
def __init__(self, log, fgen, **kwargs):
|
def __init__(
|
||||||
|
self,
|
||||||
|
log: "NamedLogger",
|
||||||
|
fgen: Generator[dict[str, Any], None, None],
|
||||||
|
**kwargs: Any
|
||||||
|
):
|
||||||
|
super(StreamTar, self).__init__(log, fgen)
|
||||||
|
|
||||||
self.ci = 0
|
self.ci = 0
|
||||||
self.co = 0
|
self.co = 0
|
||||||
self.qfile = QFile()
|
self.qfile = QFile()
|
||||||
self.log = log
|
self.errf: dict[str, Any] = {}
|
||||||
self.fgen = fgen
|
|
||||||
self.errf = None
|
|
||||||
|
|
||||||
# python 3.8 changed to PAX_FORMAT as default,
|
# python 3.8 changed to PAX_FORMAT as default,
|
||||||
# waste of space and don't care about the new features
|
# waste of space and don't care about the new features
|
||||||
fmt = tarfile.GNU_FORMAT
|
fmt = tarfile.GNU_FORMAT
|
||||||
self.tar = tarfile.open(fileobj=self.qfile, mode="w|", format=fmt)
|
self.tar = tarfile.open(fileobj=self.qfile, mode="w|", format=fmt) # type: ignore
|
||||||
|
|
||||||
w = threading.Thread(target=self._gen, name="star-gen")
|
Daemon(self._gen, "star-gen")
|
||||||
w.daemon = True
|
|
||||||
w.start()
|
|
||||||
|
|
||||||
def gen(self):
|
def gen(self) -> Generator[Optional[bytes], None, None]:
|
||||||
while True:
|
try:
|
||||||
buf = self.qfile.q.get()
|
while True:
|
||||||
if not buf:
|
buf = self.qfile.q.get()
|
||||||
break
|
if not buf:
|
||||||
|
break
|
||||||
|
|
||||||
self.co += len(buf)
|
self.co += len(buf)
|
||||||
yield buf
|
yield buf
|
||||||
|
|
||||||
yield None
|
yield None
|
||||||
if self.errf:
|
finally:
|
||||||
bos.unlink(self.errf["ap"])
|
if self.errf:
|
||||||
|
bos.unlink(self.errf["ap"])
|
||||||
|
|
||||||
def ser(self, f):
|
def ser(self, f: dict[str, Any]) -> None:
|
||||||
name = f["vp"]
|
name = f["vp"]
|
||||||
src = f["ap"]
|
src = f["ap"]
|
||||||
fsi = f["st"]
|
fsi = f["st"]
|
||||||
|
|
||||||
|
if stat.S_ISDIR(fsi.st_mode):
|
||||||
|
return
|
||||||
|
|
||||||
inf = tarfile.TarInfo(name=name)
|
inf = tarfile.TarInfo(name=name)
|
||||||
inf.mode = fsi.st_mode
|
inf.mode = fsi.st_mode
|
||||||
inf.size = fsi.st_size
|
inf.size = fsi.st_size
|
||||||
@@ -76,20 +91,21 @@ class StreamTar(object):
|
|||||||
inf.gid = 0
|
inf.gid = 0
|
||||||
|
|
||||||
self.ci += inf.size
|
self.ci += inf.size
|
||||||
with open(fsenc(src), "rb", 512 * 1024) as f:
|
with open(fsenc(src), "rb", 512 * 1024) as fo:
|
||||||
self.tar.addfile(inf, f)
|
self.tar.addfile(inf, fo)
|
||||||
|
|
||||||
def _gen(self):
|
def _gen(self) -> None:
|
||||||
errors = []
|
errors = []
|
||||||
for f in self.fgen:
|
for f in self.fgen:
|
||||||
if "err" in f:
|
if "err" in f:
|
||||||
errors.append([f["vp"], f["err"]])
|
errors.append((f["vp"], f["err"]))
|
||||||
continue
|
continue
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.ser(f)
|
self.ser(f)
|
||||||
except Exception as ex:
|
except:
|
||||||
errors.append([f["vp"], repr(ex)])
|
ex = min_ex(5, True).replace("\n", "\n-- ")
|
||||||
|
errors.append((f["vp"], ex))
|
||||||
|
|
||||||
if errors:
|
if errors:
|
||||||
self.errf, txt = errdesc(errors)
|
self.errf, txt = errdesc(errors)
|
||||||
|
|||||||
5
copyparty/stolen/dnslib/README.md
Normal file
5
copyparty/stolen/dnslib/README.md
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
`dnslib` but heavily simplified/feature-stripped
|
||||||
|
|
||||||
|
L: MIT
|
||||||
|
Copyright (c) 2010 - 2017 Paul Chakravarti
|
||||||
|
https://github.com/paulc/dnslib/
|
||||||
11
copyparty/stolen/dnslib/__init__.py
Normal file
11
copyparty/stolen/dnslib/__init__.py
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
"""
|
||||||
|
L: MIT
|
||||||
|
Copyright (c) 2010 - 2017 Paul Chakravarti
|
||||||
|
https://github.com/paulc/dnslib/tree/0.9.23
|
||||||
|
"""
|
||||||
|
|
||||||
|
from .dns import *
|
||||||
|
|
||||||
|
version = "0.9.23"
|
||||||
41
copyparty/stolen/dnslib/bimap.py
Normal file
41
copyparty/stolen/dnslib/bimap.py
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
import types
|
||||||
|
|
||||||
|
|
||||||
|
class BimapError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class Bimap(object):
|
||||||
|
def __init__(self, name, forward, error=AttributeError):
|
||||||
|
self.name = name
|
||||||
|
self.error = error
|
||||||
|
self.forward = forward.copy()
|
||||||
|
self.reverse = dict([(v, k) for (k, v) in list(forward.items())])
|
||||||
|
|
||||||
|
def get(self, k, default=None):
|
||||||
|
try:
|
||||||
|
return self.forward[k]
|
||||||
|
except KeyError:
|
||||||
|
return default or str(k)
|
||||||
|
|
||||||
|
def __getitem__(self, k):
|
||||||
|
try:
|
||||||
|
return self.forward[k]
|
||||||
|
except KeyError:
|
||||||
|
if isinstance(self.error, types.FunctionType):
|
||||||
|
return self.error(self.name, k, True)
|
||||||
|
else:
|
||||||
|
raise self.error("%s: Invalid forward lookup: [%s]" % (self.name, k))
|
||||||
|
|
||||||
|
def __getattr__(self, k):
|
||||||
|
try:
|
||||||
|
if k == "__wrapped__":
|
||||||
|
raise AttributeError()
|
||||||
|
return self.reverse[k]
|
||||||
|
except KeyError:
|
||||||
|
if isinstance(self.error, types.FunctionType):
|
||||||
|
return self.error(self.name, k, False)
|
||||||
|
else:
|
||||||
|
raise self.error("%s: Invalid reverse lookup: [%s]" % (self.name, k))
|
||||||
15
copyparty/stolen/dnslib/bit.py
Normal file
15
copyparty/stolen/dnslib/bit.py
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
from __future__ import print_function
|
||||||
|
|
||||||
|
|
||||||
|
def get_bits(data, offset, bits=1):
|
||||||
|
mask = ((1 << bits) - 1) << offset
|
||||||
|
return (data & mask) >> offset
|
||||||
|
|
||||||
|
|
||||||
|
def set_bits(data, value, offset, bits=1):
|
||||||
|
mask = ((1 << bits) - 1) << offset
|
||||||
|
clear = 0xFFFF ^ mask
|
||||||
|
data = (data & clear) | ((value << offset) & mask)
|
||||||
|
return data
|
||||||
56
copyparty/stolen/dnslib/buffer.py
Normal file
56
copyparty/stolen/dnslib/buffer.py
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
import binascii
|
||||||
|
import struct
|
||||||
|
|
||||||
|
|
||||||
|
class BufferError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class Buffer(object):
|
||||||
|
def __init__(self, data=b""):
|
||||||
|
self.data = bytearray(data)
|
||||||
|
self.offset = 0
|
||||||
|
|
||||||
|
def remaining(self):
|
||||||
|
return len(self.data) - self.offset
|
||||||
|
|
||||||
|
def get(self, length):
|
||||||
|
if length > self.remaining():
|
||||||
|
raise BufferError(
|
||||||
|
"Not enough bytes [offset=%d,remaining=%d,requested=%d]"
|
||||||
|
% (self.offset, self.remaining(), length)
|
||||||
|
)
|
||||||
|
start = self.offset
|
||||||
|
end = self.offset + length
|
||||||
|
self.offset += length
|
||||||
|
return bytes(self.data[start:end])
|
||||||
|
|
||||||
|
def hex(self):
|
||||||
|
return binascii.hexlify(self.data)
|
||||||
|
|
||||||
|
def pack(self, fmt, *args):
|
||||||
|
self.offset += struct.calcsize(fmt)
|
||||||
|
self.data += struct.pack(fmt, *args)
|
||||||
|
|
||||||
|
def append(self, s):
|
||||||
|
self.offset += len(s)
|
||||||
|
self.data += s
|
||||||
|
|
||||||
|
def update(self, ptr, fmt, *args):
|
||||||
|
s = struct.pack(fmt, *args)
|
||||||
|
self.data[ptr : ptr + len(s)] = s
|
||||||
|
|
||||||
|
def unpack(self, fmt):
|
||||||
|
try:
|
||||||
|
data = self.get(struct.calcsize(fmt))
|
||||||
|
return struct.unpack(fmt, data)
|
||||||
|
except struct.error:
|
||||||
|
raise BufferError(
|
||||||
|
"Error unpacking struct '%s' <%s>"
|
||||||
|
% (fmt, binascii.hexlify(data).decode())
|
||||||
|
)
|
||||||
|
|
||||||
|
def __len__(self):
|
||||||
|
return len(self.data)
|
||||||
775
copyparty/stolen/dnslib/dns.py
Normal file
775
copyparty/stolen/dnslib/dns.py
Normal file
@@ -0,0 +1,775 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
from __future__ import print_function
|
||||||
|
|
||||||
|
import binascii
|
||||||
|
from itertools import chain
|
||||||
|
|
||||||
|
from .bimap import Bimap, BimapError
|
||||||
|
from .bit import get_bits, set_bits
|
||||||
|
from .buffer import BufferError
|
||||||
|
from .label import DNSBuffer, DNSLabel
|
||||||
|
from .ranges import IP4, IP6, H, I, check_bytes
|
||||||
|
|
||||||
|
|
||||||
|
class DNSError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def unknown_qtype(name, key, forward):
|
||||||
|
if forward:
|
||||||
|
try:
|
||||||
|
return "TYPE%d" % (key,)
|
||||||
|
except:
|
||||||
|
raise DNSError("%s: Invalid forward lookup: [%s]" % (name, key))
|
||||||
|
else:
|
||||||
|
if key.startswith("TYPE"):
|
||||||
|
try:
|
||||||
|
return int(key[4:])
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
raise DNSError("%s: Invalid reverse lookup: [%s]" % (name, key))
|
||||||
|
|
||||||
|
|
||||||
|
QTYPE = Bimap(
|
||||||
|
"QTYPE",
|
||||||
|
{1: "A", 12: "PTR", 16: "TXT", 28: "AAAA", 33: "SRV", 47: "NSEC", 255: "ANY"},
|
||||||
|
unknown_qtype,
|
||||||
|
)
|
||||||
|
|
||||||
|
CLASS = Bimap("CLASS", {1: "IN", 254: "None", 255: "*", 0x8001: "F_IN"}, DNSError)
|
||||||
|
|
||||||
|
QR = Bimap("QR", {0: "QUERY", 1: "RESPONSE"}, DNSError)
|
||||||
|
|
||||||
|
RCODE = Bimap(
|
||||||
|
"RCODE",
|
||||||
|
{
|
||||||
|
0: "NOERROR",
|
||||||
|
1: "FORMERR",
|
||||||
|
2: "SERVFAIL",
|
||||||
|
3: "NXDOMAIN",
|
||||||
|
4: "NOTIMP",
|
||||||
|
5: "REFUSED",
|
||||||
|
6: "YXDOMAIN",
|
||||||
|
7: "YXRRSET",
|
||||||
|
8: "NXRRSET",
|
||||||
|
9: "NOTAUTH",
|
||||||
|
10: "NOTZONE",
|
||||||
|
},
|
||||||
|
DNSError,
|
||||||
|
)
|
||||||
|
|
||||||
|
OPCODE = Bimap(
|
||||||
|
"OPCODE", {0: "QUERY", 1: "IQUERY", 2: "STATUS", 4: "NOTIFY", 5: "UPDATE"}, DNSError
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def label(label, origin=None):
|
||||||
|
if label.endswith("."):
|
||||||
|
return DNSLabel(label)
|
||||||
|
else:
|
||||||
|
return (origin if isinstance(origin, DNSLabel) else DNSLabel(origin)).add(label)
|
||||||
|
|
||||||
|
|
||||||
|
class DNSRecord(object):
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, packet) -> "DNSRecord":
|
||||||
|
buffer = DNSBuffer(packet)
|
||||||
|
try:
|
||||||
|
header = DNSHeader.parse(buffer)
|
||||||
|
questions = []
|
||||||
|
rr = []
|
||||||
|
auth = []
|
||||||
|
ar = []
|
||||||
|
for i in range(header.q):
|
||||||
|
questions.append(DNSQuestion.parse(buffer))
|
||||||
|
for i in range(header.a):
|
||||||
|
rr.append(RR.parse(buffer))
|
||||||
|
for i in range(header.auth):
|
||||||
|
auth.append(RR.parse(buffer))
|
||||||
|
for i in range(header.ar):
|
||||||
|
ar.append(RR.parse(buffer))
|
||||||
|
return cls(header, questions, rr, auth=auth, ar=ar)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError(
|
||||||
|
"Error unpacking DNSRecord [offset=%d]: %s" % (buffer.offset, e)
|
||||||
|
)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def question(cls, qname, qtype="A", qclass="IN"):
|
||||||
|
return DNSRecord(
|
||||||
|
q=DNSQuestion(qname, getattr(QTYPE, qtype), getattr(CLASS, qclass))
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self, header=None, questions=None, rr=None, q=None, a=None, auth=None, ar=None
|
||||||
|
) -> None:
|
||||||
|
self.header = header or DNSHeader()
|
||||||
|
self.questions: list[DNSQuestion] = questions or []
|
||||||
|
self.rr: list[RR] = rr or []
|
||||||
|
self.auth: list[RR] = auth or []
|
||||||
|
self.ar: list[RR] = ar or []
|
||||||
|
|
||||||
|
if q:
|
||||||
|
self.questions.append(q)
|
||||||
|
if a:
|
||||||
|
self.rr.append(a)
|
||||||
|
self.set_header_qa()
|
||||||
|
|
||||||
|
def reply(self, ra=1, aa=1):
|
||||||
|
return DNSRecord(
|
||||||
|
DNSHeader(id=self.header.id, bitmap=self.header.bitmap, qr=1, ra=ra, aa=aa),
|
||||||
|
q=self.q,
|
||||||
|
)
|
||||||
|
|
||||||
|
def add_question(self, *q) -> None:
|
||||||
|
self.questions.extend(q)
|
||||||
|
self.set_header_qa()
|
||||||
|
|
||||||
|
def add_answer(self, *rr) -> None:
|
||||||
|
self.rr.extend(rr)
|
||||||
|
self.set_header_qa()
|
||||||
|
|
||||||
|
def add_auth(self, *auth) -> None:
|
||||||
|
self.auth.extend(auth)
|
||||||
|
self.set_header_qa()
|
||||||
|
|
||||||
|
def add_ar(self, *ar) -> None:
|
||||||
|
self.ar.extend(ar)
|
||||||
|
self.set_header_qa()
|
||||||
|
|
||||||
|
def set_header_qa(self) -> None:
|
||||||
|
self.header.q = len(self.questions)
|
||||||
|
self.header.a = len(self.rr)
|
||||||
|
self.header.auth = len(self.auth)
|
||||||
|
self.header.ar = len(self.ar)
|
||||||
|
|
||||||
|
def get_q(self):
|
||||||
|
return self.questions[0] if self.questions else DNSQuestion()
|
||||||
|
|
||||||
|
q = property(get_q)
|
||||||
|
|
||||||
|
def get_a(self):
|
||||||
|
return self.rr[0] if self.rr else RR()
|
||||||
|
|
||||||
|
a = property(get_a)
|
||||||
|
|
||||||
|
def pack(self) -> bytes:
|
||||||
|
self.set_header_qa()
|
||||||
|
buffer = DNSBuffer()
|
||||||
|
self.header.pack(buffer)
|
||||||
|
for q in self.questions:
|
||||||
|
q.pack(buffer)
|
||||||
|
for rr in self.rr:
|
||||||
|
rr.pack(buffer)
|
||||||
|
for auth in self.auth:
|
||||||
|
auth.pack(buffer)
|
||||||
|
for ar in self.ar:
|
||||||
|
ar.pack(buffer)
|
||||||
|
return buffer.data
|
||||||
|
|
||||||
|
def truncate(self):
|
||||||
|
return DNSRecord(DNSHeader(id=self.header.id, bitmap=self.header.bitmap, tc=1))
|
||||||
|
|
||||||
|
def format(self, prefix="", sort=False):
|
||||||
|
s = sorted if sort else lambda x: x
|
||||||
|
sections = [repr(self.header)]
|
||||||
|
sections.extend(s([repr(q) for q in self.questions]))
|
||||||
|
sections.extend(s([repr(rr) for rr in self.rr]))
|
||||||
|
sections.extend(s([repr(rr) for rr in self.auth]))
|
||||||
|
sections.extend(s([repr(rr) for rr in self.ar]))
|
||||||
|
return prefix + ("\n" + prefix).join(sections)
|
||||||
|
|
||||||
|
short = format
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return self.format()
|
||||||
|
|
||||||
|
__str__ = __repr__
|
||||||
|
|
||||||
|
|
||||||
|
class DNSHeader(object):
|
||||||
|
id = H("id")
|
||||||
|
bitmap = H("bitmap")
|
||||||
|
q = H("q")
|
||||||
|
a = H("a")
|
||||||
|
auth = H("auth")
|
||||||
|
ar = H("ar")
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, buffer):
|
||||||
|
try:
|
||||||
|
(id, bitmap, q, a, auth, ar) = buffer.unpack("!HHHHHH")
|
||||||
|
return cls(id, bitmap, q, a, auth, ar)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError(
|
||||||
|
"Error unpacking DNSHeader [offset=%d]: %s" % (buffer.offset, e)
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(self, id=None, bitmap=None, q=0, a=0, auth=0, ar=0, **args) -> None:
|
||||||
|
self.id = id if id else 0
|
||||||
|
if bitmap is None:
|
||||||
|
self.bitmap = 0
|
||||||
|
else:
|
||||||
|
self.bitmap = bitmap
|
||||||
|
self.q = q
|
||||||
|
self.a = a
|
||||||
|
self.auth = auth
|
||||||
|
self.ar = ar
|
||||||
|
for k, v in args.items():
|
||||||
|
if k.lower() == "qr":
|
||||||
|
self.qr = v
|
||||||
|
elif k.lower() == "opcode":
|
||||||
|
self.opcode = v
|
||||||
|
elif k.lower() == "aa":
|
||||||
|
self.aa = v
|
||||||
|
elif k.lower() == "tc":
|
||||||
|
self.tc = v
|
||||||
|
elif k.lower() == "rd":
|
||||||
|
self.rd = v
|
||||||
|
elif k.lower() == "ra":
|
||||||
|
self.ra = v
|
||||||
|
elif k.lower() == "z":
|
||||||
|
self.z = v
|
||||||
|
elif k.lower() == "ad":
|
||||||
|
self.ad = v
|
||||||
|
elif k.lower() == "cd":
|
||||||
|
self.cd = v
|
||||||
|
elif k.lower() == "rcode":
|
||||||
|
self.rcode = v
|
||||||
|
|
||||||
|
def get_qr(self):
|
||||||
|
return get_bits(self.bitmap, 15)
|
||||||
|
|
||||||
|
def set_qr(self, val):
|
||||||
|
self.bitmap = set_bits(self.bitmap, val, 15)
|
||||||
|
|
||||||
|
qr = property(get_qr, set_qr)
|
||||||
|
|
||||||
|
def get_opcode(self):
|
||||||
|
return get_bits(self.bitmap, 11, 4)
|
||||||
|
|
||||||
|
def set_opcode(self, val):
|
||||||
|
self.bitmap = set_bits(self.bitmap, val, 11, 4)
|
||||||
|
|
||||||
|
opcode = property(get_opcode, set_opcode)
|
||||||
|
|
||||||
|
def get_aa(self):
|
||||||
|
return get_bits(self.bitmap, 10)
|
||||||
|
|
||||||
|
def set_aa(self, val):
|
||||||
|
self.bitmap = set_bits(self.bitmap, val, 10)
|
||||||
|
|
||||||
|
aa = property(get_aa, set_aa)
|
||||||
|
|
||||||
|
def get_tc(self):
|
||||||
|
return get_bits(self.bitmap, 9)
|
||||||
|
|
||||||
|
def set_tc(self, val):
|
||||||
|
self.bitmap = set_bits(self.bitmap, val, 9)
|
||||||
|
|
||||||
|
tc = property(get_tc, set_tc)
|
||||||
|
|
||||||
|
def get_rd(self):
|
||||||
|
return get_bits(self.bitmap, 8)
|
||||||
|
|
||||||
|
def set_rd(self, val):
|
||||||
|
self.bitmap = set_bits(self.bitmap, val, 8)
|
||||||
|
|
||||||
|
rd = property(get_rd, set_rd)
|
||||||
|
|
||||||
|
def get_ra(self):
|
||||||
|
return get_bits(self.bitmap, 7)
|
||||||
|
|
||||||
|
def set_ra(self, val):
|
||||||
|
self.bitmap = set_bits(self.bitmap, val, 7)
|
||||||
|
|
||||||
|
ra = property(get_ra, set_ra)
|
||||||
|
|
||||||
|
def get_z(self):
|
||||||
|
return get_bits(self.bitmap, 6)
|
||||||
|
|
||||||
|
def set_z(self, val):
|
||||||
|
self.bitmap = set_bits(self.bitmap, val, 6)
|
||||||
|
|
||||||
|
z = property(get_z, set_z)
|
||||||
|
|
||||||
|
def get_ad(self):
|
||||||
|
return get_bits(self.bitmap, 5)
|
||||||
|
|
||||||
|
def set_ad(self, val):
|
||||||
|
self.bitmap = set_bits(self.bitmap, val, 5)
|
||||||
|
|
||||||
|
ad = property(get_ad, set_ad)
|
||||||
|
|
||||||
|
def get_cd(self):
|
||||||
|
return get_bits(self.bitmap, 4)
|
||||||
|
|
||||||
|
def set_cd(self, val):
|
||||||
|
self.bitmap = set_bits(self.bitmap, val, 4)
|
||||||
|
|
||||||
|
cd = property(get_cd, set_cd)
|
||||||
|
|
||||||
|
def get_rcode(self):
|
||||||
|
return get_bits(self.bitmap, 0, 4)
|
||||||
|
|
||||||
|
def set_rcode(self, val):
|
||||||
|
self.bitmap = set_bits(self.bitmap, val, 0, 4)
|
||||||
|
|
||||||
|
rcode = property(get_rcode, set_rcode)
|
||||||
|
|
||||||
|
def pack(self, buffer):
|
||||||
|
buffer.pack("!HHHHHH", self.id, self.bitmap, self.q, self.a, self.auth, self.ar)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
f = [
|
||||||
|
self.aa and "AA",
|
||||||
|
self.tc and "TC",
|
||||||
|
self.rd and "RD",
|
||||||
|
self.ra and "RA",
|
||||||
|
self.z and "Z",
|
||||||
|
self.ad and "AD",
|
||||||
|
self.cd and "CD",
|
||||||
|
]
|
||||||
|
if OPCODE.get(self.opcode) == "UPDATE":
|
||||||
|
f1 = "zo"
|
||||||
|
f2 = "pr"
|
||||||
|
f3 = "up"
|
||||||
|
f4 = "ad"
|
||||||
|
else:
|
||||||
|
f1 = "q"
|
||||||
|
f2 = "a"
|
||||||
|
f3 = "ns"
|
||||||
|
f4 = "ar"
|
||||||
|
return (
|
||||||
|
"<DNS Header: id=0x%x type=%s opcode=%s flags=%s "
|
||||||
|
"rcode='%s' %s=%d %s=%d %s=%d %s=%d>"
|
||||||
|
% (
|
||||||
|
self.id,
|
||||||
|
QR.get(self.qr),
|
||||||
|
OPCODE.get(self.opcode),
|
||||||
|
",".join(filter(None, f)),
|
||||||
|
RCODE.get(self.rcode),
|
||||||
|
f1,
|
||||||
|
self.q,
|
||||||
|
f2,
|
||||||
|
self.a,
|
||||||
|
f3,
|
||||||
|
self.auth,
|
||||||
|
f4,
|
||||||
|
self.ar,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
__str__ = __repr__
|
||||||
|
|
||||||
|
|
||||||
|
class DNSQuestion(object):
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, buffer):
|
||||||
|
try:
|
||||||
|
qname = buffer.decode_name()
|
||||||
|
qtype, qclass = buffer.unpack("!HH")
|
||||||
|
return cls(qname, qtype, qclass)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError(
|
||||||
|
"Error unpacking DNSQuestion [offset=%d]: %s" % (buffer.offset, e)
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(self, qname=None, qtype=1, qclass=1) -> None:
|
||||||
|
self.qname = qname
|
||||||
|
self.qtype = qtype
|
||||||
|
self.qclass = qclass
|
||||||
|
|
||||||
|
def set_qname(self, qname):
|
||||||
|
if isinstance(qname, DNSLabel):
|
||||||
|
self._qname = qname
|
||||||
|
else:
|
||||||
|
self._qname = DNSLabel(qname)
|
||||||
|
|
||||||
|
def get_qname(self):
|
||||||
|
return self._qname
|
||||||
|
|
||||||
|
qname = property(get_qname, set_qname)
|
||||||
|
|
||||||
|
def pack(self, buffer):
|
||||||
|
buffer.encode_name(self.qname)
|
||||||
|
buffer.pack("!HH", self.qtype, self.qclass)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return "<DNS Question: '%s' qtype=%s qclass=%s>" % (
|
||||||
|
self.qname,
|
||||||
|
QTYPE.get(self.qtype),
|
||||||
|
CLASS.get(self.qclass),
|
||||||
|
)
|
||||||
|
|
||||||
|
__str__ = __repr__
|
||||||
|
|
||||||
|
|
||||||
|
class RR(object):
|
||||||
|
rtype = H("rtype")
|
||||||
|
rclass = H("rclass")
|
||||||
|
ttl = I("ttl")
|
||||||
|
rdlength = H("rdlength")
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, buffer):
|
||||||
|
try:
|
||||||
|
rname = buffer.decode_name()
|
||||||
|
rtype, rclass, ttl, rdlength = buffer.unpack("!HHIH")
|
||||||
|
if rdlength:
|
||||||
|
rdata = RDMAP.get(QTYPE.get(rtype), RD).parse(buffer, rdlength)
|
||||||
|
else:
|
||||||
|
rdata = ""
|
||||||
|
return cls(rname, rtype, rclass, ttl, rdata)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError("Error unpacking RR [offset=%d]: %s" % (buffer.offset, e))
|
||||||
|
|
||||||
|
def __init__(self, rname=None, rtype=1, rclass=1, ttl=0, rdata=None) -> None:
|
||||||
|
self.rname = rname
|
||||||
|
self.rtype = rtype
|
||||||
|
self.rclass = rclass
|
||||||
|
self.ttl = ttl
|
||||||
|
self.rdata = rdata
|
||||||
|
|
||||||
|
def set_rname(self, rname):
|
||||||
|
if isinstance(rname, DNSLabel):
|
||||||
|
self._rname = rname
|
||||||
|
else:
|
||||||
|
self._rname = DNSLabel(rname)
|
||||||
|
|
||||||
|
def get_rname(self):
|
||||||
|
return self._rname
|
||||||
|
|
||||||
|
rname = property(get_rname, set_rname)
|
||||||
|
|
||||||
|
def pack(self, buffer):
|
||||||
|
buffer.encode_name(self.rname)
|
||||||
|
buffer.pack("!HHI", self.rtype, self.rclass, self.ttl)
|
||||||
|
rdlength_ptr = buffer.offset
|
||||||
|
buffer.pack("!H", 0)
|
||||||
|
start = buffer.offset
|
||||||
|
self.rdata.pack(buffer)
|
||||||
|
end = buffer.offset
|
||||||
|
buffer.update(rdlength_ptr, "!H", end - start)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return "<DNS RR: '%s' rtype=%s rclass=%s ttl=%d rdata='%s'>" % (
|
||||||
|
self.rname,
|
||||||
|
QTYPE.get(self.rtype),
|
||||||
|
CLASS.get(self.rclass),
|
||||||
|
self.ttl,
|
||||||
|
self.rdata,
|
||||||
|
)
|
||||||
|
|
||||||
|
__str__ = __repr__
|
||||||
|
|
||||||
|
|
||||||
|
class RD(object):
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, buffer, length):
|
||||||
|
try:
|
||||||
|
data = buffer.get(length)
|
||||||
|
return cls(data)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError("Error unpacking RD [offset=%d]: %s" % (buffer.offset, e))
|
||||||
|
|
||||||
|
def __init__(self, data=b"") -> None:
|
||||||
|
check_bytes("data", data)
|
||||||
|
self.data = bytes(data)
|
||||||
|
|
||||||
|
def pack(self, buffer):
|
||||||
|
buffer.append(self.data)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
if len(self.data) > 0:
|
||||||
|
return "\\# %d %s" % (
|
||||||
|
len(self.data),
|
||||||
|
binascii.hexlify(self.data).decode().upper(),
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
return "\\# 0"
|
||||||
|
|
||||||
|
attrs = ("data",)
|
||||||
|
|
||||||
|
|
||||||
|
def _force_bytes(x):
|
||||||
|
if isinstance(x, bytes):
|
||||||
|
return x
|
||||||
|
else:
|
||||||
|
return x.encode()
|
||||||
|
|
||||||
|
|
||||||
|
class TXT(RD):
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, buffer, length):
|
||||||
|
try:
|
||||||
|
data = list()
|
||||||
|
start_bo = buffer.offset
|
||||||
|
now_length = 0
|
||||||
|
while buffer.offset < start_bo + length:
|
||||||
|
(txtlength,) = buffer.unpack("!B")
|
||||||
|
|
||||||
|
if now_length + txtlength < length:
|
||||||
|
now_length += txtlength
|
||||||
|
data.append(buffer.get(txtlength))
|
||||||
|
else:
|
||||||
|
raise DNSError(
|
||||||
|
"Invalid TXT record: len(%d) > RD len(%d)" % (txtlength, length)
|
||||||
|
)
|
||||||
|
return cls(data)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError("Error unpacking TXT [offset=%d]: %s" % (buffer.offset, e))
|
||||||
|
|
||||||
|
def __init__(self, data) -> None:
|
||||||
|
if type(data) in (tuple, list):
|
||||||
|
self.data = [_force_bytes(x) for x in data]
|
||||||
|
else:
|
||||||
|
self.data = [_force_bytes(data)]
|
||||||
|
if any([len(x) > 255 for x in self.data]):
|
||||||
|
raise DNSError("TXT record too long: %s" % self.data)
|
||||||
|
|
||||||
|
def pack(self, buffer):
|
||||||
|
for ditem in self.data:
|
||||||
|
if len(ditem) > 255:
|
||||||
|
raise DNSError("TXT record too long: %s" % ditem)
|
||||||
|
buffer.pack("!B", len(ditem))
|
||||||
|
buffer.append(ditem)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return ",".join([repr(x) for x in self.data])
|
||||||
|
|
||||||
|
|
||||||
|
class A(RD):
|
||||||
|
|
||||||
|
data = IP4("data")
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, buffer, length):
|
||||||
|
try:
|
||||||
|
data = buffer.unpack("!BBBB")
|
||||||
|
return cls(data)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError("Error unpacking A [offset=%d]: %s" % (buffer.offset, e))
|
||||||
|
|
||||||
|
def __init__(self, data) -> None:
|
||||||
|
if type(data) in (tuple, list):
|
||||||
|
self.data = tuple(data)
|
||||||
|
else:
|
||||||
|
self.data = tuple(map(int, data.rstrip(".").split(".")))
|
||||||
|
|
||||||
|
def pack(self, buffer):
|
||||||
|
buffer.pack("!BBBB", *self.data)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return "%d.%d.%d.%d" % self.data
|
||||||
|
|
||||||
|
|
||||||
|
def _parse_ipv6(a):
|
||||||
|
l, _, r = a.partition("::")
|
||||||
|
l_groups = list(chain(*[divmod(int(x, 16), 256) for x in l.split(":") if x]))
|
||||||
|
r_groups = list(chain(*[divmod(int(x, 16), 256) for x in r.split(":") if x]))
|
||||||
|
zeros = [0] * (16 - len(l_groups) - len(r_groups))
|
||||||
|
return tuple(l_groups + zeros + r_groups)
|
||||||
|
|
||||||
|
|
||||||
|
def _format_ipv6(a):
|
||||||
|
left = []
|
||||||
|
right = []
|
||||||
|
current = "left"
|
||||||
|
for i in range(0, 16, 2):
|
||||||
|
group = (a[i] << 8) + a[i + 1]
|
||||||
|
if current == "left":
|
||||||
|
if group == 0 and i < 14:
|
||||||
|
if (a[i + 2] << 8) + a[i + 3] == 0:
|
||||||
|
current = "right"
|
||||||
|
else:
|
||||||
|
left.append("0")
|
||||||
|
else:
|
||||||
|
left.append("%x" % group)
|
||||||
|
else:
|
||||||
|
if group == 0 and len(right) == 0:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
right.append("%x" % group)
|
||||||
|
if len(left) < 8:
|
||||||
|
return ":".join(left) + "::" + ":".join(right)
|
||||||
|
else:
|
||||||
|
return ":".join(left)
|
||||||
|
|
||||||
|
|
||||||
|
class AAAA(RD):
|
||||||
|
data = IP6("data")
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, buffer, length):
|
||||||
|
try:
|
||||||
|
data = buffer.unpack("!16B")
|
||||||
|
return cls(data)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError("Error unpacking AAAA [offset=%d]: %s" % (buffer.offset, e))
|
||||||
|
|
||||||
|
def __init__(self, data) -> None:
|
||||||
|
if type(data) in (tuple, list):
|
||||||
|
self.data = tuple(data)
|
||||||
|
else:
|
||||||
|
self.data = _parse_ipv6(data)
|
||||||
|
|
||||||
|
def pack(self, buffer):
|
||||||
|
buffer.pack("!16B", *self.data)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return _format_ipv6(self.data)
|
||||||
|
|
||||||
|
|
||||||
|
class CNAME(RD):
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, buffer, length):
|
||||||
|
try:
|
||||||
|
label = buffer.decode_name()
|
||||||
|
return cls(label)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError("Error unpacking CNAME [offset=%d]: %s" % (buffer.offset, e))
|
||||||
|
|
||||||
|
def __init__(self, label=None) -> None:
|
||||||
|
self.label = label
|
||||||
|
|
||||||
|
def set_label(self, label):
|
||||||
|
if isinstance(label, DNSLabel):
|
||||||
|
self._label = label
|
||||||
|
else:
|
||||||
|
self._label = DNSLabel(label)
|
||||||
|
|
||||||
|
def get_label(self):
|
||||||
|
return self._label
|
||||||
|
|
||||||
|
label = property(get_label, set_label)
|
||||||
|
|
||||||
|
def pack(self, buffer):
|
||||||
|
buffer.encode_name(self.label)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return "%s" % (self.label)
|
||||||
|
|
||||||
|
attrs = ("label",)
|
||||||
|
|
||||||
|
|
||||||
|
class PTR(CNAME):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class SRV(RD):
|
||||||
|
priority = H("priority")
|
||||||
|
weight = H("weight")
|
||||||
|
port = H("port")
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, buffer, length):
|
||||||
|
try:
|
||||||
|
priority, weight, port = buffer.unpack("!HHH")
|
||||||
|
target = buffer.decode_name()
|
||||||
|
return cls(priority, weight, port, target)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError("Error unpacking SRV [offset=%d]: %s" % (buffer.offset, e))
|
||||||
|
|
||||||
|
def __init__(self, priority=0, weight=0, port=0, target=None) -> None:
|
||||||
|
self.priority = priority
|
||||||
|
self.weight = weight
|
||||||
|
self.port = port
|
||||||
|
self.target = target
|
||||||
|
|
||||||
|
def set_target(self, target):
|
||||||
|
if isinstance(target, DNSLabel):
|
||||||
|
self._target = target
|
||||||
|
else:
|
||||||
|
self._target = DNSLabel(target)
|
||||||
|
|
||||||
|
def get_target(self):
|
||||||
|
return self._target
|
||||||
|
|
||||||
|
target = property(get_target, set_target)
|
||||||
|
|
||||||
|
def pack(self, buffer):
|
||||||
|
buffer.pack("!HHH", self.priority, self.weight, self.port)
|
||||||
|
buffer.encode_name(self.target)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return "%d %d %d %s" % (self.priority, self.weight, self.port, self.target)
|
||||||
|
|
||||||
|
attrs = ("priority", "weight", "port", "target")
|
||||||
|
|
||||||
|
|
||||||
|
def decode_type_bitmap(type_bitmap):
|
||||||
|
rrlist = []
|
||||||
|
buf = DNSBuffer(type_bitmap)
|
||||||
|
while buf.remaining():
|
||||||
|
winnum, winlen = buf.unpack("BB")
|
||||||
|
bitmap = bytearray(buf.get(winlen))
|
||||||
|
for (pos, value) in enumerate(bitmap):
|
||||||
|
for i in range(8):
|
||||||
|
if (value << i) & 0x80:
|
||||||
|
bitpos = (256 * winnum) + (8 * pos) + i
|
||||||
|
rrlist.append(QTYPE[bitpos])
|
||||||
|
return rrlist
|
||||||
|
|
||||||
|
|
||||||
|
def encode_type_bitmap(rrlist):
|
||||||
|
rrlist = sorted([getattr(QTYPE, rr) for rr in rrlist])
|
||||||
|
buf = DNSBuffer()
|
||||||
|
curWindow = rrlist[0] // 256
|
||||||
|
bitmap = bytearray(32)
|
||||||
|
n = len(rrlist) - 1
|
||||||
|
for i, rr in enumerate(rrlist):
|
||||||
|
v = rr - curWindow * 256
|
||||||
|
bitmap[v // 8] |= 1 << (7 - v % 8)
|
||||||
|
|
||||||
|
if i == n or rrlist[i + 1] >= (curWindow + 1) * 256:
|
||||||
|
while bitmap[-1] == 0:
|
||||||
|
bitmap = bitmap[:-1]
|
||||||
|
buf.pack("BB", curWindow, len(bitmap))
|
||||||
|
buf.append(bitmap)
|
||||||
|
|
||||||
|
if i != n:
|
||||||
|
curWindow = rrlist[i + 1] // 256
|
||||||
|
bitmap = bytearray(32)
|
||||||
|
|
||||||
|
return buf.data
|
||||||
|
|
||||||
|
|
||||||
|
class NSEC(RD):
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, buffer, length):
|
||||||
|
try:
|
||||||
|
end = buffer.offset + length
|
||||||
|
name = buffer.decode_name()
|
||||||
|
rrlist = decode_type_bitmap(buffer.get(end - buffer.offset))
|
||||||
|
return cls(name, rrlist)
|
||||||
|
except (BufferError, BimapError) as e:
|
||||||
|
raise DNSError("Error unpacking NSEC [offset=%d]: %s" % (buffer.offset, e))
|
||||||
|
|
||||||
|
def __init__(self, label, rrlist) -> None:
|
||||||
|
self.label = label
|
||||||
|
self.rrlist = rrlist
|
||||||
|
|
||||||
|
def set_label(self, label):
|
||||||
|
if isinstance(label, DNSLabel):
|
||||||
|
self._label = label
|
||||||
|
else:
|
||||||
|
self._label = DNSLabel(label)
|
||||||
|
|
||||||
|
def get_label(self):
|
||||||
|
return self._label
|
||||||
|
|
||||||
|
label = property(get_label, set_label)
|
||||||
|
|
||||||
|
def pack(self, buffer):
|
||||||
|
buffer.encode_name(self.label)
|
||||||
|
buffer.append(encode_type_bitmap(self.rrlist))
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return "%s %s" % (self.label, " ".join(self.rrlist))
|
||||||
|
|
||||||
|
attrs = ("label", "rrlist")
|
||||||
|
|
||||||
|
|
||||||
|
RDMAP = {"A": A, "AAAA": AAAA, "TXT": TXT, "PTR": PTR, "SRV": SRV, "NSEC": NSEC}
|
||||||
154
copyparty/stolen/dnslib/label.py
Normal file
154
copyparty/stolen/dnslib/label.py
Normal file
@@ -0,0 +1,154 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
from __future__ import print_function
|
||||||
|
|
||||||
|
import re
|
||||||
|
|
||||||
|
from .bit import get_bits, set_bits
|
||||||
|
from .buffer import Buffer, BufferError
|
||||||
|
|
||||||
|
LDH = set(range(33, 127))
|
||||||
|
ESCAPE = re.compile(r"\\([0-9][0-9][0-9])")
|
||||||
|
|
||||||
|
|
||||||
|
class DNSLabelError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class DNSLabel(object):
|
||||||
|
def __init__(self, label):
|
||||||
|
if type(label) == DNSLabel:
|
||||||
|
self.label = label.label
|
||||||
|
elif type(label) in (list, tuple):
|
||||||
|
self.label = tuple(label)
|
||||||
|
else:
|
||||||
|
if not label or label in (b".", "."):
|
||||||
|
self.label = ()
|
||||||
|
elif type(label) is not bytes:
|
||||||
|
if type("") != type(b""):
|
||||||
|
|
||||||
|
label = ESCAPE.sub(lambda m: chr(int(m[1])), label)
|
||||||
|
self.label = tuple(label.encode("idna").rstrip(b".").split(b"."))
|
||||||
|
else:
|
||||||
|
if type("") == type(b""):
|
||||||
|
|
||||||
|
label = ESCAPE.sub(lambda m: chr(int(m.groups()[0])), label)
|
||||||
|
self.label = tuple(label.rstrip(b".").split(b"."))
|
||||||
|
|
||||||
|
def add(self, name):
|
||||||
|
new = DNSLabel(name)
|
||||||
|
if self.label:
|
||||||
|
new.label += self.label
|
||||||
|
return new
|
||||||
|
|
||||||
|
def idna(self):
|
||||||
|
return ".".join([s.decode("idna") for s in self.label]) + "."
|
||||||
|
|
||||||
|
def _decode(self, s):
|
||||||
|
if set(s).issubset(LDH):
|
||||||
|
|
||||||
|
return s.decode()
|
||||||
|
else:
|
||||||
|
|
||||||
|
return "".join([(chr(c) if (c in LDH) else "\\%03d" % c) for c in s])
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return ".".join([self._decode(bytearray(s)) for s in self.label]) + "."
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return "<DNSLabel: '%s'>" % str(self)
|
||||||
|
|
||||||
|
def __hash__(self):
|
||||||
|
return hash(tuple(map(lambda x: x.lower(), self.label)))
|
||||||
|
|
||||||
|
def __ne__(self, other):
|
||||||
|
return not self == other
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
if type(other) != DNSLabel:
|
||||||
|
return self.__eq__(DNSLabel(other))
|
||||||
|
else:
|
||||||
|
return [l.lower() for l in self.label] == [l.lower() for l in other.label]
|
||||||
|
|
||||||
|
def __len__(self):
|
||||||
|
return len(b".".join(self.label))
|
||||||
|
|
||||||
|
|
||||||
|
class DNSBuffer(Buffer):
|
||||||
|
def __init__(self, data=b""):
|
||||||
|
super(DNSBuffer, self).__init__(data)
|
||||||
|
self.names = {}
|
||||||
|
|
||||||
|
def decode_name(self, last=-1):
|
||||||
|
label = []
|
||||||
|
done = False
|
||||||
|
while not done:
|
||||||
|
(length,) = self.unpack("!B")
|
||||||
|
if get_bits(length, 6, 2) == 3:
|
||||||
|
|
||||||
|
self.offset -= 1
|
||||||
|
pointer = get_bits(self.unpack("!H")[0], 0, 14)
|
||||||
|
save = self.offset
|
||||||
|
if last == save:
|
||||||
|
raise BufferError(
|
||||||
|
"Recursive pointer in DNSLabel [offset=%d,pointer=%d,length=%d]"
|
||||||
|
% (self.offset, pointer, len(self.data))
|
||||||
|
)
|
||||||
|
if pointer < self.offset:
|
||||||
|
self.offset = pointer
|
||||||
|
else:
|
||||||
|
|
||||||
|
raise BufferError(
|
||||||
|
"Invalid pointer in DNSLabel [offset=%d,pointer=%d,length=%d]"
|
||||||
|
% (self.offset, pointer, len(self.data))
|
||||||
|
)
|
||||||
|
label.extend(self.decode_name(save).label)
|
||||||
|
self.offset = save
|
||||||
|
done = True
|
||||||
|
else:
|
||||||
|
if length > 0:
|
||||||
|
l = self.get(length)
|
||||||
|
try:
|
||||||
|
l.decode()
|
||||||
|
except UnicodeDecodeError:
|
||||||
|
raise BufferError("Invalid label <%s>" % l)
|
||||||
|
label.append(l)
|
||||||
|
else:
|
||||||
|
done = True
|
||||||
|
return DNSLabel(label)
|
||||||
|
|
||||||
|
def encode_name(self, name):
|
||||||
|
if not isinstance(name, DNSLabel):
|
||||||
|
name = DNSLabel(name)
|
||||||
|
if len(name) > 253:
|
||||||
|
raise DNSLabelError("Domain label too long: %r" % name)
|
||||||
|
name = list(name.label)
|
||||||
|
while name:
|
||||||
|
if tuple(name) in self.names:
|
||||||
|
|
||||||
|
pointer = self.names[tuple(name)]
|
||||||
|
pointer = set_bits(pointer, 3, 14, 2)
|
||||||
|
self.pack("!H", pointer)
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
self.names[tuple(name)] = self.offset
|
||||||
|
element = name.pop(0)
|
||||||
|
if len(element) > 63:
|
||||||
|
raise DNSLabelError("Label component too long: %r" % element)
|
||||||
|
self.pack("!B", len(element))
|
||||||
|
self.append(element)
|
||||||
|
self.append(b"\x00")
|
||||||
|
|
||||||
|
def encode_name_nocompress(self, name):
|
||||||
|
if not isinstance(name, DNSLabel):
|
||||||
|
name = DNSLabel(name)
|
||||||
|
if len(name) > 253:
|
||||||
|
raise DNSLabelError("Domain label too long: %r" % name)
|
||||||
|
name = list(name.label)
|
||||||
|
while name:
|
||||||
|
element = name.pop(0)
|
||||||
|
if len(element) > 63:
|
||||||
|
raise DNSLabelError("Label component too long: %r" % element)
|
||||||
|
self.pack("!B", len(element))
|
||||||
|
self.append(element)
|
||||||
|
self.append(b"\x00")
|
||||||
105
copyparty/stolen/dnslib/lex.py
Normal file
105
copyparty/stolen/dnslib/lex.py
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
from __future__ import print_function
|
||||||
|
|
||||||
|
import collections
|
||||||
|
|
||||||
|
try:
|
||||||
|
from StringIO import StringIO
|
||||||
|
except ImportError:
|
||||||
|
from io import StringIO
|
||||||
|
|
||||||
|
|
||||||
|
class Lexer(object):
|
||||||
|
|
||||||
|
escape_chars = "\\"
|
||||||
|
escape = {"n": "\n", "t": "\t", "r": "\r"}
|
||||||
|
|
||||||
|
def __init__(self, f, debug=False):
|
||||||
|
if hasattr(f, "read"):
|
||||||
|
self.f = f
|
||||||
|
elif type(f) == str:
|
||||||
|
self.f = StringIO(f)
|
||||||
|
elif type(f) == bytes:
|
||||||
|
self.f = StringIO(f.decode())
|
||||||
|
else:
|
||||||
|
raise ValueError("Invalid input")
|
||||||
|
self.debug = debug
|
||||||
|
self.q = collections.deque()
|
||||||
|
self.state = self.lexStart
|
||||||
|
self.escaped = False
|
||||||
|
self.eof = False
|
||||||
|
|
||||||
|
def __iter__(self):
|
||||||
|
return self.parse()
|
||||||
|
|
||||||
|
def next_token(self):
|
||||||
|
if self.debug:
|
||||||
|
print("STATE", self.state)
|
||||||
|
(tok, self.state) = self.state()
|
||||||
|
return tok
|
||||||
|
|
||||||
|
def parse(self):
|
||||||
|
while self.state is not None and not self.eof:
|
||||||
|
tok = self.next_token()
|
||||||
|
if tok:
|
||||||
|
yield tok
|
||||||
|
|
||||||
|
def read(self, n=1):
|
||||||
|
s = ""
|
||||||
|
while self.q and n > 0:
|
||||||
|
s += self.q.popleft()
|
||||||
|
n -= 1
|
||||||
|
s += self.f.read(n)
|
||||||
|
if s == "":
|
||||||
|
self.eof = True
|
||||||
|
if self.debug:
|
||||||
|
print("Read: >%s<" % repr(s))
|
||||||
|
return s
|
||||||
|
|
||||||
|
def peek(self, n=1):
|
||||||
|
s = ""
|
||||||
|
i = 0
|
||||||
|
while len(self.q) > i and n > 0:
|
||||||
|
s += self.q[i]
|
||||||
|
i += 1
|
||||||
|
n -= 1
|
||||||
|
r = self.f.read(n)
|
||||||
|
if n > 0 and r == "":
|
||||||
|
self.eof = True
|
||||||
|
self.q.extend(r)
|
||||||
|
if self.debug:
|
||||||
|
print("Peek : >%s<" % repr(s + r))
|
||||||
|
return s + r
|
||||||
|
|
||||||
|
def pushback(self, s):
|
||||||
|
p = collections.deque(s)
|
||||||
|
p.extend(self.q)
|
||||||
|
self.q = p
|
||||||
|
|
||||||
|
def readescaped(self):
|
||||||
|
c = self.read(1)
|
||||||
|
if c in self.escape_chars:
|
||||||
|
self.escaped = True
|
||||||
|
n = self.peek(3)
|
||||||
|
if n.isdigit():
|
||||||
|
n = self.read(3)
|
||||||
|
if self.debug:
|
||||||
|
print("Escape: >%s<" % n)
|
||||||
|
return chr(int(n, 8))
|
||||||
|
elif n[0] in "x":
|
||||||
|
x = self.read(3)
|
||||||
|
if self.debug:
|
||||||
|
print("Escape: >%s<" % x)
|
||||||
|
return chr(int(x[1:], 16))
|
||||||
|
else:
|
||||||
|
c = self.read(1)
|
||||||
|
if self.debug:
|
||||||
|
print("Escape: >%s<" % c)
|
||||||
|
return self.escape.get(c, c)
|
||||||
|
else:
|
||||||
|
self.escaped = False
|
||||||
|
return c
|
||||||
|
|
||||||
|
def lexStart(self):
|
||||||
|
return (None, None)
|
||||||
81
copyparty/stolen/dnslib/ranges.py
Normal file
81
copyparty/stolen/dnslib/ranges.py
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
|
if sys.version_info < (3,):
|
||||||
|
int_types = (
|
||||||
|
int,
|
||||||
|
long,
|
||||||
|
)
|
||||||
|
byte_types = (str, bytearray)
|
||||||
|
else:
|
||||||
|
int_types = (int,)
|
||||||
|
byte_types = (bytes, bytearray)
|
||||||
|
|
||||||
|
|
||||||
|
def check_instance(name, val, types):
|
||||||
|
if not isinstance(val, types):
|
||||||
|
raise ValueError(
|
||||||
|
"Attribute '%s' must be instance of %s [%s]" % (name, types, type(val))
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def check_bytes(name, val):
|
||||||
|
return check_instance(name, val, byte_types)
|
||||||
|
|
||||||
|
|
||||||
|
def range_property(attr, min, max):
|
||||||
|
def getter(obj):
|
||||||
|
return getattr(obj, "_%s" % attr)
|
||||||
|
|
||||||
|
def setter(obj, val):
|
||||||
|
if isinstance(val, int_types) and min <= val <= max:
|
||||||
|
setattr(obj, "_%s" % attr, val)
|
||||||
|
else:
|
||||||
|
raise ValueError(
|
||||||
|
"Attribute '%s' must be between %d-%d [%s]" % (attr, min, max, val)
|
||||||
|
)
|
||||||
|
|
||||||
|
return property(getter, setter)
|
||||||
|
|
||||||
|
|
||||||
|
def B(attr):
|
||||||
|
return range_property(attr, 0, 255)
|
||||||
|
|
||||||
|
|
||||||
|
def H(attr):
|
||||||
|
return range_property(attr, 0, 65535)
|
||||||
|
|
||||||
|
|
||||||
|
def I(attr):
|
||||||
|
return range_property(attr, 0, 4294967295)
|
||||||
|
|
||||||
|
|
||||||
|
def ntuple_range(attr, n, min, max):
|
||||||
|
f = lambda x: isinstance(x, int_types) and min <= x <= max
|
||||||
|
|
||||||
|
def getter(obj):
|
||||||
|
return getattr(obj, "_%s" % attr)
|
||||||
|
|
||||||
|
def setter(obj, val):
|
||||||
|
if len(val) != n:
|
||||||
|
raise ValueError(
|
||||||
|
"Attribute '%s' must be tuple with %d elements [%s]" % (attr, n, val)
|
||||||
|
)
|
||||||
|
if all(map(f, val)):
|
||||||
|
setattr(obj, "_%s" % attr, val)
|
||||||
|
else:
|
||||||
|
raise ValueError(
|
||||||
|
"Attribute '%s' elements must be between %d-%d [%s]"
|
||||||
|
% (attr, min, max, val)
|
||||||
|
)
|
||||||
|
|
||||||
|
return property(getter, setter)
|
||||||
|
|
||||||
|
|
||||||
|
def IP4(attr):
|
||||||
|
return ntuple_range(attr, 4, 0, 255)
|
||||||
|
|
||||||
|
|
||||||
|
def IP6(attr):
|
||||||
|
return ntuple_range(attr, 16, 0, 255)
|
||||||
5
copyparty/stolen/ifaddr/README.md
Normal file
5
copyparty/stolen/ifaddr/README.md
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
`ifaddr` with py2.7 support enabled by make-sfx.sh which strips py3 hints using strip_hints and removes the `^if True:` blocks
|
||||||
|
|
||||||
|
L: BSD-2-Clause
|
||||||
|
Copyright (c) 2014 Stefan C. Mueller
|
||||||
|
https://github.com/pydron/ifaddr/
|
||||||
21
copyparty/stolen/ifaddr/__init__.py
Normal file
21
copyparty/stolen/ifaddr/__init__.py
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
"""
|
||||||
|
L: BSD-2-Clause
|
||||||
|
Copyright (c) 2014 Stefan C. Mueller
|
||||||
|
https://github.com/pydron/ifaddr/tree/0.2.0
|
||||||
|
"""
|
||||||
|
|
||||||
|
import os
|
||||||
|
|
||||||
|
from ._shared import IP, Adapter
|
||||||
|
|
||||||
|
if os.name == "nt":
|
||||||
|
from ._win32 import get_adapters
|
||||||
|
elif os.name == "posix":
|
||||||
|
from ._posix import get_adapters
|
||||||
|
else:
|
||||||
|
raise RuntimeError("Unsupported Operating System: %s" % os.name)
|
||||||
|
|
||||||
|
__all__ = ["Adapter", "IP", "get_adapters"]
|
||||||
84
copyparty/stolen/ifaddr/_posix.py
Normal file
84
copyparty/stolen/ifaddr/_posix.py
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import collections
|
||||||
|
import ctypes.util
|
||||||
|
import os
|
||||||
|
import socket
|
||||||
|
|
||||||
|
import ipaddress
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Iterable, Optional
|
||||||
|
|
||||||
|
from . import _shared as shared
|
||||||
|
from ._shared import U
|
||||||
|
|
||||||
|
|
||||||
|
class ifaddrs(ctypes.Structure):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
ifaddrs._fields_ = [
|
||||||
|
("ifa_next", ctypes.POINTER(ifaddrs)),
|
||||||
|
("ifa_name", ctypes.c_char_p),
|
||||||
|
("ifa_flags", ctypes.c_uint),
|
||||||
|
("ifa_addr", ctypes.POINTER(shared.sockaddr)),
|
||||||
|
("ifa_netmask", ctypes.POINTER(shared.sockaddr)),
|
||||||
|
]
|
||||||
|
|
||||||
|
libc = ctypes.CDLL(ctypes.util.find_library("socket" if os.uname()[0] == "SunOS" else "c"), use_errno=True) # type: ignore
|
||||||
|
|
||||||
|
|
||||||
|
def get_adapters(include_unconfigured: bool = False) -> Iterable[shared.Adapter]:
|
||||||
|
|
||||||
|
addr0 = addr = ctypes.POINTER(ifaddrs)()
|
||||||
|
retval = libc.getifaddrs(ctypes.byref(addr))
|
||||||
|
if retval != 0:
|
||||||
|
eno = ctypes.get_errno()
|
||||||
|
raise OSError(eno, os.strerror(eno))
|
||||||
|
|
||||||
|
ips = collections.OrderedDict()
|
||||||
|
|
||||||
|
def add_ip(adapter_name: str, ip: Optional[shared.IP]) -> None:
|
||||||
|
if adapter_name not in ips:
|
||||||
|
index = None # type: Optional[int]
|
||||||
|
try:
|
||||||
|
# Mypy errors on this when the Windows CI runs:
|
||||||
|
# error: Module has no attribute "if_nametoindex"
|
||||||
|
index = socket.if_nametoindex(adapter_name) # type: ignore
|
||||||
|
except (OSError, AttributeError):
|
||||||
|
pass
|
||||||
|
ips[adapter_name] = shared.Adapter(
|
||||||
|
adapter_name, adapter_name, [], index=index
|
||||||
|
)
|
||||||
|
if ip is not None:
|
||||||
|
ips[adapter_name].ips.append(ip)
|
||||||
|
|
||||||
|
while addr:
|
||||||
|
name = addr[0].ifa_name.decode(encoding="UTF-8")
|
||||||
|
ip_addr = shared.sockaddr_to_ip(addr[0].ifa_addr)
|
||||||
|
if ip_addr:
|
||||||
|
if addr[0].ifa_netmask and not addr[0].ifa_netmask[0].sa_familiy:
|
||||||
|
addr[0].ifa_netmask[0].sa_familiy = addr[0].ifa_addr[0].sa_familiy
|
||||||
|
netmask = shared.sockaddr_to_ip(addr[0].ifa_netmask)
|
||||||
|
if isinstance(netmask, tuple):
|
||||||
|
netmaskStr = U(netmask[0])
|
||||||
|
prefixlen = shared.ipv6_prefixlength(ipaddress.IPv6Address(netmaskStr))
|
||||||
|
else:
|
||||||
|
if netmask is None:
|
||||||
|
t = "sockaddr_to_ip({}) returned None"
|
||||||
|
raise Exception(t.format(addr[0].ifa_netmask))
|
||||||
|
|
||||||
|
netmaskStr = U("0.0.0.0/" + netmask)
|
||||||
|
prefixlen = ipaddress.IPv4Network(netmaskStr).prefixlen
|
||||||
|
ip = shared.IP(ip_addr, prefixlen, name)
|
||||||
|
add_ip(name, ip)
|
||||||
|
else:
|
||||||
|
if include_unconfigured:
|
||||||
|
add_ip(name, None)
|
||||||
|
addr = addr[0].ifa_next
|
||||||
|
|
||||||
|
libc.freeifaddrs(addr0)
|
||||||
|
|
||||||
|
return ips.values()
|
||||||
203
copyparty/stolen/ifaddr/_shared.py
Normal file
203
copyparty/stolen/ifaddr/_shared.py
Normal file
@@ -0,0 +1,203 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import ctypes
|
||||||
|
import platform
|
||||||
|
import socket
|
||||||
|
import sys
|
||||||
|
|
||||||
|
import ipaddress
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Callable, List, Optional, Union
|
||||||
|
|
||||||
|
|
||||||
|
PY2 = sys.version_info < (3,)
|
||||||
|
if not PY2:
|
||||||
|
U: Callable[[str], str] = str
|
||||||
|
else:
|
||||||
|
U = unicode # noqa: F821 # pylint: disable=undefined-variable,self-assigning-variable
|
||||||
|
|
||||||
|
|
||||||
|
class Adapter(object):
|
||||||
|
"""
|
||||||
|
Represents a network interface device controller (NIC), such as a
|
||||||
|
network card. An adapter can have multiple IPs.
|
||||||
|
|
||||||
|
On Linux aliasing (multiple IPs per physical NIC) is implemented
|
||||||
|
by creating 'virtual' adapters, each represented by an instance
|
||||||
|
of this class. Each of those 'virtual' adapters can have both
|
||||||
|
a IPv4 and an IPv6 IP address.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self, name: str, nice_name: str, ips: List["IP"], index: Optional[int] = None
|
||||||
|
) -> None:
|
||||||
|
|
||||||
|
#: Unique name that identifies the adapter in the system.
|
||||||
|
#: On Linux this is of the form of `eth0` or `eth0:1`, on
|
||||||
|
#: Windows it is a UUID in string representation, such as
|
||||||
|
#: `{846EE342-7039-11DE-9D20-806E6F6E6963}`.
|
||||||
|
self.name = name
|
||||||
|
|
||||||
|
#: Human readable name of the adpater. On Linux this
|
||||||
|
#: is currently the same as :attr:`name`. On Windows
|
||||||
|
#: this is the name of the device.
|
||||||
|
self.nice_name = nice_name
|
||||||
|
|
||||||
|
#: List of :class:`ifaddr.IP` instances in the order they were
|
||||||
|
#: reported by the system.
|
||||||
|
self.ips = ips
|
||||||
|
|
||||||
|
#: Adapter index as used by some API (e.g. IPv6 multicast group join).
|
||||||
|
self.index = index
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
return "Adapter(name={name}, nice_name={nice_name}, ips={ips}, index={index})".format(
|
||||||
|
name=repr(self.name),
|
||||||
|
nice_name=repr(self.nice_name),
|
||||||
|
ips=repr(self.ips),
|
||||||
|
index=repr(self.index),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
if True:
|
||||||
|
# Type of an IPv4 address (a string in "xxx.xxx.xxx.xxx" format)
|
||||||
|
_IPv4Address = str
|
||||||
|
|
||||||
|
# Type of an IPv6 address (a three-tuple `(ip, flowinfo, scope_id)`)
|
||||||
|
_IPv6Address = tuple[str, int, int]
|
||||||
|
|
||||||
|
|
||||||
|
class IP(object):
|
||||||
|
"""
|
||||||
|
Represents an IP address of an adapter.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self, ip: Union[_IPv4Address, _IPv6Address], network_prefix: int, nice_name: str
|
||||||
|
) -> None:
|
||||||
|
|
||||||
|
#: IP address. For IPv4 addresses this is a string in
|
||||||
|
#: "xxx.xxx.xxx.xxx" format. For IPv6 addresses this
|
||||||
|
#: is a three-tuple `(ip, flowinfo, scope_id)`, where
|
||||||
|
#: `ip` is a string in the usual collon separated
|
||||||
|
#: hex format.
|
||||||
|
self.ip = ip
|
||||||
|
|
||||||
|
#: Number of bits of the IP that represent the
|
||||||
|
#: network. For a `255.255.255.0` netmask, this
|
||||||
|
#: number would be `24`.
|
||||||
|
self.network_prefix = network_prefix
|
||||||
|
|
||||||
|
#: Human readable name for this IP.
|
||||||
|
#: On Linux is this currently the same as the adapter name.
|
||||||
|
#: On Windows this is the name of the network connection
|
||||||
|
#: as configured in the system control panel.
|
||||||
|
self.nice_name = nice_name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_IPv4(self) -> bool:
|
||||||
|
"""
|
||||||
|
Returns `True` if this IP is an IPv4 address and `False`
|
||||||
|
if it is an IPv6 address.
|
||||||
|
"""
|
||||||
|
return not isinstance(self.ip, tuple)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_IPv6(self) -> bool:
|
||||||
|
"""
|
||||||
|
Returns `True` if this IP is an IPv6 address and `False`
|
||||||
|
if it is an IPv4 address.
|
||||||
|
"""
|
||||||
|
return isinstance(self.ip, tuple)
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
return "IP(ip={ip}, network_prefix={network_prefix}, nice_name={nice_name})".format(
|
||||||
|
ip=repr(self.ip),
|
||||||
|
network_prefix=repr(self.network_prefix),
|
||||||
|
nice_name=repr(self.nice_name),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
if platform.system() == "Darwin" or "BSD" in platform.system():
|
||||||
|
|
||||||
|
# BSD derived systems use marginally different structures
|
||||||
|
# than either Linux or Windows.
|
||||||
|
# I still keep it in `shared` since we can use
|
||||||
|
# both structures equally.
|
||||||
|
|
||||||
|
class sockaddr(ctypes.Structure):
|
||||||
|
_fields_ = [
|
||||||
|
("sa_len", ctypes.c_uint8),
|
||||||
|
("sa_familiy", ctypes.c_uint8),
|
||||||
|
("sa_data", ctypes.c_uint8 * 14),
|
||||||
|
]
|
||||||
|
|
||||||
|
class sockaddr_in(ctypes.Structure):
|
||||||
|
_fields_ = [
|
||||||
|
("sa_len", ctypes.c_uint8),
|
||||||
|
("sa_familiy", ctypes.c_uint8),
|
||||||
|
("sin_port", ctypes.c_uint16),
|
||||||
|
("sin_addr", ctypes.c_uint8 * 4),
|
||||||
|
("sin_zero", ctypes.c_uint8 * 8),
|
||||||
|
]
|
||||||
|
|
||||||
|
class sockaddr_in6(ctypes.Structure):
|
||||||
|
_fields_ = [
|
||||||
|
("sa_len", ctypes.c_uint8),
|
||||||
|
("sa_familiy", ctypes.c_uint8),
|
||||||
|
("sin6_port", ctypes.c_uint16),
|
||||||
|
("sin6_flowinfo", ctypes.c_uint32),
|
||||||
|
("sin6_addr", ctypes.c_uint8 * 16),
|
||||||
|
("sin6_scope_id", ctypes.c_uint32),
|
||||||
|
]
|
||||||
|
|
||||||
|
else:
|
||||||
|
|
||||||
|
class sockaddr(ctypes.Structure): # type: ignore
|
||||||
|
_fields_ = [("sa_familiy", ctypes.c_uint16), ("sa_data", ctypes.c_uint8 * 14)]
|
||||||
|
|
||||||
|
class sockaddr_in(ctypes.Structure): # type: ignore
|
||||||
|
_fields_ = [
|
||||||
|
("sin_familiy", ctypes.c_uint16),
|
||||||
|
("sin_port", ctypes.c_uint16),
|
||||||
|
("sin_addr", ctypes.c_uint8 * 4),
|
||||||
|
("sin_zero", ctypes.c_uint8 * 8),
|
||||||
|
]
|
||||||
|
|
||||||
|
class sockaddr_in6(ctypes.Structure): # type: ignore
|
||||||
|
_fields_ = [
|
||||||
|
("sin6_familiy", ctypes.c_uint16),
|
||||||
|
("sin6_port", ctypes.c_uint16),
|
||||||
|
("sin6_flowinfo", ctypes.c_uint32),
|
||||||
|
("sin6_addr", ctypes.c_uint8 * 16),
|
||||||
|
("sin6_scope_id", ctypes.c_uint32),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
def sockaddr_to_ip(
|
||||||
|
sockaddr_ptr: "ctypes.pointer[sockaddr]",
|
||||||
|
) -> Optional[Union[_IPv4Address, _IPv6Address]]:
|
||||||
|
if sockaddr_ptr:
|
||||||
|
if sockaddr_ptr[0].sa_familiy == socket.AF_INET:
|
||||||
|
ipv4 = ctypes.cast(sockaddr_ptr, ctypes.POINTER(sockaddr_in))
|
||||||
|
ippacked = bytes(bytearray(ipv4[0].sin_addr))
|
||||||
|
ip = U(ipaddress.ip_address(ippacked))
|
||||||
|
return ip
|
||||||
|
elif sockaddr_ptr[0].sa_familiy == socket.AF_INET6:
|
||||||
|
ipv6 = ctypes.cast(sockaddr_ptr, ctypes.POINTER(sockaddr_in6))
|
||||||
|
flowinfo = ipv6[0].sin6_flowinfo
|
||||||
|
ippacked = bytes(bytearray(ipv6[0].sin6_addr))
|
||||||
|
ip = U(ipaddress.ip_address(ippacked))
|
||||||
|
scope_id = ipv6[0].sin6_scope_id
|
||||||
|
return (ip, flowinfo, scope_id)
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
def ipv6_prefixlength(address: ipaddress.IPv6Address) -> int:
|
||||||
|
prefix_length = 0
|
||||||
|
for i in range(address.max_prefixlen):
|
||||||
|
if int(address) >> i & 1:
|
||||||
|
prefix_length = prefix_length + 1
|
||||||
|
return prefix_length
|
||||||
135
copyparty/stolen/ifaddr/_win32.py
Normal file
135
copyparty/stolen/ifaddr/_win32.py
Normal file
@@ -0,0 +1,135 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import ctypes
|
||||||
|
from ctypes import wintypes
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Iterable, List
|
||||||
|
|
||||||
|
from . import _shared as shared
|
||||||
|
|
||||||
|
NO_ERROR = 0
|
||||||
|
ERROR_BUFFER_OVERFLOW = 111
|
||||||
|
MAX_ADAPTER_NAME_LENGTH = 256
|
||||||
|
MAX_ADAPTER_DESCRIPTION_LENGTH = 128
|
||||||
|
MAX_ADAPTER_ADDRESS_LENGTH = 8
|
||||||
|
AF_UNSPEC = 0
|
||||||
|
|
||||||
|
|
||||||
|
class SOCKET_ADDRESS(ctypes.Structure):
|
||||||
|
_fields_ = [
|
||||||
|
("lpSockaddr", ctypes.POINTER(shared.sockaddr)),
|
||||||
|
("iSockaddrLength", wintypes.INT),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class IP_ADAPTER_UNICAST_ADDRESS(ctypes.Structure):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
IP_ADAPTER_UNICAST_ADDRESS._fields_ = [
|
||||||
|
("Length", wintypes.ULONG),
|
||||||
|
("Flags", wintypes.DWORD),
|
||||||
|
("Next", ctypes.POINTER(IP_ADAPTER_UNICAST_ADDRESS)),
|
||||||
|
("Address", SOCKET_ADDRESS),
|
||||||
|
("PrefixOrigin", ctypes.c_uint),
|
||||||
|
("SuffixOrigin", ctypes.c_uint),
|
||||||
|
("DadState", ctypes.c_uint),
|
||||||
|
("ValidLifetime", wintypes.ULONG),
|
||||||
|
("PreferredLifetime", wintypes.ULONG),
|
||||||
|
("LeaseLifetime", wintypes.ULONG),
|
||||||
|
("OnLinkPrefixLength", ctypes.c_uint8),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class IP_ADAPTER_ADDRESSES(ctypes.Structure):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
IP_ADAPTER_ADDRESSES._fields_ = [
|
||||||
|
("Length", wintypes.ULONG),
|
||||||
|
("IfIndex", wintypes.DWORD),
|
||||||
|
("Next", ctypes.POINTER(IP_ADAPTER_ADDRESSES)),
|
||||||
|
("AdapterName", ctypes.c_char_p),
|
||||||
|
("FirstUnicastAddress", ctypes.POINTER(IP_ADAPTER_UNICAST_ADDRESS)),
|
||||||
|
("FirstAnycastAddress", ctypes.c_void_p),
|
||||||
|
("FirstMulticastAddress", ctypes.c_void_p),
|
||||||
|
("FirstDnsServerAddress", ctypes.c_void_p),
|
||||||
|
("DnsSuffix", ctypes.c_wchar_p),
|
||||||
|
("Description", ctypes.c_wchar_p),
|
||||||
|
("FriendlyName", ctypes.c_wchar_p),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
iphlpapi = ctypes.windll.LoadLibrary("Iphlpapi") # type: ignore
|
||||||
|
|
||||||
|
|
||||||
|
def enumerate_interfaces_of_adapter(
|
||||||
|
nice_name: str, address: IP_ADAPTER_UNICAST_ADDRESS
|
||||||
|
) -> Iterable[shared.IP]:
|
||||||
|
|
||||||
|
# Iterate through linked list and fill list
|
||||||
|
addresses = [] # type: List[IP_ADAPTER_UNICAST_ADDRESS]
|
||||||
|
while True:
|
||||||
|
addresses.append(address)
|
||||||
|
if not address.Next:
|
||||||
|
break
|
||||||
|
address = address.Next[0]
|
||||||
|
|
||||||
|
for address in addresses:
|
||||||
|
ip = shared.sockaddr_to_ip(address.Address.lpSockaddr)
|
||||||
|
if ip is None:
|
||||||
|
t = "sockaddr_to_ip({}) returned None"
|
||||||
|
raise Exception(t.format(address.Address.lpSockaddr))
|
||||||
|
|
||||||
|
network_prefix = address.OnLinkPrefixLength
|
||||||
|
yield shared.IP(ip, network_prefix, nice_name)
|
||||||
|
|
||||||
|
|
||||||
|
def get_adapters(include_unconfigured: bool = False) -> Iterable[shared.Adapter]:
|
||||||
|
|
||||||
|
# Call GetAdaptersAddresses() with error and buffer size handling
|
||||||
|
|
||||||
|
addressbuffersize = wintypes.ULONG(15 * 1024)
|
||||||
|
retval = ERROR_BUFFER_OVERFLOW
|
||||||
|
while retval == ERROR_BUFFER_OVERFLOW:
|
||||||
|
addressbuffer = ctypes.create_string_buffer(addressbuffersize.value)
|
||||||
|
retval = iphlpapi.GetAdaptersAddresses(
|
||||||
|
wintypes.ULONG(AF_UNSPEC),
|
||||||
|
wintypes.ULONG(0),
|
||||||
|
None,
|
||||||
|
ctypes.byref(addressbuffer),
|
||||||
|
ctypes.byref(addressbuffersize),
|
||||||
|
)
|
||||||
|
if retval != NO_ERROR:
|
||||||
|
raise ctypes.WinError() # type: ignore
|
||||||
|
|
||||||
|
# Iterate through adapters fill array
|
||||||
|
address_infos = [] # type: List[IP_ADAPTER_ADDRESSES]
|
||||||
|
address_info = IP_ADAPTER_ADDRESSES.from_buffer(addressbuffer)
|
||||||
|
while True:
|
||||||
|
address_infos.append(address_info)
|
||||||
|
if not address_info.Next:
|
||||||
|
break
|
||||||
|
address_info = address_info.Next[0]
|
||||||
|
|
||||||
|
# Iterate through unicast addresses
|
||||||
|
result = [] # type: List[shared.Adapter]
|
||||||
|
for adapter_info in address_infos:
|
||||||
|
|
||||||
|
# We don't expect non-ascii characters here, so encoding shouldn't matter
|
||||||
|
name = adapter_info.AdapterName.decode()
|
||||||
|
nice_name = adapter_info.Description
|
||||||
|
index = adapter_info.IfIndex
|
||||||
|
|
||||||
|
if adapter_info.FirstUnicastAddress:
|
||||||
|
ips = enumerate_interfaces_of_adapter(
|
||||||
|
adapter_info.FriendlyName, adapter_info.FirstUnicastAddress[0]
|
||||||
|
)
|
||||||
|
ips = list(ips)
|
||||||
|
result.append(shared.Adapter(name, nice_name, ips, index=index))
|
||||||
|
elif include_unconfigured:
|
||||||
|
result.append(shared.Adapter(name, nice_name, [], index=index))
|
||||||
|
|
||||||
|
return result
|
||||||
591
copyparty/stolen/qrcodegen.py
Normal file
591
copyparty/stolen/qrcodegen.py
Normal file
@@ -0,0 +1,591 @@
|
|||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
# modified copy of Project Nayuki's qrcodegen (MIT-licensed);
|
||||||
|
# https://github.com/nayuki/QR-Code-generator/blob/daa3114/python/qrcodegen.py
|
||||||
|
# the original ^ is extremely well commented so refer to that for explanations
|
||||||
|
|
||||||
|
# hacks: binary-only, auto-ecc, render, py2-compat
|
||||||
|
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import collections
|
||||||
|
import itertools
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from collections.abc import Sequence
|
||||||
|
|
||||||
|
from typing import Callable, List, Optional, Tuple, Union
|
||||||
|
|
||||||
|
|
||||||
|
def num_char_count_bits(ver: int) -> int:
|
||||||
|
return 16 if (ver + 7) // 17 else 8
|
||||||
|
|
||||||
|
|
||||||
|
class Ecc(object):
|
||||||
|
ordinal: int
|
||||||
|
formatbits: int
|
||||||
|
|
||||||
|
def __init__(self, i: int, fb: int) -> None:
|
||||||
|
self.ordinal = i
|
||||||
|
self.formatbits = fb
|
||||||
|
|
||||||
|
LOW: "Ecc"
|
||||||
|
MEDIUM: "Ecc"
|
||||||
|
QUARTILE: "Ecc"
|
||||||
|
HIGH: "Ecc"
|
||||||
|
|
||||||
|
|
||||||
|
Ecc.LOW = Ecc(0, 1)
|
||||||
|
Ecc.MEDIUM = Ecc(1, 0)
|
||||||
|
Ecc.QUARTILE = Ecc(2, 3)
|
||||||
|
Ecc.HIGH = Ecc(3, 2)
|
||||||
|
|
||||||
|
|
||||||
|
class QrSegment(object):
|
||||||
|
@staticmethod
|
||||||
|
def make_seg(data: Union[bytes, Sequence[int]]) -> "QrSegment":
|
||||||
|
bb = _BitBuffer()
|
||||||
|
for b in data:
|
||||||
|
bb.append_bits(b, 8)
|
||||||
|
return QrSegment(len(data), bb)
|
||||||
|
|
||||||
|
numchars: int # num bytes, not the same as the data's bit length
|
||||||
|
bitdata: List[int] # The data bits of this segment
|
||||||
|
|
||||||
|
def __init__(self, numch: int, bitdata: Sequence[int]) -> None:
|
||||||
|
if numch < 0:
|
||||||
|
raise ValueError()
|
||||||
|
self.numchars = numch
|
||||||
|
self.bitdata = list(bitdata)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_total_bits(segs: Sequence["QrSegment"], ver: int) -> Optional[int]:
|
||||||
|
result = 0
|
||||||
|
for seg in segs:
|
||||||
|
ccbits: int = num_char_count_bits(ver)
|
||||||
|
if seg.numchars >= (1 << ccbits):
|
||||||
|
return None # segment length doesn't fit the field's bit width
|
||||||
|
result += 4 + ccbits + len(seg.bitdata)
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
class QrCode(object):
|
||||||
|
@staticmethod
|
||||||
|
def encode_binary(data: Union[bytes, Sequence[int]]) -> "QrCode":
|
||||||
|
return QrCode.encode_segments([QrSegment.make_seg(data)])
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def encode_segments(
|
||||||
|
segs: Sequence[QrSegment],
|
||||||
|
ecl: Ecc = Ecc.LOW,
|
||||||
|
minver: int = 2,
|
||||||
|
maxver: int = 40,
|
||||||
|
mask: int = -1,
|
||||||
|
) -> "QrCode":
|
||||||
|
for ver in range(minver, maxver + 1):
|
||||||
|
datacapacitybits: int = QrCode._get_num_data_codewords(ver, ecl) * 8
|
||||||
|
datausedbits: Optional[int] = QrSegment.get_total_bits(segs, ver)
|
||||||
|
if (datausedbits is not None) and (datausedbits <= datacapacitybits):
|
||||||
|
break
|
||||||
|
|
||||||
|
assert datausedbits
|
||||||
|
|
||||||
|
for newecl in (
|
||||||
|
Ecc.MEDIUM,
|
||||||
|
Ecc.QUARTILE,
|
||||||
|
Ecc.HIGH,
|
||||||
|
):
|
||||||
|
if datausedbits <= QrCode._get_num_data_codewords(ver, newecl) * 8:
|
||||||
|
ecl = newecl
|
||||||
|
|
||||||
|
# Concatenate all segments to create the data bit string
|
||||||
|
bb = _BitBuffer()
|
||||||
|
for seg in segs:
|
||||||
|
bb.append_bits(4, 4)
|
||||||
|
bb.append_bits(seg.numchars, num_char_count_bits(ver))
|
||||||
|
bb.extend(seg.bitdata)
|
||||||
|
assert len(bb) == datausedbits
|
||||||
|
|
||||||
|
# Add terminator and pad up to a byte if applicable
|
||||||
|
datacapacitybits = QrCode._get_num_data_codewords(ver, ecl) * 8
|
||||||
|
assert len(bb) <= datacapacitybits
|
||||||
|
bb.append_bits(0, min(4, datacapacitybits - len(bb)))
|
||||||
|
bb.append_bits(0, -len(bb) % 8)
|
||||||
|
assert len(bb) % 8 == 0
|
||||||
|
|
||||||
|
# Pad with alternating bytes until data capacity is reached
|
||||||
|
for padbyte in itertools.cycle((0xEC, 0x11)):
|
||||||
|
if len(bb) >= datacapacitybits:
|
||||||
|
break
|
||||||
|
bb.append_bits(padbyte, 8)
|
||||||
|
|
||||||
|
# Pack bits into bytes in big endian
|
||||||
|
datacodewords = bytearray([0] * (len(bb) // 8))
|
||||||
|
for (i, bit) in enumerate(bb):
|
||||||
|
datacodewords[i >> 3] |= bit << (7 - (i & 7))
|
||||||
|
|
||||||
|
return QrCode(ver, ecl, datacodewords, mask)
|
||||||
|
|
||||||
|
ver: int
|
||||||
|
size: int # w/h; 21..177 (ver * 4 + 17)
|
||||||
|
ecclvl: Ecc
|
||||||
|
mask: int # 0..7
|
||||||
|
modules: List[List[bool]]
|
||||||
|
unmaskable: List[List[bool]]
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
ver: int,
|
||||||
|
ecclvl: Ecc,
|
||||||
|
datacodewords: Union[bytes, Sequence[int]],
|
||||||
|
msk: int,
|
||||||
|
) -> None:
|
||||||
|
self.ver = ver
|
||||||
|
self.size = ver * 4 + 17
|
||||||
|
self.ecclvl = ecclvl
|
||||||
|
|
||||||
|
self.modules = [[False] * self.size for _ in range(self.size)]
|
||||||
|
self.unmaskable = [[False] * self.size for _ in range(self.size)]
|
||||||
|
|
||||||
|
# Compute ECC, draw modules
|
||||||
|
self._draw_function_patterns()
|
||||||
|
allcodewords: bytes = self._add_ecc_and_interleave(bytearray(datacodewords))
|
||||||
|
self._draw_codewords(allcodewords)
|
||||||
|
|
||||||
|
if msk == -1: # automask
|
||||||
|
minpenalty: int = 1 << 32
|
||||||
|
for i in range(8):
|
||||||
|
self._apply_mask(i)
|
||||||
|
self._draw_format_bits(i)
|
||||||
|
penalty = self._get_penalty_score()
|
||||||
|
if penalty < minpenalty:
|
||||||
|
msk = i
|
||||||
|
minpenalty = penalty
|
||||||
|
self._apply_mask(i) # xor/undo
|
||||||
|
|
||||||
|
assert 0 <= msk <= 7
|
||||||
|
self.mask = msk
|
||||||
|
self._apply_mask(msk) # Apply the final choice of mask
|
||||||
|
self._draw_format_bits(msk) # Overwrite old format bits
|
||||||
|
|
||||||
|
def render(self, zoom=1, pad=4) -> str:
|
||||||
|
tab = self.modules
|
||||||
|
sz = self.size
|
||||||
|
if sz % 2 and zoom == 1:
|
||||||
|
tab.append([False] * sz)
|
||||||
|
|
||||||
|
tab = [[False] * sz] * pad + tab + [[False] * sz] * pad
|
||||||
|
tab = [[False] * pad + x + [False] * pad for x in tab]
|
||||||
|
|
||||||
|
rows: list[str] = []
|
||||||
|
if zoom == 1:
|
||||||
|
for y in range(0, len(tab), 2):
|
||||||
|
row = ""
|
||||||
|
for x in range(len(tab[y])):
|
||||||
|
v = 2 if tab[y][x] else 0
|
||||||
|
v += 1 if tab[y + 1][x] else 0
|
||||||
|
row += " ▄▀█"[v]
|
||||||
|
rows.append(row)
|
||||||
|
else:
|
||||||
|
for tr in tab:
|
||||||
|
row = ""
|
||||||
|
for zb in tr:
|
||||||
|
row += " █"[int(zb)] * 2
|
||||||
|
rows.append(row)
|
||||||
|
|
||||||
|
return "\n".join(rows)
|
||||||
|
|
||||||
|
def _draw_function_patterns(self) -> None:
|
||||||
|
# Draw horizontal and vertical timing patterns
|
||||||
|
for i in range(self.size):
|
||||||
|
self._set_function_module(6, i, i % 2 == 0)
|
||||||
|
self._set_function_module(i, 6, i % 2 == 0)
|
||||||
|
|
||||||
|
# Draw 3 finder patterns (all corners except bottom right; overwrites some timing modules)
|
||||||
|
self._draw_finder_pattern(3, 3)
|
||||||
|
self._draw_finder_pattern(self.size - 4, 3)
|
||||||
|
self._draw_finder_pattern(3, self.size - 4)
|
||||||
|
|
||||||
|
# Draw numerous alignment patterns
|
||||||
|
alignpatpos: List[int] = self._get_alignment_pattern_positions()
|
||||||
|
numalign: int = len(alignpatpos)
|
||||||
|
skips: Sequence[Tuple[int, int]] = (
|
||||||
|
(0, 0),
|
||||||
|
(0, numalign - 1),
|
||||||
|
(numalign - 1, 0),
|
||||||
|
)
|
||||||
|
for i in range(numalign):
|
||||||
|
for j in range(numalign):
|
||||||
|
if (i, j) not in skips: # avoid finder corners
|
||||||
|
self._draw_alignment_pattern(alignpatpos[i], alignpatpos[j])
|
||||||
|
|
||||||
|
# draw config data with dummy mask value; ctor overwrites it
|
||||||
|
self._draw_format_bits(0)
|
||||||
|
self._draw_ver()
|
||||||
|
|
||||||
|
def _draw_format_bits(self, mask: int) -> None:
|
||||||
|
# Calculate error correction code and pack bits; ecclvl is uint2, mask is uint3
|
||||||
|
data: int = self.ecclvl.formatbits << 3 | mask
|
||||||
|
rem: int = data
|
||||||
|
for _ in range(10):
|
||||||
|
rem = (rem << 1) ^ ((rem >> 9) * 0x537)
|
||||||
|
bits: int = (data << 10 | rem) ^ 0x5412 # uint15
|
||||||
|
assert bits >> 15 == 0
|
||||||
|
|
||||||
|
# first copy
|
||||||
|
for i in range(0, 6):
|
||||||
|
self._set_function_module(8, i, _get_bit(bits, i))
|
||||||
|
self._set_function_module(8, 7, _get_bit(bits, 6))
|
||||||
|
self._set_function_module(8, 8, _get_bit(bits, 7))
|
||||||
|
self._set_function_module(7, 8, _get_bit(bits, 8))
|
||||||
|
for i in range(9, 15):
|
||||||
|
self._set_function_module(14 - i, 8, _get_bit(bits, i))
|
||||||
|
|
||||||
|
# second copy
|
||||||
|
for i in range(0, 8):
|
||||||
|
self._set_function_module(self.size - 1 - i, 8, _get_bit(bits, i))
|
||||||
|
for i in range(8, 15):
|
||||||
|
self._set_function_module(8, self.size - 15 + i, _get_bit(bits, i))
|
||||||
|
self._set_function_module(8, self.size - 8, True) # Always dark
|
||||||
|
|
||||||
|
def _draw_ver(self) -> None:
|
||||||
|
if self.ver < 7:
|
||||||
|
return
|
||||||
|
|
||||||
|
# Calculate error correction code and pack bits
|
||||||
|
rem: int = self.ver # ver is uint6, 7..40
|
||||||
|
for _ in range(12):
|
||||||
|
rem = (rem << 1) ^ ((rem >> 11) * 0x1F25)
|
||||||
|
bits: int = self.ver << 12 | rem # uint18
|
||||||
|
assert bits >> 18 == 0
|
||||||
|
|
||||||
|
# Draw two copies
|
||||||
|
for i in range(18):
|
||||||
|
bit: bool = _get_bit(bits, i)
|
||||||
|
a: int = self.size - 11 + i % 3
|
||||||
|
b: int = i // 3
|
||||||
|
self._set_function_module(a, b, bit)
|
||||||
|
self._set_function_module(b, a, bit)
|
||||||
|
|
||||||
|
def _draw_finder_pattern(self, x: int, y: int) -> None:
|
||||||
|
for dy in range(-4, 5):
|
||||||
|
for dx in range(-4, 5):
|
||||||
|
xx, yy = x + dx, y + dy
|
||||||
|
if (0 <= xx < self.size) and (0 <= yy < self.size):
|
||||||
|
# Chebyshev/infinity norm
|
||||||
|
self._set_function_module(
|
||||||
|
xx, yy, max(abs(dx), abs(dy)) not in (2, 4)
|
||||||
|
)
|
||||||
|
|
||||||
|
def _draw_alignment_pattern(self, x: int, y: int) -> None:
|
||||||
|
for dy in range(-2, 3):
|
||||||
|
for dx in range(-2, 3):
|
||||||
|
self._set_function_module(x + dx, y + dy, max(abs(dx), abs(dy)) != 1)
|
||||||
|
|
||||||
|
def _set_function_module(self, x: int, y: int, isdark: bool) -> None:
|
||||||
|
self.modules[y][x] = isdark
|
||||||
|
self.unmaskable[y][x] = True
|
||||||
|
|
||||||
|
def _add_ecc_and_interleave(self, data: bytearray) -> bytes:
|
||||||
|
ver: int = self.ver
|
||||||
|
assert len(data) == QrCode._get_num_data_codewords(ver, self.ecclvl)
|
||||||
|
|
||||||
|
# Calculate parameter numbers
|
||||||
|
numblocks: int = QrCode._NUM_ERROR_CORRECTION_BLOCKS[self.ecclvl.ordinal][ver]
|
||||||
|
blockecclen: int = QrCode._ECC_CODEWORDS_PER_BLOCK[self.ecclvl.ordinal][ver]
|
||||||
|
rawcodewords: int = QrCode._get_num_raw_data_modules(ver) // 8
|
||||||
|
numshortblocks: int = numblocks - rawcodewords % numblocks
|
||||||
|
shortblocklen: int = rawcodewords // numblocks
|
||||||
|
|
||||||
|
# Split data into blocks and append ECC to each block
|
||||||
|
blocks: List[bytes] = []
|
||||||
|
rsdiv: bytes = QrCode._reed_solomon_compute_divisor(blockecclen)
|
||||||
|
k: int = 0
|
||||||
|
for i in range(numblocks):
|
||||||
|
dat: bytearray = data[
|
||||||
|
k : k + shortblocklen - blockecclen + (0 if i < numshortblocks else 1)
|
||||||
|
]
|
||||||
|
k += len(dat)
|
||||||
|
ecc: bytes = QrCode._reed_solomon_compute_remainder(dat, rsdiv)
|
||||||
|
if i < numshortblocks:
|
||||||
|
dat.append(0)
|
||||||
|
blocks.append(dat + ecc)
|
||||||
|
assert k == len(data)
|
||||||
|
|
||||||
|
# Interleave (not concatenate) the bytes from every block into a single sequence
|
||||||
|
result = bytearray()
|
||||||
|
for i in range(len(blocks[0])):
|
||||||
|
for (j, blk) in enumerate(blocks):
|
||||||
|
# Skip the padding byte in short blocks
|
||||||
|
if (i != shortblocklen - blockecclen) or (j >= numshortblocks):
|
||||||
|
result.append(blk[i])
|
||||||
|
assert len(result) == rawcodewords
|
||||||
|
return result
|
||||||
|
|
||||||
|
def _draw_codewords(self, data: bytes) -> None:
|
||||||
|
assert len(data) == QrCode._get_num_raw_data_modules(self.ver) // 8
|
||||||
|
|
||||||
|
i: int = 0 # Bit index into the data
|
||||||
|
for right in range(self.size - 1, 0, -2):
|
||||||
|
# idx of right column in each column pair
|
||||||
|
if right <= 6:
|
||||||
|
right -= 1
|
||||||
|
for vert in range(self.size): # Vertical counter
|
||||||
|
for j in range(2):
|
||||||
|
x: int = right - j
|
||||||
|
upward: bool = (right + 1) & 2 == 0
|
||||||
|
y: int = (self.size - 1 - vert) if upward else vert
|
||||||
|
if (not self.unmaskable[y][x]) and (i < len(data) * 8):
|
||||||
|
self.modules[y][x] = _get_bit(data[i >> 3], 7 - (i & 7))
|
||||||
|
i += 1
|
||||||
|
# any remainder bits (0..7) were set 0/false/light by ctor
|
||||||
|
|
||||||
|
assert i == len(data) * 8
|
||||||
|
|
||||||
|
def _apply_mask(self, mask: int) -> None:
|
||||||
|
masker: Callable[[int, int], int] = QrCode._MASK_PATTERNS[mask]
|
||||||
|
for y in range(self.size):
|
||||||
|
for x in range(self.size):
|
||||||
|
self.modules[y][x] ^= (masker(x, y) == 0) and (
|
||||||
|
not self.unmaskable[y][x]
|
||||||
|
)
|
||||||
|
|
||||||
|
def _get_penalty_score(self) -> int:
|
||||||
|
result: int = 0
|
||||||
|
size: int = self.size
|
||||||
|
modules: List[List[bool]] = self.modules
|
||||||
|
|
||||||
|
# Adjacent modules in row having same color, and finder-like patterns
|
||||||
|
for y in range(size):
|
||||||
|
runcolor: bool = False
|
||||||
|
runx: int = 0
|
||||||
|
runhistory = collections.deque([0] * 7, 7)
|
||||||
|
for x in range(size):
|
||||||
|
if modules[y][x] == runcolor:
|
||||||
|
runx += 1
|
||||||
|
if runx == 5:
|
||||||
|
result += QrCode._PENALTY_N1
|
||||||
|
elif runx > 5:
|
||||||
|
result += 1
|
||||||
|
else:
|
||||||
|
self._finder_penalty_add_history(runx, runhistory)
|
||||||
|
if not runcolor:
|
||||||
|
result += (
|
||||||
|
self._finder_penalty_count_patterns(runhistory)
|
||||||
|
* QrCode._PENALTY_N3
|
||||||
|
)
|
||||||
|
runcolor = modules[y][x]
|
||||||
|
runx = 1
|
||||||
|
result += (
|
||||||
|
self._finder_penalty_terminate_and_count(runcolor, runx, runhistory)
|
||||||
|
* QrCode._PENALTY_N3
|
||||||
|
)
|
||||||
|
|
||||||
|
# Adjacent modules in column having same color, and finder-like patterns
|
||||||
|
for x in range(size):
|
||||||
|
runcolor = False
|
||||||
|
runy = 0
|
||||||
|
runhistory = collections.deque([0] * 7, 7)
|
||||||
|
for y in range(size):
|
||||||
|
if modules[y][x] == runcolor:
|
||||||
|
runy += 1
|
||||||
|
if runy == 5:
|
||||||
|
result += QrCode._PENALTY_N1
|
||||||
|
elif runy > 5:
|
||||||
|
result += 1
|
||||||
|
else:
|
||||||
|
self._finder_penalty_add_history(runy, runhistory)
|
||||||
|
if not runcolor:
|
||||||
|
result += (
|
||||||
|
self._finder_penalty_count_patterns(runhistory)
|
||||||
|
* QrCode._PENALTY_N3
|
||||||
|
)
|
||||||
|
runcolor = modules[y][x]
|
||||||
|
runy = 1
|
||||||
|
result += (
|
||||||
|
self._finder_penalty_terminate_and_count(runcolor, runy, runhistory)
|
||||||
|
* QrCode._PENALTY_N3
|
||||||
|
)
|
||||||
|
|
||||||
|
# 2*2 blocks of modules having same color
|
||||||
|
for y in range(size - 1):
|
||||||
|
for x in range(size - 1):
|
||||||
|
if (
|
||||||
|
modules[y][x]
|
||||||
|
== modules[y][x + 1]
|
||||||
|
== modules[y + 1][x]
|
||||||
|
== modules[y + 1][x + 1]
|
||||||
|
):
|
||||||
|
result += QrCode._PENALTY_N2
|
||||||
|
|
||||||
|
# Balance of dark and light modules
|
||||||
|
dark: int = sum((1 if cell else 0) for row in modules for cell in row)
|
||||||
|
total: int = size ** 2 # Note that size is odd, so dark/total != 1/2
|
||||||
|
|
||||||
|
# Compute the smallest integer k >= 0 such that (45-5k)% <= dark/total <= (55+5k)%
|
||||||
|
k: int = (abs(dark * 20 - total * 10) + total - 1) // total - 1
|
||||||
|
assert 0 <= k <= 9
|
||||||
|
result += k * QrCode._PENALTY_N4
|
||||||
|
assert 0 <= result <= 2568888
|
||||||
|
# ^ Non-tight upper bound based on default values of PENALTY_N1, ..., N4
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
def _get_alignment_pattern_positions(self) -> List[int]:
|
||||||
|
ver: int = self.ver
|
||||||
|
if ver == 1:
|
||||||
|
return []
|
||||||
|
|
||||||
|
numalign: int = ver // 7 + 2
|
||||||
|
step: int = (
|
||||||
|
26
|
||||||
|
if (ver == 32)
|
||||||
|
else (ver * 4 + numalign * 2 + 1) // (numalign * 2 - 2) * 2
|
||||||
|
)
|
||||||
|
result: List[int] = [
|
||||||
|
(self.size - 7 - i * step) for i in range(numalign - 1)
|
||||||
|
] + [6]
|
||||||
|
return list(reversed(result))
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _get_num_raw_data_modules(ver: int) -> int:
|
||||||
|
result: int = (16 * ver + 128) * ver + 64
|
||||||
|
if ver >= 2:
|
||||||
|
numalign: int = ver // 7 + 2
|
||||||
|
result -= (25 * numalign - 10) * numalign - 55
|
||||||
|
if ver >= 7:
|
||||||
|
result -= 36
|
||||||
|
assert 208 <= result <= 29648
|
||||||
|
return result
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _get_num_data_codewords(ver: int, ecl: Ecc) -> int:
|
||||||
|
return (
|
||||||
|
QrCode._get_num_raw_data_modules(ver) // 8
|
||||||
|
- QrCode._ECC_CODEWORDS_PER_BLOCK[ecl.ordinal][ver]
|
||||||
|
* QrCode._NUM_ERROR_CORRECTION_BLOCKS[ecl.ordinal][ver]
|
||||||
|
)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _reed_solomon_compute_divisor(degree: int) -> bytes:
|
||||||
|
if not (1 <= degree <= 255):
|
||||||
|
raise ValueError("Degree out of range")
|
||||||
|
|
||||||
|
# Polynomial coefficients are stored from highest to lowest power, excluding the leading term which is always 1.
|
||||||
|
# For example the polynomial x^3 + 255x^2 + 8x + 93 is stored as the uint8 array [255, 8, 93].
|
||||||
|
result = bytearray([0] * (degree - 1) + [1]) # start with monomial x^0
|
||||||
|
|
||||||
|
# Compute the product polynomial (x - r^0) * (x - r^1) * (x - r^2) * ... * (x - r^{degree-1}),
|
||||||
|
# and drop the highest monomial term which is always 1x^degree.
|
||||||
|
# Note that r = 0x02, which is a generator element of this field GF(2^8/0x11D).
|
||||||
|
root: int = 1
|
||||||
|
for _ in range(degree):
|
||||||
|
# Multiply the current product by (x - r^i)
|
||||||
|
for j in range(degree):
|
||||||
|
result[j] = QrCode._reed_solomon_multiply(result[j], root)
|
||||||
|
if j + 1 < degree:
|
||||||
|
result[j] ^= result[j + 1]
|
||||||
|
root = QrCode._reed_solomon_multiply(root, 0x02)
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _reed_solomon_compute_remainder(data: bytes, divisor: bytes) -> bytes:
|
||||||
|
result = bytearray([0] * len(divisor))
|
||||||
|
for b in data: # Polynomial division
|
||||||
|
factor: int = b ^ result.pop(0)
|
||||||
|
result.append(0)
|
||||||
|
for (i, coef) in enumerate(divisor):
|
||||||
|
result[i] ^= QrCode._reed_solomon_multiply(coef, factor)
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _reed_solomon_multiply(x: int, y: int) -> int:
|
||||||
|
if (x >> 8 != 0) or (y >> 8 != 0):
|
||||||
|
raise ValueError("Byte out of range")
|
||||||
|
z: int = 0 # Russian peasant multiplication
|
||||||
|
for i in reversed(range(8)):
|
||||||
|
z = (z << 1) ^ ((z >> 7) * 0x11D)
|
||||||
|
z ^= ((y >> i) & 1) * x
|
||||||
|
assert z >> 8 == 0
|
||||||
|
return z
|
||||||
|
|
||||||
|
def _finder_penalty_count_patterns(self, runhistory: collections.deque[int]) -> int:
|
||||||
|
n: int = runhistory[1]
|
||||||
|
assert n <= self.size * 3
|
||||||
|
core: bool = (
|
||||||
|
n > 0
|
||||||
|
and (runhistory[2] == runhistory[4] == runhistory[5] == n)
|
||||||
|
and runhistory[3] == n * 3
|
||||||
|
)
|
||||||
|
return (
|
||||||
|
1 if (core and runhistory[0] >= n * 4 and runhistory[6] >= n) else 0
|
||||||
|
) + (1 if (core and runhistory[6] >= n * 4 and runhistory[0] >= n) else 0)
|
||||||
|
|
||||||
|
def _finder_penalty_terminate_and_count(
|
||||||
|
self,
|
||||||
|
currentruncolor: bool,
|
||||||
|
currentrunlength: int,
|
||||||
|
runhistory: collections.deque[int],
|
||||||
|
) -> int:
|
||||||
|
if currentruncolor: # Terminate dark run
|
||||||
|
self._finder_penalty_add_history(currentrunlength, runhistory)
|
||||||
|
currentrunlength = 0
|
||||||
|
currentrunlength += self.size # Add light border to final run
|
||||||
|
self._finder_penalty_add_history(currentrunlength, runhistory)
|
||||||
|
return self._finder_penalty_count_patterns(runhistory)
|
||||||
|
|
||||||
|
def _finder_penalty_add_history(
|
||||||
|
self, currentrunlength: int, runhistory: collections.deque[int]
|
||||||
|
) -> None:
|
||||||
|
if runhistory[0] == 0:
|
||||||
|
currentrunlength += self.size # Add light border to initial run
|
||||||
|
|
||||||
|
runhistory.appendleft(currentrunlength)
|
||||||
|
|
||||||
|
_PENALTY_N1: int = 3
|
||||||
|
_PENALTY_N2: int = 3
|
||||||
|
_PENALTY_N3: int = 40
|
||||||
|
_PENALTY_N4: int = 10
|
||||||
|
|
||||||
|
# fmt: off
|
||||||
|
_ECC_CODEWORDS_PER_BLOCK: Sequence[Sequence[int]] = (
|
||||||
|
(-1, 7, 10, 15, 20, 26, 18, 20, 24, 30, 18, 20, 24, 26, 30, 22, 24, 28, 30, 28, 28, 28, 28, 30, 30, 26, 28, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30), # noqa: E241 # L
|
||||||
|
(-1, 10, 16, 26, 18, 24, 16, 18, 22, 22, 26, 30, 22, 22, 24, 24, 28, 28, 26, 26, 26, 26, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28), # noqa: E241 # M
|
||||||
|
(-1, 13, 22, 18, 26, 18, 24, 18, 22, 20, 24, 28, 26, 24, 20, 30, 24, 28, 28, 26, 30, 28, 30, 30, 30, 30, 28, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30), # noqa: E241 # Q
|
||||||
|
(-1, 17, 28, 22, 16, 22, 28, 26, 26, 24, 28, 24, 28, 22, 24, 24, 30, 28, 28, 26, 28, 30, 24, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30)) # noqa: E241 # H
|
||||||
|
|
||||||
|
_NUM_ERROR_CORRECTION_BLOCKS: Sequence[Sequence[int]] = (
|
||||||
|
(-1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 4, 4, 4, 4, 4, 6, 6, 6, 6, 7, 8, 8, 9, 9, 10, 12, 12, 12, 13, 14, 15, 16, 17, 18, 19, 19, 20, 21, 22, 24, 25), # noqa: E241 # L
|
||||||
|
(-1, 1, 1, 1, 2, 2, 4, 4, 4, 5, 5, 5, 8, 9, 9, 10, 10, 11, 13, 14, 16, 17, 17, 18, 20, 21, 23, 25, 26, 28, 29, 31, 33, 35, 37, 38, 40, 43, 45, 47, 49), # noqa: E241 # M
|
||||||
|
(-1, 1, 1, 2, 2, 4, 4, 6, 6, 8, 8, 8, 10, 12, 16, 12, 17, 16, 18, 21, 20, 23, 23, 25, 27, 29, 34, 34, 35, 38, 40, 43, 45, 48, 51, 53, 56, 59, 62, 65, 68), # noqa: E241 # Q
|
||||||
|
(-1, 1, 1, 2, 4, 4, 4, 5, 6, 8, 8, 11, 11, 16, 16, 18, 16, 19, 21, 25, 25, 25, 34, 30, 32, 35, 37, 40, 42, 45, 48, 51, 54, 57, 60, 63, 66, 70, 74, 77, 81)) # noqa: E241 # H
|
||||||
|
# fmt: on
|
||||||
|
|
||||||
|
_MASK_PATTERNS: Sequence[Callable[[int, int], int]] = (
|
||||||
|
(lambda x, y: (x + y) % 2),
|
||||||
|
(lambda x, y: y % 2),
|
||||||
|
(lambda x, y: x % 3),
|
||||||
|
(lambda x, y: (x + y) % 3),
|
||||||
|
(lambda x, y: (x // 3 + y // 2) % 2),
|
||||||
|
(lambda x, y: x * y % 2 + x * y % 3),
|
||||||
|
(lambda x, y: (x * y % 2 + x * y % 3) % 2),
|
||||||
|
(lambda x, y: ((x + y) % 2 + x * y % 3) % 2),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class _BitBuffer(list): # type: ignore
|
||||||
|
def append_bits(self, val: int, n: int) -> None:
|
||||||
|
if (n < 0) or (val >> n != 0):
|
||||||
|
raise ValueError("Value out of range")
|
||||||
|
|
||||||
|
self.extend(((val >> i) & 1) for i in reversed(range(n)))
|
||||||
|
|
||||||
|
|
||||||
|
def _get_bit(x: int, i: int) -> bool:
|
||||||
|
return (x >> i) & 1 != 0
|
||||||
|
|
||||||
|
|
||||||
|
class DataTooLongError(ValueError):
|
||||||
|
pass
|
||||||
@@ -12,27 +12,16 @@ Original source: misc/python/surrogateescape.py in https://bitbucket.org/haypo/m
|
|||||||
|
|
||||||
# This code is released under the Python license and the BSD 2-clause license
|
# This code is released under the Python license and the BSD 2-clause license
|
||||||
|
|
||||||
import platform
|
|
||||||
import codecs
|
import codecs
|
||||||
|
import platform
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
PY3 = sys.version_info[0] > 2
|
PY3 = sys.version_info > (3,)
|
||||||
WINDOWS = platform.system() == "Windows"
|
WINDOWS = platform.system() == "Windows"
|
||||||
FS_ERRORS = "surrogateescape"
|
FS_ERRORS = "surrogateescape"
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
def u(text):
|
from typing import Any
|
||||||
if PY3:
|
|
||||||
return text
|
|
||||||
else:
|
|
||||||
return text.decode("unicode_escape")
|
|
||||||
|
|
||||||
|
|
||||||
def b(data):
|
|
||||||
if PY3:
|
|
||||||
return data.encode("latin1")
|
|
||||||
else:
|
|
||||||
return data
|
|
||||||
|
|
||||||
|
|
||||||
if PY3:
|
if PY3:
|
||||||
@@ -43,7 +32,7 @@ else:
|
|||||||
bytes_chr = chr
|
bytes_chr = chr
|
||||||
|
|
||||||
|
|
||||||
def surrogateescape_handler(exc):
|
def surrogateescape_handler(exc: Any) -> tuple[str, int]:
|
||||||
"""
|
"""
|
||||||
Pure Python implementation of the PEP 383: the "surrogateescape" error
|
Pure Python implementation of the PEP 383: the "surrogateescape" error
|
||||||
handler of Python 3. Undecodable bytes will be replaced by a Unicode
|
handler of Python 3. Undecodable bytes will be replaced by a Unicode
|
||||||
@@ -74,7 +63,7 @@ class NotASurrogateError(Exception):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
def replace_surrogate_encode(mystring):
|
def replace_surrogate_encode(mystring: str) -> str:
|
||||||
"""
|
"""
|
||||||
Returns a (unicode) string, not the more logical bytes, because the codecs
|
Returns a (unicode) string, not the more logical bytes, because the codecs
|
||||||
register_error functionality expects this.
|
register_error functionality expects this.
|
||||||
@@ -100,7 +89,7 @@ def replace_surrogate_encode(mystring):
|
|||||||
return str().join(decoded)
|
return str().join(decoded)
|
||||||
|
|
||||||
|
|
||||||
def replace_surrogate_decode(mybytes):
|
def replace_surrogate_decode(mybytes: bytes) -> str:
|
||||||
"""
|
"""
|
||||||
Returns a (unicode) string
|
Returns a (unicode) string
|
||||||
"""
|
"""
|
||||||
@@ -121,7 +110,7 @@ def replace_surrogate_decode(mybytes):
|
|||||||
return str().join(decoded)
|
return str().join(decoded)
|
||||||
|
|
||||||
|
|
||||||
def encodefilename(fn):
|
def encodefilename(fn: str) -> bytes:
|
||||||
if FS_ENCODING == "ascii":
|
if FS_ENCODING == "ascii":
|
||||||
# ASCII encoder of Python 2 expects that the error handler returns a
|
# ASCII encoder of Python 2 expects that the error handler returns a
|
||||||
# Unicode string encodable to ASCII, whereas our surrogateescape error
|
# Unicode string encodable to ASCII, whereas our surrogateescape error
|
||||||
@@ -161,14 +150,11 @@ def encodefilename(fn):
|
|||||||
return fn.encode(FS_ENCODING, FS_ERRORS)
|
return fn.encode(FS_ENCODING, FS_ERRORS)
|
||||||
|
|
||||||
|
|
||||||
def decodefilename(fn):
|
def decodefilename(fn: bytes) -> str:
|
||||||
return fn.decode(FS_ENCODING, FS_ERRORS)
|
return fn.decode(FS_ENCODING, FS_ERRORS)
|
||||||
|
|
||||||
|
|
||||||
FS_ENCODING = sys.getfilesystemencoding()
|
FS_ENCODING = sys.getfilesystemencoding()
|
||||||
# FS_ENCODING = "ascii"; fn = b("[abc\xff]"); encoded = u("[abc\udcff]")
|
|
||||||
# FS_ENCODING = 'cp932'; fn = b('[abc\x81\x00]'); encoded = u('[abc\udc81\x00]')
|
|
||||||
# FS_ENCODING = 'UTF-8'; fn = b('[abc\xff]'); encoded = u('[abc\udcff]')
|
|
||||||
|
|
||||||
|
|
||||||
if WINDOWS and not PY3:
|
if WINDOWS and not PY3:
|
||||||
@@ -181,7 +167,7 @@ if WINDOWS and not PY3:
|
|||||||
FS_ENCODING = codecs.lookup(FS_ENCODING).name
|
FS_ENCODING = codecs.lookup(FS_ENCODING).name
|
||||||
|
|
||||||
|
|
||||||
def register_surrogateescape():
|
def register_surrogateescape() -> None:
|
||||||
"""
|
"""
|
||||||
Registers the surrogateescape error handler on Python 2 (only)
|
Registers the surrogateescape error handler on Python 2 (only)
|
||||||
"""
|
"""
|
||||||
|
|||||||
@@ -1,14 +1,32 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import time
|
|
||||||
import tempfile
|
import tempfile
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
|
|
||||||
from .bos import bos
|
from .bos import bos
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from typing import Any, Generator, Optional
|
||||||
|
|
||||||
def errdesc(errors):
|
from .util import NamedLogger
|
||||||
|
|
||||||
|
|
||||||
|
class StreamArc(object):
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
log: "NamedLogger",
|
||||||
|
fgen: Generator[dict[str, Any], None, None],
|
||||||
|
**kwargs: Any
|
||||||
|
):
|
||||||
|
self.log = log
|
||||||
|
self.fgen = fgen
|
||||||
|
|
||||||
|
def gen(self) -> Generator[Optional[bytes], None, None]:
|
||||||
|
raise Exception("override me")
|
||||||
|
|
||||||
|
|
||||||
|
def errdesc(errors: list[tuple[str, str]]) -> tuple[dict[str, Any], list[str]]:
|
||||||
report = ["copyparty failed to add the following files to the archive:", ""]
|
report = ["copyparty failed to add the following files to the archive:", ""]
|
||||||
|
|
||||||
for fn, err in errors:
|
for fn, err in errors:
|
||||||
|
|||||||
@@ -1,56 +1,142 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import argparse
|
||||||
|
import base64
|
||||||
|
import calendar
|
||||||
|
import errno
|
||||||
|
import gzip
|
||||||
|
import logging
|
||||||
import os
|
import os
|
||||||
import sys
|
import re
|
||||||
import time
|
|
||||||
import shlex
|
import shlex
|
||||||
import string
|
|
||||||
import signal
|
import signal
|
||||||
import socket
|
import socket
|
||||||
|
import string
|
||||||
|
import sys
|
||||||
import threading
|
import threading
|
||||||
|
import time
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
import calendar
|
|
||||||
|
|
||||||
from .__init__ import E, PY2, WINDOWS, ANYWIN, MACOS, VT100, unicode
|
# from inspect import currentframe
|
||||||
from .util import mp, start_log_thrs, start_stackmon, min_ex, ansi_re
|
# print(currentframe().f_lineno)
|
||||||
|
|
||||||
|
|
||||||
|
if True: # pylint: disable=using-constant-test
|
||||||
|
from types import FrameType
|
||||||
|
|
||||||
|
import typing
|
||||||
|
from typing import Any, Optional, Union
|
||||||
|
|
||||||
|
from .__init__ import ANYWIN, EXE, MACOS, TYPE_CHECKING, VT100, EnvParams, unicode
|
||||||
from .authsrv import AuthSrv
|
from .authsrv import AuthSrv
|
||||||
from .tcpsrv import TcpSrv
|
|
||||||
from .up2k import Up2k
|
|
||||||
from .th_srv import ThumbSrv, HAVE_PIL, HAVE_WEBP
|
|
||||||
from .mtag import HAVE_FFMPEG, HAVE_FFPROBE
|
from .mtag import HAVE_FFMPEG, HAVE_FFPROBE
|
||||||
|
from .tcpsrv import TcpSrv
|
||||||
|
from .th_srv import HAVE_PIL, HAVE_VIPS, HAVE_WEBP, ThumbSrv
|
||||||
|
from .up2k import Up2k
|
||||||
|
from .util import (
|
||||||
|
FFMPEG_URL,
|
||||||
|
VERSIONS,
|
||||||
|
Daemon,
|
||||||
|
Garda,
|
||||||
|
HLog,
|
||||||
|
HMaccas,
|
||||||
|
alltrace,
|
||||||
|
ansi_re,
|
||||||
|
min_ex,
|
||||||
|
mp,
|
||||||
|
pybin,
|
||||||
|
start_log_thrs,
|
||||||
|
start_stackmon,
|
||||||
|
)
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
try:
|
||||||
|
from .mdns import MDNS
|
||||||
|
from .ssdp import SSDPd
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
class SvcHub(object):
|
class SvcHub(object):
|
||||||
"""
|
"""
|
||||||
Hosts all services which cannot be parallelized due to reliance on monolithic resources.
|
Hosts all services which cannot be parallelized due to reliance on monolithic resources.
|
||||||
Creates a Broker which does most of the heavy stuff; hosted services can use this to perform work:
|
Creates a Broker which does most of the heavy stuff; hosted services can use this to perform work:
|
||||||
hub.broker.put(want_reply, destination, args_list).
|
hub.broker.<say|ask>(destination, args_list).
|
||||||
|
|
||||||
Either BrokerThr (plain threads) or BrokerMP (multiprocessing) is used depending on configuration.
|
Either BrokerThr (plain threads) or BrokerMP (multiprocessing) is used depending on configuration.
|
||||||
Nothing is returned synchronously; if you want any value returned from the call,
|
Nothing is returned synchronously; if you want any value returned from the call,
|
||||||
put() can return a queue (if want_reply=True) which has a blocking get() with the response.
|
put() can return a queue (if want_reply=True) which has a blocking get() with the response.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, args, argv, printed):
|
def __init__(
|
||||||
|
self,
|
||||||
|
args: argparse.Namespace,
|
||||||
|
dargs: argparse.Namespace,
|
||||||
|
argv: list[str],
|
||||||
|
printed: str,
|
||||||
|
) -> None:
|
||||||
self.args = args
|
self.args = args
|
||||||
|
self.dargs = dargs
|
||||||
self.argv = argv
|
self.argv = argv
|
||||||
self.logf = None
|
self.E: EnvParams = args.E
|
||||||
|
self.logf: Optional[typing.TextIO] = None
|
||||||
|
self.logf_base_fn = ""
|
||||||
self.stop_req = False
|
self.stop_req = False
|
||||||
self.reload_req = False
|
|
||||||
self.stopping = False
|
self.stopping = False
|
||||||
|
self.stopped = False
|
||||||
|
self.reload_req = False
|
||||||
self.reloading = False
|
self.reloading = False
|
||||||
self.stop_cond = threading.Condition()
|
self.stop_cond = threading.Condition()
|
||||||
|
self.nsigs = 3
|
||||||
self.retcode = 0
|
self.retcode = 0
|
||||||
self.httpsrv_up = 0
|
self.httpsrv_up = 0
|
||||||
|
|
||||||
self.log_mutex = threading.Lock()
|
self.log_mutex = threading.Lock()
|
||||||
self.next_day = 0
|
self.next_day = 0
|
||||||
|
self.tstack = 0.0
|
||||||
|
|
||||||
|
self.iphash = HMaccas(os.path.join(self.E.cfg, "iphash"), 8)
|
||||||
|
|
||||||
|
# for non-http clients (ftp)
|
||||||
|
self.bans: dict[str, int] = {}
|
||||||
|
self.gpwd = Garda(self.args.ban_pw)
|
||||||
|
self.g404 = Garda(self.args.ban_404)
|
||||||
|
|
||||||
|
if args.sss or args.s >= 3:
|
||||||
|
args.ss = True
|
||||||
|
args.no_dav = True
|
||||||
|
args.no_logues = True
|
||||||
|
args.no_readme = True
|
||||||
|
args.lo = args.lo or "cpp-%Y-%m%d-%H%M%S.txt.xz"
|
||||||
|
args.ls = args.ls or "**,*,ln,p,r"
|
||||||
|
|
||||||
|
if args.ss or args.s >= 2:
|
||||||
|
args.s = True
|
||||||
|
args.unpost = 0
|
||||||
|
args.no_del = True
|
||||||
|
args.no_mv = True
|
||||||
|
args.hardlink = True
|
||||||
|
args.vague_403 = True
|
||||||
|
args.ban_404 = "50,60,1440"
|
||||||
|
args.nih = True
|
||||||
|
|
||||||
|
if args.s:
|
||||||
|
args.dotpart = True
|
||||||
|
args.no_thumb = True
|
||||||
|
args.no_mtag_ff = True
|
||||||
|
args.no_robots = True
|
||||||
|
args.force_js = True
|
||||||
|
|
||||||
self.log = self._log_disabled if args.q else self._log_enabled
|
self.log = self._log_disabled if args.q else self._log_enabled
|
||||||
if args.lo:
|
if args.lo:
|
||||||
self._setup_logfile(printed)
|
self._setup_logfile(printed)
|
||||||
|
|
||||||
|
lg = logging.getLogger()
|
||||||
|
lh = HLog(self.log)
|
||||||
|
lg.handlers = [lh]
|
||||||
|
lg.setLevel(logging.DEBUG)
|
||||||
|
|
||||||
if args.stackmon:
|
if args.stackmon:
|
||||||
start_stackmon(args.stackmon, 0)
|
start_stackmon(args.stackmon, 0)
|
||||||
|
|
||||||
@@ -59,39 +145,76 @@ class SvcHub(object):
|
|||||||
|
|
||||||
if not args.use_fpool and args.j != 1:
|
if not args.use_fpool and args.j != 1:
|
||||||
args.no_fpool = True
|
args.no_fpool = True
|
||||||
m = "multithreading enabled with -j {}, so disabling fpool -- this can reduce upload performance on some filesystems"
|
t = "multithreading enabled with -j {}, so disabling fpool -- this can reduce upload performance on some filesystems"
|
||||||
self.log("root", m.format(args.j))
|
self.log("root", t.format(args.j))
|
||||||
|
|
||||||
if not args.no_fpool and args.j != 1:
|
if not args.no_fpool and args.j != 1:
|
||||||
m = "WARNING: --use-fpool combined with multithreading is untested and can probably cause undefined behavior"
|
t = "WARNING: --use-fpool combined with multithreading is untested and can probably cause undefined behavior"
|
||||||
if ANYWIN:
|
if ANYWIN:
|
||||||
m = 'windows cannot do multithreading without --no-fpool, so enabling that -- note that upload performance will suffer if you have microsoft defender "real-time protection" enabled, so you probably want to use -j 1 instead'
|
t = 'windows cannot do multithreading without --no-fpool, so enabling that -- note that upload performance will suffer if you have microsoft defender "real-time protection" enabled, so you probably want to use -j 1 instead'
|
||||||
args.no_fpool = True
|
args.no_fpool = True
|
||||||
|
|
||||||
self.log("root", m, c=3)
|
self.log("root", t, c=3)
|
||||||
|
|
||||||
|
bri = "zy"[args.theme % 2 :][:1]
|
||||||
|
ch = "abcdefghijklmnopqrstuvwx"[int(args.theme / 2)]
|
||||||
|
args.theme = "{0}{1} {0} {1}".format(ch, bri)
|
||||||
|
|
||||||
|
if args.log_fk:
|
||||||
|
args.log_fk = re.compile(args.log_fk)
|
||||||
|
|
||||||
# initiate all services to manage
|
# initiate all services to manage
|
||||||
self.asrv = AuthSrv(self.args, self.log)
|
self.asrv = AuthSrv(self.args, self.log, dargs=self.dargs)
|
||||||
|
|
||||||
|
if args.cgen:
|
||||||
|
self.asrv.cgen()
|
||||||
|
|
||||||
|
if args.exit == "cfg":
|
||||||
|
sys.exit(0)
|
||||||
|
|
||||||
if args.ls:
|
if args.ls:
|
||||||
self.asrv.dbg_ls()
|
self.asrv.dbg_ls()
|
||||||
|
|
||||||
|
if not ANYWIN:
|
||||||
|
self._setlimits()
|
||||||
|
|
||||||
|
self.log("root", "max clients: {}".format(self.args.nc))
|
||||||
|
|
||||||
|
if not self._process_config():
|
||||||
|
raise Exception("bad config")
|
||||||
|
|
||||||
self.tcpsrv = TcpSrv(self)
|
self.tcpsrv = TcpSrv(self)
|
||||||
self.up2k = Up2k(self)
|
self.up2k = Up2k(self)
|
||||||
|
|
||||||
self.thumbsrv = None
|
decs = {k: 1 for k in self.args.th_dec.split(",")}
|
||||||
if not args.no_thumb:
|
if not HAVE_VIPS:
|
||||||
if HAVE_PIL:
|
decs.pop("vips", None)
|
||||||
if not HAVE_WEBP:
|
if not HAVE_PIL:
|
||||||
args.th_no_webp = True
|
decs.pop("pil", None)
|
||||||
msg = "setting --th-no-webp because either libwebp is not available or your Pillow is too old"
|
if not HAVE_FFMPEG or not HAVE_FFPROBE:
|
||||||
self.log("thumb", msg, c=3)
|
decs.pop("ff", None)
|
||||||
|
|
||||||
|
self.args.th_dec = list(decs.keys())
|
||||||
|
self.thumbsrv = None
|
||||||
|
want_ff = False
|
||||||
|
if not args.no_thumb:
|
||||||
|
t = ", ".join(self.args.th_dec) or "(None available)"
|
||||||
|
self.log("thumb", "decoder preference: {}".format(t))
|
||||||
|
|
||||||
|
if "pil" in self.args.th_dec and not HAVE_WEBP:
|
||||||
|
msg = "disabling webp thumbnails because either libwebp is not available or your Pillow is too old"
|
||||||
|
self.log("thumb", msg, c=3)
|
||||||
|
|
||||||
|
if self.args.th_dec:
|
||||||
self.thumbsrv = ThumbSrv(self)
|
self.thumbsrv = ThumbSrv(self)
|
||||||
else:
|
else:
|
||||||
msg = "need Pillow to create thumbnails; for example:\n{}{} -m pip install --user Pillow\n"
|
want_ff = True
|
||||||
self.log(
|
msg = "need either Pillow, pyvips, or FFmpeg to create thumbnails; for example:\n{0}{1} -m pip install --user Pillow\n{0}{1} -m pip install --user pyvips\n{0}apt install ffmpeg"
|
||||||
"thumb", msg.format(" " * 37, os.path.basename(sys.executable)), c=3
|
msg = msg.format(" " * 37, os.path.basename(pybin))
|
||||||
)
|
if EXE:
|
||||||
|
msg = "copyparty.exe cannot use Pillow or pyvips; need ffprobe.exe and ffmpeg.exe to create thumbnails"
|
||||||
|
|
||||||
|
self.log("thumb", msg, c=3)
|
||||||
|
|
||||||
if not args.no_acode and args.no_thumb:
|
if not args.no_acode and args.no_thumb:
|
||||||
msg = "setting --no-acode because --no-thumb (sorry)"
|
msg = "setting --no-acode because --no-thumb (sorry)"
|
||||||
@@ -102,53 +225,177 @@ class SvcHub(object):
|
|||||||
msg = "setting --no-acode because either FFmpeg or FFprobe is not available"
|
msg = "setting --no-acode because either FFmpeg or FFprobe is not available"
|
||||||
self.log("thumb", msg, c=6)
|
self.log("thumb", msg, c=6)
|
||||||
args.no_acode = True
|
args.no_acode = True
|
||||||
|
want_ff = True
|
||||||
|
|
||||||
|
if want_ff and ANYWIN:
|
||||||
|
self.log("thumb", "download FFmpeg to fix it:\033[0m " + FFMPEG_URL, 3)
|
||||||
|
|
||||||
args.th_poke = min(args.th_poke, args.th_maxage, args.ac_maxage)
|
args.th_poke = min(args.th_poke, args.th_maxage, args.ac_maxage)
|
||||||
|
|
||||||
|
zms = ""
|
||||||
|
if not args.https_only:
|
||||||
|
zms += "d"
|
||||||
|
if not args.http_only:
|
||||||
|
zms += "D"
|
||||||
|
|
||||||
|
if args.ftp or args.ftps:
|
||||||
|
from .ftpd import Ftpd
|
||||||
|
|
||||||
|
self.ftpd = Ftpd(self)
|
||||||
|
zms += "f" if args.ftp else "F"
|
||||||
|
|
||||||
|
if args.smb:
|
||||||
|
# impacket.dcerpc is noisy about listen timeouts
|
||||||
|
sto = socket.getdefaulttimeout()
|
||||||
|
socket.setdefaulttimeout(None)
|
||||||
|
|
||||||
|
from .smbd import SMB
|
||||||
|
|
||||||
|
self.smbd = SMB(self)
|
||||||
|
socket.setdefaulttimeout(sto)
|
||||||
|
self.smbd.start()
|
||||||
|
zms += "s"
|
||||||
|
|
||||||
|
if not args.zms:
|
||||||
|
args.zms = zms
|
||||||
|
|
||||||
|
self.zc_ngen = 0
|
||||||
|
self.mdns: Optional["MDNS"] = None
|
||||||
|
self.ssdp: Optional["SSDPd"] = None
|
||||||
|
|
||||||
# decide which worker impl to use
|
# decide which worker impl to use
|
||||||
if self.check_mp_enable():
|
if self.check_mp_enable():
|
||||||
from .broker_mp import BrokerMp as Broker
|
from .broker_mp import BrokerMp as Broker
|
||||||
else:
|
else:
|
||||||
from .broker_thr import BrokerThr as Broker
|
from .broker_thr import BrokerThr as Broker # type: ignore
|
||||||
|
|
||||||
self.broker = Broker(self)
|
self.broker = Broker(self)
|
||||||
|
|
||||||
def thr_httpsrv_up(self):
|
def thr_httpsrv_up(self) -> None:
|
||||||
time.sleep(5)
|
time.sleep(1 if self.args.ign_ebind_all else 5)
|
||||||
expected = self.broker.num_workers * self.tcpsrv.nsrv
|
expected = self.broker.num_workers * self.tcpsrv.nsrv
|
||||||
failed = expected - self.httpsrv_up
|
failed = expected - self.httpsrv_up
|
||||||
if not failed:
|
if not failed:
|
||||||
return
|
return
|
||||||
|
|
||||||
if self.args.ign_ebind_all:
|
if self.args.ign_ebind_all:
|
||||||
|
if not self.tcpsrv.srv:
|
||||||
|
for _ in range(self.broker.num_workers):
|
||||||
|
self.broker.say("cb_httpsrv_up")
|
||||||
return
|
return
|
||||||
|
|
||||||
if self.args.ign_ebind and self.tcpsrv.srv:
|
if self.args.ign_ebind and self.tcpsrv.srv:
|
||||||
return
|
return
|
||||||
|
|
||||||
m = "{}/{} workers failed to start"
|
t = "{}/{} workers failed to start"
|
||||||
m = m.format(failed, expected)
|
t = t.format(failed, expected)
|
||||||
self.log("root", m, 1)
|
self.log("root", t, 1)
|
||||||
|
|
||||||
self.retcode = 1
|
self.retcode = 1
|
||||||
|
self.sigterm()
|
||||||
|
|
||||||
|
def sigterm(self) -> None:
|
||||||
os.kill(os.getpid(), signal.SIGTERM)
|
os.kill(os.getpid(), signal.SIGTERM)
|
||||||
|
|
||||||
def cb_httpsrv_up(self):
|
def cb_httpsrv_up(self) -> None:
|
||||||
self.httpsrv_up += 1
|
self.httpsrv_up += 1
|
||||||
if self.httpsrv_up != self.broker.num_workers:
|
if self.httpsrv_up != self.broker.num_workers:
|
||||||
return
|
return
|
||||||
|
|
||||||
time.sleep(0.1) # purely cosmetic dw
|
time.sleep(0.1) # purely cosmetic dw
|
||||||
self.log("root", "workers OK\n")
|
if self.tcpsrv.qr:
|
||||||
|
self.log("qr-code", self.tcpsrv.qr)
|
||||||
|
else:
|
||||||
|
self.log("root", "workers OK\n")
|
||||||
|
|
||||||
self.up2k.init_vols()
|
self.up2k.init_vols()
|
||||||
|
|
||||||
thr = threading.Thread(target=self.sd_notify, name="sd-notify")
|
Daemon(self.sd_notify, "sd-notify")
|
||||||
thr.daemon = True
|
|
||||||
thr.start()
|
|
||||||
|
|
||||||
def _logname(self):
|
def _process_config(self) -> bool:
|
||||||
|
al = self.args
|
||||||
|
|
||||||
|
al.zm_on = al.zm_on or al.z_on
|
||||||
|
al.zs_on = al.zs_on or al.z_on
|
||||||
|
al.zm_off = al.zm_off or al.z_off
|
||||||
|
al.zs_off = al.zs_off or al.z_off
|
||||||
|
ns = "zm_on zm_off zs_on zs_off acao acam"
|
||||||
|
for n in ns.split(" "):
|
||||||
|
vs = getattr(al, n).split(",")
|
||||||
|
vs = [x.strip() for x in vs]
|
||||||
|
vs = [x for x in vs if x]
|
||||||
|
setattr(al, n, vs)
|
||||||
|
|
||||||
|
ns = "acao acam"
|
||||||
|
for n in ns.split(" "):
|
||||||
|
vs = getattr(al, n)
|
||||||
|
vd = {zs: 1 for zs in vs}
|
||||||
|
setattr(al, n, vd)
|
||||||
|
|
||||||
|
ns = "acao"
|
||||||
|
for n in ns.split(" "):
|
||||||
|
vs = getattr(al, n)
|
||||||
|
vs = [x.lower() for x in vs]
|
||||||
|
setattr(al, n, vs)
|
||||||
|
|
||||||
|
R = al.rp_loc
|
||||||
|
if "//" in R or ":" in R:
|
||||||
|
t = "found URL in --rp-loc; it should be just the location, for example /foo/bar"
|
||||||
|
raise Exception(t)
|
||||||
|
|
||||||
|
al.R = R = R.strip("/")
|
||||||
|
al.SR = "/" + R if R else ""
|
||||||
|
al.RS = R + "/" if R else ""
|
||||||
|
al.SRS = "/" + R + "/" if R else "/"
|
||||||
|
|
||||||
|
if al.rsp_jtr:
|
||||||
|
al.rsp_slp = 0.000001
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def _setlimits(self) -> None:
|
||||||
|
try:
|
||||||
|
import resource
|
||||||
|
|
||||||
|
soft, hard = [
|
||||||
|
x if x > 0 else 1024 * 1024
|
||||||
|
for x in list(resource.getrlimit(resource.RLIMIT_NOFILE))
|
||||||
|
]
|
||||||
|
except:
|
||||||
|
self.log("root", "failed to read rlimits from os", 6)
|
||||||
|
return
|
||||||
|
|
||||||
|
if not soft or not hard:
|
||||||
|
t = "got bogus rlimits from os ({}, {})"
|
||||||
|
self.log("root", t.format(soft, hard), 6)
|
||||||
|
return
|
||||||
|
|
||||||
|
want = self.args.nc * 4
|
||||||
|
new_soft = min(hard, want)
|
||||||
|
if new_soft < soft:
|
||||||
|
return
|
||||||
|
|
||||||
|
# t = "requesting rlimit_nofile({}), have {}"
|
||||||
|
# self.log("root", t.format(new_soft, soft), 6)
|
||||||
|
|
||||||
|
try:
|
||||||
|
import resource
|
||||||
|
|
||||||
|
resource.setrlimit(resource.RLIMIT_NOFILE, (new_soft, hard))
|
||||||
|
soft = new_soft
|
||||||
|
except:
|
||||||
|
t = "rlimit denied; max open files: {}"
|
||||||
|
self.log("root", t.format(soft), 3)
|
||||||
|
return
|
||||||
|
|
||||||
|
if soft < want:
|
||||||
|
t = "max open files: {} (wanted {} for -nc {})"
|
||||||
|
self.log("root", t.format(soft, want, self.args.nc), 3)
|
||||||
|
self.args.nc = min(self.args.nc, soft // 2)
|
||||||
|
|
||||||
|
def _logname(self) -> str:
|
||||||
dt = datetime.utcnow()
|
dt = datetime.utcnow()
|
||||||
fn = self.args.lo
|
fn = str(self.args.lo)
|
||||||
for fs in "YmdHMS":
|
for fs in "YmdHMS":
|
||||||
fs = "%" + fs
|
fs = "%" + fs
|
||||||
if fs in fn:
|
if fs in fn:
|
||||||
@@ -156,7 +403,7 @@ class SvcHub(object):
|
|||||||
|
|
||||||
return fn
|
return fn
|
||||||
|
|
||||||
def _setup_logfile(self, printed):
|
def _setup_logfile(self, printed: str) -> None:
|
||||||
base_fn = fn = sel_fn = self._logname()
|
base_fn = fn = sel_fn = self._logname()
|
||||||
if fn != self.args.lo:
|
if fn != self.args.lo:
|
||||||
ctr = 0
|
ctr = 0
|
||||||
@@ -169,18 +416,18 @@ class SvcHub(object):
|
|||||||
fn = sel_fn
|
fn = sel_fn
|
||||||
|
|
||||||
try:
|
try:
|
||||||
import lzma
|
if fn.lower().endswith(".xz"):
|
||||||
|
import lzma
|
||||||
lh = lzma.open(fn, "wt", encoding="utf-8", errors="replace", preset=0)
|
|
||||||
|
|
||||||
|
lh = lzma.open(fn, "wt", encoding="utf-8", errors="replace", preset=0)
|
||||||
|
else:
|
||||||
|
lh = open(fn, "wt", encoding="utf-8", errors="replace")
|
||||||
except:
|
except:
|
||||||
import codecs
|
import codecs
|
||||||
|
|
||||||
lh = codecs.open(fn, "w", encoding="utf-8", errors="replace")
|
lh = codecs.open(fn, "w", encoding="utf-8", errors="replace")
|
||||||
|
|
||||||
lh.base_fn = base_fn
|
argv = [pybin] + self.argv
|
||||||
|
|
||||||
argv = [sys.executable] + self.argv
|
|
||||||
if hasattr(shlex, "quote"):
|
if hasattr(shlex, "quote"):
|
||||||
argv = [shlex.quote(x) for x in argv]
|
argv = [shlex.quote(x) for x in argv]
|
||||||
else:
|
else:
|
||||||
@@ -188,16 +435,20 @@ class SvcHub(object):
|
|||||||
|
|
||||||
msg = "[+] opened logfile [{}]\n".format(fn)
|
msg = "[+] opened logfile [{}]\n".format(fn)
|
||||||
printed += msg
|
printed += msg
|
||||||
lh.write("t0: {:.3f}\nargv: {}\n\n{}".format(E.t0, " ".join(argv), printed))
|
t = "t0: {:.3f}\nargv: {}\n\n{}"
|
||||||
|
lh.write(t.format(self.E.t0, " ".join(argv), printed))
|
||||||
self.logf = lh
|
self.logf = lh
|
||||||
|
self.logf_base_fn = base_fn
|
||||||
print(msg, end="")
|
print(msg, end="")
|
||||||
|
|
||||||
def run(self):
|
def run(self) -> None:
|
||||||
self.tcpsrv.run()
|
self.tcpsrv.run()
|
||||||
|
if getattr(self.args, "z_chk", 0) and (
|
||||||
|
getattr(self.args, "zm", False) or getattr(self.args, "zs", False)
|
||||||
|
):
|
||||||
|
Daemon(self.tcpsrv.netmon, "netmon")
|
||||||
|
|
||||||
thr = threading.Thread(target=self.thr_httpsrv_up)
|
Daemon(self.thr_httpsrv_up, "sig-hsrv-up2")
|
||||||
thr.daemon = True
|
|
||||||
thr.start()
|
|
||||||
|
|
||||||
sigs = [signal.SIGINT, signal.SIGTERM]
|
sigs = [signal.SIGINT, signal.SIGTERM]
|
||||||
if not ANYWIN:
|
if not ANYWIN:
|
||||||
@@ -212,9 +463,7 @@ class SvcHub(object):
|
|||||||
# never lucky
|
# never lucky
|
||||||
if ANYWIN:
|
if ANYWIN:
|
||||||
# msys-python probably fine but >msys-python
|
# msys-python probably fine but >msys-python
|
||||||
thr = threading.Thread(target=self.stop_thr, name="svchub-sig")
|
Daemon(self.stop_thr, "svchub-sig")
|
||||||
thr.daemon = True
|
|
||||||
thr.start()
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
while not self.stop_req:
|
while not self.stop_req:
|
||||||
@@ -223,21 +472,48 @@ class SvcHub(object):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
self.shutdown()
|
self.shutdown()
|
||||||
thr.join()
|
# cant join; eats signals on win10
|
||||||
|
while not self.stopped:
|
||||||
|
time.sleep(0.1)
|
||||||
else:
|
else:
|
||||||
self.stop_thr()
|
self.stop_thr()
|
||||||
|
|
||||||
def reload(self):
|
def start_zeroconf(self) -> None:
|
||||||
|
self.zc_ngen += 1
|
||||||
|
|
||||||
|
if getattr(self.args, "zm", False):
|
||||||
|
try:
|
||||||
|
from .mdns import MDNS
|
||||||
|
|
||||||
|
if self.mdns:
|
||||||
|
self.mdns.stop(True)
|
||||||
|
|
||||||
|
self.mdns = MDNS(self, self.zc_ngen)
|
||||||
|
Daemon(self.mdns.run, "mdns")
|
||||||
|
except:
|
||||||
|
self.log("root", "mdns startup failed;\n" + min_ex(), 3)
|
||||||
|
|
||||||
|
if getattr(self.args, "zs", False):
|
||||||
|
try:
|
||||||
|
from .ssdp import SSDPd
|
||||||
|
|
||||||
|
if self.ssdp:
|
||||||
|
self.ssdp.stop()
|
||||||
|
|
||||||
|
self.ssdp = SSDPd(self, self.zc_ngen)
|
||||||
|
Daemon(self.ssdp.run, "ssdp")
|
||||||
|
except:
|
||||||
|
self.log("root", "ssdp startup failed;\n" + min_ex(), 3)
|
||||||
|
|
||||||
|
def reload(self) -> str:
|
||||||
if self.reloading:
|
if self.reloading:
|
||||||
return "cannot reload; already in progress"
|
return "cannot reload; already in progress"
|
||||||
|
|
||||||
self.reloading = True
|
self.reloading = True
|
||||||
t = threading.Thread(target=self._reload)
|
Daemon(self._reload, "reloading")
|
||||||
t.daemon = True
|
|
||||||
t.start()
|
|
||||||
return "reload initiated"
|
return "reload initiated"
|
||||||
|
|
||||||
def _reload(self):
|
def _reload(self) -> None:
|
||||||
self.log("root", "reload scheduled")
|
self.log("root", "reload scheduled")
|
||||||
with self.up2k.mutex:
|
with self.up2k.mutex:
|
||||||
self.asrv.reload()
|
self.asrv.reload()
|
||||||
@@ -246,7 +522,7 @@ class SvcHub(object):
|
|||||||
|
|
||||||
self.reloading = False
|
self.reloading = False
|
||||||
|
|
||||||
def stop_thr(self):
|
def stop_thr(self) -> None:
|
||||||
while not self.stop_req:
|
while not self.stop_req:
|
||||||
with self.stop_cond:
|
with self.stop_cond:
|
||||||
self.stop_cond.wait(9001)
|
self.stop_cond.wait(9001)
|
||||||
@@ -257,11 +533,32 @@ class SvcHub(object):
|
|||||||
|
|
||||||
self.shutdown()
|
self.shutdown()
|
||||||
|
|
||||||
def signal_handler(self, sig, frame):
|
def kill9(self, delay: float = 0.0) -> None:
|
||||||
if self.stopping:
|
if delay > 0.01:
|
||||||
return
|
time.sleep(delay)
|
||||||
|
print("component stuck; issuing sigkill")
|
||||||
|
time.sleep(0.1)
|
||||||
|
|
||||||
if sig == signal.SIGUSR1:
|
if ANYWIN:
|
||||||
|
os.system("taskkill /f /pid {}".format(os.getpid()))
|
||||||
|
else:
|
||||||
|
os.kill(os.getpid(), signal.SIGKILL)
|
||||||
|
|
||||||
|
def signal_handler(self, sig: int, frame: Optional[FrameType]) -> None:
|
||||||
|
if self.stopping:
|
||||||
|
if self.nsigs <= 0:
|
||||||
|
try:
|
||||||
|
threading.Thread(target=self.pr, args=("OMBO BREAKER",)).start()
|
||||||
|
time.sleep(0.1)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
self.kill9()
|
||||||
|
else:
|
||||||
|
self.nsigs -= 1
|
||||||
|
return
|
||||||
|
|
||||||
|
if not ANYWIN and sig == signal.SIGUSR1:
|
||||||
self.reload_req = True
|
self.reload_req = True
|
||||||
else:
|
else:
|
||||||
self.stop_req = True
|
self.stop_req = True
|
||||||
@@ -269,7 +566,7 @@ class SvcHub(object):
|
|||||||
with self.stop_cond:
|
with self.stop_cond:
|
||||||
self.stop_cond.notify_all()
|
self.stop_cond.notify_all()
|
||||||
|
|
||||||
def shutdown(self):
|
def shutdown(self) -> None:
|
||||||
if self.stopping:
|
if self.stopping:
|
||||||
return
|
return
|
||||||
|
|
||||||
@@ -282,11 +579,19 @@ class SvcHub(object):
|
|||||||
|
|
||||||
ret = 1
|
ret = 1
|
||||||
try:
|
try:
|
||||||
with self.log_mutex:
|
self.pr("OPYTHAT")
|
||||||
print("OPYTHAT")
|
slp = 0.0
|
||||||
|
|
||||||
|
if self.mdns:
|
||||||
|
Daemon(self.mdns.stop)
|
||||||
|
slp = time.time() + 0.5
|
||||||
|
|
||||||
|
if self.ssdp:
|
||||||
|
Daemon(self.ssdp.stop)
|
||||||
|
slp = time.time() + 0.5
|
||||||
|
|
||||||
self.tcpsrv.shutdown()
|
|
||||||
self.broker.shutdown()
|
self.broker.shutdown()
|
||||||
|
self.tcpsrv.shutdown()
|
||||||
self.up2k.shutdown()
|
self.up2k.shutdown()
|
||||||
if self.thumbsrv:
|
if self.thumbsrv:
|
||||||
self.thumbsrv.shutdown()
|
self.thumbsrv.shutdown()
|
||||||
@@ -297,35 +602,47 @@ class SvcHub(object):
|
|||||||
break
|
break
|
||||||
|
|
||||||
if n == 3:
|
if n == 3:
|
||||||
print("waiting for thumbsrv (10sec)...")
|
self.pr("waiting for thumbsrv (10sec)...")
|
||||||
|
|
||||||
print("nailed it", end="")
|
if hasattr(self, "smbd"):
|
||||||
|
slp = max(slp, time.time() + 0.5)
|
||||||
|
Daemon(self.kill9, a=(1,))
|
||||||
|
Daemon(self.smbd.stop)
|
||||||
|
|
||||||
|
while time.time() < slp:
|
||||||
|
time.sleep(0.1)
|
||||||
|
|
||||||
|
self.pr("nailed it", end="")
|
||||||
ret = self.retcode
|
ret = self.retcode
|
||||||
|
except:
|
||||||
|
self.pr("\033[31m[ error during shutdown ]\n{}\033[0m".format(min_ex()))
|
||||||
|
raise
|
||||||
finally:
|
finally:
|
||||||
if self.args.wintitle:
|
if self.args.wintitle:
|
||||||
print("\033]0;\033\\", file=sys.stderr, end="")
|
print("\033]0;\033\\", file=sys.stderr, end="")
|
||||||
sys.stderr.flush()
|
sys.stderr.flush()
|
||||||
|
|
||||||
print("\033[0m")
|
self.pr("\033[0m")
|
||||||
if self.logf:
|
if self.logf:
|
||||||
self.logf.close()
|
self.logf.close()
|
||||||
|
|
||||||
|
self.stopped = True
|
||||||
sys.exit(ret)
|
sys.exit(ret)
|
||||||
|
|
||||||
def _log_disabled(self, src, msg, c=0):
|
def _log_disabled(self, src: str, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
if not self.logf:
|
if not self.logf:
|
||||||
return
|
return
|
||||||
|
|
||||||
with self.log_mutex:
|
with self.log_mutex:
|
||||||
ts = datetime.utcnow().strftime("%Y-%m%d-%H%M%S.%f")[:-3]
|
ts = datetime.utcnow().strftime("%Y-%m%d-%H%M%S.%f")[:-3]
|
||||||
self.logf.write("@{} [{}] {}\n".format(ts, src, msg))
|
self.logf.write("@{} [{}\033[0m] {}\n".format(ts, src, msg))
|
||||||
|
|
||||||
now = time.time()
|
now = time.time()
|
||||||
if now >= self.next_day:
|
if now >= self.next_day:
|
||||||
self._set_next_day()
|
self._set_next_day()
|
||||||
|
|
||||||
def _set_next_day(self):
|
def _set_next_day(self) -> None:
|
||||||
if self.next_day and self.logf and self.logf.base_fn != self._logname():
|
if self.next_day and self.logf and self.logf_base_fn != self._logname():
|
||||||
self.logf.close()
|
self.logf.close()
|
||||||
self._setup_logfile("")
|
self._setup_logfile("")
|
||||||
|
|
||||||
@@ -339,7 +656,7 @@ class SvcHub(object):
|
|||||||
dt = dt.replace(hour=0, minute=0, second=0)
|
dt = dt.replace(hour=0, minute=0, second=0)
|
||||||
self.next_day = calendar.timegm(dt.utctimetuple())
|
self.next_day = calendar.timegm(dt.utctimetuple())
|
||||||
|
|
||||||
def _log_enabled(self, src, msg, c=0):
|
def _log_enabled(self, src: str, msg: str, c: Union[int, str] = 0) -> None:
|
||||||
"""handles logging from all components"""
|
"""handles logging from all components"""
|
||||||
with self.log_mutex:
|
with self.log_mutex:
|
||||||
now = time.time()
|
now = time.time()
|
||||||
@@ -357,7 +674,7 @@ class SvcHub(object):
|
|||||||
src = ansi_re.sub("", src)
|
src = ansi_re.sub("", src)
|
||||||
elif c:
|
elif c:
|
||||||
if isinstance(c, int):
|
if isinstance(c, int):
|
||||||
msg = "\033[3{}m{}".format(c, msg)
|
msg = "\033[3{}m{}\033[0m".format(c, msg)
|
||||||
elif "\033" not in c:
|
elif "\033" not in c:
|
||||||
msg = "\033[{}m{}\033[0m".format(c, msg)
|
msg = "\033[{}m{}\033[0m".format(c, msg)
|
||||||
else:
|
else:
|
||||||
@@ -372,38 +689,45 @@ class SvcHub(object):
|
|||||||
print(msg.encode("utf-8", "replace").decode(), end="")
|
print(msg.encode("utf-8", "replace").decode(), end="")
|
||||||
except:
|
except:
|
||||||
print(msg.encode("ascii", "replace").decode(), end="")
|
print(msg.encode("ascii", "replace").decode(), end="")
|
||||||
|
except OSError as ex:
|
||||||
|
if ex.errno != errno.EPIPE:
|
||||||
|
raise
|
||||||
|
|
||||||
if self.logf:
|
if self.logf:
|
||||||
self.logf.write(msg)
|
self.logf.write(msg)
|
||||||
|
|
||||||
def check_mp_support(self):
|
def pr(self, *a: Any, **ka: Any) -> None:
|
||||||
vmin = sys.version_info[1]
|
try:
|
||||||
if WINDOWS:
|
with self.log_mutex:
|
||||||
msg = "need python 3.3 or newer for multiprocessing;"
|
print(*a, **ka)
|
||||||
if PY2 or vmin < 3:
|
except OSError as ex:
|
||||||
return msg
|
if ex.errno != errno.EPIPE:
|
||||||
elif MACOS:
|
raise
|
||||||
|
|
||||||
|
def check_mp_support(self) -> str:
|
||||||
|
if MACOS:
|
||||||
return "multiprocessing is wonky on mac osx;"
|
return "multiprocessing is wonky on mac osx;"
|
||||||
else:
|
elif sys.version_info < (3, 3):
|
||||||
msg = "need python 3.3+ for multiprocessing;"
|
return "need python 3.3 or newer for multiprocessing;"
|
||||||
if PY2 or vmin < 3:
|
|
||||||
return msg
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
x = mp.Queue(1)
|
x: mp.Queue[tuple[str, str]] = mp.Queue(1)
|
||||||
x.put(["foo", "bar"])
|
x.put(("foo", "bar"))
|
||||||
if x.get()[0] != "foo":
|
if x.get()[0] != "foo":
|
||||||
raise Exception()
|
raise Exception()
|
||||||
except:
|
except:
|
||||||
return "multiprocessing is not supported on your platform;"
|
return "multiprocessing is not supported on your platform;"
|
||||||
|
|
||||||
return None
|
return ""
|
||||||
|
|
||||||
def check_mp_enable(self):
|
def check_mp_enable(self) -> bool:
|
||||||
if self.args.j == 1:
|
if self.args.j == 1:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
if mp.cpu_count() <= 1:
|
try:
|
||||||
|
if mp.cpu_count() <= 1:
|
||||||
|
raise Exception()
|
||||||
|
except:
|
||||||
self.log("svchub", "only one CPU detected; multiprocessing disabled")
|
self.log("svchub", "only one CPU detected; multiprocessing disabled")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@@ -422,21 +746,34 @@ class SvcHub(object):
|
|||||||
self.log("svchub", "cannot efficiently use multiple CPU cores")
|
self.log("svchub", "cannot efficiently use multiple CPU cores")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def sd_notify(self):
|
def sd_notify(self) -> None:
|
||||||
try:
|
try:
|
||||||
addr = os.getenv("NOTIFY_SOCKET")
|
zb = os.getenv("NOTIFY_SOCKET")
|
||||||
if not addr:
|
if not zb:
|
||||||
return
|
return
|
||||||
|
|
||||||
addr = unicode(addr)
|
addr = unicode(zb)
|
||||||
if addr.startswith("@"):
|
if addr.startswith("@"):
|
||||||
addr = "\0" + addr[1:]
|
addr = "\0" + addr[1:]
|
||||||
|
|
||||||
m = "".join(x for x in addr if x in string.printable)
|
t = "".join(x for x in addr if x in string.printable)
|
||||||
self.log("sd_notify", m)
|
self.log("sd_notify", t)
|
||||||
|
|
||||||
sck = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
|
sck = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
|
||||||
sck.connect(addr)
|
sck.connect(addr)
|
||||||
sck.sendall(b"READY=1")
|
sck.sendall(b"READY=1")
|
||||||
except:
|
except:
|
||||||
self.log("sd_notify", min_ex())
|
self.log("sd_notify", min_ex())
|
||||||
|
|
||||||
|
def log_stacks(self) -> None:
|
||||||
|
td = time.time() - self.tstack
|
||||||
|
if td < 300:
|
||||||
|
self.log("stacks", "cooldown {}".format(td))
|
||||||
|
return
|
||||||
|
|
||||||
|
self.tstack = time.time()
|
||||||
|
zs = "{}\n{}".format(VERSIONS, alltrace())
|
||||||
|
zb = zs.encode("utf-8", "replace")
|
||||||
|
zb = gzip.compress(zb)
|
||||||
|
zs = base64.b64encode(zb).decode("ascii")
|
||||||
|
self.log("stacks", zs)
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user