mirror of
				https://github.com/9001/copyparty.git
				synced 2025-10-26 17:43:44 +00:00 
			
		
		
		
	Compare commits
	
		
			1067 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | ca6ec9c5c7 | ||
|  | 633b1f0a78 | ||
|  | 6136b9bf9c | ||
|  | 524a3ba566 | ||
|  | 58580320f9 | ||
|  | 759b0a994d | ||
|  | d2800473e4 | ||
|  | f5b1a2065e | ||
|  | 5e62532295 | ||
|  | c1bee96c40 | ||
|  | f273253a2b | ||
|  | 012bbcf770 | ||
|  | b54cb47b2e | ||
|  | 1b15f43745 | ||
|  | 96771bf1bd | ||
|  | 580078bddb | ||
|  | c5c7080ec6 | ||
|  | 408339b51d | ||
|  | 02e3d44998 | ||
|  | 156f13ded1 | ||
|  | d288467cb7 | ||
|  | 21662c9f3f | ||
|  | 9149fe6cdd | ||
|  | 9a146192b7 | ||
|  | 3a9d3b7b61 | ||
|  | f03f0973ab | ||
|  | 7ec0881e8c | ||
|  | 59e1ab42ff | ||
|  | 722216b901 | ||
|  | bd8f3dc368 | ||
|  | 33cd94a141 | ||
|  | 053ac74734 | ||
|  | cced99fafa | ||
|  | a009ff53f7 | ||
|  | ca16c4108d | ||
|  | d1b6c67dc3 | ||
|  | a61f8133d5 | ||
|  | 38d797a544 | ||
|  | 16c1877f50 | ||
|  | da5f15a778 | ||
|  | 396c64ecf7 | ||
|  | 252c3a7985 | ||
|  | a3ecbf0ae7 | ||
|  | 314327d8f2 | ||
|  | bfacd06929 | ||
|  | 4f5e8f8cf5 | ||
|  | 1fbb4c09cc | ||
|  | b332e1992b | ||
|  | 5955940b82 | ||
|  | 231a03bcfd | ||
|  | bc85723657 | ||
|  | be32b743c6 | ||
|  | 83c9843059 | ||
|  | 11cf43626d | ||
|  | a6dc5e2ce3 | ||
|  | 38593a0394 | ||
|  | 95309afeea | ||
|  | c2bf6fe2a3 | ||
|  | 99ac324fbd | ||
|  | 5562de330f | ||
|  | 95014236ac | ||
|  | 6aa7386138 | ||
|  | 3226a1f588 | ||
|  | b4cf890cd8 | ||
|  | ce09e323af | ||
|  | 941aedb177 | ||
|  | 87a0d502a3 | ||
|  | cab7c1b0b8 | ||
|  | d5892341b6 | ||
|  | 646557a43e | ||
|  | ed8d34ab43 | ||
|  | 5e34463c77 | ||
|  | 1b14eb7959 | ||
|  | ed48c2d0ed | ||
|  | 26fe84b660 | ||
|  | 5938230270 | ||
|  | 1a33a047fa | ||
|  | 43a8bcefb9 | ||
|  | 2e740e513f | ||
|  | 8a21a86b61 | ||
|  | f600116205 | ||
|  | 1c03705de8 | ||
|  | f7e461fac6 | ||
|  | 03ce6c97ff | ||
|  | ffd9e76e07 | ||
|  | fc49cb1e67 | ||
|  | f5712d9f25 | ||
|  | 161d57bdda | ||
|  | bae0d440bf | ||
|  | fff052dde1 | ||
|  | 73b06eaa02 | ||
|  | 08a8ebed17 | ||
|  | 74d07426b3 | ||
|  | 69a2bba99a | ||
|  | 4d685d78ee | ||
|  | 5845ec3f49 | ||
|  | 13373426fe | ||
|  | 8e55551a06 | ||
|  | 12a3f0ac31 | ||
|  | 18e33edc88 | ||
|  | c72c5ad4ee | ||
|  | 0fbc81ab2f | ||
|  | af0a34cf82 | ||
|  | b4590c5398 | ||
|  | f787a66230 | ||
|  | b21a99fd62 | ||
|  | eb16306cde | ||
|  | 7bc23687e3 | ||
|  | e1eaa057f2 | ||
|  | 97c264ca3e | ||
|  | cf848ab1f7 | ||
|  | cf83f9b0fd | ||
|  | d98e361083 | ||
|  | ce7f5309c7 | ||
|  | 75c485ced7 | ||
|  | 9c6e2ec012 | ||
|  | 1a02948a61 | ||
|  | 8b05ba4ba1 | ||
|  | 21e2874cb7 | ||
|  | 360ed5c46c | ||
|  | 5099bc365d | ||
|  | 12986da147 | ||
|  | 23e72797bc | ||
|  | ac7b6f8f55 | ||
|  | 981b9ff11e | ||
|  | 4186906f4c | ||
|  | 0850d24e0c | ||
|  | 7ab8334c96 | ||
|  | a4d7329ab7 | ||
|  | 3f4eae6bce | ||
|  | 518cf4be57 | ||
|  | 71096182be | ||
|  | 6452e927ea | ||
|  | bc70cfa6f0 | ||
|  | 2b6e5ebd2d | ||
|  | c761bd799a | ||
|  | 2f7c2fdee4 | ||
|  | 70a76ec343 | ||
|  | 7c3f64abf2 | ||
|  | f5f38f195c | ||
|  | 7e84f4f015 | ||
|  | 4802f8cf07 | ||
|  | cc05e67d8f | ||
|  | 2b6b174517 | ||
|  | a1d05e6e12 | ||
|  | f95ceb6a9b | ||
|  | 8f91b0726d | ||
|  | 97807f4383 | ||
|  | 5f42237f2c | ||
|  | 68289cfa54 | ||
|  | 42ea30270f | ||
|  | ebbbbf3d82 | ||
|  | 27516e2d16 | ||
|  | 84bb6f915e | ||
|  | 46752f758a | ||
|  | 34c4c22e61 | ||
|  | af2d0b8421 | ||
|  | 638b05a49a | ||
|  | 7a13e8a7fc | ||
|  | d9fa74711d | ||
|  | 41867f578f | ||
|  | 0bf41ed4ef | ||
|  | d080b4a731 | ||
|  | ca4232ada9 | ||
|  | ad348f91c9 | ||
|  | 990f915f42 | ||
|  | 53d720217b | ||
|  | 7a06ff480d | ||
|  | 3ef551f788 | ||
|  | f0125cdc36 | ||
|  | ed5f6736df | ||
|  | 15d8be0fae | ||
|  | 46f3e61360 | ||
|  | 87ad8c98d4 | ||
|  | 9bbdc4100f | ||
|  | c80307e8ff | ||
|  | c1d77e1041 | ||
|  | d9e83650dc | ||
|  | f6d635acd9 | ||
|  | 0dbd8a01ff | ||
|  | 8d755d41e0 | ||
|  | 190473bd32 | ||
|  | 030d1ec254 | ||
|  | 5a2b91a084 | ||
|  | a50a05e4e7 | ||
|  | 6cb5a87c79 | ||
|  | b9f89ca552 | ||
|  | 26c9fd5dea | ||
|  | e81a9b6fe0 | ||
|  | 452450e451 | ||
|  | 419dd2d1c7 | ||
|  | ee86b06676 | ||
|  | 953183f16d | ||
|  | 228f71708b | ||
|  | 621471a7cb | ||
|  | 8b58e951e3 | ||
|  | 1db489a0aa | ||
|  | be65c3c6cf | ||
|  | 46e7fa31fe | ||
|  | 66e21bd499 | ||
|  | 8cab4c01fd | ||
|  | d52038366b | ||
|  | 4fcfd87f5b | ||
|  | f893c6baa4 | ||
|  | 9a45549b66 | ||
|  | ae3a01038b | ||
|  | e47a2a4ca2 | ||
|  | 95ea6d5f78 | ||
|  | 7d290f6b8f | ||
|  | 9db617ed5a | ||
|  | 514456940a | ||
|  | 33feefd9cd | ||
|  | 65e14cf348 | ||
|  | 1d61bcc4f3 | ||
|  | c38bbaca3c | ||
|  | 246d245ebc | ||
|  | f269a710e2 | ||
|  | 051998429c | ||
|  | 432cdd640f | ||
|  | 9ed9b0964e | ||
|  | 6a97b3526d | ||
|  | 451d757996 | ||
|  | f9e9eba3b1 | ||
|  | 2a9a6aebd9 | ||
|  | adbb6c449e | ||
|  | 3993605324 | ||
|  | 0ae574ec2c | ||
|  | c56ded828c | ||
|  | 02c7061945 | ||
|  | 9209e44cd3 | ||
|  | ebed37394e | ||
|  | 4c7a2a7ec3 | ||
|  | 0a25a88a34 | ||
|  | 6aa9025347 | ||
|  | a918cc67eb | ||
|  | 08f4695283 | ||
|  | 44e76d5eeb | ||
|  | cfa36fd279 | ||
|  | 3d4166e006 | ||
|  | 07bac1c592 | ||
|  | 755f2ce1ba | ||
|  | cca2844deb | ||
|  | 24a2f760b7 | ||
|  | 79bbd8fe38 | ||
|  | 35dce1e3e4 | ||
|  | f886fdf913 | ||
|  | 4476f2f0da | ||
|  | 160f161700 | ||
|  | c164fc58a2 | ||
|  | 0c625a4e62 | ||
|  | bf3941cf7a | ||
|  | 3649e8288a | ||
|  | 9a45e26026 | ||
|  | e65f127571 | ||
|  | 3bfc699787 | ||
|  | 955318428a | ||
|  | f6279b356a | ||
|  | 4cc3cdc989 | ||
|  | f9aa20a3ad | ||
|  | 129d33f1a0 | ||
|  | 1ad7a3f378 | ||
|  | b533be8818 | ||
|  | fb729e5166 | ||
|  | d337ecdb20 | ||
|  | 5f1f0a48b0 | ||
|  | e0f1cb94a5 | ||
|  | a362ee2246 | ||
|  | 19f23c686e | ||
|  | 23b20ff4a6 | ||
|  | 72574da834 | ||
|  | d5a79455d1 | ||
|  | 070d4b9da9 | ||
|  | 0ace22fffe | ||
|  | 9e483d7694 | ||
|  | 26458b7a06 | ||
|  | b6a4604952 | ||
|  | af752fbbc2 | ||
|  | 279c9d706a | ||
|  | 806e7b5530 | ||
|  | f3dc6a217b | ||
|  | 7671d791fa | ||
|  | 8cd84608a5 | ||
|  | 980c6fc810 | ||
|  | fb40a484c5 | ||
|  | daa9dedcaa | ||
|  | 0d634345ac | ||
|  | e648252479 | ||
|  | 179d7a9ad8 | ||
|  | 19bc962ad5 | ||
|  | 27cce086c6 | ||
|  | fec0c620d4 | ||
|  | 05a1a31cab | ||
|  | d020527c6f | ||
|  | 4451485664 | ||
|  | a4e1a3738a | ||
|  | 4339dbeb8d | ||
|  | 5b0605774c | ||
|  | e3684e25f8 | ||
|  | 1359213196 | ||
|  | 03efc6a169 | ||
|  | 15b5982211 | ||
|  | 0eb3a5d387 | ||
|  | 7f8777389c | ||
|  | 4eb20f10ad | ||
|  | daa11df558 | ||
|  | 1bb0db30a0 | ||
|  | 02910b0020 | ||
|  | 23b8901c9c | ||
|  | 99f6ed0cd7 | ||
|  | 890c310880 | ||
|  | 0194eeb31f | ||
|  | f9be4c62b1 | ||
|  | 027e8c18f1 | ||
|  | 4a3bb35a95 | ||
|  | 4bfb0d4494 | ||
|  | 7e0ef03a1e | ||
|  | f7dbd95a54 | ||
|  | 515ee2290b | ||
|  | b0c78910bb | ||
|  | f4ca62b664 | ||
|  | 8eb8043a3d | ||
|  | 3e8541362a | ||
|  | 789724e348 | ||
|  | 5125b9532f | ||
|  | ebc9de02b0 | ||
|  | ec788fa491 | ||
|  | 9b5e264574 | ||
|  | 57c297274b | ||
|  | e9bf092317 | ||
|  | d173887324 | ||
|  | 99820d854c | ||
|  | 62df0a0eb2 | ||
|  | 600e9ac947 | ||
|  | 3ca41be2b4 | ||
|  | 5c7debd900 | ||
|  | 7fa5b23ce3 | ||
|  | ff82738aaf | ||
|  | bf5ee9d643 | ||
|  | 72a8593ecd | ||
|  | bc3bbe07d4 | ||
|  | c7cb64bfef | ||
|  | 629f537d06 | ||
|  | 9e988041b8 | ||
|  | f9a8b5c9d7 | ||
|  | b9c3538253 | ||
|  | 2bc0cdf017 | ||
|  | 02a91f60d4 | ||
|  | fae83da197 | ||
|  | 0fe4aa6418 | ||
|  | 21a51bf0dc | ||
|  | bcb353cc30 | ||
|  | 6af4508518 | ||
|  | 6a559bc28a | ||
|  | 0f5026cd20 | ||
|  | a91b80a311 | ||
|  | ec534701c8 | ||
|  | af5169f67f | ||
|  | 18676c5e65 | ||
|  | e2df6fda7b | ||
|  | e9ae9782fe | ||
|  | 016dba4ca9 | ||
|  | 39c7ef305f | ||
|  | 849c1dc848 | ||
|  | 61414014fe | ||
|  | 578a915884 | ||
|  | eacafb8a63 | ||
|  | 4446760f74 | ||
|  | 6da2a083f9 | ||
|  | 8837c8f822 | ||
|  | bac301ed66 | ||
|  | 061db3906d | ||
|  | fd7df5c952 | ||
|  | a270019147 | ||
|  | 55e0209901 | ||
|  | 2b255fbbed | ||
|  | 8a2345a0fb | ||
|  | bfa9f535aa | ||
|  | f757623ad8 | ||
|  | 3c7465e268 | ||
|  | 108665fc4f | ||
|  | ed519c9138 | ||
|  | 2dd2e2c57e | ||
|  | 6c3a976222 | ||
|  | 80cc26bd95 | ||
|  | 970fb84fd8 | ||
|  | 20cbcf6931 | ||
|  | 8fcde2a579 | ||
|  | b32d1f8ad3 | ||
|  | 03513e0cb1 | ||
|  | e041a2b197 | ||
|  | d7d625be2a | ||
|  | 4121266678 | ||
|  | 22971a6be4 | ||
|  | efbf8d7e0d | ||
|  | 397396ea4a | ||
|  | e59b077c21 | ||
|  | 4bc39f3084 | ||
|  | 21c3570786 | ||
|  | 2f85c1fb18 | ||
|  | 1e27a4c2df | ||
|  | 456f575637 | ||
|  | 51546c9e64 | ||
|  | 83b4b70ef4 | ||
|  | a5120d4f6f | ||
|  | c95941e14f | ||
|  | 0dd531149d | ||
|  | 67da1b5219 | ||
|  | 919bd16437 | ||
|  | ecead109ab | ||
|  | 765294c263 | ||
|  | d6b5351207 | ||
|  | a2009bcc6b | ||
|  | 12709a8a0a | ||
|  | c055baefd2 | ||
|  | 56522599b5 | ||
|  | 664f53b75d | ||
|  | 87200d9f10 | ||
|  | 5c3d0b6520 | ||
|  | bd49979f4a | ||
|  | 7e606cdd9f | ||
|  | 8b4b7fa794 | ||
|  | 05345ddf8b | ||
|  | 66adb470ad | ||
|  | e15c8fd146 | ||
|  | 0f09b98a39 | ||
|  | b4d6f4e24d | ||
|  | 3217fa625b | ||
|  | e719ff8a47 | ||
|  | 9fcf528d45 | ||
|  | 1ddbf5a158 | ||
|  | 64bf4574b0 | ||
|  | 5649d26077 | ||
|  | 92f923effe | ||
|  | 0d46d548b9 | ||
|  | 062df3f0c3 | ||
|  | 789fb53b8e | ||
|  | 351db5a18f | ||
|  | aabbd271c8 | ||
|  | aae8e0171e | ||
|  | 45827a2458 | ||
|  | 726030296f | ||
|  | 6659ab3881 | ||
|  | c6a103609e | ||
|  | c6b3f035e5 | ||
|  | 2b0a7e378e | ||
|  | b75ce909c8 | ||
|  | 229c3f5dab | ||
|  | ec73094506 | ||
|  | c7650c9326 | ||
|  | d94c6d4e72 | ||
|  | 3cc8760733 | ||
|  | a2f6973495 | ||
|  | f8648fa651 | ||
|  | 177aa038df | ||
|  | e0a14ec881 | ||
|  | 9366512f2f | ||
|  | ea38b8041a | ||
|  | f1870daf0d | ||
|  | 9722441aad | ||
|  | 9d014087f4 | ||
|  | 83b4038b85 | ||
|  | 1e0a448feb | ||
|  | fb81de3b36 | ||
|  | aa4f352301 | ||
|  | f1a1c2ea45 | ||
|  | 6249bd4163 | ||
|  | 2579dc64ce | ||
|  | 356512270a | ||
|  | bed27f2b43 | ||
|  | 54013d861b | ||
|  | ec100210dc | ||
|  | 3ab1acf32c | ||
|  | 8c28266418 | ||
|  | 7f8b8dcb92 | ||
|  | 6dd39811d4 | ||
|  | 35e2138e3e | ||
|  | 239b4e9fe6 | ||
|  | 2fcd0e7e72 | ||
|  | 357347ce3a | ||
|  | 36dc1107fb | ||
|  | 0a3bbc4b4a | ||
|  | 855b93dcf6 | ||
|  | 89b79ba267 | ||
|  | f5651b7d94 | ||
|  | 1881019ede | ||
|  | caba4e974c | ||
|  | bc3c9613bc | ||
|  | 15a3ee252e | ||
|  | be055961ae | ||
|  | e3031bdeec | ||
|  | 75917b9f7c | ||
|  | 910732e02c | ||
|  | 264b497681 | ||
|  | 372b949622 | ||
|  | 789a602914 | ||
|  | 093e955100 | ||
|  | c32a89bebf | ||
|  | c0bebe9f9f | ||
|  | 57579b2fe5 | ||
|  | 51d14a6b4d | ||
|  | c50f1b64e5 | ||
|  | 98aaab02c5 | ||
|  | 0fc7973d8b | ||
|  | 10362aa02e | ||
|  | 0a8e759fe6 | ||
|  | d70981cdd1 | ||
|  | e08c03b886 | ||
|  | 56086e8984 | ||
|  | 1aa9033022 | ||
|  | 076e103d53 | ||
|  | 38c00ea8fc | ||
|  | 415757af43 | ||
|  | e72ed8c0ed | ||
|  | 32f9c6b5bb | ||
|  | 6251584ef6 | ||
|  | f3e413bc28 | ||
|  | 6f6cc8f3f8 | ||
|  | 8b081e9e69 | ||
|  | c8a510d10e | ||
|  | 6f834f6679 | ||
|  | cf2d6650ac | ||
|  | cd52dea488 | ||
|  | 6ea75df05d | ||
|  | 4846e1e8d6 | ||
|  | fc024f789d | ||
|  | 473e773aea | ||
|  | 48a2e1a353 | ||
|  | 6da63fbd79 | ||
|  | 5bec37fcee | ||
|  | 3fd0ba0a31 | ||
|  | 241a143366 | ||
|  | a537064da7 | ||
|  | f3dfd24c92 | ||
|  | fa0a7f50bb | ||
|  | 44a78a7e21 | ||
|  | 6b75cbf747 | ||
|  | e7b18ab9fe | ||
|  | aa12830015 | ||
|  | f156e00064 | ||
|  | d53c212516 | ||
|  | ca27f8587c | ||
|  | 88ce008e16 | ||
|  | 081d2cc5d7 | ||
|  | 60ac68d000 | ||
|  | fbe656957d | ||
|  | 5534c78c17 | ||
|  | a45a53fdce | ||
|  | 972a56e738 | ||
|  | 5e03b3ca38 | ||
|  | 1078d933b4 | ||
|  | d6bf300d80 | ||
|  | a359d64d44 | ||
|  | 22396e8c33 | ||
|  | 5ded5a4516 | ||
|  | 79c7639aaf | ||
|  | 5bbf875385 | ||
|  | 5e159432af | ||
|  | 1d6ae409f6 | ||
|  | 9d729d3d1a | ||
|  | 4dd5d4e1b7 | ||
|  | acd8149479 | ||
|  | b97a1088fa | ||
|  | b77bed3324 | ||
|  | a2b7c85a1f | ||
|  | b28533f850 | ||
|  | bd8c7e538a | ||
|  | 89e48cff24 | ||
|  | ae90a7b7b6 | ||
|  | 6fc1be04da | ||
|  | 0061d29534 | ||
|  | a891f34a93 | ||
|  | d6a1e62a95 | ||
|  | cda36ea8b4 | ||
|  | 909a76434a | ||
|  | 39348ef659 | ||
|  | 99d30edef3 | ||
|  | b63ab15bf9 | ||
|  | 485cb4495c | ||
|  | df018eb1f2 | ||
|  | 49aa47a9b8 | ||
|  | 7d20eb202a | ||
|  | c533da9129 | ||
|  | 5cba31a814 | ||
|  | 1d824cb26c | ||
|  | 83b903d60e | ||
|  | 9c8ccabe8e | ||
|  | b1f2c4e70d | ||
|  | 273ca0c8da | ||
|  | d6f516b34f | ||
|  | 83127858ca | ||
|  | d89329757e | ||
|  | 49ffec5320 | ||
|  | 2eaae2b66a | ||
|  | ea4441e25c | ||
|  | e5f34042f9 | ||
|  | 271096874a | ||
|  | 8efd780a72 | ||
|  | 41bcf7308d | ||
|  | d102bb3199 | ||
|  | d0bed95415 | ||
|  | 2528729971 | ||
|  | 292c18b3d0 | ||
|  | 0be7c5e2d8 | ||
|  | eb5aaddba4 | ||
|  | d8fd82bcb5 | ||
|  | 97be495861 | ||
|  | 8b53c159fc | ||
|  | 81e281f703 | ||
|  | 3948214050 | ||
|  | c5e9a643e7 | ||
|  | d25881d5c3 | ||
|  | 38d8d9733f | ||
|  | 118ebf668d | ||
|  | a86f09fa46 | ||
|  | dd4fb35c8f | ||
|  | 621eb4cf95 | ||
|  | deea66ad0b | ||
|  | bf99445377 | ||
|  | 7b54a63396 | ||
|  | 0fcb015f9a | ||
|  | 0a22b1ffb6 | ||
|  | 68cecc52ab | ||
|  | 53657ccfff | ||
|  | 96223fda01 | ||
|  | 374ff3433e | ||
|  | 5d63949e98 | ||
|  | 6b065d507d | ||
|  | e79997498a | ||
|  | f7ee02ec35 | ||
|  | 69dc433e1c | ||
|  | c880cd848c | ||
|  | 5752b6db48 | ||
|  | b36f905eab | ||
|  | 483dd527c6 | ||
|  | e55678e28f | ||
|  | 3f4a8b9d6f | ||
|  | 02a856ecb4 | ||
|  | 4dff726310 | ||
|  | cbc449036f | ||
|  | 8f53152220 | ||
|  | bbb1e165d6 | ||
|  | fed8d94885 | ||
|  | 58040cc0ed | ||
|  | 03d692db66 | ||
|  | 903f8e8453 | ||
|  | 405ae1308e | ||
|  | 8a0f583d71 | ||
|  | b6d7017491 | ||
|  | 0f0217d203 | ||
|  | a203e33347 | ||
|  | 3b8f697dd4 | ||
|  | 78ba16f722 | ||
|  | 0fcfe79994 | ||
|  | c0e6df4b63 | ||
|  | 322abdcb43 | ||
|  | 31100787ce | ||
|  | c57d721be4 | ||
|  | 3b5a03e977 | ||
|  | ed807ee43e | ||
|  | 073c130ae6 | ||
|  | 8810e0be13 | ||
|  | f93016ab85 | ||
|  | b19cf260c2 | ||
|  | db03e1e7eb | ||
|  | e0d975e36a | ||
|  | cfeb15259f | ||
|  | 3b3f8fc8fb | ||
|  | 88bd2c084c | ||
|  | bd367389b0 | ||
|  | 58ba71a76f | ||
|  | d03e34d55d | ||
|  | 24f239a46c | ||
|  | 2c0826f85a | ||
|  | c061461d01 | ||
|  | e7982a04fe | ||
|  | 33b91a7513 | ||
|  | 9bb1323e44 | ||
|  | e62bb807a5 | ||
|  | 3fc0d2cc4a | ||
|  | 0c786b0766 | ||
|  | 68c7528911 | ||
|  | 26e18ae800 | ||
|  | c30dc0b546 | ||
|  | f94aa46a11 | ||
|  | 403261a293 | ||
|  | c7d9cbb11f | ||
|  | 57e1c53cbb | ||
|  | 0754b553dd | ||
|  | 50661d941b | ||
|  | c5db7c1a0c | ||
|  | 2cef5365f7 | ||
|  | fbc4e94007 | ||
|  | 037ed5a2ad | ||
|  | 69dfa55705 | ||
|  | a79a5c4e3e | ||
|  | 7e80eabfe6 | ||
|  | 375b72770d | ||
|  | e2dd683def | ||
|  | 9eba50c6e4 | ||
|  | 5a579dba52 | ||
|  | e86c719575 | ||
|  | 0e87f35547 | ||
|  | b6d3d791a5 | ||
|  | c9c3302664 | ||
|  | c3e4d65b80 | ||
|  | 27a03510c5 | ||
|  | ed7727f7cb | ||
|  | 127ec10c0d | ||
|  | 5a9c0ad225 | ||
|  | 7e8daf650e | ||
|  | 0cf737b4ce | ||
|  | 74635e0113 | ||
|  | e5c4f49901 | ||
|  | e4654ee7f1 | ||
|  | e5d05c05ed | ||
|  | 73c4f99687 | ||
|  | 28c12ef3bf | ||
|  | eed82dbb54 | ||
|  | 2c4b4ab928 | ||
|  | 505a8fc6f6 | ||
|  | e4801d9b06 | ||
|  | 04f1b2cf3a | ||
|  | c06d928bb5 | ||
|  | ab09927e7b | ||
|  | 779437db67 | ||
|  | 28cbdb652e | ||
|  | 2b2415a7d8 | ||
|  | 746a8208aa | ||
|  | a2a041a98a | ||
|  | 10b436e449 | ||
|  | 4d62b34786 | ||
|  | 0546210687 | ||
|  | f8c11faada | ||
|  | 16d6e9be1f | ||
|  | aff8185f2e | ||
|  | 217d15fe81 | ||
|  | 171e93c201 | ||
|  | acc1d2e9e3 | ||
|  | 49c2f37154 | ||
|  | 69e54497aa | ||
|  | 9aa1885669 | ||
|  | 4418508513 | ||
|  | e897df3b34 | ||
|  | 8cd97ab0e7 | ||
|  | bf4949353d | ||
|  | 98a944f7cc | ||
|  | 7c10f81c92 | ||
|  | 126ecc55c3 | ||
|  | 1034a51bd2 | ||
|  | a2657887cc | ||
|  | c14b17bfaf | ||
|  | 59ebc795e7 | ||
|  | 8e128d917e | ||
|  | ea762b05e0 | ||
|  | db374b19f1 | ||
|  | ab3839ef36 | ||
|  | 9886c442f2 | ||
|  | c8d1926d52 | ||
|  | a6bd699e52 | ||
|  | 12143f2702 | ||
|  | 480705dee9 | ||
|  | 781d5094f4 | ||
|  | 5615cb94cd | ||
|  | 302302a2ac | ||
|  | 9761b4e3e9 | ||
|  | 0cf6924dca | ||
|  | 5fd81e9f90 | ||
|  | 52bf6f892b | ||
|  | f3cce232a4 | ||
|  | 53d3c8b28e | ||
|  | 83fec3cca7 | ||
|  | 3cefc99b7d | ||
|  | 3a38dcbc05 | ||
|  | 7ff08bce57 | ||
|  | fd490af434 | ||
|  | 1195b8f17e | ||
|  | 28dce13776 | ||
|  | 431f20177a | ||
|  | 87aff54d9d | ||
|  | f50462de82 | ||
|  | 9bda8c7eb6 | ||
|  | e83c63d239 | ||
|  | b38533b0cc | ||
|  | 5ccca3fbd5 | ||
|  | 9e850fc3ab | ||
|  | ffbfcd7e00 | ||
|  | 5ea7590748 | ||
|  | 290c3bc2bb | ||
|  | b12131e91c | ||
|  | 3b354447b0 | ||
|  | d09ec6feaa | ||
|  | 21405c3fda | ||
|  | 13e5c96cab | ||
|  | 426687b75e | ||
|  | c8f59fb978 | ||
|  | 871dde79a9 | ||
|  | e14d81bc6f | ||
|  | 514d046d1f | ||
|  | 4ed9528d36 | ||
|  | 625560e642 | ||
|  | 73ebd917d1 | ||
|  | cd3e0afad2 | ||
|  | d8d1f94a86 | ||
|  | 00dfd8cfd1 | ||
|  | 273de6db31 | ||
|  | c6c0eeb0ff | ||
|  | e70c74a3b5 | ||
|  | f7d939eeab | ||
|  | e815c091b9 | ||
|  | 963529b7cf | ||
|  | 638a52374d | ||
|  | d9d42b7aa2 | ||
|  | ec7e5f36a2 | ||
|  | 56110883ea | ||
|  | 7f8d7d6006 | ||
|  | 49e4fb7e12 | ||
|  | 8dbbea473f | ||
|  | 3d375d5114 | ||
|  | f3eae67d97 | ||
|  | 40c1b19235 | ||
|  | ccaf0ab159 | ||
|  | d07f147423 | ||
|  | f5cb9f92b9 | ||
|  | f991f74983 | ||
|  | 6b3295059e | ||
|  | b18a07ae6b | ||
|  | 8ab03dabda | ||
|  | 5e760e35dc | ||
|  | afbfa04514 | ||
|  | 7aace470c5 | ||
|  | b4acb24f6a | ||
|  | bcee8a4934 | ||
|  | 36b0718542 | ||
|  | 9a92bca45d | ||
|  | b07445a363 | ||
|  | a62ec0c27e | ||
|  | 57e3a2d382 | ||
|  | b61022b374 | ||
|  | a3e2b2ec87 | ||
|  | a83d3f8801 | ||
|  | 90c5f2b9d2 | ||
|  | 4885653c07 | ||
|  | 21e1cd87ca | ||
|  | 81f82e8e9f | ||
|  | c0e31851da | ||
|  | 6599c3eced | ||
|  | 5d6c61a861 | ||
|  | 1a5c66edd3 | ||
|  | deae9fe95a | ||
|  | abd65c6334 | ||
|  | 8137a99904 | ||
|  | 6f6f9c1f74 | ||
|  | 7b575f716f | ||
|  | 6ba6ea3572 | ||
|  | 9a22ad5ea3 | ||
|  | beaab9778e | ||
|  | f327bdb6b4 | ||
|  | ae180e0f5f | ||
|  | e3f1d19756 | ||
|  | 93c2bd6ef6 | ||
|  | 4d0e5ff6db | ||
|  | 0893f06919 | ||
|  | 46b6abde3f | ||
|  | 0696610dee | ||
|  | edf0d3684c | ||
|  | 7af159f5f6 | ||
|  | 7f2cb6764a | ||
|  | 96495a9bf1 | ||
|  | b2fafec5fc | ||
|  | 0850b8ae2b | ||
|  | 8a68a96c57 | ||
|  | d3aae8ed6a | ||
|  | c62ebadda8 | ||
|  | ffcee6d390 | ||
|  | de32838346 | ||
|  | b9a4e47ea2 | ||
|  | 57d994422d | ||
|  | 6ecd745323 | ||
|  | bd769f5bdb | ||
|  | 2381692aba | ||
|  | 24fdada0a0 | ||
|  | bb5169710a | ||
|  | 9cde2352f3 | ||
|  | 482dd7a938 | ||
|  | bddcc69438 | ||
|  | 19d4540630 | ||
|  | 4f5f6c81f5 | ||
|  | 7e4c1238ba | ||
|  | f7196ac773 | ||
|  | 7a7c832000 | ||
|  | 2b4ccdbebb | ||
|  | 0d16b49489 | ||
|  | 768405b691 | ||
|  | da01413b7b | ||
|  | 914e22c53e | ||
|  | 43a23bf733 | ||
|  | 92bb00c6d2 | ||
|  | b0b97a2648 | ||
|  | 2c452fe323 | ||
|  | ad73d0c77d | ||
|  | 7f9bf1c78c | ||
|  | 61a6bc3a65 | ||
|  | 46e10b0e9f | ||
|  | 8441206e26 | ||
|  | 9fdc5ee748 | ||
|  | 00ff133387 | ||
|  | 96164cb934 | ||
|  | 82fb21ae69 | ||
|  | 89d4a2b4c4 | ||
|  | fc0c7ff374 | ||
|  | 5148c4f2e9 | ||
|  | c3b59f7bcf | ||
|  | 61e148202b | ||
|  | 8a4e0739bc | ||
|  | f75c5f2fe5 | ||
|  | 81d5859588 | ||
|  | 721886bb7a | ||
|  | b23c272820 | ||
|  | cd02bfea7a | ||
|  | 6774bd88f9 | ||
|  | 1046a4f376 | ||
|  | 8081f9ddfd | ||
|  | fa656577d1 | ||
|  | b14b86990f | ||
|  | 2a6dd7b512 | ||
|  | feebdee88b | ||
|  | 99d9277f5d | ||
|  | 9af64d6156 | ||
|  | 5e3775c1af | ||
|  | 2d2e8a3da7 | ||
|  | b2a560b76f | ||
|  | 39397a489d | ||
|  | ff593a0904 | ||
|  | f12789cf44 | ||
|  | 4f8cf2fc87 | ||
|  | fda98730ac | ||
|  | 06c6ddffb6 | ||
|  | d29f0c066c | ||
|  | c9e4de3346 | ||
|  | ca0b97f72d | ||
|  | b38f20b408 | ||
|  | 05b1dbaf56 | ||
|  | b8481e32ba | ||
|  | 9c03c65e07 | ||
|  | d8ed006b9b | ||
|  | 63c0623a5e | ||
|  | fd84506db0 | ||
|  | d8bcb44e44 | ||
|  | 56a26b0916 | ||
|  | efcf1d6b90 | ||
|  | 9f578bfec6 | ||
|  | 1f170d7d28 | ||
|  | 5ae14cf9be | ||
|  | aaf9d53be9 | ||
|  | 75c73f7ba7 | ||
|  | b6dba8beee | ||
|  | 94521cdc1a | ||
|  | 3365b1c355 | ||
|  | 6c957c4923 | ||
|  | 833997f04c | ||
|  | 68d51e4037 | ||
|  | ce274d2011 | ||
|  | 280778ed43 | ||
|  | 0f558ecbbf | ||
|  | 58f9e05d93 | ||
|  | 1ec981aea7 | ||
|  | 2a90286a7c | ||
|  | 12d25d09b2 | ||
|  | a039fae1a4 | ||
|  | 322b9abadc | ||
|  | 0aaf954cea | ||
|  | c2d22aa3d1 | ||
|  | 6934c75bba | ||
|  | c58cf78f86 | ||
|  | 7f0de790ab | ||
|  | d4bb4e3a73 | ||
|  | d25612d038 | ||
|  | 116b2351b0 | ||
|  | 69b83dfdc4 | ||
|  | 3b1839c2ce | ||
|  | 13742ebdf8 | ||
|  | 634657bea1 | ||
|  | 46e70d50b7 | ||
|  | d64e9b85a7 | ||
|  | fb853edbe3 | ||
|  | cc076c1be1 | ||
|  | 98cc9a6755 | ||
|  | 7bd2b9c23a | ||
|  | de724a1ff3 | ||
|  | 2163055dae | ||
|  | 93ed0fc10b | ||
|  | 0d98cefd40 | ||
|  | d58988a033 | ||
|  | 2acfab1e3f | ||
|  | b915dfe9a6 | ||
|  | 25bd5a823e | ||
|  | 1c35de4716 | ||
|  | 4c00435a0a | ||
|  | 844e3079a8 | ||
|  | 4778cb5b2c | ||
|  | ec5d60b919 | ||
|  | e1f4b960e8 | ||
|  | 669e46da54 | ||
|  | ba94cc5df7 | ||
|  | d08245c3df | ||
|  | 5c18d12cbf | ||
|  | 580a42dec7 | ||
|  | 29286e159b | ||
|  | 19bcf90e9f | ||
|  | dae9c00742 | ||
|  | 35324ceb7c | ||
|  | 5aadd47199 | ||
|  | 7d9057cc62 | ||
|  | c4b322b883 | ||
|  | 19b09c898a | ||
|  | eafe2098b6 | ||
|  | 2bc6a20d71 | ||
|  | 8b502a7235 | ||
|  | 37567844af | ||
|  | 2f6c4e0e34 | ||
|  | 1c7cc4cb2b | ||
|  | f83db3648e | ||
|  | b164aa00d4 | ||
|  | a2d866d0c2 | ||
|  | 2dfe4ac4c6 | ||
|  | db65d05cb5 | ||
|  | 300c0194c7 | ||
|  | 37a0d2b087 | ||
|  | a4959300ea | ||
|  | 223657e5f8 | ||
|  | 0c53de6767 | ||
|  | 9c309b1498 | ||
|  | 1aa1b34c80 | ||
|  | 755a2ee023 | ||
|  | 69d3359e47 | ||
|  | a90c49b8fb | ||
|  | b1222edb27 | ||
|  | b967a92f69 | ||
|  | 90a5cb5e59 | ||
|  | 7aba9cb76b | ||
|  | f550a8171d | ||
|  | 82e568d4c9 | ||
|  | 7b2a4a3d59 | ||
|  | 0265455cd1 | ||
|  | afafc886a4 | ||
|  | 8a959f6ac4 | ||
|  | 1c3aa0d2c5 | ||
|  | 79b7d3316a | ||
|  | fa7768583a | ||
|  | faf49f6c15 | ||
|  | 765af31b83 | ||
|  | b6a3c52d67 | ||
|  | b025c2f660 | ||
|  | e559a7c878 | ||
|  | 5c8855aafd | ||
|  | b5fc537b89 | ||
|  | 14899d3a7c | ||
|  | 0ea7881652 | ||
|  | ec29b59d1e | ||
|  | 9405597c15 | ||
|  | 82441978c6 | ||
|  | e0e6291bdb | ||
|  | b2b083fd0a | ||
|  | f8a51b68e7 | ||
|  | e0a19108e5 | ||
|  | 770ea68ca8 | ||
|  | ce36c52baf | 
							
								
								
									
										12
									
								
								.eslintrc.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.eslintrc.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| { | ||||
|     "env": { | ||||
|         "browser": true, | ||||
|         "es2021": true | ||||
|     }, | ||||
|     "extends": "eslint:recommended", | ||||
|     "parserOptions": { | ||||
|         "ecmaVersion": 12 | ||||
|     }, | ||||
|     "rules": { | ||||
|     } | ||||
| } | ||||
							
								
								
									
										2
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,6 @@ | ||||
| * text eol=lf | ||||
|  | ||||
| *.reg text eol=crlf | ||||
|  | ||||
| *.png binary | ||||
| *.gif binary | ||||
|   | ||||
							
								
								
									
										40
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| --- | ||||
| name: Bug report | ||||
| about: Create a report to help us improve | ||||
| title: '' | ||||
| labels: bug | ||||
| assignees: '9001' | ||||
|  | ||||
| --- | ||||
|  | ||||
| NOTE: | ||||
| all of the below are optional, consider them as inspiration, delete and rewrite at will, thx md | ||||
|  | ||||
|  | ||||
| **Describe the bug** | ||||
| a description of what the bug is | ||||
|  | ||||
| **To Reproduce** | ||||
| List of steps to reproduce the issue, or, if it's hard to reproduce, then at least a detailed explanation of what you did to run into it | ||||
|  | ||||
| **Expected behavior** | ||||
| a description of what you expected to happen | ||||
|  | ||||
| **Screenshots** | ||||
| if applicable, add screenshots to help explain your problem, such as the kickass crashpage :^) | ||||
|  | ||||
| **Server details** | ||||
| if the issue is possibly on the server-side, then mention some of the following: | ||||
| * server OS / version:  | ||||
| * python version:  | ||||
| * copyparty arguments:  | ||||
| * filesystem (`lsblk -f` on linux):  | ||||
|  | ||||
| **Client details** | ||||
| if the issue is possibly on the client-side, then mention some of the following: | ||||
| * the device type and model:  | ||||
| * OS version:  | ||||
| * browser version:  | ||||
|  | ||||
| **Additional context** | ||||
| any other context about the problem here | ||||
							
								
								
									
										22
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| --- | ||||
| name: Feature request | ||||
| about: Suggest an idea for this project | ||||
| title: '' | ||||
| labels: enhancement | ||||
| assignees: '9001' | ||||
|  | ||||
| --- | ||||
|  | ||||
| all of the below are optional, consider them as inspiration, delete and rewrite at will | ||||
|  | ||||
| **is your feature request related to a problem? Please describe.** | ||||
| a description of what the problem is, for example, `I'm always frustrated when [...]` or `Why is it not possible to [...]` | ||||
|  | ||||
| **Describe the idea / solution you'd like** | ||||
| a description of what you want to happen | ||||
|  | ||||
| **Describe any alternatives you've considered** | ||||
| a description of any alternative solutions or features you've considered | ||||
|  | ||||
| **Additional context** | ||||
| add any other context or screenshots about the feature request here | ||||
							
								
								
									
										10
									
								
								.github/ISSUE_TEMPLATE/something-else.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								.github/ISSUE_TEMPLATE/something-else.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| --- | ||||
| name: Something else | ||||
| about: "┐(゚∀゚)┌" | ||||
| title: '' | ||||
| labels: '' | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
|  | ||||
							
								
								
									
										7
									
								
								.github/branch-rename.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								.github/branch-rename.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| modernize your local checkout of the repo like so, | ||||
| ```sh | ||||
| git branch -m master hovudstraum | ||||
| git fetch origin | ||||
| git branch -u origin/hovudstraum hovudstraum | ||||
| git remote set-head origin -a | ||||
| ``` | ||||
							
								
								
									
										28
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										28
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @@ -10,12 +10,24 @@ | ||||
|             "cwd": "${workspaceFolder}", | ||||
|             "args": [ | ||||
|                 //"-nw", | ||||
|                 "-a", | ||||
|                 "ed:wark", | ||||
|                 "-v", | ||||
|                 "srv::r:aed" | ||||
|                 "-ed", | ||||
|                 "-emp", | ||||
|                 "-e2dsa", | ||||
|                 "-e2ts", | ||||
|                 "-mtp", | ||||
|                 ".bpm=f,bin/mtag/audio-bpm.py", | ||||
|                 "-aed:wark", | ||||
|                 "-vsrv::r:aed:cnodupe", | ||||
|                 "-vdist:dist:r" | ||||
|             ] | ||||
|         }, | ||||
|         { | ||||
|             "name": "No debug", | ||||
|             "preLaunchTask": "no_dbg", | ||||
|             "type": "python", | ||||
|             //"request": "attach", "port": 42069 | ||||
|             // fork: nc -l 42069 </dev/null | ||||
|         }, | ||||
|         { | ||||
|             "name": "Run active unit test", | ||||
|             "type": "python", | ||||
| @@ -28,5 +40,13 @@ | ||||
|                 "${file}" | ||||
|             ] | ||||
|         }, | ||||
|         { | ||||
|             "name": "Python: Current File", | ||||
|             "type": "python", | ||||
|             "request": "launch", | ||||
|             "program": "${file}", | ||||
|             "console": "integratedTerminal", | ||||
|             "justMyCode": false | ||||
|         }, | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										45
									
								
								.vscode/launch.py
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								.vscode/launch.py
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| # takes arguments from launch.json | ||||
| # is used by no_dbg in tasks.json | ||||
| # launches 10x faster than mspython debugpy | ||||
| # and is stoppable with ^C | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
|  | ||||
| print(sys.executable) | ||||
|  | ||||
| import shlex | ||||
| import jstyleson | ||||
| import subprocess as sp | ||||
|  | ||||
|  | ||||
| with open(".vscode/launch.json", "r", encoding="utf-8") as f: | ||||
|     tj = f.read() | ||||
|  | ||||
| oj = jstyleson.loads(tj) | ||||
| argv = oj["configurations"][0]["args"] | ||||
|  | ||||
| try: | ||||
|     sargv = " ".join([shlex.quote(x) for x in argv]) | ||||
|     print(sys.executable + " -m copyparty " + sargv + "\n") | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| argv = [os.path.expanduser(x) if x.startswith("~") else x for x in argv] | ||||
|  | ||||
| if re.search(" -j ?[0-9]", " ".join(argv)): | ||||
|     argv = [sys.executable, "-m", "copyparty"] + argv | ||||
|     sp.check_call(argv) | ||||
| else: | ||||
|     sys.path.insert(0, os.getcwd()) | ||||
|     from copyparty.__main__ import main as copyparty | ||||
|  | ||||
|     try: | ||||
|         copyparty(["a"] + argv) | ||||
|     except SystemExit as ex: | ||||
|         if ex.code: | ||||
|             raise | ||||
|  | ||||
| print("\n\033[32mokke\033[0m") | ||||
| sys.exit(1) | ||||
							
								
								
									
										14
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -37,7 +37,7 @@ | ||||
|     "python.linting.banditEnabled": true, | ||||
|     "python.linting.flake8Args": [ | ||||
|         "--max-line-length=120", | ||||
|         "--ignore=E722,F405,E203,W503,W293", | ||||
|         "--ignore=E722,F405,E203,W503,W293,E402", | ||||
|     ], | ||||
|     "python.linting.banditArgs": [ | ||||
|         "--ignore=B104" | ||||
| @@ -50,11 +50,9 @@ | ||||
|     "files.associations": { | ||||
|         "*.makefile": "makefile" | ||||
|     }, | ||||
|     "editor.codeActionsOnSaveTimeout": 9001, | ||||
|     "editor.formatOnSaveTimeout": 9001, | ||||
|     // | ||||
|     //  things you may wanna edit: | ||||
|     // | ||||
|     "python.pythonPath": ".venv/bin/python", | ||||
|     //"python.linting.enabled": true, | ||||
|     "python.formatting.blackArgs": [ | ||||
|         "-t", | ||||
|         "py27" | ||||
|     ], | ||||
|     "python.linting.enabled": true, | ||||
| } | ||||
							
								
								
									
										18
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| { | ||||
|     "version": "2.0.0", | ||||
|     "tasks": [ | ||||
|         { | ||||
|             "label": "pre", | ||||
|             "command": "true;rm -rf inc/* inc/.hist/;mkdir -p inc;", | ||||
|             "type": "shell" | ||||
|         }, | ||||
|         { | ||||
|             "label": "no_dbg", | ||||
|             "type": "shell", | ||||
|             "command": "${config:python.pythonPath}", | ||||
|             "args": [ | ||||
|                 ".vscode/launch.py" | ||||
|             ] | ||||
|         } | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										24
									
								
								CODE_OF_CONDUCT.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								CODE_OF_CONDUCT.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| in the words of Abraham Lincoln: | ||||
|  | ||||
| > Be excellent to each other... and... PARTY ON, DUDES! | ||||
|  | ||||
| more specifically I'll paraphrase some examples from a german automotive corporation as they cover all the bases without being too wordy | ||||
|  | ||||
| ## Examples of unacceptable behavior | ||||
| * intimidation, harassment, trolling | ||||
| * insulting, derogatory, harmful or prejudicial comments | ||||
| * posting private information without permission | ||||
| * political or personal attacks | ||||
|  | ||||
| ## Examples of expected behavior | ||||
| * being nice, friendly, welcoming, inclusive, mindful and empathetic | ||||
| * acting considerate, modest, respectful | ||||
| * using polite and inclusive language | ||||
| * criticize constructively and accept constructive criticism | ||||
| * respect different points of view | ||||
|  | ||||
| ## finally and even more specifically, | ||||
| * parse opinions and feedback objectively without prejudice | ||||
|   * it's the message that matters, not who said it | ||||
|  | ||||
| aaand that's how you say `be nice` in a way that fills half a floppy w | ||||
							
								
								
									
										3
									
								
								CONTRIBUTING.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								CONTRIBUTING.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| * do something cool | ||||
|  | ||||
| really tho, send a PR or an issue or whatever, all appreciated, anything goes, just behave aight | ||||
							
								
								
									
										934
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										934
									
								
								README.md
									
									
									
									
									
								
							| @@ -6,102 +6,916 @@ | ||||
|  | ||||
| ## summary | ||||
|  | ||||
| turn your phone or raspi into a portable file server with resumable uploads/downloads using IE6 or any other browser | ||||
| turn your phone or raspi into a portable file server with resumable uploads/downloads using *any* web browser | ||||
|  | ||||
| * server runs on anything with `py2.7` or `py3.2+` | ||||
| * *resumable* uploads need `firefox 12+` / `chrome 6+` / `safari 6+` / `IE 10+` | ||||
| * server only needs `py2.7` or `py3.3+`, all dependencies optional | ||||
| * browse/upload with IE4 / netscape4.0 on win3.11 (heh) | ||||
| * *resumable* uploads need `firefox 34+` / `chrome 41+` / `safari 7+` for full speed | ||||
| * code standard: `black` | ||||
|  | ||||
| 📷 **screenshots:** [browser](#the-browser) // [upload](#uploading) // [unpost](#unpost) // [thumbnails](#thumbnails) // [search](#searching) // [fsearch](#file-search) // [zip-DL](#zip-downloads) // [md-viewer](#markdown-viewer) // [ie4](#browser-support) | ||||
|  | ||||
|  | ||||
| ## readme toc | ||||
|  | ||||
| * top | ||||
|     * **[quickstart](#quickstart)** - download **[copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py)** and you're all set! | ||||
|         * [on servers](#on-servers) - you may also want these, especially on servers | ||||
|         * [on debian](#on-debian) - recommended additional steps on debian | ||||
|     * [notes](#notes) - general notes | ||||
|     * [status](#status) - feature summary | ||||
|     * [testimonials](#testimonials) - small collection of user feedback | ||||
| * [bugs](#bugs) | ||||
|     * [general bugs](#general-bugs) | ||||
|     * [not my bugs](#not-my-bugs) | ||||
| * [accounts and volumes](#accounts-and-volumes) - per-folder, per-user permissions | ||||
| * [the browser](#the-browser) - accessing a copyparty server using a web-browser | ||||
|     * [tabs](#tabs) - the main tabs in the ui | ||||
|     * [hotkeys](#hotkeys) - the browser has the following hotkeys | ||||
|     * [navpane](#navpane) - switching between breadcrumbs or navpane | ||||
|     * [thumbnails](#thumbnails) - press `g` to toggle grid-view instead of the file listing | ||||
|     * [zip downloads](#zip-downloads) - download folders (or file selections) as `zip` or `tar` files | ||||
|     * [uploading](#uploading) - drag files/folders into the web-browser to upload | ||||
|         * [file-search](#file-search) - dropping files into the browser also lets you see if they exist on the server | ||||
|         * [unpost](#unpost) - undo/delete accidental uploads | ||||
|     * [file manager](#file-manager) - cut/paste, rename, and delete files/folders (if you have permission) | ||||
|     * [batch rename](#batch-rename) - select some files and press `F2` to bring up the rename UI | ||||
|     * [markdown viewer](#markdown-viewer) - and there are *two* editors | ||||
|     * [other tricks](#other-tricks) | ||||
|     * [searching](#searching) - search by size, date, path/name, mp3-tags, ... | ||||
| * [server config](#server-config) | ||||
|     * [file indexing](#file-indexing) | ||||
|     * [upload rules](#upload-rules) - set upload rules using volume flags | ||||
|     * [compress uploads](#compress-uploads) - files can be autocompressed on upload | ||||
|     * [database location](#database-location) - in-volume (`.hist/up2k.db`, default) or somewhere else | ||||
|     * [metadata from audio files](#metadata-from-audio-files) - set `-e2t` to index tags on upload | ||||
|     * [file parser plugins](#file-parser-plugins) - provide custom parsers to index additional tags | ||||
|     * [complete examples](#complete-examples) | ||||
| * [browser support](#browser-support) - TLDR: yes | ||||
| * [client examples](#client-examples) - interact with copyparty using non-browser clients | ||||
| * [up2k](#up2k) - quick outline of the up2k protocol, see [uploading](#uploading) for the web-client | ||||
|     * [why chunk-hashes](#why-chunk-hashes) - a single sha512 would be better, right? | ||||
| * [performance](#performance) - defaults are usually fine - expect `8 GiB/s` download, `1 GiB/s` upload | ||||
| * [security](#security) - some notes on hardening | ||||
|     * [gotchas](#gotchas) - behavior that might be unexpected | ||||
| * [dependencies](#dependencies) - mandatory deps | ||||
|     * [optional dependencies](#optional-dependencies) - install these to enable bonus features | ||||
|     * [install recommended deps](#install-recommended-deps) | ||||
|     * [optional gpl stuff](#optional-gpl-stuff) | ||||
| * [sfx](#sfx) - there are two self-contained "binaries" | ||||
|     * [sfx repack](#sfx-repack) - reduce the size of an sfx by removing features | ||||
| * [install on android](#install-on-android) | ||||
| * [building](#building) | ||||
|     * [dev env setup](#dev-env-setup) | ||||
|     * [just the sfx](#just-the-sfx) | ||||
|     * [complete release](#complete-release) | ||||
| * [todo](#todo) - roughly sorted by priority | ||||
|     * [discarded ideas](#discarded-ideas) | ||||
|  | ||||
|  | ||||
| ## quickstart | ||||
|  | ||||
| download **[copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py)** and you're all set! | ||||
|  | ||||
| running the sfx without arguments (for example doubleclicking it on Windows) will give everyone read/write access to the current folder; see `-h` for help if you want [accounts and volumes](#accounts-and-volumes) etc | ||||
|  | ||||
| some recommended options: | ||||
| * `-e2dsa` enables general [file indexing](#file-indexing) | ||||
| * `-e2ts` enables audio metadata indexing (needs either FFprobe or Mutagen), see [optional dependencies](#optional-dependencies) | ||||
| * `-v /mnt/music:/music:r:rw,foo -a foo:bar` shares `/mnt/music` as `/music`, `r`eadable by anyone, and read-write for user `foo`, password `bar` | ||||
|   * replace `:r:rw,foo` with `:r,foo` to only make the folder readable by `foo` and nobody else | ||||
|   * see [accounts and volumes](#accounts-and-volumes) for the syntax and other access levels (`r`ead, `w`rite, `m`ove, `d`elete) | ||||
| * `--ls '**,*,ln,p,r'` to crash on startup if any of the volumes contain a symlink which point outside the volume, as that could give users unintended access | ||||
|  | ||||
|  | ||||
| ### on servers | ||||
|  | ||||
| you may also want these, especially on servers: | ||||
|  | ||||
| * [contrib/systemd/copyparty.service](contrib/systemd/copyparty.service) to run copyparty as a systemd service | ||||
| * [contrib/systemd/prisonparty.service](contrib/systemd/prisonparty.service) to run it in a chroot (for extra security) | ||||
| * [contrib/nginx/copyparty.conf](contrib/nginx/copyparty.conf) to reverse-proxy behind nginx (for better https) | ||||
|  | ||||
|  | ||||
| ### on debian | ||||
|  | ||||
| recommended additional steps on debian  which enable audio metadata and thumbnails (from images and videos): | ||||
|  | ||||
| * as root, run the following:   | ||||
|   `apt install python3 python3-pip python3-dev ffmpeg` | ||||
|  | ||||
| * then, as the user which will be running copyparty (so hopefully not root), run this:   | ||||
|   `python3 -m pip install --user -U Pillow pillow-avif-plugin` | ||||
|  | ||||
| (skipped `pyheif-pillow-opener` because apparently debian is too old to build it) | ||||
|  | ||||
|  | ||||
| ## notes | ||||
|  | ||||
| * iPhone/iPad: use Firefox to download files | ||||
| * Android-Chrome: set max "parallel uploads" for 200% upload speed (android bug) | ||||
| * Android-Firefox: takes a while to select files (in order to avoid the above android-chrome issue) | ||||
| * Desktop-Firefox: may use gigabytes of RAM if your connection is great and your files are massive | ||||
| general notes: | ||||
| * paper-printing is affected by dark/light-mode! use lightmode for color, darkmode for grayscale | ||||
|   * because no browsers currently implement the media-query to do this properly orz | ||||
|  | ||||
| browser-specific: | ||||
| * iPhone/iPad: use Firefox to download files | ||||
| * Android-Chrome: increase "parallel uploads" for higher speed (android bug) | ||||
| * Android-Firefox: takes a while to select files (their fix for ☝️) | ||||
| * Desktop-Firefox: ~~may use gigabytes of RAM if your files are massive~~ *seems to be OK now* | ||||
| * Desktop-Firefox: may stop you from deleting files you've uploaded until you visit `about:memory` and click `Minimize memory usage` | ||||
|  | ||||
|  | ||||
| ## status | ||||
|  | ||||
| * [x] sanic multipart parser | ||||
| * [x] load balancer (multiprocessing) | ||||
| * [x] upload (plain multipart, ie6 support) | ||||
| * [x] upload (js, resumable, multithreaded) | ||||
| * [x] download | ||||
| * [x] browser | ||||
| * [x] media player | ||||
| * [ ] thumbnails | ||||
| * [ ] download as zip | ||||
| * [x] volumes | ||||
| * [x] accounts | ||||
| * [x] markdown viewer | ||||
| * [x] markdown editor | ||||
| feature summary | ||||
|  | ||||
| summary: it works! you can use it! (but technically not even close to beta) | ||||
| * backend stuff | ||||
|   * ☑ sanic multipart parser | ||||
|   * ☑ multiprocessing (actual multithreading) | ||||
|   * ☑ volumes (mountpoints) | ||||
|   * ☑ [accounts](#accounts-and-volumes) | ||||
| * upload | ||||
|   * ☑ basic: plain multipart, ie6 support | ||||
|   * ☑ [up2k](#uploading): js, resumable, multithreaded | ||||
|   * ☑ stash: simple PUT filedropper | ||||
|   * ☑ [unpost](#unpost): undo/delete accidental uploads | ||||
|   * ☑ symlink/discard existing files (content-matching) | ||||
| * download | ||||
|   * ☑ single files in browser | ||||
|   * ☑ [folders as zip / tar files](#zip-downloads) | ||||
|   * ☑ FUSE client (read-only) | ||||
| * browser | ||||
|   * ☑ [navpane](#navpane) (directory tree sidebar) | ||||
|   * ☑ file manager (cut/paste, delete, [batch-rename](#batch-rename)) | ||||
|   * ☑ audio player (with OS media controls) | ||||
|   * ☑ image gallery with webm player | ||||
|   * ☑ [thumbnails](#thumbnails) | ||||
|     * ☑ ...of images using Pillow | ||||
|     * ☑ ...of videos using FFmpeg | ||||
|     * ☑ cache eviction (max-age; maybe max-size eventually) | ||||
|   * ☑ SPA (browse while uploading) | ||||
|     * if you use the navpane to navigate, not folders in the file list | ||||
| * server indexing | ||||
|   * ☑ [locate files by contents](#file-search) | ||||
|   * ☑ search by name/path/date/size | ||||
|   * ☑ [search by ID3-tags etc.](#searching) | ||||
| * markdown | ||||
|   * ☑ [viewer](#markdown-viewer) | ||||
|   * ☑ editor (sure why not) | ||||
|  | ||||
|  | ||||
| ## testimonials | ||||
|  | ||||
| small collection of user feedback | ||||
|  | ||||
| `good enough`, `surprisingly correct`, `certified good software`, `just works`, `why` | ||||
|  | ||||
|  | ||||
| # bugs | ||||
|  | ||||
| * Windows: python 3.7 and older cannot read tags with FFprobe, so use Mutagen or upgrade | ||||
| * Windows: python 2.7 cannot index non-ascii filenames with `-e2d` | ||||
| * Windows: python 2.7 cannot handle filenames with mojibake | ||||
| * `--th-ff-jpg` may fix video thumbnails on some FFmpeg versions (macos, some linux) | ||||
|  | ||||
| ## general bugs | ||||
|  | ||||
| * all volumes must exist / be available on startup; up2k (mtp especially) gets funky otherwise | ||||
| * probably more, pls let me know | ||||
|  | ||||
| ## not my bugs | ||||
|  | ||||
| * Windows: folders cannot be accessed if the name ends with `.` | ||||
|   * python or windows bug | ||||
|  | ||||
| * Windows: msys2-python 3.8.6 occasionally throws `RuntimeError: release unlocked lock` when leaving a scoped mutex in up2k | ||||
|   * this is an msys2 bug, the regular windows edition of python is fine | ||||
|  | ||||
| * VirtualBox: sqlite throws `Disk I/O Error` when running in a VM and the up2k database is in a vboxsf | ||||
|   * use `--hist` or the `hist` volflag (`-v [...]:c,hist=/tmp/foo`) to place the db inside the vm instead | ||||
|  | ||||
|  | ||||
| # accounts and volumes | ||||
|  | ||||
| per-folder, per-user permissions | ||||
| * `-a usr:pwd` adds account `usr` with password `pwd` | ||||
| * `-v .::r` adds current-folder `.` as the webroot, `r`eadable by anyone | ||||
|   * the syntax is `-v src:dst:perm:perm:...` so local-path, url-path, and one or more permissions to set | ||||
|   * granting the same permissions to multiple accounts:   | ||||
|     `-v .::r,usr1,usr2:rw,usr3,usr4` = usr1/2 read-only, 3/4 read-write | ||||
|  | ||||
| permissions: | ||||
| * `r` (read): browse folder contents, download files, download as zip/tar | ||||
| * `w` (write): upload files, move files *into* this folder | ||||
| * `m` (move): move files/folders *from* this folder | ||||
| * `d` (delete): delete files/folders | ||||
|  | ||||
| examples: | ||||
| * add accounts named u1, u2, u3 with passwords p1, p2, p3: `-a u1:p1 -a u2:p2 -a u3:p3` | ||||
| * make folder `/srv` the root of the filesystem, read-only by anyone: `-v /srv::r` | ||||
| * make folder `/mnt/music` available at `/music`, read-only for u1 and u2, read-write for u3: `-v /mnt/music:music:r,u1,u2:rw,u3` | ||||
|   * unauthorized users accessing the webroot can see that the `music` folder exists, but cannot open it | ||||
| * make folder `/mnt/incoming` available at `/inc`, write-only for u1, read-move for u2: `-v /mnt/incoming:inc:w,u1:rm,u2` | ||||
|   * unauthorized users accessing the webroot can see that the `inc` folder exists, but cannot open it | ||||
|   * `u1` can open the `inc` folder, but cannot see the contents, only upload new files to it | ||||
|   * `u2` can browse it and move files *from* `/inc` into any folder where `u2` has write-access | ||||
|  | ||||
|  | ||||
| # the browser | ||||
|  | ||||
| accessing a copyparty server using a web-browser | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## tabs | ||||
|  | ||||
| the main tabs in the ui | ||||
| * `[🔎]` [search](#searching) by size, date, path/name, mp3-tags ... | ||||
| * `[🧯]` [unpost](#unpost): undo/delete accidental uploads | ||||
| * `[🚀]` and `[🎈]` are the [uploaders](#uploading) | ||||
| * `[📂]` mkdir: create directories | ||||
| * `[📝]` new-md: create a new markdown document | ||||
| * `[📟]` send-msg: either to server-log or into textfiles if `--urlform save` | ||||
| * `[🎺]` audio-player config options | ||||
| * `[⚙️]` general client config options | ||||
|  | ||||
|  | ||||
| ## hotkeys | ||||
|  | ||||
| the browser has the following hotkeys  (always qwerty) | ||||
| * `B` toggle breadcrumbs / [navpane](#navpane) | ||||
| * `I/K` prev/next folder | ||||
| * `M` parent folder (or unexpand current) | ||||
| * `G` toggle list / [grid view](#thumbnails) | ||||
| * `T` toggle thumbnails / icons | ||||
| * `ctrl-X` cut selected files/folders | ||||
| * `ctrl-V` paste | ||||
| * `F2` [rename](#batch-rename) selected file/folder | ||||
| * when a file/folder is selected (in not-grid-view): | ||||
|   * `Up/Down` move cursor | ||||
|   * shift+`Up/Down` select and move cursor | ||||
|   * ctrl+`Up/Down` move cursor and scroll viewport | ||||
|   * `Space` toggle file selection | ||||
|   * `Ctrl-A` toggle select all | ||||
| * when playing audio: | ||||
|   * `J/L` prev/next song | ||||
|   * `U/O` skip 10sec back/forward | ||||
|   * `0..9` jump to 0%..90% | ||||
|   * `P` play/pause (also starts playing the folder) | ||||
| * when viewing images / playing videos: | ||||
|   * `J/L, Left/Right` prev/next file | ||||
|   * `Home/End` first/last file | ||||
|   * `S` toggle selection | ||||
|   * `R` rotate clockwise (shift=ccw) | ||||
|   * `Esc` close viewer | ||||
|   * videos: | ||||
|     * `U/O` skip 10sec back/forward | ||||
|     * `P/K/Space` play/pause | ||||
|     * `F` fullscreen | ||||
|     * `C` continue playing next video | ||||
|     * `V` loop | ||||
|     * `M` mute | ||||
| * when the navpane is open: | ||||
|   * `A/D` adjust tree width | ||||
| * in the [grid view](#thumbnails): | ||||
|   * `S` toggle multiselect | ||||
|   * shift+`A/D` zoom | ||||
| * in the markdown editor: | ||||
|   * `^s` save | ||||
|   * `^h` header | ||||
|   * `^k` autoformat table | ||||
|   * `^u` jump to next unicode character | ||||
|   * `^e` toggle editor / preview | ||||
|   * `^up, ^down` jump paragraphs | ||||
|  | ||||
|  | ||||
| ## navpane | ||||
|  | ||||
| switching between breadcrumbs or navpane | ||||
|  | ||||
| click the `🌲` or pressing the `B` hotkey to toggle between breadcrumbs path (default), or a navpane (tree-browser sidebar thing) | ||||
|  | ||||
| * `[-]` and `[+]` (or hotkeys `A`/`D`) adjust the size | ||||
| * `[v]` jumps to the currently open folder | ||||
| * `[a]` toggles automatic widening as you go deeper | ||||
|  | ||||
|  | ||||
| ## thumbnails | ||||
|  | ||||
| press `g` to toggle grid-view instead of the file listing,  and `t` toggles icons / thumbnails | ||||
|  | ||||
|  | ||||
|  | ||||
| it does static images with Pillow and uses FFmpeg for video files, so you may want to `--no-thumb` or maybe just `--no-vthumb` depending on how dangerous your users are | ||||
|  | ||||
| images with the following names (see `--th-covers`) become the thumbnail of the folder they're in: `folder.png`, `folder.jpg`, `cover.png`, `cover.jpg` | ||||
|  | ||||
| in the grid/thumbnail view, if the audio player panel is open, songs will start playing when clicked | ||||
|  | ||||
|  | ||||
| ## zip downloads | ||||
|  | ||||
| download folders (or file selections) as `zip` or `tar` files | ||||
|  | ||||
| select which type of archive you want in the `[⚙️] config` tab: | ||||
|  | ||||
| | name | url-suffix | description | | ||||
| |--|--|--| | ||||
| | `tar` | `?tar` | plain gnutar, works great with `curl \| tar -xv` | | ||||
| | `zip` | `?zip=utf8` | works everywhere, glitchy filenames on win7 and older | | ||||
| | `zip_dos` | `?zip` | traditional cp437 (no unicode) to fix glitchy filenames | | ||||
| | `zip_crc` | `?zip=crc` | cp437 with crc32 computed early for truly ancient software | | ||||
|  | ||||
| * hidden files (dotfiles) are excluded unless `-ed` | ||||
|   * `up2k.db` and `dir.txt` is always excluded | ||||
| * `zip_crc` will take longer to download since the server has to read each file twice | ||||
|   * this is only to support MS-DOS PKZIP v2.04g (october 1993) and older | ||||
|     * how are you accessing copyparty actually | ||||
|  | ||||
| you can also zip a selection of files or folders by clicking them in the browser, that brings up a selection editor and zip button in the bottom right | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## uploading | ||||
|  | ||||
| drag files/folders into the web-browser to upload | ||||
|  | ||||
| this initiates an upload using `up2k`; there are two uploaders available: | ||||
| * `[🎈] bup`, the basic uploader, supports almost every browser since netscape 4.0 | ||||
| * `[🚀] up2k`, the fancy one | ||||
|  | ||||
| you can also undo/delete uploads by using `[🧯]` [unpost](#unpost) | ||||
|  | ||||
| up2k has several advantages: | ||||
| * you can drop folders into the browser (files are added recursively) | ||||
| * files are processed in chunks, and each chunk is checksummed | ||||
|   * uploads autoresume if they are interrupted by network issues | ||||
|   * uploads resume if you reboot your browser or pc, just upload the same files again | ||||
|   * server detects any corruption; the client reuploads affected chunks | ||||
|   * the client doesn't upload anything that already exists on the server | ||||
| * much higher speeds than ftp/scp/tarpipe on some internet connections (mainly american ones) thanks to parallel connections | ||||
| * the last-modified timestamp of the file is preserved | ||||
|  | ||||
| see [up2k](#up2k) for details on how it works | ||||
|  | ||||
|  | ||||
|  | ||||
| **protip:** you can avoid scaring away users with [docs/minimal-up2k.html](docs/minimal-up2k.html) which makes it look [much simpler](https://user-images.githubusercontent.com/241032/118311195-dd6ca380-b4ef-11eb-86f3-75a3ff2e1332.png) | ||||
|  | ||||
| the up2k UI is the epitome of polished inutitive experiences: | ||||
| * "parallel uploads" specifies how many chunks to upload at the same time | ||||
| * `[🏃]` analysis of other files should continue while one is uploading | ||||
| * `[💭]` ask for confirmation before files are added to the queue | ||||
| * `[💤]` sync uploading between other copyparty browser-tabs so only one is active | ||||
| * `[🔎]` switch between upload and [file-search](#file-search) mode | ||||
|   * ignore `[🔎]` if you add files by dragging them into the browser | ||||
|  | ||||
| and then theres the tabs below it, | ||||
| * `[ok]` is the files which completed successfully | ||||
| * `[ng]` is the ones that failed / got rejected (already exists, ...) | ||||
| * `[done]` shows a combined list of `[ok]` and `[ng]`, chronological order | ||||
| * `[busy]` files which are currently hashing, pending-upload, or uploading | ||||
|   * plus up to 3 entries each from `[done]` and `[que]` for context | ||||
| * `[que]` is all the files that are still queued | ||||
|  | ||||
| note that since up2k has to read each file twice, `[🎈 bup]` can *theoretically* be up to 2x faster in some extreme cases (files bigger than your ram, combined with an internet connection faster than the read-speed of your HDD) | ||||
|  | ||||
| if you are resuming a massive upload and want to skip hashing the files which already finished, you can enable `turbo` in the `[⚙️] config` tab, but please read the tooltip on that button | ||||
|  | ||||
|  | ||||
| ### file-search | ||||
|  | ||||
| dropping files into the browser also lets you see if they exist on the server | ||||
|  | ||||
|  | ||||
|  | ||||
| when you drag/drop files into the browser, you will see two dropzones: `Upload` and `Search` | ||||
|  | ||||
| > on a phone? toggle the `[🔎]` switch green before tapping the big yellow Search button to select your files | ||||
|  | ||||
| the files will be hashed on the client-side, and each hash is sent to the server, which checks if that file exists somewhere | ||||
|  | ||||
| files go into `[ok]` if they exist (and you get a link to where it is), otherwise they land in `[ng]` | ||||
| * the main reason filesearch is combined with the uploader is cause the code was too spaghetti to separate it out somewhere else, this is no longer the case but now i've warmed up to the idea too much | ||||
|  | ||||
| adding the same file multiple times is blocked, so if you first search for a file and then decide to upload it, you have to click the `[cleanup]` button to discard `[done]` files (or just refresh the page) | ||||
|  | ||||
|  | ||||
| ### unpost | ||||
|  | ||||
| undo/delete accidental uploads | ||||
|  | ||||
|  | ||||
|  | ||||
| you can unpost even if you don't have regular move/delete access, however only for files uploaded within the past `--unpost` seconds (default 12 hours) and the server must be running with `-e2d` | ||||
|  | ||||
|  | ||||
| ## file manager | ||||
|  | ||||
| cut/paste, rename, and delete files/folders (if you have permission) | ||||
|  | ||||
| file selection: click somewhere on the line (not the link itsef), then: | ||||
| * `space` to toggle | ||||
| * `up/down` to move | ||||
| * `shift-up/down` to move-and-select | ||||
| * `ctrl-shift-up/down` to also scroll | ||||
|  | ||||
| * cut: select some files and `ctrl-x` | ||||
| * paste: `ctrl-v` in another folder | ||||
| * rename: `F2` | ||||
|  | ||||
| you can move files across browser tabs (cut in one tab, paste in another) | ||||
|  | ||||
|  | ||||
| ## batch rename | ||||
|  | ||||
| select some files and press `F2` to bring up the rename UI | ||||
|  | ||||
|  | ||||
|  | ||||
| quick explanation of the buttons,   | ||||
| * `[✅ apply rename]` confirms and begins renaming | ||||
| * `[❌ cancel]` aborts and closes the rename window | ||||
| * `[↺ reset]` reverts any filename changes back to the original name | ||||
| * `[decode]` does a URL-decode on the filename, fixing stuff like `&` and `%20` | ||||
| * `[advanced]` toggles advanced mode | ||||
|  | ||||
| advanced mode: rename files based on rules to decide the new names, based on the original name (regex), or based on the tags collected from the file (artist/title/...), or a mix of both | ||||
|  | ||||
| in advanced mode,   | ||||
| * `[case]` toggles case-sensitive regex | ||||
| * `regex` is the regex pattern to apply to the original filename; any files which don't match will be skipped | ||||
| * `format` is the new filename, taking values from regex capturing groups and/or from file tags | ||||
|   * very loosely based on foobar2000 syntax | ||||
| * `presets` lets you save rename rules for later | ||||
|  | ||||
| available functions: | ||||
| * `$lpad(text, length, pad_char)` | ||||
| * `$rpad(text, length, pad_char)` | ||||
|  | ||||
| so, | ||||
|  | ||||
| say you have a file named [`meganeko - Eclipse - 07 Sirius A.mp3`](https://www.youtube.com/watch?v=-dtb0vDPruI) (absolutely fantastic album btw) and the tags are: `Album:Eclipse`, `Artist:meganeko`, `Title:Sirius A`, `tn:7` | ||||
|  | ||||
| you could use just regex to rename it: | ||||
| * `regex` = `(.*) - (.*) - ([0-9]{2}) (.*)` | ||||
| * `format` = `(3). (1) - (4)` | ||||
| * `output` = `07. meganeko - Sirius A.mp3` | ||||
|  | ||||
| or you could use just tags: | ||||
| * `format` = `$lpad((tn),2,0). (artist) - (title).(ext)` | ||||
| * `output` = `7. meganeko - Sirius A.mp3` | ||||
|  | ||||
| or a mix of both: | ||||
| * `regex` = ` - ([0-9]{2}) ` | ||||
| * `format` = `(1). (artist) - (title).(ext)` | ||||
| * `output` = `07. meganeko - Sirius A.mp3` | ||||
|  | ||||
| the metadata keys you can use in the format field are the ones in the file-browser table header (whatever is collected with `-mte` and `-mtp`) | ||||
|  | ||||
|  | ||||
| ## markdown viewer | ||||
|  | ||||
| and there are *two* editors | ||||
|  | ||||
|  | ||||
|  | ||||
| * the document preview has a max-width which is the same as an A4 paper when printed | ||||
|  | ||||
|  | ||||
| ## other tricks | ||||
|  | ||||
| * you can link a particular timestamp in an audio file by adding it to the URL, such as `&20` / `&20s` / `&1m20` / `&t=1:20` after the `.../#af-c8960dab` | ||||
|  | ||||
| * if you are using media hotkeys to switch songs and are getting tired of seeing the OSD popup which Windows doesn't let you disable, consider https://ocv.me/dev/?media-osd-bgone.ps1 | ||||
|  | ||||
| * click the bottom-left `π` to open a javascript prompt for debugging | ||||
|  | ||||
| * files named `.prologue.html` / `.epilogue.html` will be rendered before/after directory listings unless `--no-logues` | ||||
|  | ||||
| * files named `README.md` / `readme.md` will be rendered after directory listings unless `--no-readme` (but `.epilogue.html` takes precedence) | ||||
|  | ||||
|  | ||||
| ## searching | ||||
|  | ||||
| search by size, date, path/name, mp3-tags, ... | ||||
|  | ||||
|  | ||||
|  | ||||
| when started with `-e2dsa` copyparty will scan/index all your files. This avoids duplicates on upload, and also makes the volumes searchable through the web-ui: | ||||
| * make search queries by `size`/`date`/`directory-path`/`filename`, or... | ||||
| * drag/drop a local file to see if the same contents exist somewhere on the server, see [file-search](#file-search) | ||||
|  | ||||
| path/name queries are space-separated, AND'ed together, and words are negated with a `-` prefix, so for example: | ||||
| * path: `shibayan -bossa` finds all files where one of the folders contain `shibayan` but filters out any results where `bossa` exists somewhere in the path | ||||
| * name: `demetori styx` gives you [good stuff](https://www.youtube.com/watch?v=zGh0g14ZJ8I&list=PL3A147BD151EE5218&index=9) | ||||
|  | ||||
| add the argument `-e2ts` to also scan/index tags from music files, which brings us over to: | ||||
|  | ||||
|  | ||||
| # server config | ||||
|  | ||||
| ## file indexing | ||||
|  | ||||
| file indexing relies on two database tables, the up2k filetree (`-e2d`) and the metadata tags (`-e2t`), stored in `.hist/up2k.db`. Configuration can be done through arguments, volume flags, or a mix of both. | ||||
|  | ||||
| through arguments: | ||||
| * `-e2d` enables file indexing on upload | ||||
| * `-e2ds` also scans writable folders for new files on startup | ||||
| * `-e2dsa` also scans all mounted volumes (including readonly ones) | ||||
| * `-e2t` enables metadata indexing on upload | ||||
| * `-e2ts` also scans for tags in all files that don't have tags yet | ||||
| * `-e2tsr` also deletes all existing tags, doing a full reindex | ||||
|  | ||||
| the same arguments can be set as volume flags, in addition to `d2d` and `d2t` for disabling: | ||||
| * `-v ~/music::r:c,e2dsa:c,e2tsr` does a full reindex of everything on startup | ||||
| * `-v ~/music::r:c,d2d` disables **all** indexing, even if any `-e2*` are on | ||||
| * `-v ~/music::r:c,d2t` disables all `-e2t*` (tags), does not affect `-e2d*` | ||||
|  | ||||
| note: | ||||
| * the parser currently can't handle `c,e2dsa,e2tsr` so you have to `c,e2dsa:c,e2tsr` | ||||
| * `e2tsr` is probably always overkill, since `e2ds`/`e2dsa` would pick up any file modifications and `e2ts` would then reindex those, unless there is a new copyparty version with new parsers and the release note says otherwise | ||||
| * the rescan button in the admin panel has no effect unless the volume has `-e2ds` or higher | ||||
|  | ||||
| to save some time, you can choose to only index filename/path/size/last-modified (and not the hash of the file contents) by setting `--no-hash` or the volume-flag `:c,dhash`, this has the following consequences: | ||||
| * initial indexing is way faster, especially when the volume is on a network disk | ||||
| * makes it impossible to [file-search](#file-search) | ||||
| * if someone uploads the same file contents, the upload will not be detected as a dupe, so it will not get symlinked or rejected | ||||
|  | ||||
| if you set `--no-hash`, you can enable hashing for specific volumes using flag `:c,ehash` | ||||
|  | ||||
|  | ||||
| ## upload rules | ||||
|  | ||||
| set upload rules using volume flags,  some examples: | ||||
|  | ||||
| * `:c,sz=1k-3m` sets allowed filesize between 1 KiB and 3 MiB inclusive (suffixes: b, k, m, g) | ||||
| * `:c,nosub` disallow uploading into subdirectories; goes well with `rotn` and `rotf`: | ||||
| * `:c,rotn=1000,2` moves uploads into subfolders, up to 1000 files in each folder before making a new one, two levels deep (must be at least 1) | ||||
| * `:c,rotf=%Y/%m/%d/%H` enforces files to be uploaded into a structure of subfolders according to that date format | ||||
|   * if someone uploads to `/foo/bar` the path would be rewritten to `/foo/bar/2021/08/06/23` for example | ||||
|   * but the actual value is not verified, just the structure, so the uploader can choose any values which conform to the format string | ||||
|     * just to avoid additional complexity in up2k which is enough of a mess already | ||||
| * `:c,lifetime=300` delete uploaded files when they become 5 minutes old | ||||
|  | ||||
| you can also set transaction limits which apply per-IP and per-volume, but these assume `-j 1` (default) otherwise the limits will be off, for example `-j 4` would allow anywhere between 1x and 4x the limits you set depending on which processing node the client gets routed to | ||||
|  | ||||
| * `:c,maxn=250,3600` allows 250 files over 1 hour from each IP (tracked per-volume) | ||||
| * `:c,maxb=1g,300` allows 1 GiB total over 5 minutes from each IP (tracked per-volume) | ||||
|  | ||||
|  | ||||
| ## compress uploads | ||||
|  | ||||
| files can be autocompressed on upload,  either on user-request (if config allows) or forced by server-config | ||||
|  | ||||
| * volume flag `gz` allows gz compression | ||||
| * volume flag `xz` allows lzma compression | ||||
| * volume flag `pk` **forces** compression on all files | ||||
| * url parameter `pk` requests compression with server-default algorithm | ||||
| * url parameter `gz` or `xz` requests compression with a specific algorithm | ||||
| * url parameter `xz` requests xz compression | ||||
|  | ||||
| things to note, | ||||
| * the `gz` and `xz` arguments take a single optional argument, the compression level (range 0 to 9) | ||||
| * the `pk` volume flag takes the optional argument `ALGORITHM,LEVEL` which will then be forced for all uploads, for example `gz,9` or `xz,0` | ||||
| * default compression is gzip level 9 | ||||
| * all upload methods except up2k are supported | ||||
| * the files will be indexed after compression, so dupe-detection and file-search will not work as expected | ||||
|  | ||||
| some examples, | ||||
|  | ||||
|  | ||||
| ## database location | ||||
|  | ||||
| in-volume (`.hist/up2k.db`, default) or somewhere else | ||||
|  | ||||
| copyparty creates a subfolder named `.hist` inside each volume where it stores the database, thumbnails, and some other stuff | ||||
|  | ||||
| this can instead be kept in a single place using the `--hist` argument, or the `hist=` volume flag, or a mix of both: | ||||
| * `--hist ~/.cache/copyparty -v ~/music::r:c,hist=-` sets `~/.cache/copyparty` as the default place to put volume info, but `~/music` gets the regular `.hist` subfolder (`-` restores default behavior) | ||||
|  | ||||
| note: | ||||
| * markdown edits are always stored in a local `.hist` subdirectory | ||||
| * on windows the volflag path is cyglike, so `/c/temp` means `C:\temp` but use regular paths for `--hist` | ||||
|   * you can use cygpaths for volumes too, `-v C:\Users::r` and `-v /c/users::r` both work | ||||
|  | ||||
|  | ||||
| ## metadata from audio files | ||||
|  | ||||
| set `-e2t` to index tags on upload | ||||
|  | ||||
| `-mte` decides which tags to index and display in the browser (and also the display order), this can be changed per-volume: | ||||
| * `-v ~/music::r:c,mte=title,artist` indexes and displays *title* followed by *artist* | ||||
|  | ||||
| if you add/remove a tag from `mte` you will need to run with `-e2tsr` once to rebuild the database, otherwise only new files will be affected | ||||
|  | ||||
| but instead of using `-mte`, `-mth` is a better way to hide tags in the browser: these tags will not be displayed by default, but they still get indexed and become searchable, and users can choose to unhide them in the `[⚙️] config` pane | ||||
|  | ||||
| `-mtm` can be used to add or redefine a metadata mapping, say you have media files with `foo` and `bar` tags and you want them to display as `qux` in the browser (preferring `foo` if both are present), then do `-mtm qux=foo,bar` and now you can `-mte artist,title,qux` | ||||
|  | ||||
| tags that start with a `.` such as `.bpm` and `.dur`(ation) indicate numeric value | ||||
|  | ||||
| see the beautiful mess of a dictionary in [mtag.py](https://github.com/9001/copyparty/blob/hovudstraum/copyparty/mtag.py) for the default mappings (should cover mp3,opus,flac,m4a,wav,aif,) | ||||
|  | ||||
| `--no-mutagen` disables Mutagen and uses FFprobe instead, which... | ||||
| * is about 20x slower than Mutagen | ||||
| * catches a few tags that Mutagen doesn't | ||||
|   * melodic key, video resolution, framerate, pixfmt | ||||
| * avoids pulling any GPL code into copyparty | ||||
| * more importantly runs FFprobe on incoming files which is bad if your FFmpeg has a cve | ||||
|  | ||||
|  | ||||
| ## file parser plugins | ||||
|  | ||||
| provide custom parsers to index additional tags | ||||
|  | ||||
| copyparty can invoke external programs to collect additional metadata for files using `mtp` (either as argument or volume flag), there is a default timeout of 30sec | ||||
|  | ||||
| * `-mtp .bpm=~/bin/audio-bpm.py` will execute `~/bin/audio-bpm.py` with the audio file as argument 1 to provide the `.bpm` tag, if that does not exist in the audio metadata | ||||
| * `-mtp key=f,t5,~/bin/audio-key.py` uses `~/bin/audio-key.py` to get the `key` tag, replacing any existing metadata tag (`f,`), aborting if it takes longer than 5sec (`t5,`) | ||||
| * `-v ~/music::r:c,mtp=.bpm=~/bin/audio-bpm.py:c,mtp=key=f,t5,~/bin/audio-key.py` both as a per-volume config wow this is getting ugly | ||||
|  | ||||
| *but wait, there's more!* `-mtp` can be used for non-audio files as well using the `a` flag: `ay` only do audio files, `an` only do non-audio files, or `ad` do all files (d as in dontcare)  | ||||
|  | ||||
| * `-mtp ext=an,~/bin/file-ext.py` runs `~/bin/file-ext.py` to get the `ext` tag only if file is not audio (`an`) | ||||
| * `-mtp arch,built,ver,orig=an,eexe,edll,~/bin/exe.py` runs `~/bin/exe.py` to get properties about windows-binaries only if file is not audio (`an`) and file extension is exe or dll | ||||
|  | ||||
|  | ||||
| ## complete examples | ||||
|  | ||||
| * read-only music server with bpm and key scanning   | ||||
|   `python copyparty-sfx.py -v /mnt/nas/music:/music:r -e2dsa -e2ts -mtp .bpm=f,audio-bpm.py -mtp key=f,audio-key.py` | ||||
|  | ||||
|  | ||||
| # browser support | ||||
|  | ||||
| TLDR: yes | ||||
|  | ||||
|  | ||||
|  | ||||
| `ie` = internet-explorer, `ff` = firefox, `c` = chrome, `iOS` = iPhone/iPad, `Andr` = Android | ||||
|  | ||||
| | feature         | ie6 | ie9  | ie10 | ie11 | ff 52 | c 49 | iOS | Andr | | ||||
| | --------------- | --- | ---- | ---- | ---- | ----- | ---- | --- | ---- | | ||||
| | browse files    | yep | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | thumbnail view  |  -  | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | basic uploader  | yep | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | up2k            |  -  |  -   | `*1` | `*1` |  yep  | yep  | yep | yep  | | ||||
| | make directory  | yep | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | send message    | yep | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | set sort order  |  -  | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | zip selection   |  -  | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | file rename     |  -  | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | file cut/paste  |  -  | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | navpane         |  -  | `*2` | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | image viewer    |  -  | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | video player    |  -  | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | markdown editor |  -  |  -   | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | markdown viewer |  -  |  -   | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | play mp3/m4a    |  -  | yep  | yep  | yep  |  yep  | yep  | yep | yep  | | ||||
| | play ogg/opus   |  -  |  -   |  -   |  -   |  yep  | yep  | `*3` | yep | | ||||
| | **= feature =** | ie6 | ie9  | ie10 | ie11 | ff 52 | c 49 | iOS | Andr | | ||||
|  | ||||
| * internet explorer 6 to 8 behave the same | ||||
| * firefox 52 and chrome 49 are the final winxp versions | ||||
| * `*1` yes, but extremely slow (ie10: `1 MiB/s`, ie11: `270 KiB/s`) | ||||
| * `*2` causes a full-page refresh on each navigation | ||||
| * `*3` using a wasm decoder which consumes a bit more power | ||||
|  | ||||
| quick summary of more eccentric web-browsers trying to view a directory index: | ||||
|  | ||||
| | browser | will it blend | | ||||
| | ------- | ------------- | | ||||
| | **safari** (14.0.3/macos) | is chrome with janky wasm, so playing opus can deadlock the javascript engine | | ||||
| | **safari** (14.0.1/iOS)   | same as macos, except it recovers from the deadlocks if you poke it a bit | | ||||
| | **links** (2.21/macports) | can browse, login, upload/mkdir/msg | | ||||
| | **lynx** (2.8.9/macports) | can browse, login, upload/mkdir/msg | | ||||
| | **w3m** (0.5.3/macports)  | can browse, login, upload at 100kB/s, mkdir/msg | | ||||
| | **netsurf** (3.10/arch)   | is basically ie6 with much better css (javascript has almost no effect) |  | ||||
| | **opera** (11.60/winxp)   | OK: thumbnails, image-viewer, zip-selection, rename/cut/paste. NG: up2k, navpane, markdown, audio | | ||||
| | **ie4** and **netscape** 4.0  | can browse (text is yellow on white), upload with `?b=u` | | ||||
| | **SerenityOS** (7e98457)  | hits a page fault, works with `?b=u`, file upload not-impl | | ||||
|  | ||||
|  | ||||
| # client examples | ||||
|  | ||||
| interact with copyparty using non-browser clients | ||||
|  | ||||
| * javascript: dump some state into a file (two separate examples) | ||||
|   * `await fetch('https://127.0.0.1:3923/', {method:"PUT", body: JSON.stringify(foo)});` | ||||
|   * `var xhr = new XMLHttpRequest(); xhr.open('POST', 'https://127.0.0.1:3923/msgs?raw'); xhr.send('foo');` | ||||
|  | ||||
| * curl/wget: upload some files (post=file, chunk=stdin) | ||||
|   * `post(){ curl -b cppwd=wark -F act=bput -F f=@"$1" http://127.0.0.1:3923/;}`   | ||||
|     `post movie.mkv` | ||||
|   * `post(){ wget --header='Cookie: cppwd=wark' --post-file="$1" -O- http://127.0.0.1:3923/?raw;}`   | ||||
|     `post movie.mkv` | ||||
|   * `chunk(){ curl -b cppwd=wark -T- http://127.0.0.1:3923/;}`   | ||||
|     `chunk <movie.mkv` | ||||
|  | ||||
| * FUSE: mount a copyparty server as a local filesystem | ||||
|   * cross-platform python client available in [./bin/](bin/) | ||||
|   * [rclone](https://rclone.org/) as client can give ~5x performance, see [./docs/rclone.md](docs/rclone.md) | ||||
|  | ||||
| * sharex (screenshot utility): see [./contrib/sharex.sxcu](contrib/#sharexsxcu) | ||||
|  | ||||
| copyparty returns a truncated sha512sum of your PUT/POST as base64; you can generate the same checksum locally to verify uplaods: | ||||
|  | ||||
|     b512(){ printf "$((sha512sum||shasum -a512)|sed -E 's/ .*//;s/(..)/\\x\1/g')"|base64|tr '+/' '-_'|head -c44;} | ||||
|     b512 <movie.mkv | ||||
|  | ||||
| you can provide passwords using cookie 'cppwd=hunter2', as a url query `?pw=hunter2`, or with basic-authentication (either as the username or password) | ||||
|  | ||||
|  | ||||
| # up2k | ||||
|  | ||||
| quick outline of the up2k protocol, see [uploading](#uploading) for the web-client | ||||
| * the up2k client splits a file into an "optimal" number of chunks | ||||
|   * 1 MiB each, unless that becomes more than 256 chunks | ||||
|   * tries 1.5M, 2M, 3, 4, 6, ... until <= 256 chunks or size >= 32M | ||||
| * client posts the list of hashes, filename, size, last-modified | ||||
| * server creates the `wark`, an identifier for this upload | ||||
|   * `sha512( salt + filesize + chunk_hashes )` | ||||
|   * and a sparse file is created for the chunks to drop into | ||||
| * client uploads each chunk | ||||
|   * header entries for the chunk-hash and wark | ||||
|   * server writes chunks into place based on the hash | ||||
| * client does another handshake with the hashlist; server replies with OK or a list of chunks to reupload | ||||
|  | ||||
| up2k has saved a few uploads from becoming corrupted in-transfer already; caught an android phone on wifi redhanded in wireshark with a bitflip, however bup with https would *probably* have noticed as well (thanks to tls also functioning as an integrity check) | ||||
|  | ||||
|  | ||||
| ## why chunk-hashes | ||||
|  | ||||
| a single sha512 would be better, right? | ||||
|  | ||||
| this is due to `crypto.subtle` not providing a streaming api (or the option to seed the sha512 hasher with a starting hash) | ||||
|  | ||||
| as a result, the hashes are much less useful than they could have been (search the server by sha512, provide the sha512 in the response http headers, ...) | ||||
|  | ||||
| hashwasm would solve the streaming issue but reduces hashing speed for sha512 (xxh128 does 6 GiB/s), and it would make old browsers and [iphones](https://bugs.webkit.org/show_bug.cgi?id=228552) unsupported | ||||
|  | ||||
|  | ||||
| # performance | ||||
|  | ||||
| defaults are usually fine - expect `8 GiB/s` download, `1 GiB/s` upload | ||||
|  | ||||
| you can ignore the `cannot efficiently use multiple CPU cores` message, very unlikely to be a problem | ||||
|  | ||||
| below are some tweaks roughly ordered by usefulness: | ||||
|  | ||||
| * `-q` disables logging and can help a bunch, even when combined with `-lo` to redirect logs to file | ||||
| * `--http-only` or `--https-only` (unless you want to support both protocols) will reduce the delay before a new connection is established | ||||
| * `--hist` pointing to a fast location (ssd) will make directory listings and searches faster when `-e2d` or `-e2t` is set | ||||
| * `--no-hash` when indexing a network-disk if you don't care about the actual filehashes and only want the names/tags searchable | ||||
| * `-j` enables multiprocessing (actual multithreading) and can make copyparty perform better in cpu-intensive workloads, for example: | ||||
|   * huge amount of short-lived connections | ||||
|   * really heavy traffic (downloads/uploads) | ||||
|    | ||||
|   ...however it adds an overhead to internal communication so it might be a net loss, see if it works 4 u | ||||
|  | ||||
|  | ||||
| # security | ||||
|  | ||||
| some notes on hardening | ||||
|  | ||||
| on public copyparty instances with anonymous upload enabled: | ||||
|  | ||||
| * users can upload html/css/js which will evaluate for other visitors in a few ways, | ||||
|   * unless `--no-readme` is set: by uploading/modifying a file named `readme.md` | ||||
|   * if `move` access is granted AND none of `--no-logues`, `--no-dot-mv`, `--no-dot-ren` is set: by uploading some .html file and renaming it to `.epilogue.html` (uploading it directly is blocked) | ||||
|  | ||||
|  | ||||
| ## gotchas | ||||
|  | ||||
| behavior that might be unexpected | ||||
|  | ||||
| * users without read-access to a folder can still see the `.prologue.html` / `.epilogue.html` / `README.md` contents, for the purpose of showing a description on how to use the uploader for example | ||||
|  | ||||
|  | ||||
| # dependencies | ||||
|  | ||||
| * `jinja2` | ||||
|   * pulls in `markupsafe` as of v2.7; use jinja 2.6 on py3.2 | ||||
| mandatory deps: | ||||
| * `jinja2` (is built into the SFX) | ||||
|  | ||||
| optional, enables thumbnails: | ||||
| * `Pillow` (requires py2.7 or py3.5+) | ||||
|  | ||||
| ## optional dependencies | ||||
|  | ||||
| install these to enable bonus features | ||||
|  | ||||
| enable music tags: | ||||
| * either `mutagen` (fast, pure-python, skips a few tags, makes copyparty GPL? idk) | ||||
| * or `ffprobe` (20x slower, more accurate, possibly dangerous depending on your distro and users) | ||||
|  | ||||
| enable [thumbnails](#thumbnails) of... | ||||
| * **images:** `Pillow` (requires py2.7 or py3.5+) | ||||
| * **videos:** `ffmpeg` and `ffprobe` somewhere in `$PATH` | ||||
| * **HEIF pictures:** `pyheif-pillow-opener` (requires Linux or a C compiler) | ||||
| * **AVIF pictures:** `pillow-avif-plugin` | ||||
|  | ||||
|  | ||||
| ## install recommended deps | ||||
| ``` | ||||
| python -m pip install --user -U jinja2 mutagen Pillow | ||||
| ``` | ||||
|  | ||||
|  | ||||
| ## optional gpl stuff | ||||
|  | ||||
| some bundled tools have copyleft dependencies, see [./bin/#mtag](bin/#mtag) | ||||
|  | ||||
| these are standalone programs and will never be imported / evaluated by copyparty, and must be enabled through `-mtp` configs | ||||
|  | ||||
|  | ||||
| # sfx | ||||
|  | ||||
| currently there are two self-contained binaries: | ||||
| * `copyparty-sfx.sh` for unix (linux and osx) -- smaller, more robust | ||||
| * `copyparty-sfx.py` for windows (unix too) -- crossplatform, beta | ||||
| there are two self-contained "binaries": | ||||
| * [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) -- pure python, works everywhere, **recommended** | ||||
| * [copyparty-sfx.sh](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.sh) -- smaller, but only for linux and macos, kinda deprecated | ||||
|  | ||||
| launch either of them and it'll unpack and run copyparty, assuming you have python installed of course | ||||
| launch either of them (**use sfx.py on systemd**) and it'll unpack and run copyparty, assuming you have python installed of course | ||||
|  | ||||
| pls note that `copyparty-sfx.sh` will fail if you rename `copyparty-sfx.py` to `copyparty.py` and keep it in the same folder because `sys.path` is funky | ||||
|  | ||||
| if you don't need all the features you can repack the sfx and save a bunch of space; all you need is an sfx and a copy of this repo (nothing else to download or build, except for either msys2 or WSL if you're on windows) | ||||
| * `724K` original size as of v0.4.0 | ||||
| * `256K` after `./scripts/make-sfx.sh re no-ogv` | ||||
| * `164K` after `./scripts/make-sfx.sh re no-ogv no-cm` | ||||
|  | ||||
| ## sfx repack | ||||
|  | ||||
| reduce the size of an sfx by removing features | ||||
|  | ||||
| if you don't need all the features, you can repack the sfx and save a bunch of space; all you need is an sfx and a copy of this repo (nothing else to download or build, except if you're on windows then you need msys2 or WSL) | ||||
| * `525k` size of original sfx.py as of v0.11.30 | ||||
| * `315k` after `./scripts/make-sfx.sh re no-ogv` | ||||
| * `223k` after `./scripts/make-sfx.sh re no-ogv no-cm` | ||||
|  | ||||
| the features you can opt to drop are | ||||
| * `ogv`.js, the opus/vorbis decoder which is needed by apple devices to play foss audio files | ||||
| * `cm`/easymde, the "fancy" markdown editor | ||||
| * `ogv`.js, the opus/vorbis decoder which is needed by apple devices to play foss audio files, saves ~192k | ||||
| * `cm`/easymde, the "fancy" markdown editor, saves ~92k | ||||
| * `fnt`, source-code-pro, the monospace font, saves ~9k | ||||
| * `dd`, the custom mouse cursor for the media player tray tab, saves ~2k | ||||
|  | ||||
| for the `re`pack to work, first run one of the sfx'es once to unpack it | ||||
|  | ||||
| **note:** you can also just download and run [scripts/copyparty-repack.sh](scripts/copyparty-repack.sh) -- this will grab the latest copyparty release from github and do a `no-ogv no-cm` repack; works on linux/macos (and windows with msys2 or WSL) | ||||
|  | ||||
|  | ||||
| # install on android | ||||
|  | ||||
| install [Termux](https://termux.com/) (see [ocv.me/termux](https://ocv.me/termux/)) and then copy-paste this into Termux (long-tap) all at once: | ||||
| ```sh | ||||
| apt update && apt -y full-upgrade && termux-setup-storage && apt -y install curl && cd && curl -L https://github.com/9001/copyparty/raw/master/scripts/copyparty-android.sh > copyparty-android.sh && chmod 755 copyparty-android.sh && ./copyparty-android.sh -h | ||||
| apt update && apt -y full-upgrade && termux-setup-storage && apt -y install python && python -m ensurepip && python -m pip install -U copyparty | ||||
| echo $? | ||||
| ``` | ||||
|  | ||||
| after the initial setup (and restarting bash), you can launch copyparty at any time by running "copyparty" in Termux | ||||
| after the initial setup, you can launch copyparty at any time by running `copyparty` anywhere in Termux | ||||
|  | ||||
|  | ||||
| # dev env setup | ||||
| # building | ||||
|  | ||||
| ## dev env setup | ||||
|  | ||||
| mostly optional; if you need a working env for vscode or similar | ||||
|  | ||||
| ```sh | ||||
| python3 -m venv .venv | ||||
| . .venv/bin/activate | ||||
| pip install jinja2  # mandatory deps | ||||
| pip install Pillow  # thumbnail deps | ||||
| pip install jinja2  # mandatory | ||||
| pip install mutagen  # audio metadata | ||||
| pip install Pillow pyheif-pillow-opener pillow-avif-plugin  # thumbnails | ||||
| pip install black bandit pylint flake8  # vscode tooling | ||||
| ``` | ||||
|  | ||||
|  | ||||
| # how to release | ||||
| ## just the sfx | ||||
|  | ||||
| first grab the web-dependencies from a previous sfx (assuming you don't need to modify something in those): | ||||
|  | ||||
| ```sh | ||||
| rm -rf copyparty/web/deps | ||||
| curl -L https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py >x.py | ||||
| python3 x.py -h | ||||
| rm x.py | ||||
| mv /tmp/pe-copyparty/copyparty/web/deps/ copyparty/web/deps/ | ||||
| ``` | ||||
|  | ||||
| then build the sfx using any of the following examples: | ||||
|  | ||||
| ```sh | ||||
| ./scripts/make-sfx.sh  # both python and sh editions | ||||
| ./scripts/make-sfx.sh no-sh gz  # just python with gzip | ||||
| ``` | ||||
|  | ||||
|  | ||||
| ## complete release | ||||
|  | ||||
| also builds the sfx so skip the sfx section above | ||||
|  | ||||
| in the `scripts` folder: | ||||
|  | ||||
| * run `make -C deps-docker` to build all dependencies | ||||
| * create github release with `make-tgz-release.sh` | ||||
| * `git tag v1.2.3 && git push origin --tags` | ||||
| * upload to pypi with `make-pypi-release.(sh|bat)` | ||||
| * create github release with `make-tgz-release.sh` | ||||
| * create sfx with `make-sfx.sh` | ||||
|  | ||||
|  | ||||
| @@ -109,13 +923,33 @@ in the `scripts` folder: | ||||
|  | ||||
| roughly sorted by priority | ||||
|  | ||||
| * up2k handle filename too long | ||||
| * up2k fails on empty files? alert then stuck | ||||
| * drop onto folders | ||||
| * look into android thumbnail cache file format | ||||
| * support pillow-simd | ||||
| * nothing! currently | ||||
|  | ||||
|  | ||||
| ## discarded ideas | ||||
|  | ||||
| * reduce up2k roundtrips | ||||
|   * start from a chunk index and just go | ||||
|   * terminate client on bad data | ||||
|     * not worth the effort, just throw enough conncetions at it | ||||
| * single sha512 across all up2k chunks? | ||||
|   * crypto.subtle cannot into streaming, would have to use hashwasm, expensive | ||||
| * separate sqlite table per tag | ||||
|   * performance fixed by skipping some indexes (`+mt.k`) | ||||
| * audio fingerprinting | ||||
|   * only makes sense if there can be a wasm client and that doesn't exist yet (except for olaf which is agpl hence counts as not existing) | ||||
| * `os.copy_file_range` for up2k cloning | ||||
|   * almost never hit this path anyways | ||||
| * up2k partials ui | ||||
|   * feels like there isn't much point | ||||
| * cache sha512 chunks on client | ||||
| * symlink existing files on upload | ||||
|   * too dangerous | ||||
| * comment field | ||||
| * figure out the deal with pixel3a not being connectable as hotspot | ||||
|   * pixel3a having unpredictable 3sec latency in general :|||| | ||||
|   * nah | ||||
| * look into android thumbnail cache file format | ||||
|   * absolutely not | ||||
| * indexedDB for hashes, cfg enable/clear/sz, 2gb avail, ~9k for 1g, ~4k for 100m, 500k items before autoeviction | ||||
|   * blank hashlist when up-ok to skip handshake | ||||
|     * too many confusing side-effects | ||||
| * hls framework for Someone Else to drop code into :^) | ||||
|   * probably not, too much stuff to consider -- seeking, start at offset, task stitching (probably np-hard), conditional passthru, rate-control (especially multi-consumer), session keepalive, cache mgmt... | ||||
|   | ||||
							
								
								
									
										68
									
								
								bin/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								bin/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | ||||
| # [`copyparty-fuse.py`](copyparty-fuse.py) | ||||
| * mount a copyparty server as a local filesystem (read-only) | ||||
| * **supports Windows!** -- expect `194 MiB/s` sequential read | ||||
| * **supports Linux** -- expect `117 MiB/s` sequential read | ||||
| * **supports macos** -- expect `85 MiB/s` sequential read | ||||
|  | ||||
| filecache is default-on for windows and macos; | ||||
| * macos readsize is 64kB, so speed ~32 MiB/s without the cache | ||||
| * windows readsize varies by software; explorer=1M, pv=32k | ||||
|  | ||||
| note that copyparty should run with `-ed` to enable dotfiles (hidden otherwise) | ||||
|  | ||||
| also consider using [../docs/rclone.md](../docs/rclone.md) instead for 5x performance | ||||
|  | ||||
|  | ||||
| ## to run this on windows: | ||||
| * 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) | ||||
| * `python -m pip install --user fusepy` | ||||
| * `python ./copyparty-fuse.py n: http://192.168.1.69:3923/` | ||||
|  | ||||
| 10% faster in [msys2](https://www.msys2.org/), 700% faster if debug prints are enabled: | ||||
| * `pacman -S mingw64/mingw-w64-x86_64-python{,-pip}` | ||||
| * `/mingw64/bin/python3 -m pip install --user fusepy` | ||||
| * `/mingw64/bin/python3 ./copyparty-fuse.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)   | ||||
| (winfsp's sshfs leaks, doesn't look like winfsp itself does, should be fine) | ||||
|  | ||||
|  | ||||
|  | ||||
| # [`copyparty-fuse🅱️.py`](copyparty-fuseb.py) | ||||
| * mount a copyparty server as a local filesystem (read-only) | ||||
| * does the same thing except more correct, `samba` approves | ||||
| * **supports Linux** -- expect `18 MiB/s` (wait what) | ||||
| * **supports Macos** -- probably | ||||
|  | ||||
|  | ||||
|  | ||||
| # [`copyparty-fuse-streaming.py`](copyparty-fuse-streaming.py) | ||||
| * pretend this doesn't exist | ||||
|  | ||||
|  | ||||
|  | ||||
| # [`mtag/`](mtag/) | ||||
| * standalone programs which perform misc. file analysis | ||||
| * copyparty can Popen programs like these during file indexing to collect additional metadata | ||||
|  | ||||
|  | ||||
| # [`dbtool.py`](dbtool.py) | ||||
| upgrade utility which can show db info and help transfer data between databases, for example when a new version of copyparty is incompatible with the old DB and automatically rebuilds the DB from scratch, but you have some really expensive `-mtp` parsers and want to copy over the tags from the old db | ||||
|  | ||||
| for that example (upgrading to v0.11.20), first launch the new version of copyparty like usual, let it make a backup of the old db and rebuild the new db until the point where it starts running mtp (colored messages as it adds the mtp tags), that's when you hit CTRL-C and patch in the old mtp tags from the old db instead | ||||
|  | ||||
| so assuming you have `-mtp` parsers to provide the tags `key` and `.bpm`: | ||||
|  | ||||
| ``` | ||||
| cd /mnt/nas/music/.hist | ||||
| ~/src/copyparty/bin/dbtool.py -ls up2k.db | ||||
| ~/src/copyparty/bin/dbtool.py -src up2k.*.v3 up2k.db -cmp | ||||
| ~/src/copyparty/bin/dbtool.py -src up2k.*.v3 up2k.db -rm-mtp-flag -copy key | ||||
| ~/src/copyparty/bin/dbtool.py -src up2k.*.v3 up2k.db -rm-mtp-flag -copy .bpm -vac | ||||
| ``` | ||||
|  | ||||
|  | ||||
| # [`prisonparty.sh`](prisonparty.sh) | ||||
| * run copyparty in a chroot, preventing any accidental file access | ||||
| * creates bindmounts for /bin, /lib, and so on, see `sysdirs=` | ||||
							
								
								
									
										1100
									
								
								bin/copyparty-fuse-streaming.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1100
									
								
								bin/copyparty-fuse-streaming.py
									
									
									
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										757
									
								
								bin/copyparty-fuse.py
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										757
									
								
								bin/copyparty-fuse.py
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										592
									
								
								bin/copyparty-fuseb.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										592
									
								
								bin/copyparty-fuseb.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,592 @@ | ||||
| #!/usr/bin/env python3 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| """copyparty-fuseb: remote copyparty as a local filesystem""" | ||||
| __author__ = "ed <copyparty@ocv.me>" | ||||
| __copyright__ = 2020 | ||||
| __license__ = "MIT" | ||||
| __url__ = "https://github.com/9001/copyparty/" | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import stat | ||||
| import errno | ||||
| import struct | ||||
| import threading | ||||
| import http.client  # py2: httplib | ||||
| import urllib.parse | ||||
| from datetime import datetime | ||||
| from urllib.parse import quote_from_bytes as quote | ||||
|  | ||||
| try: | ||||
|     import fuse | ||||
|     from fuse import Fuse | ||||
|  | ||||
|     fuse.fuse_python_api = (0, 2) | ||||
|     if not hasattr(fuse, "__version__"): | ||||
|         raise Exception("your fuse-python is way old") | ||||
| except: | ||||
|     print( | ||||
|         "\n  could not import fuse; these may help:\n    python3 -m pip install --user fuse-python\n    apt install libfuse\n    modprobe fuse\n" | ||||
|     ) | ||||
|     raise | ||||
|  | ||||
|  | ||||
| """ | ||||
| mount a copyparty server (local or remote) as a filesystem | ||||
|  | ||||
| usage: | ||||
|   python ./copyparty-fuseb.py -f -o allow_other,auto_unmount,nonempty,url=http://192.168.1.69:3923 /mnt/nas | ||||
|  | ||||
| dependencies: | ||||
|   sudo apk add fuse-dev python3-dev | ||||
|   python3 -m pip install --user fuse-python | ||||
|  | ||||
| fork of copyparty-fuse.py based on fuse-python which | ||||
|   appears to be more compliant than fusepy? since this works with samba | ||||
|     (probably just my garbage code tbh) | ||||
| """ | ||||
|  | ||||
|  | ||||
| def threadless_log(msg): | ||||
|     print(msg + "\n", end="") | ||||
|  | ||||
|  | ||||
| def boring_log(msg): | ||||
|     msg = "\033[36m{:012x}\033[0m {}\n".format(threading.current_thread().ident, msg) | ||||
|     print(msg[4:], end="") | ||||
|  | ||||
|  | ||||
| def rice_tid(): | ||||
|     tid = threading.current_thread().ident | ||||
|     c = struct.unpack(b"B" * 5, struct.pack(b">Q", tid)[-5:]) | ||||
|     return "".join("\033[1;37;48;5;{}m{:02x}".format(x, x) for x in c) + "\033[0m" | ||||
|  | ||||
|  | ||||
| def fancy_log(msg): | ||||
|     print("{} {}\n".format(rice_tid(), msg), end="") | ||||
|  | ||||
|  | ||||
| def null_log(msg): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| info = fancy_log | ||||
| log = fancy_log | ||||
| dbg = fancy_log | ||||
| log = null_log | ||||
| dbg = null_log | ||||
|  | ||||
|  | ||||
| def get_tid(): | ||||
|     return threading.current_thread().ident | ||||
|  | ||||
|  | ||||
| def html_dec(txt): | ||||
|     return ( | ||||
|         txt.replace("<", "<") | ||||
|         .replace(">", ">") | ||||
|         .replace(""", '"') | ||||
|         .replace("&", "&") | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class CacheNode(object): | ||||
|     def __init__(self, tag, data): | ||||
|         self.tag = tag | ||||
|         self.data = data | ||||
|         self.ts = time.time() | ||||
|  | ||||
|  | ||||
| class Stat(fuse.Stat): | ||||
|     def __init__(self): | ||||
|         self.st_mode = 0 | ||||
|         self.st_ino = 0 | ||||
|         self.st_dev = 0 | ||||
|         self.st_nlink = 1 | ||||
|         self.st_uid = 1000 | ||||
|         self.st_gid = 1000 | ||||
|         self.st_size = 0 | ||||
|         self.st_atime = 0 | ||||
|         self.st_mtime = 0 | ||||
|         self.st_ctime = 0 | ||||
|  | ||||
|  | ||||
| class Gateway(object): | ||||
|     def __init__(self, base_url): | ||||
|         self.base_url = base_url | ||||
|  | ||||
|         ui = urllib.parse.urlparse(base_url) | ||||
|         self.web_root = ui.path.strip("/") | ||||
|         try: | ||||
|             self.web_host, self.web_port = ui.netloc.split(":") | ||||
|             self.web_port = int(self.web_port) | ||||
|         except: | ||||
|             self.web_host = ui.netloc | ||||
|             if ui.scheme == "http": | ||||
|                 self.web_port = 80 | ||||
|             elif ui.scheme == "https": | ||||
|                 raise Exception("todo") | ||||
|             else: | ||||
|                 raise Exception("bad url?") | ||||
|  | ||||
|         self.conns = {} | ||||
|  | ||||
|     def quotep(self, path): | ||||
|         # TODO: mojibake support | ||||
|         path = path.encode("utf-8", "ignore") | ||||
|         return quote(path, safe="/") | ||||
|  | ||||
|     def getconn(self, tid=None): | ||||
|         tid = tid or get_tid() | ||||
|         try: | ||||
|             return self.conns[tid] | ||||
|         except: | ||||
|             info("new conn [{}] [{}]".format(self.web_host, self.web_port)) | ||||
|  | ||||
|             conn = http.client.HTTPConnection(self.web_host, self.web_port, timeout=260) | ||||
|  | ||||
|             self.conns[tid] = conn | ||||
|             return conn | ||||
|  | ||||
|     def closeconn(self, tid=None): | ||||
|         tid = tid or get_tid() | ||||
|         try: | ||||
|             self.conns[tid].close() | ||||
|             del self.conns[tid] | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def sendreq(self, *args, **kwargs): | ||||
|         tid = get_tid() | ||||
|         try: | ||||
|             c = self.getconn(tid) | ||||
|             c.request(*list(args), **kwargs) | ||||
|             return c.getresponse() | ||||
|         except: | ||||
|             self.closeconn(tid) | ||||
|             c = self.getconn(tid) | ||||
|             c.request(*list(args), **kwargs) | ||||
|             return c.getresponse() | ||||
|  | ||||
|     def listdir(self, path): | ||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots" | ||||
|         r = self.sendreq("GET", web_path) | ||||
|         if r.status != 200: | ||||
|             self.closeconn() | ||||
|             raise Exception( | ||||
|                 "http error {} reading dir {} in {}".format( | ||||
|                     r.status, web_path, rice_tid() | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|         return self.parse_html(r) | ||||
|  | ||||
|     def download_file_range(self, path, ofs1, ofs2): | ||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?raw" | ||||
|         hdr_range = "bytes={}-{}".format(ofs1, ofs2 - 1) | ||||
|         log("downloading {}".format(hdr_range)) | ||||
|  | ||||
|         r = self.sendreq("GET", web_path, headers={"Range": hdr_range}) | ||||
|         if r.status != http.client.PARTIAL_CONTENT: | ||||
|             self.closeconn() | ||||
|             raise Exception( | ||||
|                 "http error {} reading file {} range {} in {}".format( | ||||
|                     r.status, web_path, hdr_range, rice_tid() | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|         return r.read() | ||||
|  | ||||
|     def parse_html(self, datasrc): | ||||
|         ret = [] | ||||
|         remainder = b"" | ||||
|         ptn = re.compile( | ||||
|             r"^<tr><td>(-|DIR)</td><td><a [^>]+>([^<]+)</a></td><td>([^<]+)</td><td>([^<]+)</td></tr>$" | ||||
|         ) | ||||
|  | ||||
|         while True: | ||||
|             buf = remainder + datasrc.read(4096) | ||||
|             # print('[{}]'.format(buf.decode('utf-8'))) | ||||
|             if not buf: | ||||
|                 break | ||||
|  | ||||
|             remainder = b"" | ||||
|             endpos = buf.rfind(b"\n") | ||||
|             if endpos >= 0: | ||||
|                 remainder = buf[endpos + 1 :] | ||||
|                 buf = buf[:endpos] | ||||
|  | ||||
|             lines = buf.decode("utf-8").split("\n") | ||||
|             for line in lines: | ||||
|                 m = ptn.match(line) | ||||
|                 if not m: | ||||
|                     # print(line) | ||||
|                     continue | ||||
|  | ||||
|                 ftype, fname, fsize, fdate = m.groups() | ||||
|                 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 | ||||
|  | ||||
|     def stat_dir(self, ts, sz=4096): | ||||
|         ret = Stat() | ||||
|         ret.st_mode = stat.S_IFDIR | 0o555 | ||||
|         ret.st_nlink = 2 | ||||
|         ret.st_size = sz | ||||
|         ret.st_atime = ts | ||||
|         ret.st_mtime = ts | ||||
|         ret.st_ctime = ts | ||||
|         return ret | ||||
|  | ||||
|     def stat_file(self, ts, sz): | ||||
|         ret = Stat() | ||||
|         ret.st_mode = stat.S_IFREG | 0o444 | ||||
|         ret.st_size = sz | ||||
|         ret.st_atime = ts | ||||
|         ret.st_mtime = ts | ||||
|         ret.st_ctime = ts | ||||
|         return ret | ||||
|  | ||||
|  | ||||
| class CPPF(Fuse): | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         Fuse.__init__(self, *args, **kwargs) | ||||
|  | ||||
|         self.url = None | ||||
|  | ||||
|         self.dircache = [] | ||||
|         self.dircache_mtx = threading.Lock() | ||||
|  | ||||
|         self.filecache = [] | ||||
|         self.filecache_mtx = threading.Lock() | ||||
|  | ||||
|     def init2(self): | ||||
|         # TODO figure out how python-fuse wanted this to go | ||||
|         self.gw = Gateway(self.url)  # .decode('utf-8')) | ||||
|         info("up") | ||||
|  | ||||
|     def clean_dircache(self): | ||||
|         """not threadsafe""" | ||||
|         now = time.time() | ||||
|         cutoff = 0 | ||||
|         for cn in self.dircache: | ||||
|             if now - cn.ts > 1: | ||||
|                 cutoff += 1 | ||||
|             else: | ||||
|                 break | ||||
|  | ||||
|         if cutoff > 0: | ||||
|             self.dircache = self.dircache[cutoff:] | ||||
|  | ||||
|     def get_cached_dir(self, dirpath): | ||||
|         # with self.dircache_mtx: | ||||
|         if True: | ||||
|             self.clean_dircache() | ||||
|             for cn in self.dircache: | ||||
|                 if cn.tag == dirpath: | ||||
|                     return cn | ||||
|  | ||||
|         return None | ||||
|  | ||||
|     """ | ||||
|             ,-------------------------------,  g1>=c1, g2<=c2 | ||||
|             |cache1                   cache2|  buf[g1-c1:(g1-c1)+(g2-g1)] | ||||
|             `-------------------------------' | ||||
|                     ,---------------, | ||||
|                     |get1       get2| | ||||
|                     `---------------' | ||||
|     __________________________________________________________________________ | ||||
|  | ||||
|             ,-------------------------------,  g2<=c2, (g2>=c1) | ||||
|             |cache1                   cache2|  cdr=buf[:g2-c1] | ||||
|             `-------------------------------'  dl car; g1-512K:c1 | ||||
|     ,---------------, | ||||
|     |get1       get2| | ||||
|     `---------------' | ||||
|     __________________________________________________________________________ | ||||
|  | ||||
|             ,-------------------------------,  g1>=c1, (g1<=c2) | ||||
|             |cache1                   cache2|  car=buf[c2-g1:] | ||||
|             `-------------------------------'  dl cdr; c2:c2+1M | ||||
|                                     ,---------------, | ||||
|                                     |get1       get2| | ||||
|                                     `---------------' | ||||
|     """ | ||||
|  | ||||
|     def get_cached_file(self, path, get1, get2, file_sz): | ||||
|         car = None | ||||
|         cdr = None | ||||
|         ncn = -1 | ||||
|         # with self.filecache_mtx: | ||||
|         if True: | ||||
|             dbg("cache request from {} to {}, size {}".format(get1, get2, file_sz)) | ||||
|             for cn in self.filecache: | ||||
|                 ncn += 1 | ||||
|  | ||||
|                 cache_path, cache1 = cn.tag | ||||
|                 if cache_path != path: | ||||
|                     continue | ||||
|  | ||||
|                 cache2 = cache1 + len(cn.data) | ||||
|                 if get2 <= cache1 or get1 >= cache2: | ||||
|                     continue | ||||
|  | ||||
|                 if get1 >= cache1 and get2 <= cache2: | ||||
|                     # keep cache entry alive by moving it to the end | ||||
|                     self.filecache = ( | ||||
|                         self.filecache[:ncn] + self.filecache[ncn + 1 :] + [cn] | ||||
|                     ) | ||||
|                     buf_ofs = get1 - cache1 | ||||
|                     buf_end = buf_ofs + (get2 - get1) | ||||
|                     dbg( | ||||
|                         "found all ({}, {} to {}, len {}) [{}:{}] = {}".format( | ||||
|                             ncn, | ||||
|                             cache1, | ||||
|                             cache2, | ||||
|                             len(cn.data), | ||||
|                             buf_ofs, | ||||
|                             buf_end, | ||||
|                             buf_end - buf_ofs, | ||||
|                         ) | ||||
|                     ) | ||||
|                     return cn.data[buf_ofs:buf_end] | ||||
|  | ||||
|                 if get2 < cache2: | ||||
|                     x = cn.data[: get2 - cache1] | ||||
|                     if not cdr or len(cdr) < len(x): | ||||
|                         dbg( | ||||
|                             "found car ({}, {} to {}, len {}) [:{}-{}] = [:{}] = {}".format( | ||||
|                                 ncn, | ||||
|                                 cache1, | ||||
|                                 cache2, | ||||
|                                 len(cn.data), | ||||
|                                 get2, | ||||
|                                 cache1, | ||||
|                                 get2 - cache1, | ||||
|                                 len(x), | ||||
|                             ) | ||||
|                         ) | ||||
|                         cdr = x | ||||
|  | ||||
|                     continue | ||||
|  | ||||
|                 if get1 > cache1: | ||||
|                     x = cn.data[-(cache2 - get1) :] | ||||
|                     if not car or len(car) < len(x): | ||||
|                         dbg( | ||||
|                             "found cdr ({}, {} to {}, len {}) [-({}-{}):] = [-{}:] = {}".format( | ||||
|                                 ncn, | ||||
|                                 cache1, | ||||
|                                 cache2, | ||||
|                                 len(cn.data), | ||||
|                                 cache2, | ||||
|                                 get1, | ||||
|                                 cache2 - get1, | ||||
|                                 len(x), | ||||
|                             ) | ||||
|                         ) | ||||
|                         car = x | ||||
|  | ||||
|                     continue | ||||
|  | ||||
|                 raise Exception("what") | ||||
|  | ||||
|         if car and cdr: | ||||
|             dbg("<cache> have both") | ||||
|  | ||||
|             ret = car + cdr | ||||
|             if len(ret) == get2 - get1: | ||||
|                 return ret | ||||
|  | ||||
|             raise Exception("{} + {} != {} - {}".format(len(car), len(cdr), get2, get1)) | ||||
|  | ||||
|         elif cdr: | ||||
|             h_end = get1 + (get2 - get1) - len(cdr) | ||||
|             h_ofs = h_end - 512 * 1024 | ||||
|  | ||||
|             if h_ofs < 0: | ||||
|                 h_ofs = 0 | ||||
|  | ||||
|             buf_ofs = (get2 - get1) - len(cdr) | ||||
|  | ||||
|             dbg( | ||||
|                 "<cache> cdr {}, car {}-{}={} [-{}:]".format( | ||||
|                     len(cdr), h_ofs, h_end, h_end - h_ofs, buf_ofs | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|             buf = self.gw.download_file_range(path, h_ofs, h_end) | ||||
|             ret = buf[-buf_ofs:] + cdr | ||||
|  | ||||
|         elif car: | ||||
|             h_ofs = get1 + len(car) | ||||
|             h_end = h_ofs + 1024 * 1024 | ||||
|  | ||||
|             if h_end > file_sz: | ||||
|                 h_end = file_sz | ||||
|  | ||||
|             buf_ofs = (get2 - get1) - len(car) | ||||
|  | ||||
|             dbg( | ||||
|                 "<cache> car {}, cdr {}-{}={} [:{}]".format( | ||||
|                     len(car), h_ofs, h_end, h_end - h_ofs, buf_ofs | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|             buf = self.gw.download_file_range(path, h_ofs, h_end) | ||||
|             ret = car + buf[:buf_ofs] | ||||
|  | ||||
|         else: | ||||
|             h_ofs = get1 - 256 * 1024 | ||||
|             h_end = get2 + 1024 * 1024 | ||||
|  | ||||
|             if h_ofs < 0: | ||||
|                 h_ofs = 0 | ||||
|  | ||||
|             if h_end > file_sz: | ||||
|                 h_end = file_sz | ||||
|  | ||||
|             buf_ofs = get1 - h_ofs | ||||
|             buf_end = buf_ofs + get2 - get1 | ||||
|  | ||||
|             dbg( | ||||
|                 "<cache> {}-{}={} [{}:{}]".format( | ||||
|                     h_ofs, h_end, h_end - h_ofs, buf_ofs, buf_end | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|             buf = self.gw.download_file_range(path, h_ofs, h_end) | ||||
|             ret = buf[buf_ofs:buf_end] | ||||
|  | ||||
|         cn = CacheNode([path, h_ofs], buf) | ||||
|         # with self.filecache_mtx: | ||||
|         if True: | ||||
|             if len(self.filecache) > 6: | ||||
|                 self.filecache = self.filecache[1:] + [cn] | ||||
|             else: | ||||
|                 self.filecache.append(cn) | ||||
|  | ||||
|         return ret | ||||
|  | ||||
|     def _readdir(self, path): | ||||
|         path = path.strip("/") | ||||
|         log("readdir {}".format(path)) | ||||
|  | ||||
|         ret = self.gw.listdir(path) | ||||
|  | ||||
|         # with self.dircache_mtx: | ||||
|         if True: | ||||
|             cn = CacheNode(path, ret) | ||||
|             self.dircache.append(cn) | ||||
|             self.clean_dircache() | ||||
|  | ||||
|         return ret | ||||
|  | ||||
|     def readdir(self, path, offset): | ||||
|         for e in self._readdir(path)[offset:]: | ||||
|             # log("yield [{}]".format(e[0])) | ||||
|             yield fuse.Direntry(e[0]) | ||||
|  | ||||
|     def open(self, path, flags): | ||||
|         if (flags & (os.O_RDONLY | os.O_WRONLY | os.O_RDWR)) != os.O_RDONLY: | ||||
|             return -errno.EACCES | ||||
|  | ||||
|         st = self.getattr(path) | ||||
|         try: | ||||
|             if st.st_nlink > 0: | ||||
|                 return st | ||||
|         except: | ||||
|             return st  # -int(os.errcode) | ||||
|  | ||||
|     def read(self, path, length, offset, fh=None, *args): | ||||
|         if args: | ||||
|             log("unexpected args [" + "] [".join(repr(x) for x in args) + "]") | ||||
|             raise Exception() | ||||
|  | ||||
|         path = path.strip("/") | ||||
|  | ||||
|         ofs2 = offset + length | ||||
|         log("read {} @ {} len {} end {}".format(path, offset, length, ofs2)) | ||||
|  | ||||
|         st = self.getattr(path) | ||||
|         try: | ||||
|             file_sz = st.st_size | ||||
|         except: | ||||
|             return st  # -int(os.errcode) | ||||
|  | ||||
|         if ofs2 > file_sz: | ||||
|             ofs2 = file_sz | ||||
|             log("truncate to len {} end {}".format(ofs2 - offset, ofs2)) | ||||
|  | ||||
|         if file_sz == 0 or offset >= ofs2: | ||||
|             return b"" | ||||
|  | ||||
|         # toggle cache here i suppose | ||||
|         # return self.get_cached_file(path, offset, ofs2, file_sz) | ||||
|         return self.gw.download_file_range(path, offset, ofs2) | ||||
|  | ||||
|     def getattr(self, path): | ||||
|         log("getattr [{}]".format(path)) | ||||
|  | ||||
|         path = path.strip("/") | ||||
|         try: | ||||
|             dirpath, fname = path.rsplit("/", 1) | ||||
|         except: | ||||
|             dirpath = "" | ||||
|             fname = path | ||||
|  | ||||
|         if not path: | ||||
|             ret = self.gw.stat_dir(time.time()) | ||||
|             dbg("=root") | ||||
|             return ret | ||||
|  | ||||
|         cn = self.get_cached_dir(dirpath) | ||||
|         if cn: | ||||
|             log("cache ok") | ||||
|             dents = cn.data | ||||
|         else: | ||||
|             log("cache miss") | ||||
|             dents = self._readdir(dirpath) | ||||
|  | ||||
|         for cache_name, cache_stat, _ in dents: | ||||
|             if cache_name == fname: | ||||
|                 dbg("=file") | ||||
|                 return cache_stat | ||||
|  | ||||
|         log("=404") | ||||
|         return -errno.ENOENT | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||
|  | ||||
|     server = CPPF() | ||||
|     server.parser.add_option(mountopt="url", metavar="BASE_URL", default=None) | ||||
|     server.parse(values=server, errex=1) | ||||
|     if not server.url or not str(server.url).startswith("http"): | ||||
|         print("\nerror:") | ||||
|         print("  need argument: -o url=<...>") | ||||
|         print("  need argument: mount-path") | ||||
|         print("example:") | ||||
|         print( | ||||
|             "  ./copyparty-fuseb.py -f -o allow_other,auto_unmount,nonempty,url=http://192.168.1.69:3923 /mnt/nas" | ||||
|         ) | ||||
|         sys.exit(1) | ||||
|  | ||||
|     server.init2() | ||||
|     threading.Thread(target=server.main, daemon=True).start() | ||||
|     while True: | ||||
|         time.sleep(9001) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										245
									
								
								bin/dbtool.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										245
									
								
								bin/dbtool.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,245 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import shutil | ||||
| import sqlite3 | ||||
| import argparse | ||||
|  | ||||
| DB_VER1 = 3 | ||||
| DB_VER2 = 4 | ||||
|  | ||||
|  | ||||
| def die(msg): | ||||
|     print("\033[31m\n" + msg + "\n\033[0m") | ||||
|     sys.exit(1) | ||||
|  | ||||
|  | ||||
| def read_ver(db): | ||||
|     for tab in ["ki", "kv"]: | ||||
|         try: | ||||
|             c = db.execute(r"select v from {} where k = 'sver'".format(tab)) | ||||
|         except: | ||||
|             continue | ||||
|  | ||||
|         rows = c.fetchall() | ||||
|         if rows: | ||||
|             return int(rows[0][0]) | ||||
|  | ||||
|     return "corrupt" | ||||
|  | ||||
|  | ||||
| def ls(db): | ||||
|     nfiles = next(db.execute("select count(w) from up"))[0] | ||||
|     ntags = next(db.execute("select count(w) from mt"))[0] | ||||
|     print(f"{nfiles} files") | ||||
|     print(f"{ntags} tags\n") | ||||
|  | ||||
|     print("number of occurences for each tag,") | ||||
|     print(" 'x' = file has no tags") | ||||
|     print(" 't:mtp' = the mtp flag (file not mtp processed yet)") | ||||
|     print() | ||||
|     for k, nk in db.execute("select k, count(k) from mt group by k order by k"): | ||||
|         print(f"{nk:9} {k}") | ||||
|  | ||||
|  | ||||
| def compare(n1, d1, n2, d2, verbose): | ||||
|     nt = next(d1.execute("select count(w) from up"))[0] | ||||
|     n = 0 | ||||
|     miss = 0 | ||||
|     for w1, rd, fn in d1.execute("select w, rd, fn from up"): | ||||
|         n += 1 | ||||
|         if n % 25_000 == 0: | ||||
|             m = f"\033[36mchecked {n:,} of {nt:,} files in {n1} against {n2}\033[0m" | ||||
|             print(m) | ||||
|  | ||||
|         if rd.split("/", 1)[0] == ".hist": | ||||
|             continue | ||||
|  | ||||
|         q = "select w from up where rd = ? and fn = ?" | ||||
|         hit = d2.execute(q, (rd, fn)).fetchone() | ||||
|         if not hit: | ||||
|             miss += 1 | ||||
|             if verbose: | ||||
|                 print(f"file in {n1} missing in {n2}: [{w1}] {rd}/{fn}") | ||||
|  | ||||
|     print(f" {miss} files in {n1} missing in {n2}\n") | ||||
|  | ||||
|     nt = next(d1.execute("select count(w) from mt"))[0] | ||||
|     n = 0 | ||||
|     miss = {} | ||||
|     nmiss = 0 | ||||
|     for w1, k, v in d1.execute("select * from mt"): | ||||
|  | ||||
|         n += 1 | ||||
|         if n % 100_000 == 0: | ||||
|             m = f"\033[36mchecked {n:,} of {nt:,} tags in {n1} against {n2}, so far {nmiss} missing tags\033[0m" | ||||
|             print(m) | ||||
|  | ||||
|         q = "select rd, fn from up where substr(w,1,16) = ?" | ||||
|         rd, fn = d1.execute(q, (w1,)).fetchone() | ||||
|         if rd.split("/", 1)[0] == ".hist": | ||||
|             continue | ||||
|  | ||||
|         q = "select substr(w,1,16) from up where rd = ? and fn = ?" | ||||
|         w2 = d2.execute(q, (rd, fn)).fetchone() | ||||
|         if w2: | ||||
|             w2 = w2[0] | ||||
|  | ||||
|         v2 = None | ||||
|         if w2: | ||||
|             v2 = d2.execute( | ||||
|                 "select v from mt where w = ? and +k = ?", (w2, k) | ||||
|             ).fetchone() | ||||
|             if v2: | ||||
|                 v2 = v2[0] | ||||
|  | ||||
|         # if v != v2 and v2 and k in [".bpm", "key"] and n2 == "src": | ||||
|         #    print(f"{w} [{rd}/{fn}] {k} = [{v}] / [{v2}]") | ||||
|  | ||||
|         if v2 is not None: | ||||
|             if k.startswith("."): | ||||
|                 try: | ||||
|                     diff = abs(float(v) - float(v2)) | ||||
|                     if diff > float(v) / 0.9: | ||||
|                         v2 = None | ||||
|                     else: | ||||
|                         v2 = v | ||||
|                 except: | ||||
|                     pass | ||||
|  | ||||
|             if v != v2: | ||||
|                 v2 = None | ||||
|  | ||||
|         if v2 is None: | ||||
|             nmiss += 1 | ||||
|             try: | ||||
|                 miss[k] += 1 | ||||
|             except: | ||||
|                 miss[k] = 1 | ||||
|  | ||||
|             if verbose: | ||||
|                 print(f"missing in {n2}: [{w1}] [{rd}/{fn}] {k} = {v}") | ||||
|  | ||||
|     for k, v in sorted(miss.items()): | ||||
|         if v: | ||||
|             print(f"{n1} has {v:6} more {k:<6} tags than {n2}") | ||||
|  | ||||
|     print(f"in total, {nmiss} missing tags in {n2}\n") | ||||
|  | ||||
|  | ||||
| def copy_mtp(d1, d2, tag, rm): | ||||
|     nt = next(d1.execute("select count(w) from mt where k = ?", (tag,)))[0] | ||||
|     n = 0 | ||||
|     ndone = 0 | ||||
|     for w1, k, v in d1.execute("select * from mt where k = ?", (tag,)): | ||||
|         n += 1 | ||||
|         if n % 25_000 == 0: | ||||
|             m = f"\033[36m{n:,} of {nt:,} tags checked, so far {ndone} copied\033[0m" | ||||
|             print(m) | ||||
|  | ||||
|         q = "select rd, fn from up where substr(w,1,16) = ?" | ||||
|         rd, fn = d1.execute(q, (w1,)).fetchone() | ||||
|         if rd.split("/", 1)[0] == ".hist": | ||||
|             continue | ||||
|  | ||||
|         q = "select substr(w,1,16) from up where rd = ? and fn = ?" | ||||
|         w2 = d2.execute(q, (rd, fn)).fetchone() | ||||
|         if not w2: | ||||
|             continue | ||||
|  | ||||
|         w2 = w2[0] | ||||
|         hit = d2.execute("select v from mt where w = ? and +k = ?", (w2, k)).fetchone() | ||||
|         if hit: | ||||
|             hit = hit[0] | ||||
|  | ||||
|         if hit != v: | ||||
|             ndone += 1 | ||||
|             if hit is not None: | ||||
|                 d2.execute("delete from mt where w = ? and +k = ?", (w2, k)) | ||||
|  | ||||
|             d2.execute("insert into mt values (?,?,?)", (w2, k, v)) | ||||
|             if rm: | ||||
|                 d2.execute("delete from mt where w = ? and +k = 't:mtp'", (w2,)) | ||||
|  | ||||
|     d2.commit() | ||||
|     print(f"copied {ndone} {tag} tags over") | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     os.system("") | ||||
|     print() | ||||
|  | ||||
|     ap = argparse.ArgumentParser() | ||||
|     ap.add_argument("db", help="database to work on") | ||||
|     ap.add_argument("-src", metavar="DB", type=str, help="database to copy from") | ||||
|  | ||||
|     ap2 = ap.add_argument_group("informational / read-only stuff") | ||||
|     ap2.add_argument("-v", action="store_true", help="verbose") | ||||
|     ap2.add_argument("-ls", action="store_true", help="list summary for db") | ||||
|     ap2.add_argument("-cmp", action="store_true", help="compare databases") | ||||
|  | ||||
|     ap2 = ap.add_argument_group("options which modify target db") | ||||
|     ap2.add_argument("-copy", metavar="TAG", type=str, help="mtp tag to copy over") | ||||
|     ap2.add_argument( | ||||
|         "-rm-mtp-flag", | ||||
|         action="store_true", | ||||
|         help="when an mtp tag is copied over, also mark that as done, so copyparty won't run mtp on it", | ||||
|     ) | ||||
|     ap2.add_argument("-vac", action="store_true", help="optimize DB") | ||||
|  | ||||
|     ar = ap.parse_args() | ||||
|  | ||||
|     for v in [ar.db, ar.src]: | ||||
|         if v and not os.path.exists(v): | ||||
|             die("database must exist") | ||||
|  | ||||
|     db = sqlite3.connect(ar.db) | ||||
|     ds = sqlite3.connect(ar.src) if ar.src else None | ||||
|  | ||||
|     # revert journals | ||||
|     for d, p in [[db, ar.db], [ds, ar.src]]: | ||||
|         if not d: | ||||
|             continue | ||||
|  | ||||
|         pj = "{}-journal".format(p) | ||||
|         if not os.path.exists(pj): | ||||
|             continue | ||||
|  | ||||
|         d.execute("create table foo (bar int)") | ||||
|         d.execute("drop table foo") | ||||
|  | ||||
|     if ar.copy: | ||||
|         db.close() | ||||
|         shutil.copy2(ar.db, "{}.bak.dbtool.{:x}".format(ar.db, int(time.time()))) | ||||
|         db = sqlite3.connect(ar.db) | ||||
|  | ||||
|     for d, n in [[ds, "src"], [db, "dst"]]: | ||||
|         if not d: | ||||
|             continue | ||||
|  | ||||
|         ver = read_ver(d) | ||||
|         if ver == "corrupt": | ||||
|             die("{} database appears to be corrupt, sorry") | ||||
|  | ||||
|         if ver < DB_VER1 or ver > DB_VER2: | ||||
|             m = f"{n} db is version {ver}, this tool only supports versions between {DB_VER1} and {DB_VER2}, please upgrade it with copyparty first" | ||||
|             die(m) | ||||
|  | ||||
|     if ar.ls: | ||||
|         ls(db) | ||||
|  | ||||
|     if ar.cmp: | ||||
|         if not ds: | ||||
|             die("need src db to compare against") | ||||
|  | ||||
|         compare("src", ds, "dst", db, ar.v) | ||||
|         compare("dst", db, "src", ds, ar.v) | ||||
|  | ||||
|     if ar.copy: | ||||
|         copy_mtp(ds, db, ar.copy, ar.rm_mtp_flag) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										41
									
								
								bin/mtag/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								bin/mtag/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| standalone programs which take an audio file as argument | ||||
|  | ||||
| some of these rely on libraries which are not MIT-compatible | ||||
|  | ||||
| * [audio-bpm.py](./audio-bpm.py) detects the BPM of music using the BeatRoot Vamp Plugin; imports GPL2 | ||||
| * [audio-key.py](./audio-key.py) detects the melodic key of music using the Mixxx fork of keyfinder; imports GPL3 | ||||
| * [media-hash.py](./media-hash.py) generates checksums for audio and video streams; uses FFmpeg (LGPL or GPL) | ||||
|  | ||||
|  | ||||
| # dependencies | ||||
|  | ||||
| run [`install-deps.sh`](install-deps.sh) to build/install most dependencies required by these programs (supports windows/linux/macos) | ||||
|  | ||||
| *alternatively* (or preferably) use packages from your distro instead, then you'll need at least these: | ||||
|  | ||||
| * from distro: `numpy vamp-plugin-sdk beatroot-vamp mixxx-keyfinder ffmpeg` | ||||
| * from pypy: `keyfinder vamp` | ||||
|  | ||||
|  | ||||
| # usage from copyparty | ||||
|  | ||||
| `copyparty -e2dsa -e2ts` followed by any combination of these: | ||||
| * `-mtp key=f,audio-key.py` | ||||
| * `-mtp .bpm=f,audio-bpm.py` | ||||
| * `-mtp ahash,vhash=f,media-hash.py` | ||||
|  | ||||
| * `f,` makes the detected value replace any existing values | ||||
| * the `.` in `.bpm` indicates numeric value | ||||
| * assumes the python files are in the folder you're launching copyparty from, replace the filename with a relative/absolute path if that's not the case | ||||
| * `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 | ||||
|  | ||||
| instead of affecting all volumes, you can set the options for just one volume like so: | ||||
|  | ||||
| `copyparty -v /mnt/nas/music:/music:r:c,e2dsa:c,e2ts` immediately followed by any combination of these: | ||||
|  | ||||
| * `:c,mtp=key=f,audio-key.py` | ||||
| * `:c,mtp=.bpm=f,audio-bpm.py` | ||||
| * `:c,mtp=ahash,vhash=f,media-hash.py` | ||||
							
								
								
									
										69
									
								
								bin/mtag/audio-bpm.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										69
									
								
								bin/mtag/audio-bpm.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import vamp | ||||
| import tempfile | ||||
| import numpy as np | ||||
| import subprocess as sp | ||||
|  | ||||
| from copyparty.util import fsenc | ||||
|  | ||||
| """ | ||||
| dep: vamp | ||||
| dep: beatroot-vamp | ||||
| dep: ffmpeg | ||||
| """ | ||||
|  | ||||
|  | ||||
| def det(tf): | ||||
|     # fmt: off | ||||
|     sp.check_call([ | ||||
|         "ffmpeg", | ||||
|         "-nostdin", | ||||
|         "-hide_banner", | ||||
|         "-v", "fatal", | ||||
|         "-ss", "13", | ||||
|         "-y", "-i", fsenc(sys.argv[1]), | ||||
|         "-ac", "1", | ||||
|         "-ar", "22050", | ||||
|         "-t", "300", | ||||
|         "-f", "f32le", | ||||
|         tf | ||||
|     ]) | ||||
|     # fmt: on | ||||
|  | ||||
|     with open(tf, "rb") as f: | ||||
|         d = np.fromfile(f, dtype=np.float32) | ||||
|         try: | ||||
|             # 98% accuracy on jcore | ||||
|             c = vamp.collect(d, 22050, "beatroot-vamp:beatroot") | ||||
|             cl = c["list"] | ||||
|         except: | ||||
|             # fallback; 73% accuracy | ||||
|             plug = "vamp-example-plugins:fixedtempo" | ||||
|             c = vamp.collect(d, 22050, plug, parameters={"maxdflen": 40}) | ||||
|             print(c["list"][0]["label"].split(" ")[0]) | ||||
|             return | ||||
|  | ||||
|         # throws if detection failed: | ||||
|         bpm = float(cl[-1]["timestamp"] - cl[1]["timestamp"]) | ||||
|         bpm = round(60 * ((len(cl) - 1) / bpm), 2) | ||||
|         print(f"{bpm:.2f}") | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     with tempfile.NamedTemporaryFile(suffix=".pcm", delete=False) as f: | ||||
|         f.write(b"h") | ||||
|         tf = f.name | ||||
|  | ||||
|     try: | ||||
|         det(tf) | ||||
|     except: | ||||
|         pass  # mute | ||||
|     finally: | ||||
|         os.unlink(tf) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										123
									
								
								bin/mtag/audio-key-slicing.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										123
									
								
								bin/mtag/audio-key-slicing.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,123 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import tempfile | ||||
| import subprocess as sp | ||||
|  | ||||
| import keyfinder | ||||
|  | ||||
| from copyparty.util import fsenc | ||||
|  | ||||
| """ | ||||
| dep: github/mixxxdj/libkeyfinder | ||||
| dep: pypi/keyfinder | ||||
| dep: ffmpeg | ||||
|  | ||||
| note: this is a janky edition of the regular audio-key.py, | ||||
|   slicing the files at 20sec intervals and keeping 5sec from each, | ||||
|   surprisingly accurate but still garbage (446 ok, 69 bad, 13% miss) | ||||
|  | ||||
|   it is fast tho | ||||
| """ | ||||
|  | ||||
|  | ||||
| def get_duration(): | ||||
|     # TODO provide ffprobe tags to mtp as json | ||||
|  | ||||
|     # fmt: off | ||||
|     dur = sp.check_output([ | ||||
|         "ffprobe", | ||||
|         "-hide_banner", | ||||
|         "-v", "fatal", | ||||
|         "-show_streams", | ||||
|         "-show_format", | ||||
|         fsenc(sys.argv[1]) | ||||
|     ]) | ||||
|     # fmt: on | ||||
|  | ||||
|     dur = dur.decode("ascii", "replace").split("\n") | ||||
|     dur = [x.split("=")[1] for x in dur if x.startswith("duration=")] | ||||
|     dur = [float(x) for x in dur if re.match(r"^[0-9\.,]+$", x)] | ||||
|     return list(sorted(dur))[-1] if dur else None | ||||
|  | ||||
|  | ||||
| def get_segs(dur): | ||||
|     # keep first 5s of each 20s, | ||||
|     # keep entire last segment | ||||
|     ofs = 0 | ||||
|     segs = [] | ||||
|     while True: | ||||
|         seg = [ofs, 5] | ||||
|         segs.append(seg) | ||||
|         if dur - ofs < 20: | ||||
|             seg[-1] = int(dur - seg[0]) | ||||
|             break | ||||
|  | ||||
|         ofs += 20 | ||||
|  | ||||
|     return segs | ||||
|  | ||||
|  | ||||
| def slice(tf): | ||||
|     dur = get_duration() | ||||
|     dur = min(dur, 600)  # max 10min | ||||
|     segs = get_segs(dur) | ||||
|  | ||||
|     # fmt: off | ||||
|     cmd = [ | ||||
|         "ffmpeg", | ||||
|         "-nostdin", | ||||
|         "-hide_banner", | ||||
|         "-v", "fatal", | ||||
|         "-y" | ||||
|     ] | ||||
|  | ||||
|     for seg in segs: | ||||
|         cmd.extend([ | ||||
|             "-ss", str(seg[0]), | ||||
|             "-i", fsenc(sys.argv[1]) | ||||
|         ]) | ||||
|      | ||||
|     filt = "" | ||||
|     for n, seg in enumerate(segs): | ||||
|         filt += "[{}:a:0]atrim=duration={}[a{}]; ".format(n, seg[1], n) | ||||
|      | ||||
|     prev = "a0" | ||||
|     for n in range(1, len(segs)): | ||||
|         nxt = "b{}".format(n) | ||||
|         filt += "[{}][a{}]acrossfade=d=0.5[{}]; ".format(prev, n, nxt) | ||||
|         prev = nxt | ||||
|  | ||||
|     cmd.extend([ | ||||
|         "-filter_complex", filt[:-2], | ||||
|         "-map", "[{}]".format(nxt), | ||||
|         "-sample_fmt", "s16", | ||||
|         tf | ||||
|     ]) | ||||
|     # fmt: on | ||||
|  | ||||
|     # print(cmd) | ||||
|     sp.check_call(cmd) | ||||
|  | ||||
|  | ||||
| def det(tf): | ||||
|     slice(tf) | ||||
|     print(keyfinder.key(tf).camelot()) | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     with tempfile.NamedTemporaryFile(suffix=".flac", delete=False) as f: | ||||
|         f.write(b"h") | ||||
|         tf = f.name | ||||
|  | ||||
|     try: | ||||
|         det(tf) | ||||
|     finally: | ||||
|         os.unlink(tf) | ||||
|         pass | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										54
									
								
								bin/mtag/audio-key.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										54
									
								
								bin/mtag/audio-key.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import tempfile | ||||
| import subprocess as sp | ||||
| import keyfinder | ||||
|  | ||||
| from copyparty.util import fsenc | ||||
|  | ||||
| """ | ||||
| dep: github/mixxxdj/libkeyfinder | ||||
| dep: pypi/keyfinder | ||||
| dep: ffmpeg | ||||
| """ | ||||
|  | ||||
|  | ||||
| # tried trimming the first/last 5th, bad idea, | ||||
| # misdetects 9a law field (Sphere Caliber) as 10b, | ||||
| # obvious when mixing 9a ghostly parapara ship | ||||
|  | ||||
|  | ||||
| def det(tf): | ||||
|     # fmt: off | ||||
|     sp.check_call([ | ||||
|         "ffmpeg", | ||||
|         "-nostdin", | ||||
|         "-hide_banner", | ||||
|         "-v", "fatal", | ||||
|         "-y", "-i", fsenc(sys.argv[1]), | ||||
|         "-t", "300", | ||||
|         "-sample_fmt", "s16", | ||||
|         tf | ||||
|     ]) | ||||
|     # fmt: on | ||||
|  | ||||
|     print(keyfinder.key(tf).camelot()) | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     with tempfile.NamedTemporaryFile(suffix=".flac", delete=False) as f: | ||||
|         f.write(b"h") | ||||
|         tf = f.name | ||||
|  | ||||
|     try: | ||||
|         det(tf) | ||||
|     except: | ||||
|         pass  # mute | ||||
|     finally: | ||||
|         os.unlink(tf) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										96
									
								
								bin/mtag/exe.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								bin/mtag/exe.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,96 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import sys | ||||
| import time | ||||
| import json | ||||
| import pefile | ||||
|  | ||||
| """ | ||||
| retrieve exe info, | ||||
| example for multivalue providers | ||||
| """ | ||||
|  | ||||
|  | ||||
| def unk(v): | ||||
|     return "unk({:04x})".format(v) | ||||
|  | ||||
|  | ||||
| class PE2(pefile.PE): | ||||
|     def __init__(self, *a, **ka): | ||||
|         for k in [ | ||||
|             # -- parse_data_directories: | ||||
|             "parse_import_directory", | ||||
|             "parse_export_directory", | ||||
|             # "parse_resources_directory", | ||||
|             "parse_debug_directory", | ||||
|             "parse_relocations_directory", | ||||
|             "parse_directory_tls", | ||||
|             "parse_directory_load_config", | ||||
|             "parse_delay_import_directory", | ||||
|             "parse_directory_bound_imports", | ||||
|             # -- full_load: | ||||
|             "parse_rich_header", | ||||
|         ]: | ||||
|             setattr(self, k, self.noop) | ||||
|  | ||||
|         super(PE2, self).__init__(*a, **ka) | ||||
|  | ||||
|     def noop(*a, **ka): | ||||
|         pass | ||||
|  | ||||
|  | ||||
| try: | ||||
|     pe = PE2(sys.argv[1], fast_load=False) | ||||
| except: | ||||
|     sys.exit(0) | ||||
|  | ||||
| arch = pe.FILE_HEADER.Machine | ||||
| if arch == 0x14C: | ||||
|     arch = "x86" | ||||
| elif arch == 0x8664: | ||||
|     arch = "x64" | ||||
| else: | ||||
|     arch = unk(arch) | ||||
|  | ||||
| try: | ||||
|     buildtime = time.gmtime(pe.FILE_HEADER.TimeDateStamp) | ||||
|     buildtime = time.strftime("%Y-%m-%d_%H:%M:%S", buildtime) | ||||
| except: | ||||
|     buildtime = "invalid" | ||||
|  | ||||
| ui = pe.OPTIONAL_HEADER.Subsystem | ||||
| if ui == 2: | ||||
|     ui = "GUI" | ||||
| elif ui == 3: | ||||
|     ui = "cmdline" | ||||
| else: | ||||
|     ui = unk(ui) | ||||
|  | ||||
| extra = {} | ||||
| if hasattr(pe, "FileInfo"): | ||||
|     for v1 in pe.FileInfo: | ||||
|         for v2 in v1: | ||||
|             if v2.name != "StringFileInfo": | ||||
|                 continue | ||||
|  | ||||
|             for v3 in v2.StringTable: | ||||
|                 for k, v in v3.entries.items(): | ||||
|                     v = v.decode("utf-8", "replace").strip() | ||||
|                     if not v: | ||||
|                         continue | ||||
|  | ||||
|                     if k in [b"FileVersion", b"ProductVersion"]: | ||||
|                         extra["ver"] = v | ||||
|  | ||||
|                     if k in [b"OriginalFilename", b"InternalName"]: | ||||
|                         extra["orig"] = v | ||||
|  | ||||
| r = { | ||||
|     "arch": arch, | ||||
|     "built": buildtime, | ||||
|     "ui": ui, | ||||
|     "cksum": "{:08x}".format(pe.OPTIONAL_HEADER.CheckSum), | ||||
| } | ||||
| r.update(extra) | ||||
|  | ||||
| print(json.dumps(r, indent=4)) | ||||
							
								
								
									
										9
									
								
								bin/mtag/file-ext.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								bin/mtag/file-ext.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import sys | ||||
|  | ||||
| """ | ||||
| example that just prints the file extension | ||||
| """ | ||||
|  | ||||
| print(sys.argv[1].split(".")[-1]) | ||||
							
								
								
									
										265
									
								
								bin/mtag/install-deps.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										265
									
								
								bin/mtag/install-deps.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,265 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
|  | ||||
| # install dependencies for audio-*.py | ||||
| # | ||||
| # linux: requires {python3,ffmpeg,fftw}-dev py3-{wheel,pip} py3-numpy{,-dev} vamp-sdk-dev patchelf | ||||
| # win64: requires msys2-mingw64 environment | ||||
| # macos: requires macports | ||||
| # | ||||
| # has the following manual dependencies, especially on mac: | ||||
| #   https://www.vamp-plugins.org/pack.html | ||||
| # | ||||
| # installs stuff to the following locations: | ||||
| #   ~/pe/ | ||||
| #   whatever your python uses for --user packages | ||||
| # | ||||
| # does the following terrible things: | ||||
| #   modifies the keyfinder python lib to load the .so in ~/pe | ||||
|  | ||||
|  | ||||
| linux=1 | ||||
|  | ||||
| win= | ||||
| [ ! -z "$MSYSTEM" ] || [ -e /msys2.exe ] && { | ||||
| 	[ "$MSYSTEM" = MINGW64 ] || { | ||||
| 		echo windows detected, msys2-mingw64 required | ||||
| 		exit 1 | ||||
| 	} | ||||
| 	pacman -S --needed mingw-w64-x86_64-{ffmpeg,python,python-pip,vamp-plugin-sdk} | ||||
| 	win=1 | ||||
| 	linux= | ||||
| } | ||||
|  | ||||
| mac= | ||||
| [ $(uname -s) = Darwin ] && { | ||||
| 	#pybin="$(printf '%s\n' /opt/local/bin/python* | (sed -E 's/(.*\/[^/0-9]+)([0-9]?[^/]*)$/\2 \1/' || cat) | (sort -nr || cat) | (sed -E 's/([^ ]*) (.*)/\2\1/' || cat) | grep -E '/(python|pypy)[0-9\.-]*$' | head -n 1)" | ||||
| 	pybin=/opt/local/bin/python3.9 | ||||
| 	[ -e "$pybin" ] || { | ||||
| 		echo mac detected, python3 from macports required | ||||
| 		exit 1 | ||||
| 	} | ||||
| 	pkgs='ffmpeg python39 py39-wheel' | ||||
| 	ninst=$(port installed | awk '/^  /{print$1}' | sort | uniq | grep -E '^('"$(echo "$pkgs" | tr ' ' '|')"')$' | wc -l) | ||||
| 	[ $ninst -eq 3 ] || { | ||||
| 		sudo port install $pkgs | ||||
| 	} | ||||
| 	mac=1 | ||||
| 	linux= | ||||
| } | ||||
|  | ||||
| hash -r | ||||
|  | ||||
| [ $mac ] || { | ||||
| 	command -v python3 && pybin=python3 || pybin=python | ||||
| } | ||||
|  | ||||
| $pybin -m pip install --user numpy | ||||
|  | ||||
|  | ||||
| command -v gnutar && tar() { gnutar "$@"; } | ||||
| command -v gtar && tar() { gtar "$@"; } | ||||
| command -v gsed && sed() { gsed "$@"; } | ||||
|  | ||||
|  | ||||
| need() { | ||||
| 	command -v $1 >/dev/null || { | ||||
| 		echo need $1 | ||||
| 		exit 1 | ||||
| 	} | ||||
| } | ||||
| need cmake | ||||
| need ffmpeg | ||||
| need $pybin | ||||
| #need patchelf | ||||
|  | ||||
|  | ||||
| td="$(mktemp -d)" | ||||
| cln() { | ||||
| 	rm -rf "$td" | ||||
| } | ||||
| trap cln EXIT | ||||
| cd "$td" | ||||
| pwd | ||||
|  | ||||
|  | ||||
| dl_text() { | ||||
| 	command -v curl >/dev/null && exec curl "$@" | ||||
| 	exec wget -O- "$@" | ||||
| } | ||||
| dl_files() { | ||||
| 	local yolo= ex= | ||||
| 	[ $1 = "yolo" ] && yolo=1 && ex=k && shift | ||||
| 	command -v curl >/dev/null && exec curl -${ex}JOL "$@" | ||||
| 	 | ||||
| 	[ $yolo ] && ex=--no-check-certificate | ||||
| 	exec wget --trust-server-names $ex "$@" | ||||
| } | ||||
| export -f dl_files | ||||
|  | ||||
|  | ||||
| github_tarball() { | ||||
| 	dl_text "$1" | | ||||
| 	tee json | | ||||
| 	( | ||||
| 		# prefer jq if available | ||||
| 		jq -r '.tarball_url' || | ||||
|  | ||||
| 		# fallback to awk (sorry) | ||||
| 		awk -F\" '/"tarball_url": "/ {print$4}' | ||||
| 	) | | ||||
| 	tee /dev/stderr | | ||||
| 	tr -d '\r' | tr '\n' '\0' | | ||||
| 	xargs -0 bash -c 'dl_files "$@"' _ | ||||
| } | ||||
|  | ||||
|  | ||||
| gitlab_tarball() { | ||||
| 	dl_text "$1" | | ||||
| 	tee json | | ||||
| 	( | ||||
| 		# prefer jq if available | ||||
| 		jq -r '.[0].assets.sources[]|select(.format|test("tar.gz")).url' || | ||||
|  | ||||
| 		# fallback to abomination | ||||
| 		tr \" '\n' | grep -E '\.tar\.gz$' | head -n 1 | ||||
| 	) | | ||||
| 	tee /dev/stderr | | ||||
| 	tr -d '\r' | tr '\n' '\0' | | ||||
| 	tee links | | ||||
| 	xargs -0 bash -c 'dl_files "$@"' _ | ||||
| } | ||||
|  | ||||
|  | ||||
| install_keyfinder() { | ||||
| 	# windows support: | ||||
| 	#   use msys2 in mingw-w64 mode | ||||
| 	#   pacman -S --needed mingw-w64-x86_64-{ffmpeg,python} | ||||
| 	 | ||||
| 	github_tarball https://api.github.com/repos/mixxxdj/libkeyfinder/releases/latest | ||||
|  | ||||
| 	tar -xf mixxxdj-libkeyfinder-* | ||||
| 	rm -- *.tar.gz | ||||
| 	cd mixxxdj-libkeyfinder* | ||||
| 	 | ||||
| 	h="$HOME" | ||||
| 	so="lib/libkeyfinder.so" | ||||
| 	memes=() | ||||
|  | ||||
| 	[ $win ] && | ||||
| 		so="bin/libkeyfinder.dll" && | ||||
| 		h="$(printf '%s\n' "$USERPROFILE" | tr '\\' '/')" && | ||||
| 		memes+=(-G "MinGW Makefiles" -DBUILD_TESTING=OFF) | ||||
| 	 | ||||
| 	[ $mac ] && | ||||
| 		so="lib/libkeyfinder.dylib" | ||||
|  | ||||
| 	cmake -DCMAKE_INSTALL_PREFIX="$h/pe/keyfinder" "${memes[@]}" -S . -B build | ||||
| 	cmake --build build --parallel $(nproc || echo 4) | ||||
| 	cmake --install build | ||||
|  | ||||
| 	libpath="$h/pe/keyfinder/$so" | ||||
| 	[ $linux ] && [ ! -e "$libpath" ] && | ||||
| 		so=lib64/libkeyfinder.so | ||||
| 	 | ||||
| 	libpath="$h/pe/keyfinder/$so" | ||||
| 	[ -e "$libpath" ] || { | ||||
| 		echo "so not found at $sop" | ||||
| 		exit 1 | ||||
| 	} | ||||
| 	 | ||||
| 	# rm -rf /Users/ed/Library/Python/3.9/lib/python/site-packages/*keyfinder* | ||||
| 	CFLAGS="-I$h/pe/keyfinder/include -I/opt/local/include" \ | ||||
| 	LDFLAGS="-L$h/pe/keyfinder/lib -L$h/pe/keyfinder/lib64 -L/opt/local/lib" \ | ||||
| 	PKG_CONFIG_PATH=/c/msys64/mingw64/lib/pkgconfig \ | ||||
| 	$pybin -m pip install --user keyfinder | ||||
|  | ||||
| 	pypath="$($pybin -c 'import keyfinder; print(keyfinder.__file__)')" | ||||
| 	for pyso in "${pypath%/*}"/*.so; do | ||||
| 		[ -e "$pyso" ] || break | ||||
| 		patchelf --set-rpath "${libpath%/*}" "$pyso" || | ||||
| 			echo "WARNING: patchelf failed (only fatal on musl-based distros)" | ||||
| 	done | ||||
| 	 | ||||
| 	mv "$pypath"{,.bak} | ||||
| 	( | ||||
| 		printf 'import ctypes\nctypes.cdll.LoadLibrary("%s")\n' "$libpath" | ||||
| 		cat "$pypath.bak" | ||||
| 	) >"$pypath" | ||||
|  | ||||
| 	echo | ||||
| 	echo libkeyfinder successfully installed to the following locations: | ||||
| 	echo "  $libpath" | ||||
| 	echo "  $pypath" | ||||
| } | ||||
|  | ||||
|  | ||||
| have_beatroot() { | ||||
| 	$pybin -c 'import vampyhost, sys; plugs = vampyhost.list_plugins(); sys.exit(0 if "beatroot-vamp:beatroot" in plugs else 1)' | ||||
| } | ||||
|  | ||||
|  | ||||
| install_vamp() { | ||||
| 	# windows support: | ||||
| 	#   use msys2 in mingw-w64 mode | ||||
| 	#   pacman -S --needed mingw-w64-x86_64-{ffmpeg,python,python-pip,vamp-plugin-sdk} | ||||
| 	 | ||||
| 	$pybin -m pip install --user vamp | ||||
|  | ||||
| 	have_beatroot || { | ||||
| 		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) | ||||
| 		sha512sum -c <( | ||||
| 			echo "1f444d1d58ccf565c0adfe99f1a1aa62789e19f5071e46857e2adfbc9d453037bc1c4dcb039b02c16240e9b97f444aaff3afb625c86aa2470233e711f55b6874  -" | ||||
| 		) <beatroot-vamp-v1.0.tar.gz | ||||
| 		tar -xf beatroot-vamp-v1.0.tar.gz  | ||||
| 		cd beatroot-vamp-v1.0 | ||||
| 		make -f Makefile.linux -j4 | ||||
| 		# /home/ed/vamp /home/ed/.vamp /usr/local/lib/vamp | ||||
| 		mkdir ~/vamp | ||||
| 		cp -pv beatroot-vamp.* ~/vamp/ | ||||
| 	} | ||||
| 	 | ||||
| 	have_beatroot && | ||||
| 		printf '\033[32mfound the vamp beatroot plugin, nice\033[0m\n' || | ||||
| 		printf '\033[31mWARNING: could not find the vamp beatroot plugin, please install it for optimal results\033[0m\n' | ||||
| } | ||||
|  | ||||
|  | ||||
| # not in use because it kinda segfaults, also no windows support | ||||
| install_soundtouch() { | ||||
| 	gitlab_tarball https://gitlab.com/api/v4/projects/soundtouch%2Fsoundtouch/releases | ||||
| 	 | ||||
| 	tar -xvf soundtouch-* | ||||
| 	rm -- *.tar.gz | ||||
| 	cd soundtouch-* | ||||
| 	 | ||||
| 	# https://github.com/jrising/pysoundtouch | ||||
| 	./bootstrap | ||||
| 	./configure --enable-integer-samples CXXFLAGS="-fPIC" --prefix="$HOME/pe/soundtouch" | ||||
| 	make -j$(nproc || echo 4) | ||||
| 	make install | ||||
| 	 | ||||
| 	CFLAGS=-I$HOME/pe/soundtouch/include/ \ | ||||
| 	LDFLAGS=-L$HOME/pe/soundtouch/lib \ | ||||
| 	$pybin -m pip install --user git+https://github.com/snowxmas/pysoundtouch.git | ||||
| 	 | ||||
| 	pypath="$($pybin -c 'import importlib; print(importlib.util.find_spec("soundtouch").origin)')" | ||||
| 	libpath="$(echo "$HOME/pe/soundtouch/lib/")" | ||||
| 	patchelf --set-rpath "$libpath" "$pypath" | ||||
|  | ||||
| 	echo | ||||
| 	echo soundtouch successfully installed to the following locations: | ||||
| 	echo "  $libpath" | ||||
| 	echo "  $pypath" | ||||
| } | ||||
|  | ||||
|  | ||||
| [ "$1" = keyfinder ] && { install_keyfinder; exit $?; } | ||||
| [ "$1" = soundtouch ] && { install_soundtouch; exit $?; } | ||||
| [ "$1" = vamp ] && { install_vamp; exit $?; } | ||||
|  | ||||
| echo no args provided, installing keyfinder and vamp | ||||
| install_keyfinder | ||||
| install_vamp | ||||
							
								
								
									
										73
									
								
								bin/mtag/media-hash.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								bin/mtag/media-hash.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,73 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import re | ||||
| import sys | ||||
| import json | ||||
| import time | ||||
| import base64 | ||||
| import hashlib | ||||
| import subprocess as sp | ||||
|  | ||||
| try: | ||||
|     from copyparty.util import fsenc | ||||
| except: | ||||
|  | ||||
|     def fsenc(p): | ||||
|         return p | ||||
|  | ||||
|  | ||||
| """ | ||||
| dep: ffmpeg | ||||
| """ | ||||
|  | ||||
|  | ||||
| def det(): | ||||
|     # fmt: off | ||||
|     cmd = [ | ||||
|         "ffmpeg", | ||||
|         "-nostdin", | ||||
|         "-hide_banner", | ||||
|         "-v", "fatal", | ||||
|         "-i", fsenc(sys.argv[1]), | ||||
|         "-f", "framemd5", | ||||
|         "-" | ||||
|     ] | ||||
|     # fmt: on | ||||
|  | ||||
|     p = sp.Popen(cmd, stdout=sp.PIPE) | ||||
|     # ps = io.TextIOWrapper(p.stdout, encoding="utf-8") | ||||
|     ps = p.stdout | ||||
|  | ||||
|     chans = {} | ||||
|     for ln in ps: | ||||
|         if ln.startswith(b"#stream#"): | ||||
|             break | ||||
|  | ||||
|         m = re.match(r"^#media_type ([0-9]): ([a-zA-Z])", ln.decode("utf-8")) | ||||
|         if m: | ||||
|             chans[m.group(1)] = m.group(2) | ||||
|  | ||||
|     hashers = [hashlib.sha512(), hashlib.sha512()] | ||||
|     for ln in ps: | ||||
|         n = int(ln[:1]) | ||||
|         v = ln.rsplit(b",", 1)[-1].strip() | ||||
|         hashers[n].update(v) | ||||
|  | ||||
|     r = {} | ||||
|     for k, v in chans.items(): | ||||
|         dg = hashers[int(k)].digest()[:12] | ||||
|         dg = base64.urlsafe_b64encode(dg).decode("ascii") | ||||
|         r[v[0].lower() + "hash"] = dg | ||||
|  | ||||
|     print(json.dumps(r, indent=4)) | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     try: | ||||
|         det() | ||||
|     except: | ||||
|         pass  # mute | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										39
									
								
								bin/mtag/res/yt-ipr.conf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								bin/mtag/res/yt-ipr.conf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| # example config file to use copyparty as a youtube manifest collector, | ||||
| # use with copyparty like:  python copyparty.py -c yt-ipr.conf | ||||
| # | ||||
| # see docs/example.conf for a better explanation of the syntax, but | ||||
| # newlines are block separators, so adding blank lines inside a volume definition is bad | ||||
| # (use comments as separators instead) | ||||
|  | ||||
|  | ||||
| # create user ed, password wark | ||||
| u ed:wark | ||||
|  | ||||
|  | ||||
| # create a volume at /ytm which stores files at ./srv/ytm | ||||
| ./srv/ytm | ||||
| /ytm | ||||
| # write-only, but read-write for user ed | ||||
| w | ||||
| rw ed | ||||
| # rescan the volume on startup | ||||
| c e2dsa | ||||
| # collect tags from all new files since last scan | ||||
| c e2ts | ||||
| # optionally enable compression to make the files 50% smaller | ||||
| c pk | ||||
| # only allow uploads which are between 16k and 1m large | ||||
| c sz=16k-1m | ||||
| # allow up to 10 uploads over 5 minutes from each ip | ||||
| c maxn=10,300 | ||||
| # move uploads into subfolders: YEAR-MONTH / DAY-HOUR / <upload> | ||||
| c rotf=%Y-%m/%d-%H | ||||
| # delete uploads when they are 24 hours old | ||||
| c lifetime=86400 | ||||
| # add the parser and tell copyparty what tags it can expect from it | ||||
| c mtp=yt-id,yt-title,yt-author,yt-channel,yt-views,yt-private,yt-manifest,yt-expires=bin/mtag/yt-ipr.py | ||||
| # decide which tags we want to index and in what order | ||||
| c mte=yt-id,yt-title,yt-author,yt-channel,yt-views,yt-private,yt-manifest,yt-expires | ||||
|  | ||||
|  | ||||
| # create any other volumes you'd like down here, or merge this with an existing config file | ||||
							
								
								
									
										47
									
								
								bin/mtag/res/yt-ipr.user.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								bin/mtag/res/yt-ipr.user.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| // ==UserScript== | ||||
| // @name    youtube-playerdata-hub | ||||
| // @match   https://youtube.com/* | ||||
| // @match   https://*.youtube.com/* | ||||
| // @version 1.0 | ||||
| // @grant   GM_addStyle | ||||
| // ==/UserScript== | ||||
|  | ||||
| function main() { | ||||
|     var server = 'https://127.0.0.1:3923/ytm?pw=wark', | ||||
|         interval = 60; // sec | ||||
|  | ||||
|     var sent = {}; | ||||
|     function send(txt, mf_url, desc) { | ||||
|         if (sent[mf_url]) | ||||
|             return; | ||||
|  | ||||
|         fetch(server + '&_=' + Date.now(), { method: "PUT", body: txt }); | ||||
|         console.log('[yt-pdh] yeet %d bytes, %s', txt.length, desc); | ||||
|         sent[mf_url] = 1; | ||||
|     } | ||||
|  | ||||
|     function collect() { | ||||
|         try { | ||||
|             var pd = document.querySelector('ytd-watch-flexy'); | ||||
|             if (!pd) | ||||
|                 return console.log('[yt-pdh] no video found'); | ||||
|  | ||||
|             pd = pd.playerData; | ||||
|             var mu = pd.streamingData.dashManifestUrl || pd.streamingData.hlsManifestUrl; | ||||
|             if (!mu || !mu.length) | ||||
|                 return console.log('[yt-pdh] no manifest found'); | ||||
|  | ||||
|             var desc = pd.videoDetails.videoId + ', ' + pd.videoDetails.title; | ||||
|             send(JSON.stringify(pd), mu, desc); | ||||
|         } | ||||
|         catch (ex) { | ||||
|             console.log("[yt-pdh]", ex); | ||||
|         } | ||||
|     } | ||||
|     setInterval(collect, interval * 1000); | ||||
| } | ||||
|  | ||||
| var scr = document.createElement('script'); | ||||
| scr.textContent = '(' + main.toString() + ')();'; | ||||
| (document.head || document.getElementsByTagName('head')[0]).appendChild(scr); | ||||
| console.log('[yt-pdh] a'); | ||||
							
								
								
									
										8
									
								
								bin/mtag/sleep.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								bin/mtag/sleep.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import time | ||||
| import random | ||||
|  | ||||
| v = random.random() * 6 | ||||
| time.sleep(v) | ||||
| print(f"{v:.2f}") | ||||
							
								
								
									
										198
									
								
								bin/mtag/yt-ipr.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										198
									
								
								bin/mtag/yt-ipr.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,198 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import gzip | ||||
| import json | ||||
| import base64 | ||||
| import string | ||||
| import urllib.request | ||||
| from datetime import datetime | ||||
|  | ||||
| """ | ||||
| youtube initial player response | ||||
|  | ||||
| it's probably best to use this through a config file; see res/yt-ipr.conf | ||||
|  | ||||
| but if you want to use plain arguments instead then: | ||||
|   -v srv/ytm:ytm:w:rw,ed | ||||
|        :c,e2ts:c,e2dsa | ||||
|        :c,sz=16k-1m:c,maxn=10,300:c,rotf=%Y-%m/%d-%H | ||||
|        :c,mtp=yt-id,yt-title,yt-author,yt-channel,yt-views,yt-private,yt-manifest,yt-expires=bin/mtag/yt-ipr.py | ||||
|        :c,mte=yt-id,yt-title,yt-author,yt-channel,yt-views,yt-private,yt-manifest,yt-expires | ||||
|  | ||||
| see res/yt-ipr.user.js for the example userscript to go with this | ||||
| """ | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     try: | ||||
|         with gzip.open(sys.argv[1], "rt", encoding="utf-8", errors="replace") as f: | ||||
|             txt = f.read() | ||||
|     except: | ||||
|         with open(sys.argv[1], "r", encoding="utf-8", errors="replace") as f: | ||||
|             txt = f.read() | ||||
|  | ||||
|     txt = "{" + txt.split("{", 1)[1] | ||||
|  | ||||
|     try: | ||||
|         pd = json.loads(txt) | ||||
|     except json.decoder.JSONDecodeError as ex: | ||||
|         pd = json.loads(txt[: ex.pos]) | ||||
|  | ||||
|     # print(json.dumps(pd, indent=2)) | ||||
|  | ||||
|     if "videoDetails" in pd: | ||||
|         parse_youtube(pd) | ||||
|     else: | ||||
|         parse_freg(pd) | ||||
|  | ||||
|  | ||||
| def get_expiration(url): | ||||
|     et = re.search(r"[?&]expire=([0-9]+)", url).group(1) | ||||
|     et = datetime.utcfromtimestamp(int(et)) | ||||
|     return et.strftime("%Y-%m-%d, %H:%M") | ||||
|  | ||||
|  | ||||
| def parse_youtube(pd): | ||||
|     vd = pd["videoDetails"] | ||||
|     sd = pd["streamingData"] | ||||
|  | ||||
|     et = sd["adaptiveFormats"][0]["url"] | ||||
|     et = get_expiration(et) | ||||
|  | ||||
|     mf = [] | ||||
|     if "dashManifestUrl" in sd: | ||||
|         mf.append("dash") | ||||
|     if "hlsManifestUrl" in sd: | ||||
|         mf.append("hls") | ||||
|  | ||||
|     r = { | ||||
|         "yt-id": vd["videoId"], | ||||
|         "yt-title": vd["title"], | ||||
|         "yt-author": vd["author"], | ||||
|         "yt-channel": vd["channelId"], | ||||
|         "yt-views": vd["viewCount"], | ||||
|         "yt-private": vd["isPrivate"], | ||||
|         # "yt-expires": sd["expiresInSeconds"], | ||||
|         "yt-manifest": ",".join(mf), | ||||
|         "yt-expires": et, | ||||
|     } | ||||
|     print(json.dumps(r)) | ||||
|  | ||||
|     freg_conv(pd) | ||||
|  | ||||
|  | ||||
| def parse_freg(pd): | ||||
|     md = pd["metadata"] | ||||
|     r = { | ||||
|         "yt-id": md["id"], | ||||
|         "yt-title": md["title"], | ||||
|         "yt-author": md["channelName"], | ||||
|         "yt-channel": md["channelURL"].strip("/").split("/")[-1], | ||||
|         "yt-expires": get_expiration(list(pd["video"].values())[0]), | ||||
|     } | ||||
|     print(json.dumps(r)) | ||||
|  | ||||
|  | ||||
| def freg_conv(pd): | ||||
|     # based on getURLs.js v1.5 (2021-08-07) | ||||
|     # fmt: off | ||||
|     priority = { | ||||
|         "video": [ | ||||
|             337, 315, 266, 138,  # 2160p60 | ||||
|             313, 336,  # 2160p | ||||
|             308,  # 1440p60 | ||||
|             271, 264,  # 1440p | ||||
|             335, 303, 299,  # 1080p60 | ||||
|             248, 169, 137,  # 1080p | ||||
|             334, 302, 298,  # 720p60 | ||||
|             247, 136  # 720p | ||||
|         ], | ||||
|         "audio": [ | ||||
|             251, 141, 171, 140, 250, 249, 139 | ||||
|         ] | ||||
|     } | ||||
|  | ||||
|     vid_id = pd["videoDetails"]["videoId"] | ||||
|     chan_id = pd["videoDetails"]["channelId"] | ||||
|  | ||||
|     try: | ||||
|         thumb_url = pd["microformat"]["playerMicroformatRenderer"]["thumbnail"]["thumbnails"][0]["url"] | ||||
|         start_ts = pd["microformat"]["playerMicroformatRenderer"]["liveBroadcastDetails"]["startTimestamp"] | ||||
|     except: | ||||
|         thumb_url = f"https://img.youtube.com/vi/{vid_id}/maxresdefault.jpg" | ||||
|         start_ts = "" | ||||
|  | ||||
|     # fmt: on | ||||
|  | ||||
|     metadata = { | ||||
|         "title": pd["videoDetails"]["title"], | ||||
|         "id": vid_id, | ||||
|         "channelName": pd["videoDetails"]["author"], | ||||
|         "channelURL": "https://www.youtube.com/channel/" + chan_id, | ||||
|         "description": pd["videoDetails"]["shortDescription"], | ||||
|         "thumbnailUrl": thumb_url, | ||||
|         "startTimestamp": start_ts, | ||||
|     } | ||||
|  | ||||
|     if [x for x in vid_id if x not in string.ascii_letters + string.digits + "_-"]: | ||||
|         print(f"malicious json", file=sys.stderr) | ||||
|         return | ||||
|  | ||||
|     basepath = os.path.dirname(sys.argv[1]) | ||||
|  | ||||
|     thumb_fn = f"{basepath}/{vid_id}.jpg" | ||||
|     tmp_fn = f"{thumb_fn}.{os.getpid()}" | ||||
|     if not os.path.exists(thumb_fn) and ( | ||||
|         thumb_url.startswith("https://img.youtube.com/vi/") | ||||
|         or thumb_url.startswith("https://i.ytimg.com/vi/") | ||||
|     ): | ||||
|         try: | ||||
|             with urllib.request.urlopen(thumb_url) as fi: | ||||
|                 with open(tmp_fn, "wb") as fo: | ||||
|                     fo.write(fi.read()) | ||||
|  | ||||
|             os.rename(tmp_fn, thumb_fn) | ||||
|         except: | ||||
|             if os.path.exists(tmp_fn): | ||||
|                 os.unlink(tmp_fn) | ||||
|  | ||||
|     try: | ||||
|         with open(thumb_fn, "rb") as f: | ||||
|             thumb = base64.b64encode(f.read()).decode("ascii") | ||||
|     except: | ||||
|         thumb = "/9j/4AAQSkZJRgABAQEASABIAAD/2wBDAAMCAgICAgMCAgIDAwMDBAYEBAQEBAgGBgUGCQgKCgkICQkKDA8MCgsOCwkJDRENDg8QEBEQCgwSExIQEw8QEBD/yQALCAABAAEBAREA/8wABgAQEAX/2gAIAQEAAD8A0s8g/9k=" | ||||
|  | ||||
|     metadata["thumbnail"] = "data:image/jpeg;base64," + thumb | ||||
|  | ||||
|     ret = { | ||||
|         "metadata": metadata, | ||||
|         "version": "1.5", | ||||
|         "createTime": datetime.utcnow().strftime("%Y-%m-%dT%H:%M:%SZ"), | ||||
|     } | ||||
|  | ||||
|     for stream, itags in priority.items(): | ||||
|         for itag in itags: | ||||
|             url = None | ||||
|             for afmt in pd["streamingData"]["adaptiveFormats"]: | ||||
|                 if itag == afmt["itag"]: | ||||
|                     url = afmt["url"] | ||||
|                     break | ||||
|  | ||||
|             if url: | ||||
|                 ret[stream] = {itag: url} | ||||
|                 break | ||||
|  | ||||
|     fn = f"{basepath}/{vid_id}.urls.json" | ||||
|     with open(fn, "w", encoding="utf-8", errors="replace") as f: | ||||
|         f.write(json.dumps(ret, indent=4)) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     try: | ||||
|         main() | ||||
|     except: | ||||
|         # raise | ||||
|         pass | ||||
							
								
								
									
										99
									
								
								bin/prisonparty.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								bin/prisonparty.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,99 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
| # runs copyparty (or any other program really) in a chroot | ||||
| # | ||||
| # assumption: these directories, and everything within, are owned by root | ||||
| sysdirs=( /bin /lib /lib32 /lib64 /sbin /usr ) | ||||
|  | ||||
|  | ||||
| # error-handler | ||||
| help() { cat <<'EOF' | ||||
|  | ||||
| usage: | ||||
|   ./prisonparty.sh <ROOTDIR> <UID> <GID> [VOLDIR [VOLDIR...]] -- copyparty-sfx.py [...]" | ||||
|  | ||||
| example: | ||||
|   ./prisonparty.sh /var/lib/copyparty-jail 1000 1000 /mnt/nas/music -- copyparty-sfx.py -v /mnt/nas/music::rwmd" | ||||
|  | ||||
| EOF | ||||
| exit 1 | ||||
| } | ||||
|  | ||||
|  | ||||
| # read arguments | ||||
| trap help EXIT | ||||
| jail="$(realpath "$1")"; shift | ||||
| uid="$1"; shift | ||||
| gid="$1"; shift | ||||
|  | ||||
| vols=() | ||||
| while true; do | ||||
| 	v="$1"; shift | ||||
| 	[ "$v" = -- ] && break  # end of volumes | ||||
| 	[ "$#" -eq 0 ] && break  # invalid usage | ||||
| 	vols+=( "$(realpath "$v")" ) | ||||
| done | ||||
| pybin="$1"; shift | ||||
| pybin="$(realpath "$pybin")" | ||||
| cpp="$1"; shift | ||||
| cpp="$(realpath "$cpp")" | ||||
| cppdir="$(dirname "$cpp")" | ||||
| trap - EXIT | ||||
|  | ||||
|  | ||||
| # debug/vis | ||||
| echo | ||||
| echo "chroot-dir = $jail" | ||||
| echo "user:group = $uid:$gid" | ||||
| echo " copyparty = $cpp" | ||||
| echo | ||||
| printf '\033[33m%s\033[0m\n' "copyparty can access these folders and all their subdirectories:" | ||||
| for v in "${vols[@]}"; do | ||||
| 	printf '\033[36m ├─\033[0m %s \033[36m ── added by (You)\033[0m\n' "$v" | ||||
| done | ||||
| printf '\033[36m ├─\033[0m %s \033[36m ── where the copyparty binary is\033[0m\n' "$cppdir" | ||||
| printf '\033[36m ╰─\033[0m %s \033[36m ── the folder you are currently in\033[0m\n' "$PWD" | ||||
| vols+=("$cppdir" "$PWD") | ||||
| echo | ||||
|  | ||||
|  | ||||
| # remove any trailing slashes | ||||
| jail="${jail%/}" | ||||
| cppdir="${cppdir%/}" | ||||
|  | ||||
|  | ||||
| # bind-mount system directories and volumes | ||||
| printf '%s\n' "${sysdirs[@]}" "${vols[@]}" | LC_ALL=C sort | | ||||
| while IFS= read -r v; do | ||||
| 	[ -e "$v" ] || { | ||||
| 		# printf '\033[1;31mfolder does not exist:\033[0m %s\n' "/$v" | ||||
| 		continue | ||||
| 	} | ||||
| 	i1=$(stat -c%D.%i "$v"      2>/dev/null || echo a) | ||||
| 	i2=$(stat -c%D.%i "$jail$v" 2>/dev/null || echo b) | ||||
| 	[ $i1 = $i2 ] && continue | ||||
| 	 | ||||
| 	mkdir -p "$jail$v" | ||||
| 	mount --bind "$v" "$jail$v" | ||||
| done | ||||
|  | ||||
|  | ||||
| # create a tmp | ||||
| mkdir -p "$jail/tmp" | ||||
| chmod 777 "$jail/tmp" | ||||
|  | ||||
|  | ||||
| # run copyparty | ||||
| /sbin/chroot --userspec=$uid:$gid "$jail" "$pybin" "$cpp" "$@" && rv=0 || rv=$? | ||||
|  | ||||
|  | ||||
| # cleanup if not in use | ||||
| lsof "$jail" | grep -qF "$jail" && | ||||
| 	echo "chroot is in use, will not cleanup" || | ||||
| { | ||||
| 	mount | grep -qF " on $jail" | | ||||
| 	awk '{sub(/ type .*/,"");sub(/.* on /,"");print}' | | ||||
| 	LC_ALL=C sort -r  | tee /dev/stderr | tr '\n' '\0' | xargs -r0 umount | ||||
| } | ||||
| exit $rv | ||||
| @@ -118,7 +118,7 @@ printf ']}' >> /dev/shm/$salt.hs | ||||
|  | ||||
| printf '\033[36m' | ||||
|  | ||||
| #curl "http://$target:1234$posturl/handshake.php" -H "Content-Type: text/plain;charset=UTF-8" -H "Cookie: cppwd=$passwd" --data "$(cat "/dev/shm/$salt.hs")" | tee /dev/shm/$salt.res | ||||
| #curl "http://$target:3923$posturl/handshake.php" -H "Content-Type: text/plain;charset=UTF-8" -H "Cookie: cppwd=$passwd" --data "$(cat "/dev/shm/$salt.hs")" | tee /dev/shm/$salt.res | ||||
|  | ||||
| { | ||||
|     { | ||||
| @@ -135,7 +135,7 @@ EOF | ||||
|     cat /dev/shm/$salt.hs | ||||
| } | | ||||
| tee /dev/shm/$salt.hsb | | ||||
| ncat $target 1234 | | ||||
| ncat $target 3923 | | ||||
| tee /dev/shm/$salt.hs1r | ||||
|  | ||||
| wark="$(cat /dev/shm/$salt.hs1r | getwark)" | ||||
| @@ -190,7 +190,7 @@ EOF | ||||
|     nchunk=$((nchunk+1)) | ||||
|  | ||||
| done | | ||||
| ncat $target 1234 | | ||||
| ncat $target 3923 | | ||||
| tee /dev/shm/$salt.pr | ||||
|  | ||||
| t=$(date +%s.%N) | ||||
| @@ -201,7 +201,7 @@ t=$(date +%s.%N) | ||||
|  | ||||
| printf '\033[36m' | ||||
|  | ||||
| ncat $target 1234 < /dev/shm/$salt.hsb | | ||||
| ncat $target 3923 < /dev/shm/$salt.hsb | | ||||
| tee /dev/shm/$salt.hs2r | | ||||
| grep -E '"hash": ?\[ *\]' | ||||
|  | ||||
|   | ||||
							
								
								
									
										38
									
								
								contrib/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								contrib/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| ### [`copyparty.bat`](copyparty.bat) | ||||
| * launches copyparty with no arguments (anon read+write within same folder) | ||||
| * intended for windows machines with no python.exe in PATH | ||||
| * works on windows, linux and macos | ||||
| * assumes `copyparty-sfx.py` was renamed to `copyparty.py` in the same folder as `copyparty.bat` | ||||
|  | ||||
| ### [`index.html`](index.html) | ||||
| * drop-in redirect from an httpd to copyparty | ||||
| * assumes the webserver and copyparty is running on the same server/IP | ||||
| * modify `10.13.1.1` as necessary if you wish to support browsers without javascript | ||||
|  | ||||
| ### [`sharex.sxcu`](sharex.sxcu) | ||||
| * sharex config file to upload screenshots and grab the URL | ||||
| * `RequestURL`: full URL to the target folder | ||||
| * `pw`: password (remove the `pw` line if anon-write) | ||||
|  | ||||
| however if your copyparty is behind a reverse-proxy, you may want to use [`sharex-html.sxcu`](sharex-html.sxcu) instead: | ||||
| * `RequestURL`: full URL to the target folder | ||||
| * `URL`: full URL to the root folder (with trailing slash) followed by `$regex:1|1$` | ||||
| * `pw`: password (remove `Parameters` if anon-write) | ||||
|  | ||||
| ### [`explorer-nothumbs-nofoldertypes.reg`](explorer-nothumbs-nofoldertypes.reg) | ||||
| * disables thumbnails and folder-type detection in windows explorer | ||||
| * makes it way faster (especially for slow/networked locations (such as copyparty-fuse)) | ||||
|  | ||||
| ### [`cfssl.sh`](cfssl.sh) | ||||
| * creates CA and server certificates using cfssl | ||||
| * give a 3rd argument to install it to your copyparty config | ||||
|  | ||||
| # OS integration | ||||
| init-scripts to start copyparty as a service | ||||
| * [`systemd/copyparty.service`](systemd/copyparty.service) runs the sfx normally | ||||
| * [`systemd/prisonparty.service`](systemd/prisonparty.service) runs the sfx in a chroot | ||||
| * [`openrc/copyparty`](openrc/copyparty) | ||||
|  | ||||
| # Reverse-proxy | ||||
| copyparty has basic support for running behind another webserver | ||||
| * [`nginx/copyparty.conf`](nginx/copyparty.conf) | ||||
							
								
								
									
										72
									
								
								contrib/cfssl.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										72
									
								
								contrib/cfssl.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,72 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
| # ca-name and server-name | ||||
| ca_name="$1" | ||||
| srv_name="$2" | ||||
|  | ||||
| [ -z "$srv_name" ] && { | ||||
| 	echo "need arg 1: ca name" | ||||
| 	echo "need arg 2: server name" | ||||
| 	exit 1 | ||||
| } | ||||
|  | ||||
|  | ||||
| gen_ca() { | ||||
| 	(tee /dev/stderr <<EOF | ||||
| {"CN": "$ca_name ca", | ||||
| "CA": {"expiry":"87600h", "pathlen":0}, | ||||
| "key": {"algo":"rsa", "size":4096}, | ||||
| "names": [{"O":"$ca_name ca"}]} | ||||
| EOF | ||||
| 	)| | ||||
| 	cfssl gencert -initca - | | ||||
| 	cfssljson -bare ca | ||||
| 	 | ||||
| 	mv ca-key.pem ca.key | ||||
| 	rm ca.csr | ||||
| } | ||||
|  | ||||
|  | ||||
| gen_srv() { | ||||
| 	(tee /dev/stderr <<EOF | ||||
| {"key": {"algo":"rsa", "size":4096}, | ||||
| "names": [{"O":"$ca_name - $srv_name"}]} | ||||
| EOF | ||||
| 	)| | ||||
| 	cfssl gencert -ca ca.pem -ca-key ca.key \ | ||||
| 		-profile=www -hostname="$srv_name.$ca_name" - | | ||||
| 	cfssljson -bare "$srv_name" | ||||
|  | ||||
| 	mv "$srv_name-key.pem" "$srv_name.key" | ||||
| 	rm "$srv_name.csr" | ||||
| } | ||||
|  | ||||
|  | ||||
| # create ca if not exist | ||||
| [ -e ca.key ] || | ||||
| 	gen_ca | ||||
|  | ||||
| # always create server cert | ||||
| gen_srv | ||||
|  | ||||
|  | ||||
| # dump cert info | ||||
| show() { | ||||
| 	openssl x509 -text -noout -in $1 | | ||||
| 	awk '!o; {o=0} /[0-9a-f:]{16}/{o=1}' | ||||
| } | ||||
| show ca.pem | ||||
| show "$srv_name.pem" | ||||
|  | ||||
|  | ||||
| # write cert into copyparty config | ||||
| [ -z "$3" ] || { | ||||
| 	mkdir -p ~/.config/copyparty | ||||
| 	cat "$srv_name".{key,pem} ca.pem >~/.config/copyparty/cert.pem  | ||||
| } | ||||
|  | ||||
|  | ||||
| # rm *.key *.pem | ||||
| # cfssl print-defaults config | ||||
| # cfssl print-defaults csr | ||||
							
								
								
									
										33
									
								
								contrib/copyparty.bat
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								contrib/copyparty.bat
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| exec python "$(dirname "$0")"/copyparty.py | ||||
|  | ||||
| @rem on linux, the above will execute and the script will terminate | ||||
| @rem on windows, the rest of this script will run | ||||
|  | ||||
| @echo off | ||||
| cls | ||||
|  | ||||
| set py= | ||||
| for /f %%i in ('where python 2^>nul') do ( | ||||
|     set "py=%%i" | ||||
|     goto c1 | ||||
| ) | ||||
| :c1 | ||||
|  | ||||
| if [%py%] == [] ( | ||||
|     for /f %%i in ('where /r "%localappdata%\programs\python" python 2^>nul') do ( | ||||
|         set "py=%%i" | ||||
|         goto c2 | ||||
|     ) | ||||
| ) | ||||
| :c2 | ||||
|  | ||||
| if [%py%] == [] set "py=c:\python27\python.exe" | ||||
|  | ||||
| if not exist "%py%" ( | ||||
|     echo could not find python | ||||
|     echo( | ||||
|     pause | ||||
|     exit /b | ||||
| ) | ||||
|  | ||||
| start cmd /c %py% "%~dp0\copyparty.py" | ||||
							
								
								
									
										31
									
								
								contrib/explorer-nothumbs-nofoldertypes.reg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								contrib/explorer-nothumbs-nofoldertypes.reg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| Windows Registry Editor Version 5.00 | ||||
|  | ||||
| ; this will do 3 things, all optional: | ||||
| ;  1) disable thumbnails | ||||
| ;  2) delete all existing folder type settings/detections | ||||
| ;  3) disable folder type detection (force default columns) | ||||
| ; | ||||
| ; this makes the file explorer way faster, | ||||
| ; especially on slow/networked locations | ||||
|  | ||||
|  | ||||
| ; ===================================================================== | ||||
| ; 1) disable thumbnails | ||||
|  | ||||
| [HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced] | ||||
| "IconsOnly"=dword:00000001 | ||||
|  | ||||
|  | ||||
| ; ===================================================================== | ||||
| ; 2) delete all existing folder type settings/detections | ||||
|  | ||||
| [-HKEY_CURRENT_USER\Software\Classes\Local Settings\Software\Microsoft\Windows\Shell\Bags] | ||||
|  | ||||
| [-HKEY_CURRENT_USER\Software\Classes\Local Settings\Software\Microsoft\Windows\Shell\BagMRU] | ||||
|  | ||||
|  | ||||
| ; ===================================================================== | ||||
| ; 3) disable folder type detection | ||||
|  | ||||
| [HKEY_CURRENT_USER\Software\Classes\Local Settings\Software\Microsoft\Windows\Shell\Bags\AllFolders\Shell] | ||||
| "FolderType"="NotSpecified" | ||||
							
								
								
									
										43
									
								
								contrib/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								contrib/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|  | ||||
| <head> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>⇆🎉 redirect</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<style> | ||||
|  | ||||
| html, body { | ||||
| 	font-family: sans-serif; | ||||
| } | ||||
| body { | ||||
| 	padding: 1em 2em; | ||||
| 	font-size: 1.5em; | ||||
| } | ||||
| a { | ||||
| 	font-size: 1.2em; | ||||
| 	padding: .1em; | ||||
| } | ||||
|  | ||||
| </style> | ||||
| </head> | ||||
| <body> | ||||
| 	<span id="desc">you probably want</span> <a id="redir" href="//10.13.1.1:3923/">copyparty</a> | ||||
| 	<script> | ||||
|  | ||||
| var a = document.getElementById('redir'), | ||||
| 	proto = window.location.protocol.indexOf('https') === 0 ? 'https' : 'http', | ||||
| 	loc = window.location.hostname || '127.0.0.1', | ||||
| 	port = a.getAttribute('href').split(':').pop().split('/')[0], | ||||
| 	url = proto + '://' + loc + ':' + port + '/'; | ||||
|  | ||||
| a.setAttribute('href', url); | ||||
| document.getElementById('desc').innerHTML = 'redirecting to'; | ||||
|  | ||||
| setTimeout(function() { | ||||
| 	window.location.href = url; | ||||
| }, 500); | ||||
|  | ||||
| </script> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										39
									
								
								contrib/nginx/copyparty.conf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								contrib/nginx/copyparty.conf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| # when running copyparty behind a reverse proxy, | ||||
| # the following arguments are recommended: | ||||
| # | ||||
| #   -nc 512         important, see next paragraph | ||||
| #   --http-only     lower latency on initial connection | ||||
| #   -i 127.0.0.1    only accept connections from nginx | ||||
| # | ||||
| # -nc must match or exceed the webserver's max number of concurrent clients; | ||||
| # nginx default is 512  (worker_processes 1, worker_connections 512) | ||||
| # | ||||
| # you may also consider adding -j0 for CPU-intensive configurations | ||||
| # (not that i can really think of any good examples) | ||||
|  | ||||
| upstream cpp { | ||||
| 	server 127.0.0.1:3923; | ||||
| 	keepalive 120; | ||||
| } | ||||
| server { | ||||
| 	listen 443 ssl; | ||||
| 	listen [::]:443 ssl; | ||||
|  | ||||
| 	server_name fs.example.com; | ||||
| 	 | ||||
| 	location / { | ||||
| 		proxy_pass http://cpp; | ||||
| 		proxy_redirect off; | ||||
| 		# disable buffering (next 4 lines) | ||||
| 		proxy_http_version 1.1; | ||||
| 		client_max_body_size 0; | ||||
| 		proxy_buffering off; | ||||
| 		proxy_request_buffering off; | ||||
|  | ||||
| 		proxy_set_header   Host              $host; | ||||
| 		proxy_set_header   X-Real-IP         $remote_addr; | ||||
| 		proxy_set_header   X-Forwarded-For   $proxy_add_x_forwarded_for; | ||||
| 		proxy_set_header   X-Forwarded-Proto $scheme; | ||||
| 		proxy_set_header   Connection        "Keep-Alive"; | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										18
									
								
								contrib/openrc/copyparty
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								contrib/openrc/copyparty
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| #!/sbin/openrc-run | ||||
|  | ||||
| # this will start `/usr/local/bin/copyparty-sfx.py` | ||||
| # and share '/mnt' with anonymous read+write | ||||
| # | ||||
| # installation: | ||||
| #   cp -pv copyparty /etc/init.d && rc-update add copyparty | ||||
| # | ||||
| # you may want to: | ||||
| #   change '/usr/bin/python' to another interpreter | ||||
| #   change '/mnt::rw' to another location or permission-set | ||||
|  | ||||
| name="$SVCNAME" | ||||
| command_background=true | ||||
| pidfile="/var/run/$SVCNAME.pid" | ||||
|  | ||||
| command="/usr/bin/python /usr/local/bin/copyparty-sfx.py" | ||||
| command_args="-q -v /mnt::rw" | ||||
							
								
								
									
										19
									
								
								contrib/sharex-html.sxcu
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								contrib/sharex-html.sxcu
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| { | ||||
|   "Version": "13.5.0", | ||||
|   "Name": "copyparty-html", | ||||
|   "DestinationType": "ImageUploader", | ||||
|   "RequestMethod": "POST", | ||||
|   "RequestURL": "http://127.0.0.1:3923/sharex", | ||||
|   "Parameters": { | ||||
|     "pw": "wark" | ||||
|   }, | ||||
|   "Body": "MultipartFormData", | ||||
|   "Arguments": { | ||||
|     "act": "bput" | ||||
|   }, | ||||
|   "FileFormName": "f", | ||||
|   "RegexList": [ | ||||
|     "bytes // <a href=\"/([^\"]+)\"" | ||||
|   ], | ||||
|   "URL": "http://127.0.0.1:3923/$regex:1|1$" | ||||
| } | ||||
							
								
								
									
										17
									
								
								contrib/sharex.sxcu
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								contrib/sharex.sxcu
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| { | ||||
|   "Version": "13.5.0", | ||||
|   "Name": "copyparty", | ||||
|   "DestinationType": "ImageUploader", | ||||
|   "RequestMethod": "POST", | ||||
|   "RequestURL": "http://127.0.0.1:3923/sharex", | ||||
|   "Parameters": { | ||||
|     "pw": "wark", | ||||
|     "j": null | ||||
|   }, | ||||
|   "Body": "MultipartFormData", | ||||
|   "Arguments": { | ||||
|     "act": "bput" | ||||
|   }, | ||||
|   "FileFormName": "f", | ||||
|   "URL": "$json:files[0].url$" | ||||
| } | ||||
							
								
								
									
										34
									
								
								contrib/systemd/copyparty.service
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								contrib/systemd/copyparty.service
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| # this will start `/usr/local/bin/copyparty-sfx.py` | ||||
| # and share '/mnt' with anonymous read+write | ||||
| # | ||||
| # installation: | ||||
| #   cp -pv copyparty.service /etc/systemd/system && systemctl enable --now copyparty | ||||
| # | ||||
| # you may want to: | ||||
| #   change '/usr/bin/python' to another interpreter | ||||
| #   change '/mnt::rw' to another location or permission-set | ||||
| # | ||||
| # with `Type=notify`, copyparty will signal systemd when it is ready to | ||||
| #   accept connections; correctly delaying units depending on copyparty. | ||||
| #   But note that journalctl will get the timestamps wrong due to | ||||
| #   python disabling line-buffering, so messages are out-of-order: | ||||
| #   https://user-images.githubusercontent.com/241032/126040249-cb535cc7-c599-4931-a796-a5d9af691bad.png | ||||
| # | ||||
| # enable line-buffering for realtime logging (slight performance cost): | ||||
| #   modify ExecStart and prefix it with `/usr/bin/stdbuf -oL` like so: | ||||
| #   ExecStart=/usr/bin/stdbuf -oL /usr/bin/python3 [...] | ||||
| # but some systemd versions require this instead (higher performance cost): | ||||
| #   inside the [Service] block, add the following line: | ||||
| #   Environment=PYTHONUNBUFFERED=x | ||||
|  | ||||
| [Unit] | ||||
| Description=copyparty file server | ||||
|  | ||||
| [Service] | ||||
| Type=notify | ||||
| SyslogIdentifier=copyparty | ||||
| ExecStart=/usr/bin/python3 /usr/local/bin/copyparty-sfx.py -q -v /mnt::rw | ||||
| ExecStartPre=/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf' | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
							
								
								
									
										27
									
								
								contrib/systemd/prisonparty.service
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								contrib/systemd/prisonparty.service
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| # this will start `/usr/local/bin/copyparty-sfx.py` | ||||
| # in a chroot, preventing accidental access elsewhere | ||||
| # and share '/mnt' with anonymous read+write | ||||
| # | ||||
| # installation: | ||||
| #   1) put copyparty-sfx.py and prisonparty.sh in /usr/local/bin | ||||
| #   2) cp -pv prisonparty.service /etc/systemd/system && systemctl enable --now prisonparty | ||||
| # | ||||
| # you may want to: | ||||
| #   change '/mnt::rw' to another location or permission-set | ||||
| #    (remember to change the '/mnt' chroot arg too) | ||||
| # | ||||
| # enable line-buffering for realtime logging (slight performance cost): | ||||
| #   inside the [Service] block, add the following line: | ||||
| #   Environment=PYTHONUNBUFFERED=x | ||||
|  | ||||
| [Unit] | ||||
| Description=copyparty file server | ||||
|  | ||||
| [Service] | ||||
| SyslogIdentifier=prisonparty | ||||
| WorkingDirectory=/usr/local/bin | ||||
| 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 | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
| @@ -2,12 +2,16 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import platform | ||||
| import time | ||||
| import sys | ||||
| import os | ||||
|  | ||||
| PY2 = sys.version_info[0] == 2 | ||||
| if PY2: | ||||
|     sys.dont_write_bytecode = True | ||||
|     unicode = unicode | ||||
| else: | ||||
|     unicode = str | ||||
|  | ||||
| WINDOWS = False | ||||
| if platform.system() == "Windows": | ||||
| @@ -16,21 +20,46 @@ if platform.system() == "Windows": | ||||
| VT100 = not WINDOWS or WINDOWS >= [10, 0, 14393] | ||||
| # introduced in anniversary update | ||||
|  | ||||
| ANYWIN = WINDOWS or sys.platform in ["msys"] | ||||
|  | ||||
| MACOS = platform.system() == "Darwin" | ||||
|  | ||||
|  | ||||
| def get_unix_home(): | ||||
|     try: | ||||
|         v = os.environ["XDG_CONFIG_HOME"] | ||||
|         if not v: | ||||
|             raise Exception() | ||||
|         ret = os.path.normpath(v) | ||||
|         os.listdir(ret) | ||||
|         return ret | ||||
|     except: | ||||
|         pass | ||||
|  | ||||
|     try: | ||||
|         v = os.path.expanduser("~/.config") | ||||
|         if v.startswith("~"): | ||||
|             raise Exception() | ||||
|         ret = os.path.normpath(v) | ||||
|         os.listdir(ret) | ||||
|         return ret | ||||
|     except: | ||||
|         return "/tmp" | ||||
|  | ||||
|  | ||||
| class EnvParams(object): | ||||
|     def __init__(self): | ||||
|         self.t0 = time.time() | ||||
|         self.mod = os.path.dirname(os.path.realpath(__file__)) | ||||
|         if self.mod.endswith("__init__"): | ||||
|             self.mod = os.path.dirname(self.mod) | ||||
|  | ||||
|         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 = os.path.normpath( | ||||
|                 os.getenv("XDG_CONFIG_HOME", os.path.expanduser("~/.config")) | ||||
|                 + "/copyparty" | ||||
|             ) | ||||
|             self.cfg = get_unix_home() + "/copyparty" | ||||
|  | ||||
|         self.cfg = self.cfg.replace("\\", "/") | ||||
|         try: | ||||
|   | ||||
| @@ -8,17 +8,31 @@ __copyright__ = 2019 | ||||
| __license__ = "MIT" | ||||
| __url__ = "https://github.com/9001/copyparty/" | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import shutil | ||||
| import filecmp | ||||
| import locale | ||||
| import argparse | ||||
| import threading | ||||
| import traceback | ||||
| from textwrap import dedent | ||||
|  | ||||
| from .__init__ import E, WINDOWS, VT100 | ||||
| from .__init__ import E, WINDOWS, VT100, PY2, unicode | ||||
| from .__version__ import S_VERSION, S_BUILD_DT, CODENAME | ||||
| from .svchub import SvcHub | ||||
| from .util import py_desc | ||||
| from .util import py_desc, align_tab, IMPLICATIONS, ansi_re | ||||
| from .authsrv import re_vol | ||||
|  | ||||
| HAVE_SSL = True | ||||
| try: | ||||
|     import ssl | ||||
| except: | ||||
|     HAVE_SSL = False | ||||
|  | ||||
| printed = "" | ||||
|  | ||||
|  | ||||
| class RiceFormatter(argparse.HelpFormatter): | ||||
| @@ -44,6 +58,27 @@ class RiceFormatter(argparse.HelpFormatter): | ||||
|         return "".join(indent + line + "\n" for line in text.splitlines()) | ||||
|  | ||||
|  | ||||
| class Dodge11874(RiceFormatter): | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         kwargs["width"] = 9003 | ||||
|         super(Dodge11874, self).__init__(*args, **kwargs) | ||||
|  | ||||
|  | ||||
| def lprint(*a, **ka): | ||||
|     global printed | ||||
|  | ||||
|     txt = " ".join(unicode(x) for x in a) + ka.get("end", "\n") | ||||
|     printed += txt | ||||
|     if not VT100: | ||||
|         txt = ansi_re.sub("", txt) | ||||
|  | ||||
|     print(txt, **ka) | ||||
|  | ||||
|  | ||||
| def warn(msg): | ||||
|     lprint("\033[1mwarning:\033[0;33m {}\033[0m\n".format(msg)) | ||||
|  | ||||
|  | ||||
| def ensure_locale(): | ||||
|     for x in [ | ||||
|         "en_US.UTF-8", | ||||
| @@ -52,7 +87,7 @@ def ensure_locale(): | ||||
|     ]: | ||||
|         try: | ||||
|             locale.setlocale(locale.LC_ALL, x) | ||||
|             print("Locale:", x) | ||||
|             lprint("Locale:", x) | ||||
|             break | ||||
|         except: | ||||
|             continue | ||||
| @@ -73,7 +108,7 @@ def ensure_cert(): | ||||
|  | ||||
|     try: | ||||
|         if filecmp.cmp(cert_cfg, cert_insec): | ||||
|             print( | ||||
|             lprint( | ||||
|                 "\033[33m  using default TLS certificate; https will be insecure." | ||||
|                 + "\033[36m\n  certificate location: {}\033[0m\n".format(cert_cfg) | ||||
|             ) | ||||
| @@ -84,36 +119,118 @@ def ensure_cert(): | ||||
|     # printf 'NO\n.\n.\n.\n.\ncopyparty-insecure\n.\n' | faketime '2000-01-01 00:00:00' openssl req -x509 -sha256 -newkey rsa:2048 -keyout insecure.pem -out insecure.pem -days $((($(printf %d 0x7fffffff)-$(date +%s --date=2000-01-01T00:00:00Z))/(60*60*24))) -nodes && ls -al insecure.pem && openssl x509 -in insecure.pem -text -noout | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     if WINDOWS: | ||||
|         os.system("")  # enables colors | ||||
| def configure_ssl_ver(al): | ||||
|     def terse_sslver(txt): | ||||
|         txt = txt.lower() | ||||
|         for c in ["_", "v", "."]: | ||||
|             txt = txt.replace(c, "") | ||||
|  | ||||
|     desc = py_desc().replace("[", "\033[1;30m[") | ||||
|         return txt.replace("tls10", "tls1") | ||||
|  | ||||
|     f = '\033[36mcopyparty v{} "\033[35m{}\033[36m" ({})\n{}\033[0m\n' | ||||
|     print(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc)) | ||||
|     # oh man i love openssl | ||||
|     # check this out | ||||
|     # hold my beer | ||||
|     ptn = re.compile(r"^OP_NO_(TLS|SSL)v") | ||||
|     sslver = terse_sslver(al.ssl_ver).split(",") | ||||
|     flags = [k for k in ssl.__dict__ if ptn.match(k)] | ||||
|     # SSLv2 SSLv3 TLSv1 TLSv1_1 TLSv1_2 TLSv1_3 | ||||
|     if "help" in sslver: | ||||
|         avail = [terse_sslver(x[6:]) for x in flags] | ||||
|         avail = " ".join(sorted(avail) + ["all"]) | ||||
|         lprint("\navailable ssl/tls versions:\n  " + avail) | ||||
|         sys.exit(0) | ||||
|  | ||||
|     ensure_locale() | ||||
|     ensure_cert() | ||||
|     al.ssl_flags_en = 0 | ||||
|     al.ssl_flags_de = 0 | ||||
|     for flag in sorted(flags): | ||||
|         ver = terse_sslver(flag[6:]) | ||||
|         num = getattr(ssl, flag) | ||||
|         if ver in sslver: | ||||
|             al.ssl_flags_en |= num | ||||
|         else: | ||||
|             al.ssl_flags_de |= num | ||||
|  | ||||
|     if sslver == ["all"]: | ||||
|         x = al.ssl_flags_en | ||||
|         al.ssl_flags_en = al.ssl_flags_de | ||||
|         al.ssl_flags_de = x | ||||
|  | ||||
|     for k in ["ssl_flags_en", "ssl_flags_de"]: | ||||
|         num = getattr(al, k) | ||||
|         lprint("{}: {:8x} ({})".format(k, num, num)) | ||||
|  | ||||
|     # think i need that beer now | ||||
|  | ||||
|  | ||||
| def configure_ssl_ciphers(al): | ||||
|     ctx = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH) | ||||
|     if al.ssl_ver: | ||||
|         ctx.options &= ~al.ssl_flags_en | ||||
|         ctx.options |= al.ssl_flags_de | ||||
|  | ||||
|     is_help = al.ciphers == "help" | ||||
|  | ||||
|     if al.ciphers and not is_help: | ||||
|         try: | ||||
|             ctx.set_ciphers(al.ciphers) | ||||
|         except: | ||||
|             lprint("\n\033[1;31mfailed to set ciphers\033[0m\n") | ||||
|  | ||||
|     if not hasattr(ctx, "get_ciphers"): | ||||
|         lprint("cannot read cipher list: openssl or python too old") | ||||
|     else: | ||||
|         ciphers = [x["description"] for x in ctx.get_ciphers()] | ||||
|         lprint("\n  ".join(["\nenabled ciphers:"] + align_tab(ciphers) + [""])) | ||||
|  | ||||
|     if is_help: | ||||
|         sys.exit(0) | ||||
|  | ||||
|  | ||||
| def sighandler(sig=None, frame=None): | ||||
|     msg = [""] * 5 | ||||
|     for th in threading.enumerate(): | ||||
|         msg.append(str(th)) | ||||
|         msg.extend(traceback.format_stack(sys._current_frames()[th.ident])) | ||||
|  | ||||
|     msg.append("\n") | ||||
|     print("\n".join(msg)) | ||||
|  | ||||
|  | ||||
| def run_argparse(argv, formatter): | ||||
|     ap = argparse.ArgumentParser( | ||||
|         formatter_class=RiceFormatter, | ||||
|         formatter_class=formatter, | ||||
|         prog="copyparty", | ||||
|         description="http file sharing hub v{} ({})".format(S_VERSION, S_BUILD_DT), | ||||
|         epilog=dedent( | ||||
|             """ | ||||
|     ) | ||||
|  | ||||
|     sects = [ | ||||
|         [ | ||||
|             "accounts", | ||||
|             "accounts and volumes", | ||||
|             dedent( | ||||
|                 """ | ||||
|             -a takes username:password, | ||||
|             -v takes src:dst:permset:permset:... where "permset" is | ||||
|                accesslevel followed by username (no separator) | ||||
|             -v takes src:dst:perm1:perm2:permN:volflag1:volflag2:volflagN:... | ||||
|                where "perm" is "accesslevels,username1,username2,..." | ||||
|                and "volflag" is config flags to set on this volume | ||||
|              | ||||
|             list of accesslevels: | ||||
|               "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 | ||||
|  | ||||
|             too many volflags to list here, see the other sections | ||||
|  | ||||
|             example:\033[35m | ||||
|               -a ed:hunter2 -v .::r:aed -v ../inc:dump:w:aed  \033[36m | ||||
|               -a ed:hunter2 -v .::r:rw,ed -v ../inc:dump:w:rw,ed:c,nodupe  \033[36m | ||||
|               mount current directory at "/" with | ||||
|                * r (read-only) for everyone | ||||
|                * a (read+write) for ed | ||||
|                * rw (read+write) for ed | ||||
|               mount ../inc at "/dump" with | ||||
|                * w (write-only) for everyone | ||||
|                * a (read+write) for ed  \033[0m | ||||
|                * rw (read+write) for ed | ||||
|                * reject duplicate files  \033[0m | ||||
|              | ||||
|             if no accounts or volumes are configured, | ||||
|             current folder will be read/write for everyone | ||||
| @@ -121,25 +238,317 @@ def main(): | ||||
|             consider the config file for more flexible account/volume management, | ||||
|             including dynamic reload at runtime (and being more readable w) | ||||
|             """ | ||||
|         ), | ||||
|     ) | ||||
|     ap.add_argument( | ||||
|         "-c", metavar="PATH", type=str, action="append", help="add config file" | ||||
|     ) | ||||
|     ap.add_argument("-i", metavar="IP", type=str, default="0.0.0.0", help="ip to bind") | ||||
|     ap.add_argument("-p", metavar="PORT", type=int, default=1234, help="port to bind") | ||||
|     ap.add_argument("-nc", metavar="NUM", type=int, default=16, help="max num clients") | ||||
|     ap.add_argument( | ||||
|         "-j", metavar="CORES", type=int, default=1, help="max num cpu cores" | ||||
|     ) | ||||
|     ap.add_argument("-a", metavar="ACCT", type=str, action="append", help="add account") | ||||
|     ap.add_argument("-v", metavar="VOL", type=str, action="append", help="add volume") | ||||
|     ap.add_argument("-q", action="store_true", help="quiet") | ||||
|     ap.add_argument("-ed", action="store_true", help="enable ?dots") | ||||
|     ap.add_argument("-nw", action="store_true", help="disable writes (benchmark)") | ||||
|     al = ap.parse_args() | ||||
|             ), | ||||
|         ], | ||||
|         [ | ||||
|             "flags", | ||||
|             "list of volflags", | ||||
|             dedent( | ||||
|                 """ | ||||
|             volflags are appended to volume definitions, for example, | ||||
|             to create a write-only volume with the \033[33mnodupe\033[0m and \033[32mnosub\033[0m flags: | ||||
|               \033[35m-v /mnt/inc:/inc:w\033[33m:c,nodupe\033[32m:c,nosub | ||||
|  | ||||
|     SvcHub(al).run() | ||||
|             \033[0muploads, general: | ||||
|               \033[36mnodupe\033[35m rejects existing files (instead of symlinking them) | ||||
|               \033[36mnosub\033[35m forces all uploads into the top folder of the vfs | ||||
|               \033[36mgz\033[35m allows server-side gzip of uploads with ?gz (also c,xz) | ||||
|               \033[36mpk\033[35m forces server-side compression, optional arg: xz,9 | ||||
|              | ||||
|             \033[0mupload rules: | ||||
|               \033[36mmaxn=250,600\033[35m max 250 uploads over 15min | ||||
|               \033[36mmaxb=1g,300\033[35m max 1 GiB over 5min (suffixes: b, k, m, g) | ||||
|               \033[36msz=1k-3m\033[35m allow filesizes between 1 KiB and 3MiB | ||||
|              | ||||
|             \033[0mupload rotation: | ||||
|             (moves all uploads into the specified folder structure) | ||||
|               \033[36mrotn=100,3\033[35m 3 levels of subfolders with 100 entries in each | ||||
|               \033[36mrotf=%Y-%m/%d-%H\033[35m date-formatted organizing | ||||
|               \033[36mlifetime=3600\033[35m uploads are deleted after 1 hour | ||||
|              | ||||
|             \033[0mdatabase, general: | ||||
|               \033[36me2d\033[35m sets -e2d (all -e2* args can be set using ce2* volflags) | ||||
|               \033[36md2t\033[35m disables metadata collection, overrides -e2t* | ||||
|               \033[36md2d\033[35m disables all database stuff, overrides -e2* | ||||
|               \033[36mdhash\033[35m disables file hashing on initial scans, also ehash | ||||
|               \033[36mhist=/tmp/cdb\033[35m puts thumbnails and indexes at that location | ||||
|               \033[36mscan=60\033[35m scan for new files every 60sec, same as --re-maxage | ||||
|              | ||||
|             \033[0mdatabase, audio tags: | ||||
|             "mte", "mth", "mtp", "mtm" all work the same as -mte, -mth, ... | ||||
|               \033[36mmtp=.bpm=f,audio-bpm.py\033[35m uses the "audio-bpm.py" program to | ||||
|                 generate ".bpm" tags from uploads (f = overwrite tags) | ||||
|               \033[36mmtp=ahash,vhash=media-hash.py\033[35m collects two tags at once | ||||
|             \033[0m""" | ||||
|             ), | ||||
|         ], | ||||
|         [ | ||||
|             "urlform", | ||||
|             "", | ||||
|             dedent( | ||||
|                 """ | ||||
|             values for --urlform: | ||||
|               \033[36mstash\033[35m dumps the data to file and returns length + checksum | ||||
|               \033[36msave,get\033[35m dumps to file and returns the page like a GET | ||||
|               \033[36mprint,get\033[35m prints the data in the log and returns GET | ||||
|               (leave out the ",get" to return an error instead) | ||||
|             """ | ||||
|             ), | ||||
|         ], | ||||
|         [ | ||||
|             "ls", | ||||
|             "volume inspection", | ||||
|             dedent( | ||||
|                 """ | ||||
|             \033[35m--ls USR,VOL,FLAGS | ||||
|               \033[36mUSR\033[0m is a user to browse as; * is anonymous, ** is all users | ||||
|               \033[36mVOL\033[0m is a single volume to scan, default is * (all vols) | ||||
|               \033[36mFLAG\033[0m is flags; | ||||
|                 \033[36mv\033[0m in addition to realpaths, print usernames and vpaths | ||||
|                 \033[36mln\033[0m only prints symlinks leaving the volume mountpoint | ||||
|                 \033[36mp\033[0m exits 1 if any such symlinks are found | ||||
|                 \033[36mr\033[0m resumes startup after the listing | ||||
|             examples: | ||||
|               --ls '**'          # list all files which are possible to read | ||||
|               --ls '**,*,ln'     # check for dangerous symlinks | ||||
|               --ls '**,*,ln,p,r' # check, then start normally if safe | ||||
|             """ | ||||
|             ), | ||||
|         ], | ||||
|     ] | ||||
|  | ||||
|     # fmt: off | ||||
|     u = unicode | ||||
|     ap2 = ap.add_argument_group('general options') | ||||
|     ap2.add_argument("-c", metavar="PATH", type=u, action="append", help="add config file") | ||||
|     ap2.add_argument("-nc", metavar="NUM", type=int, default=64, help="max num clients") | ||||
|     ap2.add_argument("-j", metavar="CORES", type=int, default=1, help="max num cpu cores") | ||||
|     ap2.add_argument("-a", metavar="ACCT", type=u, action="append", help="add account, USER:PASS; example [ed:wark") | ||||
|     ap2.add_argument("-v", metavar="VOL", type=u, action="append", help="add volume, SRC:DST:FLAG; example [.::r], [/mnt/nas/music:/music:r:aed") | ||||
|     ap2.add_argument("-ed", action="store_true", help="enable ?dots") | ||||
|     ap2.add_argument("-emp", action="store_true", help="enable markdown plugins") | ||||
|     ap2.add_argument("-mcr", metavar="SEC", type=int, default=60, help="md-editor mod-chk rate") | ||||
|     ap2.add_argument("--urlform", metavar="MODE", type=u, default="print,get", help="how to handle url-forms; examples: [stash], [save,get]") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('upload options') | ||||
|     ap2.add_argument("--dotpart", action="store_true", help="dotfile incomplete uploads") | ||||
|     ap2.add_argument("--sparse", metavar="MiB", type=int, default=4, help="up2k min.size threshold (mswin-only)") | ||||
|     ap2.add_argument("--unpost", metavar="SEC", type=int, default=3600*12, help="grace period where uploads can be deleted by the uploader, even without delete permissions; 0=disabled") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('network options') | ||||
|     ap2.add_argument("-i", metavar="IP", type=u, default="0.0.0.0", help="ip to bind (comma-sep.)") | ||||
|     ap2.add_argument("-p", metavar="PORT", type=u, default="3923", help="ports to bind (comma/range)") | ||||
|     ap2.add_argument("--rproxy", metavar="DEPTH", type=int, default=1, help="which ip to keep; 0 = tcp, 1 = origin (first x-fwd), 2 = cloudflare, 3 = nginx, -1 = closest proxy") | ||||
|      | ||||
|     ap2 = ap.add_argument_group('SSL/TLS options') | ||||
|     ap2.add_argument("--http-only", action="store_true", help="disable ssl/tls") | ||||
|     ap2.add_argument("--https-only", action="store_true", help="disable plaintext") | ||||
|     ap2.add_argument("--ssl-ver", metavar="LIST", type=u, help="set allowed ssl/tls versions; [help] shows available versions; default is what your python version considers safe") | ||||
|     ap2.add_argument("--ciphers", metavar="LIST", type=u, help="set allowed ssl/tls ciphers; [help] shows available ciphers") | ||||
|     ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info") | ||||
|     ap2.add_argument("--ssl-log", metavar="PATH", type=u, help="log master secrets") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('opt-outs') | ||||
|     ap2.add_argument("-nw", action="store_true", help="disable writes (benchmark)") | ||||
|     ap2.add_argument("--no-del", action="store_true", help="disable delete operations") | ||||
|     ap2.add_argument("--no-mv", action="store_true", help="disable move/rename operations") | ||||
|     ap2.add_argument("-nih", action="store_true", help="no info hostname") | ||||
|     ap2.add_argument("-nid", action="store_true", help="no info disk-usage") | ||||
|     ap2.add_argument("--no-zip", action="store_true", help="disable download as zip/tar") | ||||
|     ap2.add_argument("--no-lifetime", action="store_true", help="disable automatic deletion of uploads after a certain time (lifetime volflag)") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('safety options') | ||||
|     ap2.add_argument("--ls", metavar="U[,V[,F]]", type=u, help="scan all volumes; arguments USER,VOL,FLAGS; example [**,*,ln,p,r]") | ||||
|     ap2.add_argument("--salt", type=u, default="hunter2", help="up2k file-hash salt") | ||||
|     ap2.add_argument("--no-dot-mv", action="store_true", help="disallow moving dotfiles; makes it impossible to move folders containing dotfiles") | ||||
|     ap2.add_argument("--no-dot-ren", action="store_true", help="disallow renaming dotfiles; makes it impossible to make something a dotfile") | ||||
|     ap2.add_argument("--no-logues", action="store_true", help="disable rendering .prologue/.epilogue.html into directory listings") | ||||
|     ap2.add_argument("--no-readme", action="store_true", help="disable rendering readme.md into directory listings") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('logging options') | ||||
|     ap2.add_argument("-q", action="store_true", help="quiet") | ||||
|     ap2.add_argument("-lo", metavar="PATH", type=u, help="logfile, example: cpp-%%Y-%%m%%d-%%H%%M%%S.txt.xz") | ||||
|     ap2.add_argument("--no-voldump", action="store_true", help="do not list volumes and permissions on startup") | ||||
|     ap2.add_argument("--log-conn", action="store_true", help="print tcp-server msgs") | ||||
|     ap2.add_argument("--log-htp", action="store_true", help="print http-server threadpool scaling") | ||||
|     ap2.add_argument("--ihead", metavar="HEADER", type=u, action='append', help="dump incoming header") | ||||
|     ap2.add_argument("--lf-url", metavar="RE", type=u, default=r"^/\.cpr/|\?th=[wj]$", help="dont log URLs matching") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('admin panel options') | ||||
|     ap2.add_argument("--no-rescan", action="store_true", help="disable ?scan (volume reindexing)") | ||||
|     ap2.add_argument("--no-stack", action="store_true", help="disable ?stack (list all stacks)") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('thumbnail options') | ||||
|     ap2.add_argument("--no-thumb", action="store_true", help="disable all thumbnails") | ||||
|     ap2.add_argument("--no-vthumb", action="store_true", help="disable video thumbnails") | ||||
|     ap2.add_argument("--th-size", metavar="WxH", default="320x256", help="thumbnail res") | ||||
|     ap2.add_argument("--th-mt", metavar="CORES", type=int, default=0, help="max num cpu cores to use, 0=all") | ||||
|     ap2.add_argument("--th-no-crop", action="store_true", help="dynamic height; show full image") | ||||
|     ap2.add_argument("--th-no-jpg", action="store_true", help="disable jpg output") | ||||
|     ap2.add_argument("--th-no-webp", action="store_true", help="disable webp output") | ||||
|     ap2.add_argument("--th-ff-jpg", action="store_true", help="force jpg for video thumbs") | ||||
|     ap2.add_argument("--th-poke", metavar="SEC", type=int, default=300, help="activity labeling cooldown") | ||||
|     ap2.add_argument("--th-clean", metavar="SEC", type=int, default=43200, help="cleanup interval; 0=disabled") | ||||
|     ap2.add_argument("--th-maxage", metavar="SEC", type=int, default=604800, help="max folder age") | ||||
|     ap2.add_argument("--th-covers", metavar="N,N", type=u, default="folder.png,folder.jpg,cover.png,cover.jpg", help="folder thumbnails to stat for") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('general db options') | ||||
|     ap2.add_argument("-e2d", action="store_true", help="enable up2k database") | ||||
|     ap2.add_argument("-e2ds", action="store_true", help="enable up2k db-scanner, sets -e2d") | ||||
|     ap2.add_argument("-e2dsa", action="store_true", help="scan all folders (for search), sets -e2ds") | ||||
|     ap2.add_argument("--hist", metavar="PATH", type=u, help="where to store volume data (db, thumbs)") | ||||
|     ap2.add_argument("--no-hash", action="store_true", help="disable hashing during e2ds folder scans") | ||||
|     ap2.add_argument("--re-int", metavar="SEC", type=int, default=30, help="disk rescan check interval") | ||||
|     ap2.add_argument("--re-maxage", metavar="SEC", type=int, default=0, help="disk rescan volume interval, 0=off, can be set per-volume with the 'scan' volflag") | ||||
|     ap2.add_argument("--srch-time", metavar="SEC", type=int, default=30, help="search deadline") | ||||
|      | ||||
|     ap2 = ap.add_argument_group('metadata db options') | ||||
|     ap2.add_argument("-e2t", action="store_true", help="enable metadata indexing") | ||||
|     ap2.add_argument("-e2ts", action="store_true", help="enable metadata scanner, sets -e2t") | ||||
|     ap2.add_argument("-e2tsr", action="store_true", help="rescan all metadata, sets -e2ts") | ||||
|     ap2.add_argument("--no-mutagen", action="store_true", help="use FFprobe for tags instead") | ||||
|     ap2.add_argument("--no-mtag-mt", action="store_true", help="disable tag-read parallelism") | ||||
|     ap2.add_argument("--no-mtag-ff", action="store_true", help="never use FFprobe as tag reader") | ||||
|     ap2.add_argument("-mtm", metavar="M=t,t,t", type=u, action="append", help="add/replace metadata mapping") | ||||
|     ap2.add_argument("-mte", metavar="M,M,M", type=u, help="tags to index/display (comma-sep.)", | ||||
|         default="circle,album,.tn,artist,title,.bpm,key,.dur,.q,.vq,.aq,vc,ac,res,.fps,ahash,vhash") | ||||
|     ap2.add_argument("-mth", metavar="M,M,M", type=u, help="tags to hide by default (comma-sep.)", | ||||
|         default=".vq,.aq,vc,ac,res,.fps") | ||||
|     ap2.add_argument("-mtp", metavar="M=[f,]bin", type=u, action="append", help="read tag M using bin") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('appearance options') | ||||
|     ap2.add_argument("--css-browser", metavar="L", type=u, help="URL to additional CSS to include") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('debug options') | ||||
|     ap2.add_argument("--no-sendfile", action="store_true", help="disable sendfile") | ||||
|     ap2.add_argument("--no-scandir", action="store_true", help="disable scandir") | ||||
|     ap2.add_argument("--no-fastboot", action="store_true", help="wait for up2k indexing") | ||||
|     ap2.add_argument("--no-htp", action="store_true", help="disable httpserver threadpool, create threads as-needed instead") | ||||
|     ap2.add_argument("--stackmon", metavar="P,S", type=u, help="write stacktrace to Path every S second") | ||||
|     ap2.add_argument("--log-thrs", metavar="SEC", type=float, help="list active threads every SEC") | ||||
|     # fmt: on | ||||
|  | ||||
|     ap2 = ap.add_argument_group("help sections") | ||||
|     for k, h, _ in sects: | ||||
|         ap2.add_argument("--help-" + k, action="store_true", help=h) | ||||
|  | ||||
|     ret = ap.parse_args(args=argv[1:]) | ||||
|     for k, h, t in sects: | ||||
|         k2 = "help_" + k.replace("-", "_") | ||||
|         if vars(ret)[k2]: | ||||
|             lprint("# {} help page".format(k)) | ||||
|             lprint(t + "\033[0m") | ||||
|             sys.exit(0) | ||||
|  | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def main(argv=None): | ||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||
|     if WINDOWS: | ||||
|         os.system("rem")  # enables colors | ||||
|  | ||||
|     if argv is None: | ||||
|         argv = sys.argv | ||||
|  | ||||
|     desc = py_desc().replace("[", "\033[1;30m[") | ||||
|  | ||||
|     f = '\033[36mcopyparty v{} "\033[35m{}\033[36m" ({})\n{}\033[0m\n' | ||||
|     lprint(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc)) | ||||
|  | ||||
|     ensure_locale() | ||||
|     if HAVE_SSL: | ||||
|         ensure_cert() | ||||
|  | ||||
|     deprecated = [["-e2s", "-e2ds"]] | ||||
|     for dk, nk in deprecated: | ||||
|         try: | ||||
|             idx = argv.index(dk) | ||||
|         except: | ||||
|             continue | ||||
|  | ||||
|         msg = "\033[1;31mWARNING:\033[0;1m\n  {} \033[0;33mwas replaced with\033[0;1m {} \033[0;33mand will be removed\n\033[0m" | ||||
|         lprint(msg.format(dk, nk)) | ||||
|         argv[idx] = nk | ||||
|         time.sleep(2) | ||||
|  | ||||
|     try: | ||||
|         al = run_argparse(argv, RiceFormatter) | ||||
|     except AssertionError: | ||||
|         al = run_argparse(argv, Dodge11874) | ||||
|  | ||||
|     nstrs = [] | ||||
|     anymod = False | ||||
|     for ostr in al.v or []: | ||||
|         m = re_vol.match(ostr) | ||||
|         if not m: | ||||
|             # not our problem | ||||
|             nstrs.append(ostr) | ||||
|             continue | ||||
|  | ||||
|         src, dst, perms = m.groups() | ||||
|         na = [src, dst] | ||||
|         mod = False | ||||
|         for opt in perms.split(":"): | ||||
|             if re.match("c[^,]", opt): | ||||
|                 mod = True | ||||
|                 na.append("c," + opt[1:]) | ||||
|             elif re.sub("^[rwmd]*", "", opt) and "," not in opt: | ||||
|                 mod = True | ||||
|                 perm = opt[0] | ||||
|                 if perm == "a": | ||||
|                     perm = "rw" | ||||
|                 na.append(perm + "," + opt[1:]) | ||||
|             else: | ||||
|                 na.append(opt) | ||||
|  | ||||
|         nstr = ":".join(na) | ||||
|         nstrs.append(nstr if mod else ostr) | ||||
|         if mod: | ||||
|             msg = "\033[1;31mWARNING:\033[0;1m\n  -v {} \033[0;33mwas replaced with\033[0;1m\n  -v {} \n\033[0m" | ||||
|             lprint(msg.format(ostr, nstr)) | ||||
|             anymod = True | ||||
|  | ||||
|     if anymod: | ||||
|         al.v = nstrs | ||||
|         time.sleep(2) | ||||
|  | ||||
|     # propagate implications | ||||
|     for k1, k2 in IMPLICATIONS: | ||||
|         if getattr(al, k1): | ||||
|             setattr(al, k2, True) | ||||
|  | ||||
|     al.i = al.i.split(",") | ||||
|     try: | ||||
|         if "-" in al.p: | ||||
|             lo, hi = [int(x) for x in al.p.split("-")] | ||||
|             al.p = list(range(lo, hi + 1)) | ||||
|         else: | ||||
|             al.p = [int(x) for x in al.p.split(",")] | ||||
|     except: | ||||
|         raise Exception("invalid value for -p") | ||||
|  | ||||
|     if HAVE_SSL: | ||||
|         if al.ssl_ver: | ||||
|             configure_ssl_ver(al) | ||||
|  | ||||
|         if al.ciphers: | ||||
|             configure_ssl_ciphers(al) | ||||
|     else: | ||||
|         warn("ssl module does not exist; cannot enable https") | ||||
|  | ||||
|     if PY2 and WINDOWS and al.e2d: | ||||
|         warn( | ||||
|             "windows py2 cannot do unicode filenames with -e2d\n" | ||||
|             + "  (if you crash with codec errors then that is why)" | ||||
|         ) | ||||
|  | ||||
|     if sys.version_info < (3, 6): | ||||
|         al.no_scandir = True | ||||
|  | ||||
|     # signal.signal(signal.SIGINT, sighandler) | ||||
|  | ||||
|     SvcHub(al, argv, printed).run() | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| # coding: utf-8 | ||||
|  | ||||
| VERSION = (0, 4, 3) | ||||
| CODENAME = "NIH" | ||||
| BUILD_DT = (2020, 5, 17) | ||||
| VERSION = (1, 0, 2) | ||||
| CODENAME = "sufficient" | ||||
| BUILD_DT = (2021, 9, 9) | ||||
|  | ||||
| S_VERSION = ".".join(map(str, VERSION)) | ||||
| S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										59
									
								
								copyparty/bos/bos.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								copyparty/bos/bos.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| from ..util import fsenc, fsdec | ||||
| from . import path | ||||
|  | ||||
|  | ||||
| # 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/.$//')" | ||||
|  | ||||
|  | ||||
| def chmod(p, mode): | ||||
|     return os.chmod(fsenc(p), mode) | ||||
|  | ||||
|  | ||||
| def listdir(p="."): | ||||
|     return [fsdec(x) for x in os.listdir(fsenc(p))] | ||||
|  | ||||
|  | ||||
| def lstat(p): | ||||
|     return os.lstat(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def makedirs(name, mode=0o755, exist_ok=True): | ||||
|     bname = fsenc(name) | ||||
|     try: | ||||
|         os.makedirs(bname, mode=mode) | ||||
|     except: | ||||
|         if not exist_ok or not os.path.isdir(bname): | ||||
|             raise | ||||
|  | ||||
|  | ||||
| def mkdir(p, mode=0o755): | ||||
|     return os.mkdir(fsenc(p), mode=mode) | ||||
|  | ||||
|  | ||||
| def rename(src, dst): | ||||
|     return os.rename(fsenc(src), fsenc(dst)) | ||||
|  | ||||
|  | ||||
| def replace(src, dst): | ||||
|     return os.replace(fsenc(src), fsenc(dst)) | ||||
|  | ||||
|  | ||||
| def rmdir(p): | ||||
|     return os.rmdir(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def stat(p): | ||||
|     return os.stat(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def unlink(p): | ||||
|     return os.unlink(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def utime(p, times=None): | ||||
|     return os.utime(fsenc(p), times) | ||||
							
								
								
									
										33
									
								
								copyparty/bos/path.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								copyparty/bos/path.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| from ..util import fsenc, fsdec | ||||
|  | ||||
|  | ||||
| def abspath(p): | ||||
|     return fsdec(os.path.abspath(fsenc(p))) | ||||
|  | ||||
|  | ||||
| def exists(p): | ||||
|     return os.path.exists(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def getmtime(p): | ||||
|     return os.path.getmtime(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def getsize(p): | ||||
|     return os.path.getsize(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def isdir(p): | ||||
|     return os.path.isdir(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def islink(p): | ||||
|     return os.path.islink(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def realpath(p): | ||||
|     return fsdec(os.path.realpath(fsenc(p))) | ||||
| @@ -4,17 +4,11 @@ from __future__ import print_function, unicode_literals | ||||
| import time | ||||
| import threading | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, VT100 | ||||
| from .broker_util import try_exec | ||||
| from .broker_mpw import MpWorker | ||||
| from .util import mp | ||||
|  | ||||
|  | ||||
| if PY2 and not WINDOWS: | ||||
|     from multiprocessing.reduction import ForkingPickler | ||||
|     from StringIO import StringIO as MemesIO  # pylint: disable=import-error | ||||
|  | ||||
|  | ||||
| class BrokerMp(object): | ||||
|     """external api; manages MpWorkers""" | ||||
|  | ||||
| @@ -28,38 +22,33 @@ class BrokerMp(object): | ||||
|         self.retpend_mutex = threading.Lock() | ||||
|         self.mutex = threading.Lock() | ||||
|  | ||||
|         cores = self.args.j | ||||
|         if not cores: | ||||
|             cores = mp.cpu_count() | ||||
|  | ||||
|         self.log("broker", "booting {} subprocesses".format(cores)) | ||||
|         for n in range(cores): | ||||
|         self.num_workers = self.args.j or mp.cpu_count() | ||||
|         self.log("broker", "booting {} subprocesses".format(self.num_workers)) | ||||
|         for n in range(1, self.num_workers + 1): | ||||
|             q_pend = mp.Queue(1) | ||||
|             q_yield = 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.nid = n | ||||
|             proc.clients = {} | ||||
|             proc.workload = 0 | ||||
|  | ||||
|             thr = threading.Thread(target=self.collector, args=(proc,)) | ||||
|             thr = threading.Thread( | ||||
|                 target=self.collector, args=(proc,), name="mp-sink-{}".format(n) | ||||
|             ) | ||||
|             thr.daemon = True | ||||
|             thr.start() | ||||
|  | ||||
|             self.procs.append(proc) | ||||
|             proc.start() | ||||
|  | ||||
|         if True: | ||||
|             thr = threading.Thread(target=self.debug_load_balancer) | ||||
|             thr.daemon = True | ||||
|             thr.start() | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.log("broker", "shutting down") | ||||
|         for proc in self.procs: | ||||
|             thr = threading.Thread(target=proc.q_pend.put([0, "shutdown", []])) | ||||
|         for n, proc in enumerate(self.procs): | ||||
|             thr = threading.Thread( | ||||
|                 target=proc.q_pend.put([0, "shutdown", []]), | ||||
|                 name="mp-shutdown-{}-{}".format(n, len(self.procs)), | ||||
|             ) | ||||
|             thr.start() | ||||
|  | ||||
|         with self.mutex: | ||||
| @@ -82,20 +71,6 @@ class BrokerMp(object): | ||||
|             if dest == "log": | ||||
|                 self.log(*args) | ||||
|  | ||||
|             elif dest == "workload": | ||||
|                 with self.mutex: | ||||
|                     proc.workload = args[0] | ||||
|  | ||||
|             elif dest == "httpdrop": | ||||
|                 addr = args[0] | ||||
|  | ||||
|                 with self.mutex: | ||||
|                     del proc.clients[addr] | ||||
|                     if not proc.clients: | ||||
|                         proc.workload = 0 | ||||
|  | ||||
|                 self.hub.tcpsrv.num_clients.add(-1) | ||||
|  | ||||
|             elif dest == "retq": | ||||
|                 # response from previous ipc call | ||||
|                 with self.retpend_mutex: | ||||
| @@ -121,38 +96,12 @@ class BrokerMp(object): | ||||
|         returns a Queue object which eventually contains the response if want_retval | ||||
|         (not-impl here since nothing uses it yet) | ||||
|         """ | ||||
|         if dest == "httpconn": | ||||
|             sck, addr = args | ||||
|             sck2 = sck | ||||
|             if PY2: | ||||
|                 buf = MemesIO() | ||||
|                 ForkingPickler(buf).dump(sck) | ||||
|                 sck2 = buf.getvalue() | ||||
|         if dest == "listen": | ||||
|             for p in self.procs: | ||||
|                 p.q_pend.put([0, dest, [args[0], len(self.procs)]]) | ||||
|  | ||||
|             proc = sorted(self.procs, key=lambda x: x.workload)[0] | ||||
|             proc.q_pend.put([0, dest, [sck2, addr]]) | ||||
|  | ||||
|             with self.mutex: | ||||
|                 proc.clients[addr] = 50 | ||||
|                 proc.workload += 50 | ||||
|         elif dest == "cb_httpsrv_up": | ||||
|             self.hub.cb_httpsrv_up() | ||||
|  | ||||
|         else: | ||||
|             raise Exception("what is " + str(dest)) | ||||
|  | ||||
|     def debug_load_balancer(self): | ||||
|         fmt = "\033[1m{}\033[0;36m{:4}\033[0m " | ||||
|         if not VT100: | ||||
|             fmt = "({}{:4})" | ||||
|  | ||||
|         last = "" | ||||
|         while self.procs: | ||||
|             msg = "" | ||||
|             for proc in self.procs: | ||||
|                 msg += fmt.format(len(proc.clients), proc.workload) | ||||
|  | ||||
|             if msg != last: | ||||
|                 last = msg | ||||
|                 with self.hub.log_mutex: | ||||
|                     print(msg) | ||||
|  | ||||
|             time.sleep(0.1) | ||||
|   | ||||
| @@ -2,17 +2,13 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import sys | ||||
| import time | ||||
| import signal | ||||
| import threading | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS | ||||
| from .broker_util import ExceptionalQueue | ||||
| from .httpsrv import HttpSrv | ||||
| from .util import FAKE_MP | ||||
|  | ||||
| if PY2 and not WINDOWS: | ||||
|     import pickle  # nosec | ||||
| from copyparty.authsrv import AuthSrv | ||||
|  | ||||
|  | ||||
| class MpWorker(object): | ||||
| @@ -24,39 +20,43 @@ class MpWorker(object): | ||||
|         self.args = args | ||||
|         self.n = n | ||||
|  | ||||
|         self.log = self._log_disabled if args.q and not args.lo else self._log_enabled | ||||
|  | ||||
|         self.retpend = {} | ||||
|         self.retpend_mutex = threading.Lock() | ||||
|         self.mutex = threading.Lock() | ||||
|         self.workload_thr_active = False | ||||
|  | ||||
|         # we inherited signal_handler from parent, | ||||
|         # replace it with something harmless | ||||
|         if not FAKE_MP: | ||||
|             signal.signal(signal.SIGINT, self.signal_handler) | ||||
|             for sig in [signal.SIGINT, signal.SIGTERM]: | ||||
|                 signal.signal(sig, self.signal_handler) | ||||
|  | ||||
|         # starting to look like a good idea | ||||
|         self.asrv = AuthSrv(args, None, False) | ||||
|  | ||||
|         # instantiate all services here (TODO: inheritance?) | ||||
|         self.httpsrv = HttpSrv(self) | ||||
|         self.httpsrv.disconnect_func = self.httpdrop | ||||
|         self.httpsrv = HttpSrv(self, n) | ||||
|  | ||||
|         # on winxp and some other platforms, | ||||
|         # use thr.join() to block all signals | ||||
|         thr = threading.Thread(target=self.main) | ||||
|         thr = threading.Thread(target=self.main, name="mpw-main") | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|         thr.join() | ||||
|  | ||||
|     def signal_handler(self, signal, frame): | ||||
|     def signal_handler(self, sig, frame): | ||||
|         # print('k') | ||||
|         pass | ||||
|  | ||||
|     def log(self, src, msg): | ||||
|         self.q_yield.put([0, "log", [src, msg]]) | ||||
|     def _log_enabled(self, src, msg, c=0): | ||||
|         self.q_yield.put([0, "log", [src, msg, c]]) | ||||
|  | ||||
|     def logw(self, msg): | ||||
|         self.log("mp{}".format(self.n), msg) | ||||
|     def _log_disabled(self, src, msg, c=0): | ||||
|         pass | ||||
|  | ||||
|     def httpdrop(self, addr): | ||||
|         self.q_yield.put([0, "httpdrop", [addr]]) | ||||
|     def logw(self, msg, c=0): | ||||
|         self.log("mp{}".format(self.n), msg, c) | ||||
|  | ||||
|     def main(self): | ||||
|         while True: | ||||
| @@ -64,24 +64,13 @@ class MpWorker(object): | ||||
|  | ||||
|             # self.logw("work: [{}]".format(d[0])) | ||||
|             if dest == "shutdown": | ||||
|                 self.httpsrv.shutdown() | ||||
|                 self.logw("ok bye") | ||||
|                 sys.exit(0) | ||||
|                 return | ||||
|  | ||||
|             elif dest == "httpconn": | ||||
|                 sck, addr = args | ||||
|                 if PY2: | ||||
|                     sck = pickle.loads(sck)  # nosec | ||||
|  | ||||
|                 self.log("%s %s" % addr, "-" * 4 + "C-qpop") | ||||
|                 self.httpsrv.accept(sck, addr) | ||||
|  | ||||
|                 with self.mutex: | ||||
|                     if not self.workload_thr_active: | ||||
|                         self.workload_thr_alive = True | ||||
|                         thr = threading.Thread(target=self.thr_workload) | ||||
|                         thr.daemon = True | ||||
|                         thr.start() | ||||
|             elif dest == "listen": | ||||
|                 self.httpsrv.listen(args[0], args[1]) | ||||
|  | ||||
|             elif dest == "retq": | ||||
|                 # response from previous ipc call | ||||
| @@ -105,16 +94,3 @@ class MpWorker(object): | ||||
|  | ||||
|         self.q_yield.put([retq_id, dest, args]) | ||||
|         return retq | ||||
|  | ||||
|     def thr_workload(self): | ||||
|         """announce workloads to MpSrv (the mp controller / loadbalancer)""" | ||||
|         # avoid locking in extract_filedata by tracking difference here | ||||
|         while True: | ||||
|             time.sleep(0.2) | ||||
|             with self.mutex: | ||||
|                 if self.httpsrv.num_clients() == 0: | ||||
|                     # no clients rn, termiante thread | ||||
|                     self.workload_thr_alive = False | ||||
|                     return | ||||
|  | ||||
|             self.q_yield.put([0, "workload", [self.httpsrv.workload]]) | ||||
|   | ||||
| @@ -14,22 +14,22 @@ class BrokerThr(object): | ||||
|         self.hub = hub | ||||
|         self.log = hub.log | ||||
|         self.args = hub.args | ||||
|         self.asrv = hub.asrv | ||||
|  | ||||
|         self.mutex = threading.Lock() | ||||
|         self.num_workers = 1 | ||||
|  | ||||
|         # instantiate all services here (TODO: inheritance?) | ||||
|         self.httpsrv = HttpSrv(self) | ||||
|         self.httpsrv.disconnect_func = self.httpdrop | ||||
|         self.httpsrv = HttpSrv(self, None) | ||||
|  | ||||
|     def shutdown(self): | ||||
|         # self.log("broker", "shutting down") | ||||
|         self.httpsrv.shutdown() | ||||
|         pass | ||||
|  | ||||
|     def put(self, want_retval, dest, *args): | ||||
|         if dest == "httpconn": | ||||
|             sck, addr = args | ||||
|             self.log("%s %s" % addr, "-" * 4 + "C-qpop") | ||||
|             self.httpsrv.accept(sck, addr) | ||||
|         if dest == "listen": | ||||
|             self.httpsrv.listen(args[0], 1) | ||||
|  | ||||
|         else: | ||||
|             # new ipc invoking managed service in hub | ||||
| @@ -46,6 +46,3 @@ class BrokerThr(object): | ||||
|             retq = ExceptionalQueue(1) | ||||
|             retq.put(rv) | ||||
|             return retq | ||||
|  | ||||
|     def httpdrop(self, addr): | ||||
|         self.hub.tcpsrv.num_clients.add(-1) | ||||
|   | ||||
							
								
								
									
										1675
									
								
								copyparty/httpcli.py
									
									
									
									
									
								
							
							
						
						
									
										1675
									
								
								copyparty/httpcli.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,29 +1,24 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import ssl | ||||
| import time | ||||
| import socket | ||||
|  | ||||
| HAVE_SSL = True | ||||
| try: | ||||
|     import jinja2 | ||||
| except ImportError: | ||||
|     print( | ||||
|         """\033[1;31m | ||||
|   you do not have jinja2 installed,\033[33m | ||||
|   choose one of these:\033[0m | ||||
|    * apt install python-jinja2 | ||||
|    * python3 -m pip install --user jinja2 | ||||
|    * (try another python version, if you have one) | ||||
|    * (try copyparty.sfx instead) | ||||
| """ | ||||
|     ) | ||||
|     sys.exit(1) | ||||
|     import ssl | ||||
| except: | ||||
|     HAVE_SSL = False | ||||
|  | ||||
| from .__init__ import E | ||||
| from .util import Unrecv | ||||
| from .httpcli import HttpCli | ||||
| from .u2idx import U2idx | ||||
| from .th_cli import ThumbCli | ||||
| from .th_srv import HAVE_PIL | ||||
| from .ico import Ico | ||||
|  | ||||
|  | ||||
| class HttpConn(object): | ||||
| @@ -38,30 +33,57 @@ class HttpConn(object): | ||||
|         self.hsrv = hsrv | ||||
|  | ||||
|         self.args = hsrv.args | ||||
|         self.auth = hsrv.auth | ||||
|         self.asrv = hsrv.asrv | ||||
|         self.cert_path = hsrv.cert_path | ||||
|  | ||||
|         self.workload = 0 | ||||
|         self.log_func = hsrv.log | ||||
|         self.log_src = "{} \033[36m{}".format(addr[0], addr[1]).ljust(26) | ||||
|         enth = HAVE_PIL and not self.args.no_thumb | ||||
|         self.thumbcli = ThumbCli(hsrv.broker) if enth else None | ||||
|         self.ico = Ico(self.args) | ||||
|  | ||||
|         env = jinja2.Environment() | ||||
|         env.loader = jinja2.FileSystemLoader(os.path.join(E.mod, "web")) | ||||
|         self.tpl_mounts = env.get_template("splash.html") | ||||
|         self.tpl_browser = env.get_template("browser.html") | ||||
|         self.tpl_msg = env.get_template("msg.html") | ||||
|         self.tpl_md = env.get_template("md.html") | ||||
|         self.tpl_mde = env.get_template("mde.html") | ||||
|         self.t0 = time.time() | ||||
|         self.stopping = False | ||||
|         self.nreq = 0 | ||||
|         self.nbyte = 0 | ||||
|         self.u2idx = None | ||||
|         self.log_func = hsrv.log | ||||
|         self.lf_url = re.compile(self.args.lf_url) if self.args.lf_url else None | ||||
|         self.set_rproxy() | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.stopping = True | ||||
|         try: | ||||
|             self.s.shutdown(socket.SHUT_RDWR) | ||||
|             self.s.close() | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def set_rproxy(self, ip=None): | ||||
|         if ip is None: | ||||
|             color = 36 | ||||
|             ip = self.addr[0] | ||||
|             self.rproxy = None | ||||
|         else: | ||||
|             color = 34 | ||||
|             self.rproxy = ip | ||||
|  | ||||
|         self.ip = ip | ||||
|         self.log_src = "{} \033[{}m{}".format(ip, color, self.addr[1]).ljust(26) | ||||
|         return self.log_src | ||||
|  | ||||
|     def respath(self, res_name): | ||||
|         return os.path.join(E.mod, "web", res_name) | ||||
|  | ||||
|     def log(self, msg): | ||||
|         self.log_func(self.log_src, msg) | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func(self.log_src, msg, c) | ||||
|  | ||||
|     def run(self): | ||||
|     def get_u2idx(self): | ||||
|         if not self.u2idx: | ||||
|             self.u2idx = U2idx(self) | ||||
|  | ||||
|         return self.u2idx | ||||
|  | ||||
|     def _detect_https(self): | ||||
|         method = None | ||||
|         self.sr = None | ||||
|         if self.cert_path: | ||||
|             try: | ||||
|                 method = self.s.recv(4, socket.MSG_PEEK) | ||||
| @@ -82,20 +104,64 @@ class HttpConn(object): | ||||
|                 err = "need at least 4 bytes in the first packet; got {}".format( | ||||
|                     len(method) | ||||
|                 ) | ||||
|                 self.log(err) | ||||
|                 if method: | ||||
|                     self.log(err) | ||||
|  | ||||
|                 self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8")) | ||||
|                 return | ||||
|  | ||||
|         if method not in [None, b"GET ", b"HEAD", b"POST"]: | ||||
|         return method not in [None, b"GET ", b"HEAD", b"POST", b"PUT ", b"OPTI"] | ||||
|  | ||||
|     def run(self): | ||||
|         self.sr = None | ||||
|         if self.args.https_only: | ||||
|             is_https = True | ||||
|         elif self.args.http_only or not HAVE_SSL: | ||||
|             is_https = False | ||||
|         else: | ||||
|             is_https = self._detect_https() | ||||
|  | ||||
|         if is_https: | ||||
|             if self.sr: | ||||
|                 self.log("\033[1;31mTODO: cannot do https in jython\033[0m") | ||||
|                 self.log("TODO: cannot do https in jython", c="1;31") | ||||
|                 return | ||||
|  | ||||
|             self.log_src = self.log_src.replace("[36m", "[35m") | ||||
|             try: | ||||
|                 self.s = ssl.wrap_socket( | ||||
|                     self.s, server_side=True, certfile=self.cert_path | ||||
|                 ) | ||||
|                 ctx = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH) | ||||
|                 ctx.load_cert_chain(self.cert_path) | ||||
|                 if self.args.ssl_ver: | ||||
|                     ctx.options &= ~self.args.ssl_flags_en | ||||
|                     ctx.options |= self.args.ssl_flags_de | ||||
|                     # print(repr(ctx.options)) | ||||
|  | ||||
|                 if self.args.ssl_log: | ||||
|                     try: | ||||
|                         ctx.keylog_filename = self.args.ssl_log | ||||
|                     except: | ||||
|                         self.log("keylog failed; openssl or python too old") | ||||
|  | ||||
|                 if self.args.ciphers: | ||||
|                     ctx.set_ciphers(self.args.ciphers) | ||||
|  | ||||
|                 self.s = ctx.wrap_socket(self.s, server_side=True) | ||||
|                 msg = [ | ||||
|                     "\033[1;3{:d}m{}".format(c, s) | ||||
|                     for c, s in zip([0, 5, 0], self.s.cipher()) | ||||
|                 ] | ||||
|                 self.log(" ".join(msg) + "\033[0m") | ||||
|  | ||||
|                 if self.args.ssl_dbg and hasattr(self.s, "shared_ciphers"): | ||||
|                     overlap = [y[::-1] for y in self.s.shared_ciphers()] | ||||
|                     lines = [str(x) for x in (["TLS cipher overlap:"] + overlap)] | ||||
|                     self.log("\n".join(lines)) | ||||
|                     for k, v in [ | ||||
|                         ["compression", self.s.compression()], | ||||
|                         ["ALPN proto", self.s.selected_alpn_protocol()], | ||||
|                         ["NPN proto", self.s.selected_npn_protocol()], | ||||
|                     ]: | ||||
|                         self.log("TLS {}: {}".format(k, v or "nah")) | ||||
|  | ||||
|             except Exception as ex: | ||||
|                 em = str(ex) | ||||
|  | ||||
| @@ -104,18 +170,19 @@ class HttpConn(object): | ||||
|                     self.log("client rejected our certificate (nice)") | ||||
|  | ||||
|                 elif "ALERT_CERTIFICATE_UNKNOWN" in em: | ||||
|                     # chrome-android keeps doing this | ||||
|                     # android-chrome keeps doing this | ||||
|                     pass | ||||
|  | ||||
|                 else: | ||||
|                     self.log("\033[35mhandshake\033[0m " + em) | ||||
|                     self.log("handshake\033[0m " + em, c=5) | ||||
|  | ||||
|                 return | ||||
|  | ||||
|         if not self.sr: | ||||
|             self.sr = Unrecv(self.s) | ||||
|  | ||||
|         while True: | ||||
|         while not self.stopping: | ||||
|             self.nreq += 1 | ||||
|             cli = HttpCli(self) | ||||
|             if not cli.run(): | ||||
|                 return | ||||
|   | ||||
| @@ -2,13 +2,39 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import math | ||||
| import base64 | ||||
| import socket | ||||
| import threading | ||||
|  | ||||
| from .__init__ import E, MACOS | ||||
| try: | ||||
|     import jinja2 | ||||
| except ImportError: | ||||
|     print( | ||||
|         """\033[1;31m | ||||
|   you do not have jinja2 installed,\033[33m | ||||
|   choose one of these:\033[0m | ||||
|    * apt install python-jinja2 | ||||
|    * {} -m pip install --user jinja2 | ||||
|    * (try another python version, if you have one) | ||||
|    * (try copyparty.sfx instead) | ||||
| """.format( | ||||
|             os.path.basename(sys.executable) | ||||
|         ) | ||||
|     ) | ||||
|     sys.exit(1) | ||||
|  | ||||
| from .__init__ import E, PY2, MACOS | ||||
| from .util import spack, min_ex, start_stackmon, start_log_thrs | ||||
| from .bos import bos | ||||
| from .httpconn import HttpConn | ||||
| from .authsrv import AuthSrv | ||||
|  | ||||
| if PY2: | ||||
|     import Queue as queue | ||||
| else: | ||||
|     import queue | ||||
|  | ||||
|  | ||||
| class HttpSrv(object): | ||||
| @@ -17,38 +43,208 @@ class HttpSrv(object): | ||||
|     relying on MpSrv for performance (HttpSrv is just plain threads) | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, broker): | ||||
|     def __init__(self, broker, nid): | ||||
|         self.broker = broker | ||||
|         self.nid = nid | ||||
|         self.args = broker.args | ||||
|         self.log = broker.log | ||||
|         self.asrv = broker.asrv | ||||
|  | ||||
|         self.disconnect_func = None | ||||
|         self.name = "httpsrv" + ("-n{}-i{:x}".format(nid, os.getpid()) if nid else "") | ||||
|         self.mutex = threading.Lock() | ||||
|         self.stopping = False | ||||
|  | ||||
|         self.clients = {} | ||||
|         self.workload = 0 | ||||
|         self.workload_thr_alive = False | ||||
|         self.auth = AuthSrv(self.args, self.log) | ||||
|         self.tp_nthr = 0  # actual | ||||
|         self.tp_ncli = 0  # fading | ||||
|         self.tp_time = None  # latest worker collect | ||||
|         self.tp_q = None if self.args.no_htp else queue.LifoQueue() | ||||
|  | ||||
|         self.srvs = [] | ||||
|         self.ncli = 0  # exact | ||||
|         self.clients = {}  # laggy | ||||
|         self.nclimax = 0 | ||||
|         self.cb_ts = 0 | ||||
|         self.cb_v = 0 | ||||
|  | ||||
|         env = jinja2.Environment() | ||||
|         env.loader = jinja2.FileSystemLoader(os.path.join(E.mod, "web")) | ||||
|         self.j2 = { | ||||
|             x: env.get_template(x + ".html") | ||||
|             for x in ["splash", "browser", "browser2", "msg", "md", "mde"] | ||||
|         } | ||||
|  | ||||
|         cert_path = os.path.join(E.cfg, "cert.pem") | ||||
|         if os.path.exists(cert_path): | ||||
|         if bos.path.exists(cert_path): | ||||
|             self.cert_path = cert_path | ||||
|         else: | ||||
|             self.cert_path = None | ||||
|  | ||||
|         if self.tp_q: | ||||
|             self.start_threads(4) | ||||
|  | ||||
|             name = "httpsrv-scaler" + ("-{}".format(nid) if nid else "") | ||||
|             t = threading.Thread(target=self.thr_scaler, name=name) | ||||
|             t.daemon = True | ||||
|             t.start() | ||||
|  | ||||
|         if nid: | ||||
|             if self.args.stackmon: | ||||
|                 start_stackmon(self.args.stackmon, nid) | ||||
|  | ||||
|             if self.args.log_thrs: | ||||
|                 start_log_thrs(self.log, self.args.log_thrs, nid) | ||||
|  | ||||
|     def start_threads(self, n): | ||||
|         self.tp_nthr += n | ||||
|         if self.args.log_htp: | ||||
|             self.log(self.name, "workers += {} = {}".format(n, self.tp_nthr), 6) | ||||
|  | ||||
|         for _ in range(n): | ||||
|             thr = threading.Thread( | ||||
|                 target=self.thr_poolw, | ||||
|                 name=self.name + "-poolw", | ||||
|             ) | ||||
|             thr.daemon = True | ||||
|             thr.start() | ||||
|  | ||||
|     def stop_threads(self, n): | ||||
|         self.tp_nthr -= n | ||||
|         if self.args.log_htp: | ||||
|             self.log(self.name, "workers -= {} = {}".format(n, self.tp_nthr), 6) | ||||
|  | ||||
|         for _ in range(n): | ||||
|             self.tp_q.put(None) | ||||
|  | ||||
|     def thr_scaler(self): | ||||
|         while True: | ||||
|             time.sleep(2 if self.tp_ncli else 30) | ||||
|             with self.mutex: | ||||
|                 self.tp_ncli = max(self.ncli, self.tp_ncli - 2) | ||||
|                 if self.tp_nthr > self.tp_ncli + 8: | ||||
|                     self.stop_threads(4) | ||||
|  | ||||
|     def listen(self, sck, nlisteners): | ||||
|         ip, port = sck.getsockname() | ||||
|         self.srvs.append(sck) | ||||
|         self.nclimax = math.ceil(self.args.nc * 1.0 / nlisteners) | ||||
|         t = threading.Thread( | ||||
|             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): | ||||
|         """listens on a shared tcp server""" | ||||
|         ip, port = srv_sck.getsockname() | ||||
|         fno = srv_sck.fileno() | ||||
|         msg = "subscribed @ {}:{}  f{}".format(ip, port, fno) | ||||
|         self.log(self.name, msg) | ||||
|         self.broker.put(False, "cb_httpsrv_up") | ||||
|         while not self.stopping: | ||||
|             if self.args.log_conn: | ||||
|                 self.log(self.name, "|%sC-ncli" % ("-" * 1,), c="1;30") | ||||
|  | ||||
|             if self.ncli >= self.nclimax: | ||||
|                 self.log(self.name, "at connection limit; waiting", 3) | ||||
|                 while self.ncli >= self.nclimax: | ||||
|                     time.sleep(0.1) | ||||
|  | ||||
|             if self.args.log_conn: | ||||
|                 self.log(self.name, "|%sC-acc1" % ("-" * 2,), c="1;30") | ||||
|  | ||||
|             try: | ||||
|                 sck, addr = srv_sck.accept() | ||||
|             except (OSError, socket.error) as ex: | ||||
|                 self.log(self.name, "accept({}): {}".format(fno, ex), c=6) | ||||
|                 time.sleep(0.02) | ||||
|                 continue | ||||
|  | ||||
|             if self.args.log_conn: | ||||
|                 m = "|{}C-acc2 \033[0;36m{} \033[3{}m{}".format( | ||||
|                     "-" * 3, ip, port % 8, port | ||||
|                 ) | ||||
|                 self.log("%s %s" % addr, m, c="1;30") | ||||
|  | ||||
|             self.accept(sck, addr) | ||||
|  | ||||
|     def accept(self, sck, addr): | ||||
|         """takes an incoming tcp connection and creates a thread to handle it""" | ||||
|         self.log("%s %s" % addr, "-" * 5 + "C-cthr") | ||||
|         thr = threading.Thread(target=self.thr_client, args=(sck, addr)) | ||||
|         now = time.time() | ||||
|  | ||||
|         if now - (self.tp_time or now) > 300: | ||||
|             m = "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.tp_time = None | ||||
|             self.tp_q = None | ||||
|  | ||||
|         with self.mutex: | ||||
|             self.ncli += 1 | ||||
|             if self.tp_q: | ||||
|                 self.tp_time = self.tp_time or now | ||||
|                 self.tp_ncli = max(self.tp_ncli, self.ncli) | ||||
|                 if self.tp_nthr < self.ncli + 4: | ||||
|                     self.start_threads(8) | ||||
|  | ||||
|                 self.tp_q.put((sck, addr)) | ||||
|                 return | ||||
|  | ||||
|         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" | ||||
|             self.log(self.name, m, 1) | ||||
|  | ||||
|         thr = threading.Thread( | ||||
|             target=self.thr_client, | ||||
|             args=(sck, addr), | ||||
|             name="httpconn-{}-{}".format(addr[0].split(".", 2)[-1][-6:], addr[1]), | ||||
|         ) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|     def num_clients(self): | ||||
|         with self.mutex: | ||||
|             return len(self.clients) | ||||
|     def thr_poolw(self): | ||||
|         while True: | ||||
|             task = self.tp_q.get() | ||||
|             if not task: | ||||
|                 break | ||||
|  | ||||
|             with self.mutex: | ||||
|                 self.tp_time = None | ||||
|  | ||||
|             try: | ||||
|                 sck, addr = task | ||||
|                 me = threading.current_thread() | ||||
|                 me.name = "httpconn-{}-{}".format( | ||||
|                     addr[0].split(".", 2)[-1][-6:], addr[1] | ||||
|                 ) | ||||
|                 self.thr_client(sck, addr) | ||||
|                 me.name = self.name + "-poolw" | ||||
|             except: | ||||
|                 self.log(self.name, "thr_client: " + min_ex(), 3) | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.log("ok bye") | ||||
|         self.stopping = True | ||||
|         for srv in self.srvs: | ||||
|             try: | ||||
|                 srv.close() | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         clients = list(self.clients.keys()) | ||||
|         for cli in clients: | ||||
|             try: | ||||
|                 cli.shutdown() | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         if self.tp_q: | ||||
|             self.stop_threads(self.tp_nthr) | ||||
|             for _ in range(10): | ||||
|                 time.sleep(0.05) | ||||
|                 if self.tp_q.empty(): | ||||
|                     break | ||||
|  | ||||
|         self.log(self.name, "ok bye") | ||||
|  | ||||
|     def thr_client(self, sck, addr): | ||||
|         """thread managing one tcp client""" | ||||
| @@ -57,65 +253,69 @@ class HttpSrv(object): | ||||
|         cli = HttpConn(sck, addr, self) | ||||
|         with self.mutex: | ||||
|             self.clients[cli] = 0 | ||||
|             self.workload += 50 | ||||
|  | ||||
|             if not self.workload_thr_alive: | ||||
|                 self.workload_thr_alive = True | ||||
|                 thr = threading.Thread(target=self.thr_workload) | ||||
|                 thr.daemon = True | ||||
|                 thr.start() | ||||
|  | ||||
|         fno = sck.fileno() | ||||
|         try: | ||||
|             self.log("%s %s" % addr, "-" * 6 + "C-crun") | ||||
|             if self.args.log_conn: | ||||
|                 self.log("%s %s" % addr, "|%sC-crun" % ("-" * 4,), c="1;30") | ||||
|  | ||||
|             cli.run() | ||||
|  | ||||
|         except (OSError, socket.error) as ex: | ||||
|             if ex.errno not in [10038, 10054, 107, 57, 49, 9]: | ||||
|                 self.log( | ||||
|                     "%s %s" % addr, | ||||
|                     "run({}): {}".format(fno, ex), | ||||
|                     c=6, | ||||
|                 ) | ||||
|  | ||||
|         finally: | ||||
|             self.log("%s %s" % addr, "-" * 7 + "C-done") | ||||
|             sck = cli.s | ||||
|             if self.args.log_conn: | ||||
|                 self.log("%s %s" % addr, "|%sC-cdone" % ("-" * 5,), c="1;30") | ||||
|  | ||||
|             try: | ||||
|                 fno = sck.fileno() | ||||
|                 sck.shutdown(socket.SHUT_RDWR) | ||||
|                 sck.close() | ||||
|             except (OSError, socket.error) as ex: | ||||
|                 if not MACOS: | ||||
|                     self.log( | ||||
|                         "%s %s" % addr, | ||||
|                         "shut_rdwr err:\n  {}\n  {}".format(repr(sck), ex), | ||||
|                         "shut({}): {}".format(fno, ex), | ||||
|                         c="1;30", | ||||
|                     ) | ||||
|                 if ex.errno not in [10038, 10054, 107, 57, 9]: | ||||
|                 if ex.errno not in [10038, 10054, 107, 57, 49, 9]: | ||||
|                     # 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 | ||||
|             finally: | ||||
|                 with self.mutex: | ||||
|                     del self.clients[cli] | ||||
|                     self.ncli -= 1 | ||||
|  | ||||
|                 if self.disconnect_func: | ||||
|                     self.disconnect_func(addr)  # pylint: disable=not-callable | ||||
|     def cachebuster(self): | ||||
|         if time.time() - self.cb_ts < 1: | ||||
|             return self.cb_v | ||||
|  | ||||
|     def thr_workload(self): | ||||
|         """indicates the python interpreter workload caused by this HttpSrv""" | ||||
|         # avoid locking in extract_filedata by tracking difference here | ||||
|         while True: | ||||
|             time.sleep(0.2) | ||||
|             with self.mutex: | ||||
|                 if not self.clients: | ||||
|                     # no clients rn, termiante thread | ||||
|                     self.workload_thr_alive = False | ||||
|                     self.workload = 0 | ||||
|                     return | ||||
|         with self.mutex: | ||||
|             if time.time() - self.cb_ts < 1: | ||||
|                 return self.cb_v | ||||
|  | ||||
|             total = 0 | ||||
|             with self.mutex: | ||||
|                 for cli in self.clients.keys(): | ||||
|                     now = cli.workload | ||||
|                     delta = now - self.clients[cli] | ||||
|                     if delta < 0: | ||||
|                         # was reset in HttpCli to prevent overflow | ||||
|                         delta = now | ||||
|             v = E.t0 | ||||
|             try: | ||||
|                 with os.scandir(os.path.join(E.mod, "web")) as dh: | ||||
|                     for fh in dh: | ||||
|                         inf = fh.stat() | ||||
|                         v = max(v, inf.st_mtime) | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|                     total += delta | ||||
|                     self.clients[cli] = now | ||||
|  | ||||
|             self.workload = total | ||||
|             v = base64.urlsafe_b64encode(spack(b">xxL", int(v))) | ||||
|             self.cb_v = v.decode("ascii")[-4:] | ||||
|             self.cb_ts = time.time() | ||||
|             return self.cb_v | ||||
|   | ||||
							
								
								
									
										42
									
								
								copyparty/ico.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								copyparty/ico.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import hashlib | ||||
| import colorsys | ||||
|  | ||||
| from .__init__ import PY2 | ||||
|  | ||||
|  | ||||
| class Ico(object): | ||||
|     def __init__(self, args): | ||||
|         self.args = args | ||||
|  | ||||
|     def get(self, ext, as_thumb): | ||||
|         """placeholder to make thumbnails not break""" | ||||
|  | ||||
|         h = hashlib.md5(ext.encode("utf-8")).digest()[:2] | ||||
|         if PY2: | ||||
|             h = [ord(x) for x in h] | ||||
|  | ||||
|         c1 = colorsys.hsv_to_rgb(h[0] / 256.0, 1, 0.3) | ||||
|         c2 = colorsys.hsv_to_rgb(h[0] / 256.0, 1, 1) | ||||
|         c = list(c1) + list(c2) | ||||
|         c = [int(x * 255) for x in c] | ||||
|         c = "".join(["{:02x}".format(x) for x in c]) | ||||
|  | ||||
|         h = 30 | ||||
|         if not self.args.th_no_crop and as_thumb: | ||||
|             w, h = self.args.th_size.split("x") | ||||
|             h = int(100 / (float(w) / float(h))) | ||||
|  | ||||
|         svg = """\ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <svg version="1.1" viewBox="0 0 100 {}" xmlns="http://www.w3.org/2000/svg"><g> | ||||
| <rect width="100%" height="100%" fill="#{}" /> | ||||
| <text x="50%" y="50%" dominant-baseline="middle" text-anchor="middle" xml:space="preserve" | ||||
|   fill="#{}" font-family="monospace" font-size="14px" style="letter-spacing:.5px">{}</text> | ||||
| </g></svg> | ||||
| """ | ||||
|         svg = svg.format(h, c[:6], c[6:], ext).encode("utf-8") | ||||
|  | ||||
|         return ["image/svg+xml", svg] | ||||
							
								
								
									
										497
									
								
								copyparty/mtag.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										497
									
								
								copyparty/mtag.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,497 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import json | ||||
| import shutil | ||||
| import subprocess as sp | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, unicode | ||||
| from .util import fsenc, fsdec, uncyg, REKOBO_LKEY | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| def have_ff(cmd): | ||||
|     if PY2: | ||||
|         print("# checking {}".format(cmd)) | ||||
|         cmd = (cmd + " -version").encode("ascii").split(b" ") | ||||
|         try: | ||||
|             sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE).communicate() | ||||
|             return True | ||||
|         except: | ||||
|             return False | ||||
|     else: | ||||
|         return bool(shutil.which(cmd)) | ||||
|  | ||||
|  | ||||
| HAVE_FFMPEG = have_ff("ffmpeg") | ||||
| HAVE_FFPROBE = have_ff("ffprobe") | ||||
|  | ||||
|  | ||||
| class MParser(object): | ||||
|     def __init__(self, cmdline): | ||||
|         self.tag, args = cmdline.split("=", 1) | ||||
|         self.tags = self.tag.split(",") | ||||
|  | ||||
|         self.timeout = 30 | ||||
|         self.force = False | ||||
|         self.audio = "y" | ||||
|         self.ext = [] | ||||
|  | ||||
|         while True: | ||||
|             try: | ||||
|                 bp = os.path.expanduser(args) | ||||
|                 if WINDOWS: | ||||
|                     bp = uncyg(bp) | ||||
|  | ||||
|                 if bos.path.exists(bp): | ||||
|                     self.bin = bp | ||||
|                     return | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|             arg, args = args.split(",", 1) | ||||
|             arg = arg.lower() | ||||
|  | ||||
|             if arg.startswith("a"): | ||||
|                 self.audio = arg[1:]  # [r]equire [n]ot [d]ontcare | ||||
|                 continue | ||||
|  | ||||
|             if arg == "f": | ||||
|                 self.force = True | ||||
|                 continue | ||||
|  | ||||
|             if arg.startswith("t"): | ||||
|                 self.timeout = int(arg[1:]) | ||||
|                 continue | ||||
|  | ||||
|             if arg.startswith("e"): | ||||
|                 self.ext.append(arg[1:]) | ||||
|                 continue | ||||
|  | ||||
|             raise Exception() | ||||
|  | ||||
|  | ||||
| def ffprobe(abspath): | ||||
|     cmd = [ | ||||
|         b"ffprobe", | ||||
|         b"-hide_banner", | ||||
|         b"-show_streams", | ||||
|         b"-show_format", | ||||
|         b"--", | ||||
|         fsenc(abspath), | ||||
|     ] | ||||
|     p = sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE) | ||||
|     r = p.communicate() | ||||
|     txt = r[0].decode("utf-8", "replace") | ||||
|     return parse_ffprobe(txt) | ||||
|  | ||||
|  | ||||
| def parse_ffprobe(txt): | ||||
|     """ffprobe -show_format -show_streams""" | ||||
|     streams = [] | ||||
|     fmt = {} | ||||
|     g = None | ||||
|     for ln in [x.rstrip("\r") for x in txt.split("\n")]: | ||||
|         try: | ||||
|             k, v = ln.split("=", 1) | ||||
|             g[k] = v | ||||
|             continue | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         if ln == "[STREAM]": | ||||
|             g = {} | ||||
|             streams.append(g) | ||||
|  | ||||
|         if ln == "[FORMAT]": | ||||
|             g = {"codec_type": "format"}  # heh | ||||
|             fmt = g | ||||
|  | ||||
|     streams = [fmt] + streams | ||||
|     ret = {}  # processed | ||||
|     md = {}  # raw tags | ||||
|  | ||||
|     is_audio = fmt.get("format_name") in ["mp3", "ogg", "flac", "wav"] | ||||
|     if fmt.get("filename", "").split(".")[-1].lower() in ["m4a", "aac"]: | ||||
|         is_audio = True | ||||
|  | ||||
|     # if audio file, ensure audio stream appears first | ||||
|     if ( | ||||
|         is_audio | ||||
|         and len(streams) > 2 | ||||
|         and streams[1].get("codec_type") != "audio" | ||||
|         and streams[2].get("codec_type") == "audio" | ||||
|     ): | ||||
|         streams = [fmt, streams[2], streams[1]] + streams[3:] | ||||
|  | ||||
|     have = {} | ||||
|     for strm in streams: | ||||
|         typ = strm.get("codec_type") | ||||
|         if typ in have: | ||||
|             continue | ||||
|  | ||||
|         have[typ] = True | ||||
|         kvm = [] | ||||
|  | ||||
|         if typ == "audio": | ||||
|             kvm = [ | ||||
|                 ["codec_name", "ac"], | ||||
|                 ["channel_layout", "chs"], | ||||
|                 ["sample_rate", ".hz"], | ||||
|                 ["bit_rate", ".aq"], | ||||
|                 ["duration", ".dur"], | ||||
|             ] | ||||
|  | ||||
|         if typ == "video": | ||||
|             if strm.get("DISPOSITION:attached_pic") == "1" or is_audio: | ||||
|                 continue | ||||
|  | ||||
|             kvm = [ | ||||
|                 ["codec_name", "vc"], | ||||
|                 ["pix_fmt", "pixfmt"], | ||||
|                 ["r_frame_rate", ".fps"], | ||||
|                 ["bit_rate", ".vq"], | ||||
|                 ["width", ".resw"], | ||||
|                 ["height", ".resh"], | ||||
|                 ["duration", ".dur"], | ||||
|             ] | ||||
|  | ||||
|         if typ == "format": | ||||
|             kvm = [["duration", ".dur"], ["bit_rate", ".q"]] | ||||
|  | ||||
|         for sk, rk in kvm: | ||||
|             v = strm.get(sk) | ||||
|             if v is None: | ||||
|                 continue | ||||
|  | ||||
|             if rk.startswith("."): | ||||
|                 try: | ||||
|                     v = float(v) | ||||
|                     v2 = ret.get(rk) | ||||
|                     if v2 is None or v > v2: | ||||
|                         ret[rk] = v | ||||
|                 except: | ||||
|                     # sqlite doesnt care but the code below does | ||||
|                     if v not in ["N/A"]: | ||||
|                         ret[rk] = v | ||||
|             else: | ||||
|                 ret[rk] = v | ||||
|  | ||||
|     if ret.get("vc") == "ansi":  # shellscript | ||||
|         return {}, {} | ||||
|  | ||||
|     for strm in streams: | ||||
|         for k, v in strm.items(): | ||||
|             if not k.startswith("TAG:"): | ||||
|                 continue | ||||
|  | ||||
|             k = k[4:].strip() | ||||
|             v = v.strip() | ||||
|             if k and v and k not in md: | ||||
|                 md[k] = [v] | ||||
|  | ||||
|     for k in [".q", ".vq", ".aq"]: | ||||
|         if k in ret: | ||||
|             ret[k] /= 1000  # bit_rate=320000 | ||||
|  | ||||
|     for k in [".q", ".vq", ".aq", ".resw", ".resh"]: | ||||
|         if k in ret: | ||||
|             ret[k] = int(ret[k]) | ||||
|  | ||||
|     if ".fps" in ret: | ||||
|         fps = ret[".fps"] | ||||
|         if "/" in fps: | ||||
|             fa, fb = fps.split("/") | ||||
|             fps = int(fa) * 1.0 / int(fb) | ||||
|  | ||||
|         if fps < 1000 and fmt.get("format_name") not in ["image2", "png_pipe"]: | ||||
|             ret[".fps"] = round(fps, 3) | ||||
|         else: | ||||
|             del ret[".fps"] | ||||
|  | ||||
|     if ".dur" in ret: | ||||
|         if ret[".dur"] < 0.1: | ||||
|             del ret[".dur"] | ||||
|             if ".q" in ret: | ||||
|                 del ret[".q"] | ||||
|  | ||||
|     if ".resw" in ret and ".resh" in ret: | ||||
|         ret["res"] = "{}x{}".format(ret[".resw"], ret[".resh"]) | ||||
|  | ||||
|     ret = {k: [0, v] for k, v in ret.items()} | ||||
|  | ||||
|     return ret, md | ||||
|  | ||||
|  | ||||
| class MTag(object): | ||||
|     def __init__(self, log_func, args): | ||||
|         self.log_func = log_func | ||||
|         self.args = args | ||||
|         self.usable = True | ||||
|         self.prefer_mt = not args.no_mtag_ff | ||||
|         self.backend = "ffprobe" if args.no_mutagen else "mutagen" | ||||
|         self.can_ffprobe = ( | ||||
|             HAVE_FFPROBE | ||||
|             and not args.no_mtag_ff | ||||
|             and (not WINDOWS or sys.version_info >= (3, 8)) | ||||
|         ) | ||||
|         mappings = args.mtm | ||||
|         or_ffprobe = " or FFprobe" | ||||
|  | ||||
|         if self.backend == "mutagen": | ||||
|             self.get = self.get_mutagen | ||||
|             try: | ||||
|                 import mutagen | ||||
|             except: | ||||
|                 self.log("could not load Mutagen, trying FFprobe instead", c=3) | ||||
|                 self.backend = "ffprobe" | ||||
|  | ||||
|         if self.backend == "ffprobe": | ||||
|             self.usable = self.can_ffprobe | ||||
|             self.get = self.get_ffprobe | ||||
|             self.prefer_mt = True | ||||
|  | ||||
|             if not HAVE_FFPROBE: | ||||
|                 pass | ||||
|  | ||||
|             elif args.no_mtag_ff: | ||||
|                 msg = "found FFprobe but it was disabled by --no-mtag-ff" | ||||
|                 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: | ||||
|             msg = "need Mutagen{} to read media tags so please run this:\n{}{} -m pip install --user mutagen\n" | ||||
|             pybin = os.path.basename(sys.executable) | ||||
|             self.log(msg.format(or_ffprobe, " " * 37, pybin), c=1) | ||||
|             return | ||||
|  | ||||
|         # https://picard-docs.musicbrainz.org/downloads/MusicBrainz_Picard_Tag_Map.html | ||||
|         tagmap = { | ||||
|             "album": ["album", "talb", "\u00a9alb", "original-album", "toal"], | ||||
|             "artist": [ | ||||
|                 "artist", | ||||
|                 "tpe1", | ||||
|                 "\u00a9art", | ||||
|                 "composer", | ||||
|                 "performer", | ||||
|                 "arranger", | ||||
|                 "\u00a9wrt", | ||||
|                 "tcom", | ||||
|                 "tpe3", | ||||
|                 "original-artist", | ||||
|                 "tope", | ||||
|             ], | ||||
|             "title": ["title", "tit2", "\u00a9nam"], | ||||
|             "circle": [ | ||||
|                 "album-artist", | ||||
|                 "tpe2", | ||||
|                 "aart", | ||||
|                 "conductor", | ||||
|                 "organization", | ||||
|                 "band", | ||||
|             ], | ||||
|             ".tn": ["tracknumber", "trck", "trkn", "track"], | ||||
|             "genre": ["genre", "tcon", "\u00a9gen"], | ||||
|             "date": [ | ||||
|                 "original-release-date", | ||||
|                 "release-date", | ||||
|                 "date", | ||||
|                 "tdrc", | ||||
|                 "\u00a9day", | ||||
|                 "original-date", | ||||
|                 "original-year", | ||||
|                 "tyer", | ||||
|                 "tdor", | ||||
|                 "tory", | ||||
|                 "year", | ||||
|                 "creation-time", | ||||
|             ], | ||||
|             ".bpm": ["bpm", "tbpm", "tmpo", "tbp"], | ||||
|             "key": ["initial-key", "tkey", "key"], | ||||
|             "comment": ["comment", "comm", "\u00a9cmt", "comments", "description"], | ||||
|         } | ||||
|  | ||||
|         if mappings: | ||||
|             for k, v in [x.split("=") for x in mappings]: | ||||
|                 tagmap[k] = v.split(",") | ||||
|  | ||||
|         self.tagmap = {} | ||||
|         for k, vs in tagmap.items(): | ||||
|             vs2 = [] | ||||
|             for v in vs: | ||||
|                 if "-" not in v: | ||||
|                     vs2.append(v) | ||||
|                     continue | ||||
|  | ||||
|                 vs2.append(v.replace("-", " ")) | ||||
|                 vs2.append(v.replace("-", "_")) | ||||
|                 vs2.append(v.replace("-", "")) | ||||
|  | ||||
|             self.tagmap[k] = vs2 | ||||
|  | ||||
|         self.rmap = { | ||||
|             v: [n, k] for k, vs in self.tagmap.items() for n, v in enumerate(vs) | ||||
|         } | ||||
|         # self.get = self.compare | ||||
|  | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func("mtag", msg, c) | ||||
|  | ||||
|     def normalize_tags(self, ret, md): | ||||
|         for k, v in dict(md).items(): | ||||
|             if not v: | ||||
|                 continue | ||||
|  | ||||
|             k = k.lower().split("::")[0].strip() | ||||
|             mk = self.rmap.get(k) | ||||
|             if not mk: | ||||
|                 continue | ||||
|  | ||||
|             pref, mk = mk | ||||
|             if mk not in ret or ret[mk][0] > pref: | ||||
|                 ret[mk] = [pref, v[0]] | ||||
|  | ||||
|         # take first value | ||||
|         ret = {k: unicode(v[1]).strip() for k, v in ret.items()} | ||||
|  | ||||
|         # track 3/7 => track 3 | ||||
|         for k, v in ret.items(): | ||||
|             if k[0] == ".": | ||||
|                 v = v.split("/")[0].strip().lstrip("0") | ||||
|                 ret[k] = v or 0 | ||||
|  | ||||
|         # normalize key notation to rkeobo | ||||
|         okey = ret.get("key") | ||||
|         if okey: | ||||
|             key = okey.replace(" ", "").replace("maj", "").replace("min", "m") | ||||
|             ret["key"] = REKOBO_LKEY.get(key.lower(), okey) | ||||
|  | ||||
|         return ret | ||||
|  | ||||
|     def compare(self, abspath): | ||||
|         if abspath.endswith(".au"): | ||||
|             return {} | ||||
|  | ||||
|         print("\n" + abspath) | ||||
|         r1 = self.get_mutagen(abspath) | ||||
|         r2 = self.get_ffprobe(abspath) | ||||
|  | ||||
|         keys = {} | ||||
|         for d in [r1, r2]: | ||||
|             for k in d.keys(): | ||||
|                 keys[k] = True | ||||
|  | ||||
|         diffs = [] | ||||
|         l1 = [] | ||||
|         l2 = [] | ||||
|         for k in sorted(keys.keys()): | ||||
|             if k in [".q", ".dur"]: | ||||
|                 continue  # lenient | ||||
|  | ||||
|             v1 = r1.get(k) | ||||
|             v2 = r2.get(k) | ||||
|             if v1 == v2: | ||||
|                 print("  ", k, v1) | ||||
|             elif v1 != "0000":  # FFprobe date=0 | ||||
|                 diffs.append(k) | ||||
|                 print(" 1", k, v1) | ||||
|                 print(" 2", k, v2) | ||||
|                 if v1: | ||||
|                     l1.append(k) | ||||
|                 if v2: | ||||
|                     l2.append(k) | ||||
|  | ||||
|         if diffs: | ||||
|             raise Exception() | ||||
|  | ||||
|         return r1 | ||||
|  | ||||
|     def get_mutagen(self, abspath): | ||||
|         import mutagen | ||||
|  | ||||
|         try: | ||||
|             md = mutagen.File(fsenc(abspath), easy=True) | ||||
|             x = md.info.length | ||||
|         except Exception as ex: | ||||
|             return self.get_ffprobe(abspath) if self.can_ffprobe else {} | ||||
|  | ||||
|         sz = bos.path.getsize(abspath) | ||||
|         ret = {".q": [0, int((sz / md.info.length) / 128)]} | ||||
|  | ||||
|         for attr, k, norm in [ | ||||
|             ["codec", "ac", unicode], | ||||
|             ["channels", "chs", int], | ||||
|             ["sample_rate", ".hz", int], | ||||
|             ["bitrate", ".aq", int], | ||||
|             ["length", ".dur", int], | ||||
|         ]: | ||||
|             try: | ||||
|                 v = getattr(md.info, attr) | ||||
|             except: | ||||
|                 if k != "ac": | ||||
|                     continue | ||||
|  | ||||
|                 try: | ||||
|                     v = str(md.info).split(".")[1] | ||||
|                     if v.startswith("ogg"): | ||||
|                         v = v[3:] | ||||
|                 except: | ||||
|                     continue | ||||
|  | ||||
|             if not v: | ||||
|                 continue | ||||
|  | ||||
|             if k == ".aq": | ||||
|                 v /= 1000 | ||||
|  | ||||
|             if k == "ac" and v.startswith("mp4a.40."): | ||||
|                 v = "aac" | ||||
|  | ||||
|             ret[k] = [0, norm(v)] | ||||
|  | ||||
|         return self.normalize_tags(ret, md) | ||||
|  | ||||
|     def get_ffprobe(self, abspath): | ||||
|         ret, md = ffprobe(abspath) | ||||
|         return self.normalize_tags(ret, md) | ||||
|  | ||||
|     def get_bin(self, parsers, abspath): | ||||
|         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["PYTHONPATH"] = pypath | ||||
|  | ||||
|         ret = {} | ||||
|         for tagname, mp in parsers.items(): | ||||
|             try: | ||||
|                 cmd = [sys.executable, mp.bin, abspath] | ||||
|                 args = {"env": env, "timeout": mp.timeout} | ||||
|  | ||||
|                 if WINDOWS: | ||||
|                     args["creationflags"] = 0x4000 | ||||
|                 else: | ||||
|                     cmd = ["nice"] + cmd | ||||
|  | ||||
|                 cmd = [fsenc(x) for x in cmd] | ||||
|                 v = sp.check_output(cmd, **args).strip() | ||||
|                 if not v: | ||||
|                     continue | ||||
|  | ||||
|                 if "," not in tagname: | ||||
|                     ret[tagname] = v.decode("utf-8") | ||||
|                 else: | ||||
|                     v = json.loads(v) | ||||
|                     for tag in tagname.split(","): | ||||
|                         if tag and tag in v: | ||||
|                             ret[tag] = v[tag] | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         return ret | ||||
							
								
								
									
										100
									
								
								copyparty/star.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								copyparty/star.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import tarfile | ||||
| import threading | ||||
|  | ||||
| from .sutil import errdesc | ||||
| from .util import Queue, fsenc | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| class QFile(object): | ||||
|     """file-like object which buffers writes into a queue""" | ||||
|  | ||||
|     def __init__(self): | ||||
|         self.q = Queue(64) | ||||
|         self.bq = [] | ||||
|         self.nq = 0 | ||||
|  | ||||
|     def write(self, buf): | ||||
|         if buf is None or self.nq >= 240 * 1024: | ||||
|             self.q.put(b"".join(self.bq)) | ||||
|             self.bq = [] | ||||
|             self.nq = 0 | ||||
|  | ||||
|         if buf is None: | ||||
|             self.q.put(None) | ||||
|         else: | ||||
|             self.bq.append(buf) | ||||
|             self.nq += len(buf) | ||||
|  | ||||
|  | ||||
| class StreamTar(object): | ||||
|     """construct in-memory tar file from the given path""" | ||||
|  | ||||
|     def __init__(self, log, fgen, **kwargs): | ||||
|         self.ci = 0 | ||||
|         self.co = 0 | ||||
|         self.qfile = QFile() | ||||
|         self.log = log | ||||
|         self.fgen = fgen | ||||
|         self.errf = None | ||||
|  | ||||
|         # python 3.8 changed to PAX_FORMAT as default, | ||||
|         # waste of space and don't care about the new features | ||||
|         fmt = tarfile.GNU_FORMAT | ||||
|         self.tar = tarfile.open(fileobj=self.qfile, mode="w|", format=fmt) | ||||
|  | ||||
|         w = threading.Thread(target=self._gen, name="star-gen") | ||||
|         w.daemon = True | ||||
|         w.start() | ||||
|  | ||||
|     def gen(self): | ||||
|         while True: | ||||
|             buf = self.qfile.q.get() | ||||
|             if not buf: | ||||
|                 break | ||||
|  | ||||
|             self.co += len(buf) | ||||
|             yield buf | ||||
|  | ||||
|         yield None | ||||
|         if self.errf: | ||||
|             bos.unlink(self.errf["ap"]) | ||||
|  | ||||
|     def ser(self, f): | ||||
|         name = f["vp"] | ||||
|         src = f["ap"] | ||||
|         fsi = f["st"] | ||||
|  | ||||
|         inf = tarfile.TarInfo(name=name) | ||||
|         inf.mode = fsi.st_mode | ||||
|         inf.size = fsi.st_size | ||||
|         inf.mtime = fsi.st_mtime | ||||
|         inf.uid = 0 | ||||
|         inf.gid = 0 | ||||
|  | ||||
|         self.ci += inf.size | ||||
|         with open(fsenc(src), "rb", 512 * 1024) as f: | ||||
|             self.tar.addfile(inf, f) | ||||
|  | ||||
|     def _gen(self): | ||||
|         errors = [] | ||||
|         for f in self.fgen: | ||||
|             if "err" in f: | ||||
|                 errors.append([f["vp"], f["err"]]) | ||||
|                 continue | ||||
|  | ||||
|             try: | ||||
|                 self.ser(f) | ||||
|             except Exception as ex: | ||||
|                 errors.append([f["vp"], repr(ex)]) | ||||
|  | ||||
|         if errors: | ||||
|             self.errf, txt = errdesc(errors) | ||||
|             self.log("\n".join(([repr(self.errf)] + txt[1:]))) | ||||
|             self.ser(self.errf) | ||||
|  | ||||
|         self.tar.close() | ||||
|         self.qfile.write(None) | ||||
| @@ -1,3 +1,5 @@ | ||||
| # coding: utf-8 | ||||
|  | ||||
| """ | ||||
| This is Victor Stinner's pure-Python implementation of PEP 383: the "surrogateescape" error | ||||
| handler of Python 3. | ||||
| @@ -171,7 +173,7 @@ FS_ENCODING = sys.getfilesystemencoding() | ||||
|  | ||||
| if WINDOWS and not PY3: | ||||
|     # py2 thinks win* is mbcs, probably a bug? anyways this works | ||||
|     FS_ENCODING = 'utf-8' | ||||
|     FS_ENCODING = "utf-8" | ||||
|  | ||||
|  | ||||
| # normalize the filesystem encoding name. | ||||
|   | ||||
							
								
								
									
										28
									
								
								copyparty/sutil.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								copyparty/sutil.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import time | ||||
| import tempfile | ||||
| from datetime import datetime | ||||
|  | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| def errdesc(errors): | ||||
|     report = ["copyparty failed to add the following files to the archive:", ""] | ||||
|  | ||||
|     for fn, err in errors: | ||||
|         report.extend([" file: {}".format(fn), "error: {}".format(err), ""]) | ||||
|  | ||||
|     with tempfile.NamedTemporaryFile(prefix="copyparty-", delete=False) as tf: | ||||
|         tf_path = tf.name | ||||
|         tf.write("\r\n".join(report).encode("utf-8", "replace")) | ||||
|  | ||||
|     dt = datetime.utcnow().strftime("%Y-%m%d-%H%M%S") | ||||
|  | ||||
|     bos.chmod(tf_path, 0o444) | ||||
|     return { | ||||
|         "vp": "archive-errors-{}.txt".format(dt), | ||||
|         "ap": tf_path, | ||||
|         "st": bos.stat(tf_path), | ||||
|     }, report | ||||
| @@ -2,16 +2,23 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import shlex | ||||
| import string | ||||
| import signal | ||||
| import socket | ||||
| import threading | ||||
| from datetime import datetime, timedelta | ||||
| import calendar | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, MACOS, VT100 | ||||
| from .__init__ import E, PY2, WINDOWS, ANYWIN, MACOS, VT100, unicode | ||||
| from .util import mp, start_log_thrs, start_stackmon, min_ex, ansi_re | ||||
| from .authsrv import AuthSrv | ||||
| from .tcpsrv import TcpSrv | ||||
| from .up2k import Up2k | ||||
| from .util import mp | ||||
| from .th_srv import ThumbSrv, HAVE_PIL, HAVE_WEBP | ||||
|  | ||||
|  | ||||
| class SvcHub(object): | ||||
| @@ -25,19 +32,51 @@ class SvcHub(object): | ||||
|     put() can return a queue (if want_reply=True) which has a blocking get() with the response. | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, args): | ||||
|     def __init__(self, args, argv, printed): | ||||
|         self.args = args | ||||
|         self.argv = argv | ||||
|         self.logf = None | ||||
|         self.stop_req = False | ||||
|         self.stopping = False | ||||
|         self.stop_cond = threading.Condition() | ||||
|         self.httpsrv_up = 0 | ||||
|  | ||||
|         self.ansi_re = re.compile("\033\\[[^m]*m") | ||||
|         self.log_mutex = threading.Lock() | ||||
|         self.next_day = 0 | ||||
|  | ||||
|         self.log = self._log_disabled if args.q else self._log_enabled | ||||
|         if args.lo: | ||||
|             self._setup_logfile(printed) | ||||
|  | ||||
|         if args.stackmon: | ||||
|             start_stackmon(args.stackmon, 0) | ||||
|  | ||||
|         if args.log_thrs: | ||||
|             start_log_thrs(self.log, args.log_thrs, 0) | ||||
|  | ||||
|         # initiate all services to manage | ||||
|         self.asrv = AuthSrv(self.args, self.log) | ||||
|         if args.ls: | ||||
|             self.asrv.dbg_ls() | ||||
|  | ||||
|         self.tcpsrv = TcpSrv(self) | ||||
|         self.up2k = Up2k(self) | ||||
|  | ||||
|         self.thumbsrv = None | ||||
|         if not args.no_thumb: | ||||
|             if HAVE_PIL: | ||||
|                 if not HAVE_WEBP: | ||||
|                     args.th_no_webp = True | ||||
|                     msg = "setting --th-no-webp because either libwebp is not available or your Pillow is too old" | ||||
|                     self.log("thumb", msg, c=3) | ||||
|  | ||||
|                 self.thumbsrv = ThumbSrv(self) | ||||
|             else: | ||||
|                 msg = "need Pillow to create thumbnails; for example:\n{}{} -m pip install --user Pillow\n" | ||||
|                 self.log( | ||||
|                     "thumb", msg.format(" " * 37, os.path.basename(sys.executable)), c=3 | ||||
|                 ) | ||||
|  | ||||
|         # decide which worker impl to use | ||||
|         if self.check_mp_enable(): | ||||
|             from .broker_mp import BrokerMp as Broker | ||||
| @@ -47,75 +86,233 @@ class SvcHub(object): | ||||
|  | ||||
|         self.broker = Broker(self) | ||||
|  | ||||
|     def run(self): | ||||
|         thr = threading.Thread(target=self.tcpsrv.run) | ||||
|     def thr_httpsrv_up(self): | ||||
|         time.sleep(5) | ||||
|         failed = self.broker.num_workers - self.httpsrv_up | ||||
|         if not failed: | ||||
|             return | ||||
|  | ||||
|         m = "{}/{} workers failed to start" | ||||
|         m = m.format(failed, self.broker.num_workers) | ||||
|         self.log("root", m, 1) | ||||
|         os._exit(1) | ||||
|  | ||||
|     def cb_httpsrv_up(self): | ||||
|         self.httpsrv_up += 1 | ||||
|         if self.httpsrv_up != self.broker.num_workers: | ||||
|             return | ||||
|  | ||||
|         self.log("root", "workers OK\n") | ||||
|         self.up2k.init_vols() | ||||
|  | ||||
|         thr = threading.Thread(target=self.sd_notify, name="sd-notify") | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|         # winxp/py2.7 support: thr.join() kills signals | ||||
|         try: | ||||
|             while True: | ||||
|                 time.sleep(9001) | ||||
|     def _logname(self): | ||||
|         dt = datetime.utcnow() | ||||
|         fn = self.args.lo | ||||
|         for fs in "YmdHMS": | ||||
|             fs = "%" + fs | ||||
|             if fs in fn: | ||||
|                 fn = fn.replace(fs, dt.strftime(fs)) | ||||
|  | ||||
|         except KeyboardInterrupt: | ||||
|         return fn | ||||
|  | ||||
|     def _setup_logfile(self, printed): | ||||
|         base_fn = fn = sel_fn = self._logname() | ||||
|         if fn != self.args.lo: | ||||
|             ctr = 0 | ||||
|             # yup this is a race; if started sufficiently concurrently, two | ||||
|             # copyparties can grab the same logfile (considered and ignored) | ||||
|             while os.path.exists(sel_fn): | ||||
|                 ctr += 1 | ||||
|                 sel_fn = "{}.{}".format(fn, ctr) | ||||
|  | ||||
|         fn = sel_fn | ||||
|  | ||||
|         try: | ||||
|             import lzma | ||||
|  | ||||
|             lh = lzma.open(fn, "wt", encoding="utf-8", errors="replace", preset=0) | ||||
|  | ||||
|         except: | ||||
|             import codecs | ||||
|  | ||||
|             lh = codecs.open(fn, "w", encoding="utf-8", errors="replace") | ||||
|  | ||||
|         lh.base_fn = base_fn | ||||
|  | ||||
|         argv = [sys.executable] + self.argv | ||||
|         if hasattr(shlex, "quote"): | ||||
|             argv = [shlex.quote(x) for x in argv] | ||||
|         else: | ||||
|             argv = ['"{}"'.format(x) for x in argv] | ||||
|  | ||||
|         msg = "[+] opened logfile [{}]\n".format(fn) | ||||
|         printed += msg | ||||
|         lh.write("t0: {:.3f}\nargv: {}\n\n{}".format(E.t0, " ".join(argv), printed)) | ||||
|         self.logf = lh | ||||
|         print(msg, end="") | ||||
|  | ||||
|     def run(self): | ||||
|         self.tcpsrv.run() | ||||
|  | ||||
|         thr = threading.Thread(target=self.thr_httpsrv_up) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|         for sig in [signal.SIGINT, signal.SIGTERM]: | ||||
|             signal.signal(sig, self.signal_handler) | ||||
|  | ||||
|         # macos hangs after shutdown on sigterm with while-sleep, | ||||
|         # windows cannot ^c stop_cond (and win10 does the macos thing but winxp is fine??) | ||||
|         # linux is fine with both, | ||||
|         # never lucky | ||||
|         if ANYWIN: | ||||
|             # msys-python probably fine but >msys-python | ||||
|             thr = threading.Thread(target=self.stop_thr, name="svchub-sig") | ||||
|             thr.daemon = True | ||||
|             thr.start() | ||||
|  | ||||
|             try: | ||||
|                 while not self.stop_req: | ||||
|                     time.sleep(1) | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|             self.shutdown() | ||||
|             thr.join() | ||||
|         else: | ||||
|             self.stop_thr() | ||||
|  | ||||
|     def stop_thr(self): | ||||
|         while not self.stop_req: | ||||
|             with self.stop_cond: | ||||
|                 self.stop_cond.wait(9001) | ||||
|  | ||||
|         self.shutdown() | ||||
|  | ||||
|     def signal_handler(self, sig, frame): | ||||
|         if self.stopping: | ||||
|             return | ||||
|  | ||||
|         self.stop_req = True | ||||
|         with self.stop_cond: | ||||
|             self.stop_cond.notify_all() | ||||
|  | ||||
|     def shutdown(self): | ||||
|         if self.stopping: | ||||
|             return | ||||
|  | ||||
|         self.stopping = True | ||||
|         self.stop_req = True | ||||
|         with self.stop_cond: | ||||
|             self.stop_cond.notify_all() | ||||
|  | ||||
|         ret = 1 | ||||
|         try: | ||||
|             with self.log_mutex: | ||||
|                 print("OPYTHAT") | ||||
|  | ||||
|             self.tcpsrv.shutdown() | ||||
|             self.broker.shutdown() | ||||
|             print("nailed it") | ||||
|             self.up2k.shutdown() | ||||
|             if self.thumbsrv: | ||||
|                 self.thumbsrv.shutdown() | ||||
|  | ||||
|     def _log_disabled(self, src, msg): | ||||
|         pass | ||||
|                 for n in range(200):  # 10s | ||||
|                     time.sleep(0.05) | ||||
|                     if self.thumbsrv.stopped(): | ||||
|                         break | ||||
|  | ||||
|     def _log_enabled(self, src, msg): | ||||
|                     if n == 3: | ||||
|                         print("waiting for thumbsrv (10sec)...") | ||||
|  | ||||
|             print("nailed it", end="") | ||||
|             ret = 0 | ||||
|         finally: | ||||
|             print("\033[0m") | ||||
|             if self.logf: | ||||
|                 self.logf.close() | ||||
|  | ||||
|             sys.exit(ret) | ||||
|  | ||||
|     def _log_disabled(self, src, msg, c=0): | ||||
|         if not self.logf: | ||||
|             return | ||||
|  | ||||
|         with self.log_mutex: | ||||
|             ts = datetime.utcnow().strftime("%Y-%m%d-%H%M%S.%f")[:-3] | ||||
|             self.logf.write("@{} [{}] {}\n".format(ts, src, msg)) | ||||
|  | ||||
|             now = time.time() | ||||
|             if now >= self.next_day: | ||||
|                 self._set_next_day() | ||||
|  | ||||
|     def _set_next_day(self): | ||||
|         if self.next_day and self.logf and self.logf.base_fn != self._logname(): | ||||
|             self.logf.close() | ||||
|             self._setup_logfile("") | ||||
|  | ||||
|         dt = datetime.utcnow() | ||||
|  | ||||
|         # unix timestamp of next 00:00:00 (leap-seconds safe) | ||||
|         day_now = dt.day | ||||
|         while dt.day == day_now: | ||||
|             dt += timedelta(hours=12) | ||||
|  | ||||
|         dt = dt.replace(hour=0, minute=0, second=0) | ||||
|         self.next_day = calendar.timegm(dt.utctimetuple()) | ||||
|  | ||||
|     def _log_enabled(self, src, msg, c=0): | ||||
|         """handles logging from all components""" | ||||
|         with self.log_mutex: | ||||
|             now = time.time() | ||||
|             if now >= self.next_day: | ||||
|                 dt = datetime.utcfromtimestamp(now) | ||||
|                 print("\033[36m{}\033[0m".format(dt.strftime("%Y-%m-%d"))) | ||||
|                 print("\033[36m{}\033[0m\n".format(dt.strftime("%Y-%m-%d")), end="") | ||||
|                 self._set_next_day() | ||||
|  | ||||
|                 # unix timestamp of next 00:00:00 (leap-seconds safe) | ||||
|                 day_now = dt.day | ||||
|                 while dt.day == day_now: | ||||
|                     dt += timedelta(hours=12) | ||||
|  | ||||
|                 dt = dt.replace(hour=0, minute=0, second=0) | ||||
|                 self.next_day = calendar.timegm(dt.utctimetuple()) | ||||
|  | ||||
|             fmt = "\033[36m{} \033[33m{:21} \033[0m{}" | ||||
|             fmt = "\033[36m{} \033[33m{:21} \033[0m{}\n" | ||||
|             if not VT100: | ||||
|                 fmt = "{} {:21} {}" | ||||
|                 fmt = "{} {:21} {}\n" | ||||
|                 if "\033" in msg: | ||||
|                     msg = self.ansi_re.sub("", msg) | ||||
|                     msg = ansi_re.sub("", msg) | ||||
|                 if "\033" in src: | ||||
|                     src = self.ansi_re.sub("", src) | ||||
|                     src = ansi_re.sub("", src) | ||||
|             elif c: | ||||
|                 if isinstance(c, int): | ||||
|                     msg = "\033[3{}m{}".format(c, msg) | ||||
|                 elif "\033" not in c: | ||||
|                     msg = "\033[{}m{}\033[0m".format(c, msg) | ||||
|                 else: | ||||
|                     msg = "{}{}\033[0m".format(c, msg) | ||||
|  | ||||
|             ts = datetime.utcfromtimestamp(now).strftime("%H:%M:%S.%f")[:-3] | ||||
|             msg = fmt.format(ts, src, msg) | ||||
|             try: | ||||
|                 print(msg) | ||||
|                 print(msg, end="") | ||||
|             except UnicodeEncodeError: | ||||
|                 try: | ||||
|                     print(msg.encode("utf-8", "replace").decode()) | ||||
|                     print(msg.encode("utf-8", "replace").decode(), end="") | ||||
|                 except: | ||||
|                     print(msg.encode("ascii", "replace").decode()) | ||||
|                     print(msg.encode("ascii", "replace").decode(), end="") | ||||
|  | ||||
|             if self.logf: | ||||
|                 self.logf.write(msg) | ||||
|  | ||||
|     def check_mp_support(self): | ||||
|         vmin = sys.version_info[1] | ||||
|         if WINDOWS: | ||||
|             msg = "need python 3.3 or newer for multiprocessing;" | ||||
|             if PY2: | ||||
|                 # py2 pickler doesn't support winsock | ||||
|                 return msg | ||||
|             elif vmin < 3: | ||||
|             if PY2 or vmin < 3: | ||||
|                 return msg | ||||
|         elif MACOS: | ||||
|             return "multiprocessing is wonky on mac osx;" | ||||
|         else: | ||||
|             msg = "need python 2.7 or 3.3+ for multiprocessing;" | ||||
|             if not PY2 and vmin < 3: | ||||
|             msg = "need python 3.3+ for multiprocessing;" | ||||
|             if PY2 or vmin < 3: | ||||
|                 return msg | ||||
|  | ||||
|         try: | ||||
| @@ -147,5 +344,24 @@ class SvcHub(object): | ||||
|         if not err: | ||||
|             return True | ||||
|         else: | ||||
|             self.log("root", err) | ||||
|             self.log("svchub", err) | ||||
|             return False | ||||
|  | ||||
|     def sd_notify(self): | ||||
|         try: | ||||
|             addr = os.getenv("NOTIFY_SOCKET") | ||||
|             if not addr: | ||||
|                 return | ||||
|  | ||||
|             addr = unicode(addr) | ||||
|             if addr.startswith("@"): | ||||
|                 addr = "\0" + addr[1:] | ||||
|  | ||||
|             m = "".join(x for x in addr if x in string.printable) | ||||
|             self.log("sd_notify", m) | ||||
|  | ||||
|             sck = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM) | ||||
|             sck.connect(addr) | ||||
|             sck.sendall(b"READY=1") | ||||
|         except: | ||||
|             self.log("sd_notify", min_ex()) | ||||
|   | ||||
							
								
								
									
										275
									
								
								copyparty/szip.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										275
									
								
								copyparty/szip.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,275 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import time | ||||
| import zlib | ||||
| from datetime import datetime | ||||
|  | ||||
| from .sutil import errdesc | ||||
| from .util import yieldfile, sanitize_fn, spack, sunpack | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| def dostime2unix(buf): | ||||
|     t, d = sunpack(b"<HH", buf) | ||||
|  | ||||
|     ts = (t & 0x1F) * 2 | ||||
|     tm = (t >> 5) & 0x3F | ||||
|     th = t >> 11 | ||||
|  | ||||
|     dd = d & 0x1F | ||||
|     dm = (d >> 5) & 0xF | ||||
|     dy = (d >> 9) + 1980 | ||||
|  | ||||
|     tt = (dy, dm, dd, th, tm, ts) | ||||
|     tf = "{:04d}-{:02d}-{:02d} {:02d}:{:02d}:{:02d}" | ||||
|     iso = tf.format(*tt) | ||||
|  | ||||
|     dt = datetime.strptime(iso, "%Y-%m-%d %H:%M:%S") | ||||
|     return int(dt.timestamp()) | ||||
|  | ||||
|  | ||||
| def unixtime2dos(ts): | ||||
|     tt = time.gmtime(ts) | ||||
|     dy, dm, dd, th, tm, ts = list(tt)[:6] | ||||
|  | ||||
|     bd = ((dy - 1980) << 9) + (dm << 5) + dd | ||||
|     bt = (th << 11) + (tm << 5) + ts // 2 | ||||
|     return spack(b"<HH", bt, bd) | ||||
|  | ||||
|  | ||||
| def gen_fdesc(sz, crc32, z64): | ||||
|     ret = b"\x50\x4b\x07\x08" | ||||
|     fmt = b"<LQQ" if z64 else b"<LLL" | ||||
|     ret += spack(fmt, crc32, sz, sz) | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def gen_hdr(h_pos, fn, sz, lastmod, utf8, crc32, pre_crc): | ||||
|     """ | ||||
|     does regular file headers | ||||
|     and the central directory meme if h_pos is set | ||||
|     (h_pos = absolute position of the regular header) | ||||
|     """ | ||||
|  | ||||
|     # appnote 4.5 / zip 3.0 (2008) / unzip 6.0 (2009) says to add z64 | ||||
|     # extinfo for values which exceed H, but that becomes an off-by-one | ||||
|     # (can't tell if it was clamped or exactly maxval), make it obvious | ||||
|     z64 = sz >= 0xFFFFFFFF | ||||
|     z64v = [sz, sz] if z64 else [] | ||||
|     if h_pos and h_pos >= 0xFFFFFFFF: | ||||
|         # central, also consider ptr to original header | ||||
|         z64v.append(h_pos) | ||||
|  | ||||
|     # confusingly this doesn't bump if h_pos | ||||
|     req_ver = b"\x2d\x00" if z64 else b"\x0a\x00" | ||||
|  | ||||
|     if crc32: | ||||
|         crc32 = spack(b"<L", crc32) | ||||
|     else: | ||||
|         crc32 = b"\x00" * 4 | ||||
|  | ||||
|     if h_pos is None: | ||||
|         # 4b magic, 2b min-ver | ||||
|         ret = b"\x50\x4b\x03\x04" + req_ver | ||||
|     else: | ||||
|         # 4b magic, 2b spec-ver, 2b min-ver | ||||
|         ret = b"\x50\x4b\x01\x02\x1e\x03" + req_ver | ||||
|  | ||||
|     ret += b"\x00" if pre_crc else b"\x08"  # streaming | ||||
|     ret += b"\x08" if utf8 else b"\x00"  # appnote 6.3.2 (2007) | ||||
|  | ||||
|     # 2b compression, 4b time, 4b crc | ||||
|     ret += b"\x00\x00" + unixtime2dos(lastmod) + crc32 | ||||
|  | ||||
|     # spec says to put zeros when !crc if bit3 (streaming) | ||||
|     # however infozip does actual sz and it even works on winxp | ||||
|     # (same reasning for z64 extradata later) | ||||
|     vsz = 0xFFFFFFFF if z64 else sz | ||||
|     ret += spack(b"<LL", vsz, vsz) | ||||
|  | ||||
|     # windows support (the "?" replace below too) | ||||
|     fn = sanitize_fn(fn, "/", []) | ||||
|     bfn = fn.encode("utf-8" if utf8 else "cp437", "replace").replace(b"?", b"_") | ||||
|  | ||||
|     z64_len = len(z64v) * 8 + 4 if z64v else 0 | ||||
|     ret += spack(b"<HH", len(bfn), z64_len) | ||||
|  | ||||
|     if h_pos is not None: | ||||
|         # 2b comment, 2b diskno | ||||
|         ret += b"\x00" * 4 | ||||
|  | ||||
|         # 2b internal.attr, 4b external.attr | ||||
|         # infozip-macos: 0100 0000 a481 file:644 | ||||
|         # infozip-macos: 0100 0100 0080 file:000 | ||||
|         ret += b"\x01\x00\x00\x00\xa4\x81" | ||||
|  | ||||
|         # 4b local-header-ofs | ||||
|         ret += spack(b"<L", min(h_pos, 0xFFFFFFFF)) | ||||
|  | ||||
|     ret += bfn | ||||
|  | ||||
|     if z64v: | ||||
|         ret += spack(b"<HH" + b"Q" * len(z64v), 1, len(z64v) * 8, *z64v) | ||||
|  | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def gen_ecdr(items, cdir_pos, cdir_end): | ||||
|     """ | ||||
|     summary of all file headers, | ||||
|     usually the zipfile footer unless something clamps | ||||
|     """ | ||||
|  | ||||
|     ret = b"\x50\x4b\x05\x06" | ||||
|  | ||||
|     # 2b ndisk, 2b disk0 | ||||
|     ret += b"\x00" * 4 | ||||
|  | ||||
|     cdir_sz = cdir_end - cdir_pos | ||||
|  | ||||
|     nitems = min(0xFFFF, len(items)) | ||||
|     csz = min(0xFFFFFFFF, cdir_sz) | ||||
|     cpos = min(0xFFFFFFFF, cdir_pos) | ||||
|  | ||||
|     need_64 = nitems == 0xFFFF or 0xFFFFFFFF in [csz, cpos] | ||||
|  | ||||
|     # 2b tnfiles, 2b dnfiles, 4b dir sz, 4b dir pos | ||||
|     ret += spack(b"<HHLL", nitems, nitems, csz, cpos) | ||||
|  | ||||
|     # 2b comment length | ||||
|     ret += b"\x00\x00" | ||||
|  | ||||
|     return [ret, need_64] | ||||
|  | ||||
|  | ||||
| def gen_ecdr64(items, cdir_pos, cdir_end): | ||||
|     """ | ||||
|     z64 end of central directory | ||||
|     added when numfiles or a headerptr clamps | ||||
|     """ | ||||
|  | ||||
|     ret = b"\x50\x4b\x06\x06" | ||||
|  | ||||
|     # 8b own length from hereon | ||||
|     ret += b"\x2c" + b"\x00" * 7 | ||||
|  | ||||
|     # 2b spec-ver, 2b min-ver | ||||
|     ret += b"\x1e\x03\x2d\x00" | ||||
|  | ||||
|     # 4b ndisk, 4b disk0 | ||||
|     ret += b"\x00" * 8 | ||||
|  | ||||
|     # 8b tnfiles, 8b dnfiles, 8b dir sz, 8b dir pos | ||||
|     cdir_sz = cdir_end - cdir_pos | ||||
|     ret += spack(b"<QQQQ", len(items), len(items), cdir_sz, cdir_pos) | ||||
|  | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def gen_ecdr64_loc(ecdr64_pos): | ||||
|     """ | ||||
|     z64 end of central directory locator | ||||
|     points to ecdr64 | ||||
|     why | ||||
|     """ | ||||
|  | ||||
|     ret = b"\x50\x4b\x06\x07" | ||||
|  | ||||
|     # 4b cdisk, 8b start of ecdr64, 4b ndisks | ||||
|     ret += spack(b"<LQL", 0, ecdr64_pos, 1) | ||||
|  | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| class StreamZip(object): | ||||
|     def __init__(self, log, fgen, utf8=False, pre_crc=False): | ||||
|         self.log = log | ||||
|         self.fgen = fgen | ||||
|         self.utf8 = utf8 | ||||
|         self.pre_crc = pre_crc | ||||
|  | ||||
|         self.pos = 0 | ||||
|         self.items = [] | ||||
|  | ||||
|     def _ct(self, buf): | ||||
|         self.pos += len(buf) | ||||
|         return buf | ||||
|  | ||||
|     def ser(self, f): | ||||
|         name = f["vp"] | ||||
|         src = f["ap"] | ||||
|         st = f["st"] | ||||
|  | ||||
|         sz = st.st_size | ||||
|         ts = st.st_mtime + 1 | ||||
|  | ||||
|         crc = None | ||||
|         if self.pre_crc: | ||||
|             crc = 0 | ||||
|             for buf in yieldfile(src): | ||||
|                 crc = zlib.crc32(buf, crc) | ||||
|  | ||||
|             crc &= 0xFFFFFFFF | ||||
|  | ||||
|         h_pos = self.pos | ||||
|         buf = gen_hdr(None, name, sz, ts, self.utf8, crc, self.pre_crc) | ||||
|         yield self._ct(buf) | ||||
|  | ||||
|         crc = crc or 0 | ||||
|         for buf in yieldfile(src): | ||||
|             if not self.pre_crc: | ||||
|                 crc = zlib.crc32(buf, crc) | ||||
|  | ||||
|             yield self._ct(buf) | ||||
|  | ||||
|         crc &= 0xFFFFFFFF | ||||
|  | ||||
|         self.items.append([name, sz, ts, crc, h_pos]) | ||||
|  | ||||
|         z64 = sz >= 4 * 1024 * 1024 * 1024 | ||||
|  | ||||
|         if z64 or not self.pre_crc: | ||||
|             buf = gen_fdesc(sz, crc, z64) | ||||
|             yield self._ct(buf) | ||||
|  | ||||
|     def gen(self): | ||||
|         errors = [] | ||||
|         for f in self.fgen: | ||||
|             if "err" in f: | ||||
|                 errors.append([f["vp"], f["err"]]) | ||||
|                 continue | ||||
|  | ||||
|             try: | ||||
|                 for x in self.ser(f): | ||||
|                     yield x | ||||
|             except Exception as ex: | ||||
|                 errors.append([f["vp"], repr(ex)]) | ||||
|  | ||||
|         if errors: | ||||
|             errf, txt = errdesc(errors) | ||||
|             self.log("\n".join(([repr(errf)] + txt[1:]))) | ||||
|             for x in self.ser(errf): | ||||
|                 yield x | ||||
|  | ||||
|         cdir_pos = self.pos | ||||
|         for name, sz, ts, crc, h_pos in self.items: | ||||
|             buf = gen_hdr(h_pos, name, sz, ts, self.utf8, crc, self.pre_crc) | ||||
|             yield self._ct(buf) | ||||
|         cdir_end = self.pos | ||||
|  | ||||
|         _, need_64 = gen_ecdr(self.items, cdir_pos, cdir_end) | ||||
|         if need_64: | ||||
|             ecdir64_pos = self.pos | ||||
|             buf = gen_ecdr64(self.items, cdir_pos, cdir_end) | ||||
|             yield self._ct(buf) | ||||
|  | ||||
|             buf = gen_ecdr64_loc(ecdir64_pos) | ||||
|             yield self._ct(buf) | ||||
|  | ||||
|         ecdr, _ = gen_ecdr(self.items, cdir_pos, cdir_end) | ||||
|         yield self._ct(ecdr) | ||||
|  | ||||
|         if errors: | ||||
|             bos.unlink(errf["ap"]) | ||||
| @@ -2,10 +2,10 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import time | ||||
| import socket | ||||
|  | ||||
| from .util import chkcmd, Counter | ||||
| from .__init__ import MACOS, ANYWIN | ||||
| from .util import chkcmd | ||||
|  | ||||
|  | ||||
| class TcpSrv(object): | ||||
| @@ -19,76 +19,165 @@ class TcpSrv(object): | ||||
|         self.args = hub.args | ||||
|         self.log = hub.log | ||||
|  | ||||
|         self.num_clients = Counter() | ||||
|         self.stopping = False | ||||
|  | ||||
|         ip = "127.0.0.1" | ||||
|         eps = {ip: "local only"} | ||||
|         if self.args.i != ip: | ||||
|             eps = self.detect_interfaces(self.args.i) or {self.args.i: "external"} | ||||
|         nonlocals = [x for x in self.args.i if x != ip] | ||||
|         if nonlocals: | ||||
|             eps = self.detect_interfaces(self.args.i) | ||||
|             if not eps: | ||||
|                 for x in nonlocals: | ||||
|                     eps[x] = "external" | ||||
|  | ||||
|         msgs = [] | ||||
|         m = "available @ http://{}:{}/  (\033[33m{}\033[0m)" | ||||
|         for ip, desc in sorted(eps.items(), key=lambda x: x[1]): | ||||
|             self.log( | ||||
|                 "tcpsrv", | ||||
|                 "available @ http://{}:{}/  (\033[33m{}\033[0m)".format( | ||||
|                     ip, self.args.p, desc | ||||
|                 ), | ||||
|             ) | ||||
|             for port in sorted(self.args.p): | ||||
|                 msgs.append(m.format(ip, port, desc)) | ||||
|  | ||||
|         self.srv = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | ||||
|         self.srv.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | ||||
|         if msgs: | ||||
|             msgs[-1] += "\n" | ||||
|             for m in msgs: | ||||
|                 self.log("tcpsrv", m) | ||||
|  | ||||
|         self.srv = [] | ||||
|         for ip in self.args.i: | ||||
|             for port in self.args.p: | ||||
|                 self.srv.append(self._listen(ip, port)) | ||||
|  | ||||
|     def _listen(self, ip, port): | ||||
|         srv = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | ||||
|         srv.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | ||||
|         srv.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1) | ||||
|         try: | ||||
|             self.srv.bind((self.args.i, self.args.p)) | ||||
|             srv.bind((ip, port)) | ||||
|             return srv | ||||
|         except (OSError, socket.error) as ex: | ||||
|             if ex.errno == 98: | ||||
|                 raise Exception( | ||||
|                     "\033[1;31mport {} is busy on interface {}\033[0m".format( | ||||
|                         self.args.p, self.args.i | ||||
|                     ) | ||||
|                 ) | ||||
|  | ||||
|             if ex.errno == 99: | ||||
|                 raise Exception( | ||||
|                     "\033[1;31minterface {} does not exist\033[0m".format(self.args.i) | ||||
|                 ) | ||||
|             if ex.errno in [98, 48]: | ||||
|                 e = "\033[1;31mport {} is busy on interface {}\033[0m".format(port, ip) | ||||
|             elif ex.errno in [99, 49]: | ||||
|                 e = "\033[1;31minterface {} does not exist\033[0m".format(ip) | ||||
|             else: | ||||
|                 raise | ||||
|             raise Exception(e) | ||||
|  | ||||
|     def run(self): | ||||
|         self.srv.listen(self.args.nc) | ||||
|         for srv in self.srv: | ||||
|             srv.listen(self.args.nc) | ||||
|             ip, port = srv.getsockname() | ||||
|             fno = srv.fileno() | ||||
|             msg = "listening @ {}:{}  f{}".format(ip, port, fno) | ||||
|             self.log("tcpsrv", msg) | ||||
|             if self.args.q: | ||||
|                 print(msg) | ||||
|  | ||||
|         self.log("tcpsrv", "listening @ {0}:{1}".format(self.args.i, self.args.p)) | ||||
|  | ||||
|         while True: | ||||
|             self.log("tcpsrv", "-" * 1 + "C-ncli") | ||||
|             if self.num_clients.v >= self.args.nc: | ||||
|                 time.sleep(0.1) | ||||
|                 continue | ||||
|  | ||||
|             self.log("tcpsrv", "-" * 2 + "C-acc1") | ||||
|             sck, addr = self.srv.accept() | ||||
|             self.log("%s %s" % addr, "-" * 3 + "C-acc2") | ||||
|             self.num_clients.add() | ||||
|             self.hub.broker.put(False, "httpconn", sck, addr) | ||||
|             self.hub.broker.put(False, "listen", srv) | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.stopping = True | ||||
|         try: | ||||
|             for srv in self.srv: | ||||
|                 srv.close() | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         self.log("tcpsrv", "ok bye") | ||||
|  | ||||
|     def detect_interfaces(self, listen_ip): | ||||
|     def ips_linux(self): | ||||
|         eps = {} | ||||
|  | ||||
|         # get all ips and their interfaces | ||||
|         try: | ||||
|             ip_addr, _ = chkcmd("ip", "addr") | ||||
|             txt, _ = chkcmd(["ip", "addr"]) | ||||
|         except: | ||||
|             ip_addr = None | ||||
|             return eps | ||||
|  | ||||
|         if ip_addr: | ||||
|             r = re.compile(r"^\s+inet ([^ ]+)/.* (.*)") | ||||
|             for ln in ip_addr.split("\n"): | ||||
|                 try: | ||||
|                     ip, dev = r.match(ln.rstrip()).groups() | ||||
|                     if listen_ip in ["0.0.0.0", ip]: | ||||
|                         eps[ip] = dev | ||||
|                 except: | ||||
|                     pass | ||||
|         r = re.compile(r"^\s+inet ([^ ]+)/.* (.*)") | ||||
|         for ln in txt.split("\n"): | ||||
|             try: | ||||
|                 ip, dev = r.match(ln.rstrip()).groups() | ||||
|                 eps[ip] = dev | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def ips_macos(self): | ||||
|         eps = {} | ||||
|         try: | ||||
|             txt, _ = chkcmd(["ifconfig"]) | ||||
|         except: | ||||
|             return eps | ||||
|  | ||||
|         rdev = re.compile(r"^([^ ]+):") | ||||
|         rip = re.compile(r"^\tinet ([0-9\.]+) ") | ||||
|         dev = None | ||||
|         for ln in txt.split("\n"): | ||||
|             m = rdev.match(ln) | ||||
|             if m: | ||||
|                 dev = m.group(1) | ||||
|  | ||||
|             m = rip.match(ln) | ||||
|             if m: | ||||
|                 eps[m.group(1)] = dev | ||||
|                 dev = None | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def ips_windows_ipconfig(self): | ||||
|         eps = {} | ||||
|         try: | ||||
|             txt, _ = chkcmd(["ipconfig"]) | ||||
|         except: | ||||
|             return eps | ||||
|  | ||||
|         rdev = re.compile(r"(^[^ ].*):$") | ||||
|         rip = re.compile(r"^ +IPv?4? [^:]+: *([0-9\.]{7,15})$") | ||||
|         dev = None | ||||
|         for ln in txt.replace("\r", "").split("\n"): | ||||
|             m = rdev.match(ln) | ||||
|             if m: | ||||
|                 dev = m.group(1).split(" adapter ", 1)[-1] | ||||
|  | ||||
|             m = rip.match(ln) | ||||
|             if m and dev: | ||||
|                 eps[m.group(1)] = dev | ||||
|                 dev = None | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def ips_windows_netsh(self): | ||||
|         eps = {} | ||||
|         try: | ||||
|             txt, _ = chkcmd("netsh interface ip show address".split()) | ||||
|         except: | ||||
|             return eps | ||||
|  | ||||
|         rdev = re.compile(r'.* "([^"]+)"$') | ||||
|         rip = re.compile(r".* IP\b.*: +([0-9\.]{7,15})$") | ||||
|         dev = None | ||||
|         for ln in txt.replace("\r", "").split("\n"): | ||||
|             m = rdev.match(ln) | ||||
|             if m: | ||||
|                 dev = m.group(1) | ||||
|  | ||||
|             m = rip.match(ln) | ||||
|             if m and dev: | ||||
|                 eps[m.group(1)] = dev | ||||
|                 dev = None | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def detect_interfaces(self, listen_ips): | ||||
|         if MACOS: | ||||
|             eps = self.ips_macos() | ||||
|         elif ANYWIN: | ||||
|             eps = self.ips_windows_ipconfig()  # sees more interfaces | ||||
|             eps.update(self.ips_windows_netsh())  # has better names | ||||
|         else: | ||||
|             eps = self.ips_linux() | ||||
|  | ||||
|         if "0.0.0.0" not in listen_ips: | ||||
|             eps = {k: v for k, v in eps.items() if k in listen_ips} | ||||
|  | ||||
|         default_route = None | ||||
|         s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) | ||||
| @@ -113,11 +202,12 @@ class TcpSrv(object): | ||||
|  | ||||
|         s.close() | ||||
|  | ||||
|         if default_route and listen_ip in ["0.0.0.0", default_route]: | ||||
|             desc = "\033[32mexternal" | ||||
|             try: | ||||
|                 eps[default_route] += ", " + desc | ||||
|             except: | ||||
|                 eps[default_route] = desc | ||||
|         for lip in listen_ips: | ||||
|             if default_route and lip in ["0.0.0.0", default_route]: | ||||
|                 desc = "\033[32mexternal" | ||||
|                 try: | ||||
|                     eps[default_route] += ", " + desc | ||||
|                 except: | ||||
|                     eps[default_route] = desc | ||||
|  | ||||
|         return eps | ||||
|   | ||||
							
								
								
									
										59
									
								
								copyparty/th_cli.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								copyparty/th_cli.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
|  | ||||
| from .util import Cooldown | ||||
| from .th_srv import thumb_path, THUMBABLE, FMT_FF | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| class ThumbCli(object): | ||||
|     def __init__(self, broker): | ||||
|         self.broker = broker | ||||
|         self.args = broker.args | ||||
|         self.asrv = broker.asrv | ||||
|  | ||||
|         # cache on both sides for less broker spam | ||||
|         self.cooldown = Cooldown(self.args.th_poke) | ||||
|  | ||||
|     def get(self, ptop, rem, mtime, fmt): | ||||
|         ext = rem.rsplit(".")[-1].lower() | ||||
|         if ext not in THUMBABLE: | ||||
|             return None | ||||
|  | ||||
|         is_vid = ext in FMT_FF | ||||
|         if is_vid and self.args.no_vthumb: | ||||
|             return None | ||||
|  | ||||
|         if rem.startswith(".hist/th/") and rem.split(".")[-1] in ["webp", "jpg"]: | ||||
|             return os.path.join(ptop, rem) | ||||
|  | ||||
|         if fmt == "j" and self.args.th_no_jpg: | ||||
|             fmt = "w" | ||||
|  | ||||
|         if fmt == "w": | ||||
|             if self.args.th_no_webp or (is_vid and self.args.th_ff_jpg): | ||||
|                 fmt = "j" | ||||
|  | ||||
|         histpath = self.asrv.vfs.histtab[ptop] | ||||
|         tpath = thumb_path(histpath, rem, mtime, fmt) | ||||
|         ret = None | ||||
|         try: | ||||
|             st = bos.stat(tpath) | ||||
|             if st.st_size: | ||||
|                 ret = tpath | ||||
|             else: | ||||
|                 return None | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         if ret: | ||||
|             tdir = os.path.dirname(tpath) | ||||
|             if self.cooldown.poke(tdir): | ||||
|                 self.broker.put(False, "thumbsrv.poke", tdir) | ||||
|  | ||||
|             return ret | ||||
|  | ||||
|         x = self.broker.put(True, "thumbsrv.get", ptop, rem, mtime, fmt) | ||||
|         return x.get() | ||||
							
								
								
									
										424
									
								
								copyparty/th_srv.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										424
									
								
								copyparty/th_srv.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,424 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import time | ||||
| import shutil | ||||
| import base64 | ||||
| import hashlib | ||||
| import threading | ||||
| import subprocess as sp | ||||
|  | ||||
| from .__init__ import PY2, unicode | ||||
| from .util import fsenc, vsplit, runcmd, Queue, Cooldown, BytesIO, min_ex | ||||
| from .bos import bos | ||||
| from .mtag import HAVE_FFMPEG, HAVE_FFPROBE, ffprobe | ||||
|  | ||||
|  | ||||
| HAVE_PIL = False | ||||
| HAVE_HEIF = False | ||||
| HAVE_AVIF = False | ||||
| HAVE_WEBP = False | ||||
|  | ||||
| try: | ||||
|     from PIL import Image, ImageOps, ExifTags | ||||
|  | ||||
|     HAVE_PIL = True | ||||
|     try: | ||||
|         Image.new("RGB", (2, 2)).save(BytesIO(), format="webp") | ||||
|         HAVE_WEBP = True | ||||
|     except: | ||||
|         pass | ||||
|  | ||||
|     try: | ||||
|         from pyheif_pillow_opener import register_heif_opener | ||||
|  | ||||
|         register_heif_opener() | ||||
|         HAVE_HEIF = True | ||||
|     except: | ||||
|         pass | ||||
|  | ||||
|     try: | ||||
|         import pillow_avif | ||||
|  | ||||
|         HAVE_AVIF = True | ||||
|     except: | ||||
|         pass | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| # https://pillow.readthedocs.io/en/stable/handbook/image-file-formats.html | ||||
| # ffmpeg -formats | ||||
| FMT_PIL = "bmp dib gif icns ico jpg jpeg jp2 jpx pcx png pbm pgm ppm pnm sgi tga tif tiff webp xbm dds xpm" | ||||
| FMT_FF = "av1 asf avi flv m4v mkv mjpeg mjpg mpg mpeg mpg2 mpeg2 h264 avc mts h265 hevc mov 3gp mp4 ts mpegts nut ogv ogm rm vob webm wmv" | ||||
|  | ||||
| if HAVE_HEIF: | ||||
|     FMT_PIL += " heif heifs heic heics" | ||||
|  | ||||
| if HAVE_AVIF: | ||||
|     FMT_PIL += " avif avifs" | ||||
|  | ||||
| FMT_PIL, FMT_FF = [{x: True for x in y.split(" ") if x} for y in [FMT_PIL, FMT_FF]] | ||||
|  | ||||
|  | ||||
| THUMBABLE = {} | ||||
|  | ||||
| if HAVE_PIL: | ||||
|     THUMBABLE.update(FMT_PIL) | ||||
|  | ||||
| if HAVE_FFMPEG and HAVE_FFPROBE: | ||||
|     THUMBABLE.update(FMT_FF) | ||||
|  | ||||
|  | ||||
| def thumb_path(histpath, rem, mtime, fmt): | ||||
|     # base16 = 16 = 256 | ||||
|     # b64-lc = 38 = 1444 | ||||
|     # base64 = 64 = 4096 | ||||
|     rd, fn = vsplit(rem) | ||||
|     if rd: | ||||
|         h = hashlib.sha512(fsenc(rd)).digest() | ||||
|         b64 = base64.urlsafe_b64encode(h).decode("ascii")[:24] | ||||
|         rd = "{}/{}/".format(b64[:2], b64[2:4]).lower() + b64 | ||||
|     else: | ||||
|         rd = "top" | ||||
|  | ||||
|     # could keep original filenames but this is safer re pathlen | ||||
|     h = hashlib.sha512(fsenc(fn)).digest() | ||||
|     fn = base64.urlsafe_b64encode(h).decode("ascii")[:24] | ||||
|  | ||||
|     return "{}/th/{}/{}.{:x}.{}".format( | ||||
|         histpath, rd, fn, int(mtime), "webp" if fmt == "w" else "jpg" | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class ThumbSrv(object): | ||||
|     def __init__(self, hub): | ||||
|         self.hub = hub | ||||
|         self.asrv = hub.asrv | ||||
|         self.args = hub.args | ||||
|         self.log_func = hub.log | ||||
|  | ||||
|         res = hub.args.th_size.split("x") | ||||
|         self.res = tuple([int(x) for x in res]) | ||||
|         self.poke_cd = Cooldown(self.args.th_poke) | ||||
|  | ||||
|         self.mutex = threading.Lock() | ||||
|         self.busy = {} | ||||
|         self.stopping = False | ||||
|         self.nthr = self.args.th_mt | ||||
|         if not self.nthr: | ||||
|             self.nthr = os.cpu_count() if hasattr(os, "cpu_count") else 4 | ||||
|  | ||||
|         self.q = Queue(self.nthr * 4) | ||||
|         for n in range(self.nthr): | ||||
|             t = threading.Thread( | ||||
|                 target=self.worker, name="thumb-{}-{}".format(n, self.nthr) | ||||
|             ) | ||||
|             t.daemon = True | ||||
|             t.start() | ||||
|  | ||||
|         if not self.args.no_vthumb and (not HAVE_FFMPEG or not HAVE_FFPROBE): | ||||
|             missing = [] | ||||
|             if not HAVE_FFMPEG: | ||||
|                 missing.append("FFmpeg") | ||||
|  | ||||
|             if not HAVE_FFPROBE: | ||||
|                 missing.append("FFprobe") | ||||
|  | ||||
|             msg = "cannot create video thumbnails because some of the required programs are not available: " | ||||
|             msg += ", ".join(missing) | ||||
|             self.log(msg, c=3) | ||||
|  | ||||
|         if self.args.th_clean: | ||||
|             t = threading.Thread(target=self.cleaner, name="thumb-cleaner") | ||||
|             t.daemon = True | ||||
|             t.start() | ||||
|  | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func("thumb", msg, c) | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.stopping = True | ||||
|         for _ in range(self.nthr): | ||||
|             self.q.put(None) | ||||
|  | ||||
|     def stopped(self): | ||||
|         with self.mutex: | ||||
|             return not self.nthr | ||||
|  | ||||
|     def get(self, ptop, rem, mtime, fmt): | ||||
|         histpath = self.asrv.vfs.histtab[ptop] | ||||
|         tpath = thumb_path(histpath, rem, mtime, fmt) | ||||
|         abspath = os.path.join(ptop, rem) | ||||
|         cond = threading.Condition(self.mutex) | ||||
|         do_conv = False | ||||
|         with self.mutex: | ||||
|             try: | ||||
|                 self.busy[tpath].append(cond) | ||||
|                 self.log("wait {}".format(tpath)) | ||||
|             except: | ||||
|                 thdir = os.path.dirname(tpath) | ||||
|                 bos.makedirs(thdir) | ||||
|  | ||||
|                 inf_path = os.path.join(thdir, "dir.txt") | ||||
|                 if not bos.path.exists(inf_path): | ||||
|                     with open(inf_path, "wb") as f: | ||||
|                         f.write(fsenc(os.path.dirname(abspath))) | ||||
|  | ||||
|                 self.busy[tpath] = [cond] | ||||
|                 do_conv = True | ||||
|  | ||||
|         if do_conv: | ||||
|             self.q.put([abspath, tpath]) | ||||
|             self.log("conv {} \033[0m{}".format(tpath, abspath), c=6) | ||||
|  | ||||
|         while not self.stopping: | ||||
|             with self.mutex: | ||||
|                 if tpath not in self.busy: | ||||
|                     break | ||||
|  | ||||
|             with cond: | ||||
|                 cond.wait(3) | ||||
|  | ||||
|         try: | ||||
|             st = bos.stat(tpath) | ||||
|             if st.st_size: | ||||
|                 return tpath | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         return None | ||||
|  | ||||
|     def worker(self): | ||||
|         while not self.stopping: | ||||
|             task = self.q.get() | ||||
|             if not task: | ||||
|                 break | ||||
|  | ||||
|             abspath, tpath = task | ||||
|             ext = abspath.split(".")[-1].lower() | ||||
|             fun = None | ||||
|             if not bos.path.exists(tpath): | ||||
|                 if ext in FMT_PIL: | ||||
|                     fun = self.conv_pil | ||||
|                 elif ext in FMT_FF: | ||||
|                     fun = self.conv_ffmpeg | ||||
|  | ||||
|             if fun: | ||||
|                 try: | ||||
|                     fun(abspath, tpath) | ||||
|                 except: | ||||
|                     msg = "{} could not create thumbnail of {}\n{}" | ||||
|                     self.log(msg.format(fun.__name__, abspath, min_ex()), "1;30") | ||||
|                     with open(tpath, "wb") as _: | ||||
|                         pass | ||||
|  | ||||
|             with self.mutex: | ||||
|                 subs = self.busy[tpath] | ||||
|                 del self.busy[tpath] | ||||
|  | ||||
|             for x in subs: | ||||
|                 with x: | ||||
|                     x.notify_all() | ||||
|  | ||||
|         with self.mutex: | ||||
|             self.nthr -= 1 | ||||
|  | ||||
|     def fancy_pillow(self, im): | ||||
|         # exif_transpose is expensive (loads full image + unconditional copy) | ||||
|         r = max(*self.res) * 2 | ||||
|         im.thumbnail((r, r), resample=Image.LANCZOS) | ||||
|         try: | ||||
|             k = next(k for k, v in ExifTags.TAGS.items() if v == "Orientation") | ||||
|             exif = im.getexif() | ||||
|             rot = int(exif[k]) | ||||
|             del exif[k] | ||||
|         except: | ||||
|             rot = 1 | ||||
|  | ||||
|         rots = {8: Image.ROTATE_90, 3: Image.ROTATE_180, 6: Image.ROTATE_270} | ||||
|         if rot in rots: | ||||
|             im = im.transpose(rots[rot]) | ||||
|  | ||||
|         if self.args.th_no_crop: | ||||
|             im.thumbnail(self.res, resample=Image.LANCZOS) | ||||
|         else: | ||||
|             iw, ih = im.size | ||||
|             dw, dh = self.res | ||||
|             res = (min(iw, dw), min(ih, dh)) | ||||
|             im = ImageOps.fit(im, res, method=Image.LANCZOS) | ||||
|  | ||||
|         return im | ||||
|  | ||||
|     def conv_pil(self, abspath, tpath): | ||||
|         with Image.open(fsenc(abspath)) as im: | ||||
|             try: | ||||
|                 im = self.fancy_pillow(im) | ||||
|             except Exception as ex: | ||||
|                 self.log("fancy_pillow {}".format(ex), "1;30") | ||||
|                 im.thumbnail(self.res) | ||||
|  | ||||
|             fmts = ["RGB", "L"] | ||||
|             args = {"quality": 40} | ||||
|  | ||||
|             if tpath.endswith(".webp"): | ||||
|                 # quality 80 = pillow-default | ||||
|                 # quality 75 = ffmpeg-default | ||||
|                 # method 0 = pillow-default, fast | ||||
|                 # method 4 = ffmpeg-default | ||||
|                 # method 6 = max, slow | ||||
|                 fmts += ["RGBA", "LA"] | ||||
|                 args["method"] = 6 | ||||
|             else: | ||||
|                 # default q = 75 | ||||
|                 args["progressive"] = True | ||||
|  | ||||
|             if im.mode not in fmts: | ||||
|                 # print("conv {}".format(im.mode)) | ||||
|                 im = im.convert("RGB") | ||||
|  | ||||
|             im.save(tpath, **args) | ||||
|  | ||||
|     def conv_ffmpeg(self, abspath, tpath): | ||||
|         ret, _ = ffprobe(abspath) | ||||
|  | ||||
|         ext = abspath.rsplit(".")[-1] | ||||
|         if ext in ["h264", "h265"]: | ||||
|             seek = [] | ||||
|         else: | ||||
|             dur = ret[".dur"][1] if ".dur" in ret else 4 | ||||
|             seek = "{:.0f}".format(dur / 3) | ||||
|             seek = [b"-ss", seek.encode("utf-8")] | ||||
|  | ||||
|         scale = "scale={0}:{1}:force_original_aspect_ratio=" | ||||
|         if self.args.th_no_crop: | ||||
|             scale += "decrease,setsar=1:1" | ||||
|         else: | ||||
|             scale += "increase,crop={0}:{1},setsar=1:1" | ||||
|  | ||||
|         scale = scale.format(*list(self.res)).encode("utf-8") | ||||
|         # fmt: off | ||||
|         cmd = [ | ||||
|             b"ffmpeg", | ||||
|             b"-nostdin", | ||||
|             b"-v", b"error", | ||||
|             b"-hide_banner" | ||||
|         ] | ||||
|         cmd += seek | ||||
|         cmd += [ | ||||
|             b"-i", fsenc(abspath), | ||||
|             b"-map", b"0:v:0", | ||||
|             b"-vf", scale, | ||||
|             b"-frames:v", b"1", | ||||
|             b"-metadata:s:v:0", b"rotate=0", | ||||
|         ] | ||||
|         # fmt: on | ||||
|  | ||||
|         if tpath.endswith(".jpg"): | ||||
|             cmd += [ | ||||
|                 b"-q:v", | ||||
|                 b"6",  # default=?? | ||||
|             ] | ||||
|         else: | ||||
|             cmd += [ | ||||
|                 b"-q:v", | ||||
|                 b"50",  # default=75 | ||||
|                 b"-compression_level:v", | ||||
|                 b"6",  # default=4, 0=fast, 6=max | ||||
|             ] | ||||
|  | ||||
|         cmd += [fsenc(tpath)] | ||||
|         # self.log((b" ".join(cmd)).decode("utf-8")) | ||||
|  | ||||
|         ret, sout, serr = runcmd(cmd) | ||||
|         if ret != 0: | ||||
|             m = "FFmpeg failed (probably a corrupt video file):\n" | ||||
|             m += "\n".join(["ff: {}".format(x) for x in serr.split("\n")]) | ||||
|             self.log(m, c="1;30") | ||||
|             raise sp.CalledProcessError(ret, (cmd[0], b"...", cmd[-1])) | ||||
|  | ||||
|     def poke(self, tdir): | ||||
|         if not self.poke_cd.poke(tdir): | ||||
|             return | ||||
|  | ||||
|         ts = int(time.time()) | ||||
|         try: | ||||
|             p1 = os.path.dirname(tdir) | ||||
|             p2 = os.path.dirname(p1) | ||||
|             for dp in [tdir, p1, p2]: | ||||
|                 bos.utime(dp, (ts, ts)) | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def cleaner(self): | ||||
|         interval = self.args.th_clean | ||||
|         while True: | ||||
|             time.sleep(interval) | ||||
|             ndirs = 0 | ||||
|             for vol, histpath in self.asrv.vfs.histtab.items(): | ||||
|                 if histpath.startswith(vol): | ||||
|                     self.log("\033[Jcln {}/\033[A".format(histpath)) | ||||
|                 else: | ||||
|                     self.log("\033[Jcln {} ({})/\033[A".format(histpath, vol)) | ||||
|  | ||||
|                 ndirs += self.clean(histpath) | ||||
|  | ||||
|             self.log("\033[Jcln ok; rm {} dirs".format(ndirs)) | ||||
|  | ||||
|     def clean(self, histpath): | ||||
|         thumbpath = os.path.join(histpath, "th") | ||||
|         # self.log("cln {}".format(thumbpath)) | ||||
|         maxage = self.args.th_maxage | ||||
|         now = time.time() | ||||
|         prev_b64 = None | ||||
|         prev_fp = None | ||||
|         try: | ||||
|             ents = bos.listdir(thumbpath) | ||||
|         except: | ||||
|             return 0 | ||||
|  | ||||
|         ndirs = 0 | ||||
|         for f in sorted(ents): | ||||
|             fp = os.path.join(thumbpath, f) | ||||
|             cmp = fp.lower().replace("\\", "/") | ||||
|  | ||||
|             # "top" or b64 prefix/full (a folder) | ||||
|             if len(f) <= 3 or len(f) == 24: | ||||
|                 age = now - bos.path.getmtime(fp) | ||||
|                 if age > maxage: | ||||
|                     with self.mutex: | ||||
|                         safe = True | ||||
|                         for k in self.busy.keys(): | ||||
|                             if k.lower().replace("\\", "/").startswith(cmp): | ||||
|                                 safe = False | ||||
|                                 break | ||||
|  | ||||
|                         if safe: | ||||
|                             ndirs += 1 | ||||
|                             self.log("rm -rf [{}]".format(fp)) | ||||
|                             shutil.rmtree(fp, ignore_errors=True) | ||||
|                 else: | ||||
|                     ndirs += self.clean(fp) | ||||
|                 continue | ||||
|  | ||||
|             # thumb file | ||||
|             try: | ||||
|                 b64, ts, ext = f.split(".") | ||||
|                 if len(b64) != 24 or len(ts) != 8 or ext not in ["jpg", "webp"]: | ||||
|                     raise Exception() | ||||
|  | ||||
|                 ts = int(ts, 16) | ||||
|             except: | ||||
|                 if f != "dir.txt": | ||||
|                     self.log("foreign file in thumbs dir: [{}]".format(fp), 1) | ||||
|  | ||||
|                 continue | ||||
|  | ||||
|             if b64 == prev_b64: | ||||
|                 self.log("rm replaced [{}]".format(fp)) | ||||
|                 bos.unlink(prev_fp) | ||||
|  | ||||
|             prev_b64 = b64 | ||||
|             prev_fp = fp | ||||
|  | ||||
|         return ndirs | ||||
							
								
								
									
										290
									
								
								copyparty/u2idx.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										290
									
								
								copyparty/u2idx.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,290 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import time | ||||
| import threading | ||||
| from datetime import datetime | ||||
|  | ||||
| from .__init__ import unicode | ||||
| from .util import s3dec, Pebkac, min_ex | ||||
| from .bos import bos | ||||
| from .up2k import up2k_wark_from_hashlist | ||||
|  | ||||
|  | ||||
| try: | ||||
|     HAVE_SQLITE3 = True | ||||
|     import sqlite3 | ||||
| except: | ||||
|     HAVE_SQLITE3 = False | ||||
|  | ||||
|  | ||||
| class U2idx(object): | ||||
|     def __init__(self, conn): | ||||
|         self.log_func = conn.log_func | ||||
|         self.asrv = conn.asrv | ||||
|         self.args = conn.args | ||||
|         self.timeout = self.args.srch_time | ||||
|  | ||||
|         if not HAVE_SQLITE3: | ||||
|             self.log("your python does not have sqlite3; searching will be disabled") | ||||
|             return | ||||
|  | ||||
|         self.cur = {} | ||||
|         self.mem_cur = sqlite3.connect(":memory:") | ||||
|         self.mem_cur.execute(r"create table a (b text)") | ||||
|  | ||||
|         self.p_end = None | ||||
|         self.p_dur = 0 | ||||
|  | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func("u2idx", msg, c) | ||||
|  | ||||
|     def fsearch(self, vols, body): | ||||
|         """search by up2k hashlist""" | ||||
|         if not HAVE_SQLITE3: | ||||
|             return [] | ||||
|  | ||||
|         fsize = body["size"] | ||||
|         fhash = body["hash"] | ||||
|         wark = up2k_wark_from_hashlist(self.args.salt, fsize, fhash) | ||||
|  | ||||
|         uq = "where substr(w,1,16) = ? and w = ?" | ||||
|         uv = [wark[:16], wark] | ||||
|  | ||||
|         try: | ||||
|             return self.run_query(vols, uq, uv)[0] | ||||
|         except: | ||||
|             raise Pebkac(500, min_ex()) | ||||
|  | ||||
|     def get_cur(self, ptop): | ||||
|         if not HAVE_SQLITE3: | ||||
|             return None | ||||
|  | ||||
|         cur = self.cur.get(ptop) | ||||
|         if cur: | ||||
|             return cur | ||||
|  | ||||
|         histpath = self.asrv.vfs.histtab[ptop] | ||||
|         db_path = os.path.join(histpath, "up2k.db") | ||||
|         if not bos.path.exists(db_path): | ||||
|             return None | ||||
|  | ||||
|         cur = sqlite3.connect(db_path, 2).cursor() | ||||
|         self.cur[ptop] = cur | ||||
|         return cur | ||||
|  | ||||
|     def search(self, vols, uq): | ||||
|         """search by query params""" | ||||
|         if not HAVE_SQLITE3: | ||||
|             return [] | ||||
|  | ||||
|         q = "" | ||||
|         va = [] | ||||
|         joins = "" | ||||
|         is_key = True | ||||
|         is_size = False | ||||
|         is_date = False | ||||
|         kw_key = ["(", ")", "and ", "or ", "not "] | ||||
|         kw_val = ["==", "=", "!=", ">", ">=", "<", "<=", "like "] | ||||
|         ptn_mt = re.compile(r"^\.?[a-z_-]+$") | ||||
|         mt_ctr = 0 | ||||
|         mt_keycmp = "substr(up.w,1,16)" | ||||
|         mt_keycmp2 = None | ||||
|         ptn_lc = re.compile(r" (mt[0-9]+\.v) ([=<!>]+) \? $") | ||||
|         ptn_lcv = re.compile(r"[a-zA-Z]") | ||||
|  | ||||
|         while True: | ||||
|             uq = uq.strip() | ||||
|             if not uq: | ||||
|                 break | ||||
|  | ||||
|             ok = False | ||||
|             for kw in kw_key + kw_val: | ||||
|                 if uq.startswith(kw): | ||||
|                     is_key = kw in kw_key | ||||
|                     uq = uq[len(kw) :] | ||||
|                     ok = True | ||||
|                     q += kw | ||||
|                     break | ||||
|  | ||||
|             if ok: | ||||
|                 continue | ||||
|  | ||||
|             v, uq = (uq + " ").split(" ", 1) | ||||
|             if is_key: | ||||
|                 is_key = False | ||||
|  | ||||
|                 if v == "size": | ||||
|                     v = "up.sz" | ||||
|                     is_size = True | ||||
|  | ||||
|                 elif v == "date": | ||||
|                     v = "up.mt" | ||||
|                     is_date = True | ||||
|  | ||||
|                 elif v == "path": | ||||
|                     v = "up.rd" | ||||
|  | ||||
|                 elif v == "name": | ||||
|                     v = "up.fn" | ||||
|  | ||||
|                 elif v == "tags" or ptn_mt.match(v): | ||||
|                     mt_ctr += 1 | ||||
|                     mt_keycmp2 = "mt{}.w".format(mt_ctr) | ||||
|                     joins += "inner join mt mt{} on {} = {} ".format( | ||||
|                         mt_ctr, mt_keycmp, mt_keycmp2 | ||||
|                     ) | ||||
|                     mt_keycmp = mt_keycmp2 | ||||
|                     if v == "tags": | ||||
|                         v = "mt{0}.v".format(mt_ctr) | ||||
|                     else: | ||||
|                         v = "+mt{0}.k = '{1}' and mt{0}.v".format(mt_ctr, v) | ||||
|  | ||||
|                 else: | ||||
|                     raise Pebkac(400, "invalid key [" + v + "]") | ||||
|  | ||||
|                 q += v + " " | ||||
|                 continue | ||||
|  | ||||
|             head = "" | ||||
|             tail = "" | ||||
|  | ||||
|             if is_date: | ||||
|                 is_date = False | ||||
|                 v = v.upper().rstrip("Z").replace(",", " ").replace("T", " ") | ||||
|                 while "  " in v: | ||||
|                     v = v.replace("  ", " ") | ||||
|  | ||||
|                 for fmt in [ | ||||
|                     "%Y-%m-%d %H:%M:%S", | ||||
|                     "%Y-%m-%d %H:%M", | ||||
|                     "%Y-%m-%d %H", | ||||
|                     "%Y-%m-%d", | ||||
|                 ]: | ||||
|                     try: | ||||
|                         v = datetime.strptime(v, fmt).timestamp() | ||||
|                         break | ||||
|                     except: | ||||
|                         pass | ||||
|  | ||||
|             elif is_size: | ||||
|                 is_size = False | ||||
|                 v = int(float(v) * 1024 * 1024) | ||||
|  | ||||
|             else: | ||||
|                 if v.startswith("*"): | ||||
|                     head = "'%'||" | ||||
|                     v = v[1:] | ||||
|  | ||||
|                 if v.endswith("*"): | ||||
|                     tail = "||'%'" | ||||
|                     v = v[:-1] | ||||
|  | ||||
|             q += " {}?{} ".format(head, tail) | ||||
|             va.append(v) | ||||
|             is_key = True | ||||
|  | ||||
|             # lowercase tag searches | ||||
|             m = ptn_lc.search(q) | ||||
|             if not m or not ptn_lcv.search(unicode(v)): | ||||
|                 continue | ||||
|  | ||||
|             va.pop() | ||||
|             va.append(v.lower()) | ||||
|             q = q[: m.start()] | ||||
|  | ||||
|             field, oper = m.groups() | ||||
|             if oper in ["=", "=="]: | ||||
|                 q += " {} like ? ".format(field) | ||||
|             else: | ||||
|                 q += " lower({}) {} ? ".format(field, oper) | ||||
|  | ||||
|         try: | ||||
|             return self.run_query(vols, joins + "where " + q, va) | ||||
|         except Exception as ex: | ||||
|             raise Pebkac(500, repr(ex)) | ||||
|  | ||||
|     def run_query(self, vols, uq, uv): | ||||
|         done_flag = [] | ||||
|         self.active_id = "{:.6f}_{}".format( | ||||
|             time.time(), threading.current_thread().ident | ||||
|         ) | ||||
|         thr = threading.Thread( | ||||
|             target=self.terminator, | ||||
|             args=( | ||||
|                 self.active_id, | ||||
|                 done_flag, | ||||
|             ), | ||||
|             name="u2idx-terminator", | ||||
|         ) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|         if not uq or not uv: | ||||
|             q = "select * from up" | ||||
|             v = () | ||||
|         else: | ||||
|             q = "select up.* from up " + uq | ||||
|             v = tuple(uv) | ||||
|  | ||||
|         self.log("qs: {!r} {!r}".format(q, v)) | ||||
|  | ||||
|         ret = [] | ||||
|         lim = 1000 | ||||
|         taglist = {} | ||||
|         for (vtop, ptop, flags) in vols: | ||||
|             cur = self.get_cur(ptop) | ||||
|             if not cur: | ||||
|                 continue | ||||
|  | ||||
|             self.active_cur = cur | ||||
|  | ||||
|             sret = [] | ||||
|             c = cur.execute(q, v) | ||||
|             for hit in c: | ||||
|                 w, ts, sz, rd, fn, ip, at = hit | ||||
|                 lim -= 1 | ||||
|                 if lim <= 0: | ||||
|                     break | ||||
|  | ||||
|                 if rd.startswith("//") or fn.startswith("//"): | ||||
|                     rd, fn = s3dec(rd, fn) | ||||
|  | ||||
|                 rp = "/".join([x for x in [vtop, rd, fn] if x]) | ||||
|                 sret.append({"ts": int(ts), "sz": sz, "rp": rp, "w": w[:16]}) | ||||
|  | ||||
|             for hit in sret: | ||||
|                 w = hit["w"] | ||||
|                 del hit["w"] | ||||
|                 tags = {} | ||||
|                 q2 = "select k, v from mt where w = ? and k != 'x'" | ||||
|                 for k, v2 in cur.execute(q2, (w,)): | ||||
|                     taglist[k] = True | ||||
|                     tags[k] = v2 | ||||
|  | ||||
|                 hit["tags"] = tags | ||||
|  | ||||
|             ret.extend(sret) | ||||
|             # print("[{}] {}".format(ptop, sret)) | ||||
|  | ||||
|         done_flag.append(True) | ||||
|         self.active_id = None | ||||
|  | ||||
|         # undupe hits from multiple metadata keys | ||||
|         if len(ret) > 1: | ||||
|             ret = [ret[0]] + [ | ||||
|                 y for x, y in zip(ret[:-1], ret[1:]) if x["rp"] != y["rp"] | ||||
|             ] | ||||
|  | ||||
|         return ret, list(taglist.keys()) | ||||
|  | ||||
|     def terminator(self, identifier, done_flag): | ||||
|         for _ in range(self.timeout): | ||||
|             time.sleep(1) | ||||
|             if done_flag: | ||||
|                 return | ||||
|  | ||||
|         if identifier == self.active_id: | ||||
|             self.active_cur.connection.interrupt() | ||||
							
								
								
									
										1915
									
								
								copyparty/up2k.py
									
									
									
									
									
								
							
							
						
						
									
										1915
									
								
								copyparty/up2k.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										12
									
								
								copyparty/web/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								copyparty/web/Makefile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| # run me to zopfli all the static files | ||||
| # which should help on really slow connections | ||||
| # but then why are you using copyparty in the first place | ||||
|  | ||||
| pk: $(addsuffix .gz, $(wildcard *.js *.css)) | ||||
| un: $(addsuffix .un, $(wildcard *.gz)) | ||||
|  | ||||
| %.gz: % | ||||
| 	pigz -11 -J 34 -I 5730 $< | ||||
|  | ||||
| %.un: % | ||||
| 	pigz -d $< | ||||
							
								
								
									
										889
									
								
								copyparty/web/baguettebox.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										889
									
								
								copyparty/web/baguettebox.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,889 @@ | ||||
| /*! | ||||
|  * baguetteBox.js | ||||
|  * @author  feimosi | ||||
|  * @version 1.11.1-mod | ||||
|  * @url https://github.com/feimosi/baguetteBox.js | ||||
|  */ | ||||
|  | ||||
| window.baguetteBox = (function () { | ||||
|     'use strict'; | ||||
|  | ||||
|     var options = {}, | ||||
|         defaults = { | ||||
|             captions: true, | ||||
|             buttons: 'auto', | ||||
|             noScrollbars: false, | ||||
|             bodyClass: 'bbox-open', | ||||
|             titleTag: false, | ||||
|             async: false, | ||||
|             preload: 2, | ||||
|             animation: 'slideIn', | ||||
|             afterShow: null, | ||||
|             afterHide: null, | ||||
|             onChange: null, | ||||
|         }, | ||||
|         overlay, slider, btnPrev, btnNext, btnHelp, btnRotL, btnRotR, btnSel, btnVmode, btnClose, | ||||
|         currentGallery = [], | ||||
|         currentIndex = 0, | ||||
|         isOverlayVisible = false, | ||||
|         touch = {},  // start-pos | ||||
|         touchFlag = false,  // busy | ||||
|         re_i = /.+\.(gif|jpe?g|png|webp)(\?|$)/i, | ||||
|         re_v = /.+\.(webm|mp4)(\?|$)/i, | ||||
|         data = {},  // all galleries | ||||
|         imagesElements = [], | ||||
|         documentLastFocus = null, | ||||
|         isFullscreen = false, | ||||
|         vmute = false, | ||||
|         vloop = false, | ||||
|         vnext = false, | ||||
|         resume_mp = false; | ||||
|  | ||||
|     var onFSC = function (e) { | ||||
|         isFullscreen = !!document.fullscreenElement; | ||||
|     }; | ||||
|  | ||||
|     var overlayClickHandler = function (e) { | ||||
|         if (e.target.id.indexOf('baguette-img') !== -1) | ||||
|             hideOverlay(); | ||||
|     }; | ||||
|  | ||||
|     var touchstartHandler = function (e) { | ||||
|         touch.count = e.touches.length; | ||||
|         if (touch.count > 1) | ||||
|             touch.multitouch = true; | ||||
|  | ||||
|         touch.startX = e.changedTouches[0].pageX; | ||||
|         touch.startY = e.changedTouches[0].pageY; | ||||
|     }; | ||||
|     var touchmoveHandler = function (e) { | ||||
|         if (touchFlag || touch.multitouch) | ||||
|             return; | ||||
|  | ||||
|         e.preventDefault ? e.preventDefault() : e.returnValue = false; | ||||
|         var touchEvent = e.touches[0] || e.changedTouches[0]; | ||||
|         if (touchEvent.pageX - touch.startX > 40) { | ||||
|             touchFlag = true; | ||||
|             showPreviousImage(); | ||||
|         } else if (touchEvent.pageX - touch.startX < -40) { | ||||
|             touchFlag = true; | ||||
|             showNextImage(); | ||||
|         } else if (touch.startY - touchEvent.pageY > 100) { | ||||
|             hideOverlay(); | ||||
|         } | ||||
|     }; | ||||
|     var touchendHandler = function (e) { | ||||
|         touch.count--; | ||||
|         if (e && e.touches) | ||||
|             touch.count = e.touches.length; | ||||
|  | ||||
|         if (touch.count <= 0) | ||||
|             touch.multitouch = false; | ||||
|  | ||||
|         touchFlag = false; | ||||
|     }; | ||||
|     var contextmenuHandler = function () { | ||||
|         touchendHandler(); | ||||
|     }; | ||||
|  | ||||
|     var trapFocusInsideOverlay = function (e) { | ||||
|         if (overlay.style.display === 'block' && (overlay.contains && !overlay.contains(e.target))) { | ||||
|             e.stopPropagation(); | ||||
|             btnClose.focus(); | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     function run(selector, userOptions) { | ||||
|         buildOverlay(); | ||||
|         removeFromCache(selector); | ||||
|         return bindImageClickListeners(selector, userOptions); | ||||
|     } | ||||
|  | ||||
|     function bindImageClickListeners(selector, userOptions) { | ||||
|         var galleryNodeList = QSA(selector); | ||||
|         var selectorData = { | ||||
|             galleries: [], | ||||
|             nodeList: galleryNodeList | ||||
|         }; | ||||
|         data[selector] = selectorData; | ||||
|  | ||||
|         [].forEach.call(galleryNodeList, function (galleryElement) { | ||||
|             var tagsNodeList = []; | ||||
|             if (galleryElement.tagName === 'A') | ||||
|                 tagsNodeList = [galleryElement]; | ||||
|             else | ||||
|                 tagsNodeList = galleryElement.getElementsByTagName('a'); | ||||
|  | ||||
|             tagsNodeList = [].filter.call(tagsNodeList, function (element) { | ||||
|                 if (element.className.indexOf(userOptions && userOptions.ignoreClass) === -1) | ||||
|                     return re_i.test(element.href) || re_v.test(element.href); | ||||
|             }); | ||||
|             if (!tagsNodeList.length) | ||||
|                 return; | ||||
|  | ||||
|             var gallery = []; | ||||
|             [].forEach.call(tagsNodeList, function (imageElement, imageIndex) { | ||||
|                 var imageElementClickHandler = function (e) { | ||||
|                     if (ctrl(e)) | ||||
|                         return true; | ||||
|  | ||||
|                     e.preventDefault ? e.preventDefault() : e.returnValue = false; | ||||
|                     prepareOverlay(gallery, userOptions); | ||||
|                     showOverlay(imageIndex); | ||||
|                 }; | ||||
|                 var imageItem = { | ||||
|                     eventHandler: imageElementClickHandler, | ||||
|                     imageElement: imageElement | ||||
|                 }; | ||||
|                 bind(imageElement, 'click', imageElementClickHandler); | ||||
|                 gallery.push(imageItem); | ||||
|             }); | ||||
|             selectorData.galleries.push(gallery); | ||||
|         }); | ||||
|  | ||||
|         return selectorData.galleries; | ||||
|     } | ||||
|  | ||||
|     function clearCachedData() { | ||||
|         for (var selector in data) | ||||
|             if (data.hasOwnProperty(selector)) | ||||
|                 removeFromCache(selector); | ||||
|     } | ||||
|  | ||||
|     function removeFromCache(selector) { | ||||
|         if (!data.hasOwnProperty(selector)) | ||||
|             return; | ||||
|  | ||||
|         var galleries = data[selector].galleries; | ||||
|         [].forEach.call(galleries, function (gallery) { | ||||
|             [].forEach.call(gallery, function (imageItem) { | ||||
|                 unbind(imageItem.imageElement, 'click', imageItem.eventHandler); | ||||
|             }); | ||||
|  | ||||
|             if (currentGallery === gallery) | ||||
|                 currentGallery = []; | ||||
|         }); | ||||
|  | ||||
|         delete data[selector]; | ||||
|     } | ||||
|  | ||||
|     function buildOverlay() { | ||||
|         overlay = ebi('bbox-overlay'); | ||||
|         if (!overlay) { | ||||
|             var ctr = mknod('div'); | ||||
|             ctr.innerHTML = ( | ||||
|                 '<div id="bbox-overlay" role="dialog">' + | ||||
|                 '<div id="bbox-slider"></div>' + | ||||
|                 '<button id="bbox-prev" class="bbox-btn" type="button" aria-label="Previous"><</button>' + | ||||
|                 '<button id="bbox-next" class="bbox-btn" type="button" aria-label="Next">></button>' + | ||||
|                 '<div id="bbox-btns">' + | ||||
|                 '<button id="bbox-help" type="button">?</button>' + | ||||
|                 '<button id="bbox-rotl" type="button">↶</button>' + | ||||
|                 '<button id="bbox-rotr" type="button">↷</button>' + | ||||
|                 '<button id="bbox-tsel" type="button">sel</button>' + | ||||
|                 '<button id="bbox-vmode" type="button" tt="a"></button>' + | ||||
|                 '<button id="bbox-close" type="button" aria-label="Close">X</button>' + | ||||
|                 '</div></div>' | ||||
|             ); | ||||
|             overlay = ctr.firstChild; | ||||
|             QS('body').appendChild(overlay); | ||||
|             tt.att(overlay); | ||||
|         } | ||||
|         slider = ebi('bbox-slider'); | ||||
|         btnPrev = ebi('bbox-prev'); | ||||
|         btnNext = ebi('bbox-next'); | ||||
|         btnHelp = ebi('bbox-help'); | ||||
|         btnRotL = ebi('bbox-rotl'); | ||||
|         btnRotR = ebi('bbox-rotr'); | ||||
|         btnSel = ebi('bbox-tsel'); | ||||
|         btnVmode = ebi('bbox-vmode'); | ||||
|         btnClose = ebi('bbox-close'); | ||||
|         bindEvents(); | ||||
|     } | ||||
|  | ||||
|     function halp() { | ||||
|         if (ebi('bbox-halp')) | ||||
|             return; | ||||
|  | ||||
|         var list = [ | ||||
|             ['<b># hotkey</b>', '<b># operation</b>'], | ||||
|             ['escape', 'close'], | ||||
|             ['left, J', 'previous file'], | ||||
|             ['right, L', 'next file'], | ||||
|             ['home', 'first file'], | ||||
|             ['end', 'last file'], | ||||
|             ['R', 'rotate (shift=ccw)'], | ||||
|             ['S', 'toggle file selection'], | ||||
|             ['space, P, K', 'video: play / pause'], | ||||
|             ['U', 'video: seek 10sec back'], | ||||
|             ['P', 'video: seek 10sec ahead'], | ||||
|             ['M', 'video: toggle mute'], | ||||
|             ['V', 'video: toggle loop'], | ||||
|             ['C', 'video: toggle auto-next'], | ||||
|             ['F', 'video: toggle fullscreen'], | ||||
|         ], | ||||
|             d = mknod('table'), | ||||
|             html = ['<tbody>']; | ||||
|  | ||||
|         for (var a = 0; a < list.length; a++) | ||||
|             html.push('<tr><td>' + list[a][0] + '</td><td>' + list[a][1] + '</td></tr>'); | ||||
|  | ||||
|         d.innerHTML = html.join('\n') + '</tbody>'; | ||||
|         d.setAttribute('id', 'bbox-halp'); | ||||
|         d.onclick = function () { | ||||
|             overlay.removeChild(d); | ||||
|         }; | ||||
|         overlay.appendChild(d); | ||||
|     } | ||||
|  | ||||
|     function keyDownHandler(e) { | ||||
|         if (e.ctrlKey || e.altKey || e.metaKey || e.isComposing) | ||||
|             return; | ||||
|  | ||||
|         var k = e.code + '', v = vid(); | ||||
|  | ||||
|         if (k == "ArrowLeft" || k == "KeyJ") | ||||
|             showPreviousImage(); | ||||
|         else if (k == "ArrowRight" || k == "KeyL") | ||||
|             showNextImage(); | ||||
|         else if (k == "Escape") | ||||
|             hideOverlay(); | ||||
|         else if (k == "Home") | ||||
|             showFirstImage(e); | ||||
|         else if (k == "End") | ||||
|             showLastImage(e); | ||||
|         else if (k == "Space" || k == "KeyP" || k == "KeyK") | ||||
|             playpause(); | ||||
|         else if (k == "KeyU" || k == "KeyO") | ||||
|             relseek(k == "KeyU" ? -10 : 10); | ||||
|         else if (k == "KeyM" && v) { | ||||
|             v.muted = vmute = !vmute; | ||||
|             mp_ctl(); | ||||
|         } | ||||
|         else if (k == "KeyV" && v) { | ||||
|             vloop = !vloop; | ||||
|             vnext = vnext && !vloop; | ||||
|             setVmode(); | ||||
|         } | ||||
|         else if (k == "KeyC" && v) { | ||||
|             vnext = !vnext; | ||||
|             vloop = vloop && !vnext; | ||||
|             setVmode(); | ||||
|         } | ||||
|         else if (k == "KeyF") | ||||
|             try { | ||||
|                 if (isFullscreen) | ||||
|                     document.exitFullscreen(); | ||||
|                 else | ||||
|                     v.requestFullscreen(); | ||||
|             } | ||||
|             catch (ex) { } | ||||
|         else if (k == "KeyS") | ||||
|             tglsel(); | ||||
|         else if (k == "KeyR") | ||||
|             rotn(e.shiftKey ? -1 : 1); | ||||
|     } | ||||
|  | ||||
|     function setVmode() { | ||||
|         var v = vid(); | ||||
|         ebi('bbox-vmode').style.display = v ? '' : 'none'; | ||||
|         if (!v) | ||||
|             return; | ||||
|  | ||||
|         var msg = 'When video ends, ', tts = '', lbl; | ||||
|         if (vloop) { | ||||
|             lbl = 'Loop'; | ||||
|             msg += 'repeat it'; | ||||
|             tts = '$NHotkey: V'; | ||||
|         } | ||||
|         else if (vnext) { | ||||
|             lbl = 'Cont'; | ||||
|             msg += 'continue to next'; | ||||
|             tts = '$NHotkey: C'; | ||||
|         } | ||||
|         else { | ||||
|             lbl = 'Stop'; | ||||
|             msg += 'just stop' | ||||
|         } | ||||
|         btnVmode.setAttribute('aria-label', msg); | ||||
|         btnVmode.setAttribute('tt', msg + tts); | ||||
|         btnVmode.textContent = lbl; | ||||
|  | ||||
|         v.loop = vloop | ||||
|         if (vloop && v.paused) | ||||
|             v.play(); | ||||
|     } | ||||
|  | ||||
|     function tglVmode() { | ||||
|         if (vloop) { | ||||
|             vnext = true; | ||||
|             vloop = false; | ||||
|         } | ||||
|         else if (vnext) | ||||
|             vnext = false; | ||||
|         else | ||||
|             vloop = true; | ||||
|  | ||||
|         setVmode(); | ||||
|         if (tt.en) | ||||
|             tt.show.bind(this)(); | ||||
|     } | ||||
|  | ||||
|     function tglsel() { | ||||
|         var thumb = currentGallery[currentIndex].imageElement, | ||||
|             name = vsplit(thumb.href)[1], | ||||
|             files = msel.getall(); | ||||
|  | ||||
|         for (var a = 0; a < files.length; a++) | ||||
|             if (vsplit(files[a].vp)[1] == name) | ||||
|                 clmod(ebi(files[a].id).closest('tr'), 'sel', 't'); | ||||
|  | ||||
|         msel.selui(); | ||||
|         selbg(); | ||||
|     } | ||||
|  | ||||
|     function selbg() { | ||||
|         var img = vidimg(), | ||||
|             thumb = currentGallery[currentIndex].imageElement, | ||||
|             name = vsplit(thumb.href)[1], | ||||
|             files = msel.getsel(), | ||||
|             sel = false; | ||||
|  | ||||
|         for (var a = 0; a < files.length; a++) | ||||
|             if (vsplit(files[a].vp)[1] == name) | ||||
|                 sel = true; | ||||
|  | ||||
|         ebi('bbox-overlay').style.background = sel ? | ||||
|             'rgba(153,34,85,0.7)' : ''; | ||||
|  | ||||
|         img.style.borderRadius = sel ? '1em' : ''; | ||||
|         btnSel.style.color = sel ? '#fff' : ''; | ||||
|         btnSel.style.background = sel ? '#d48' : ''; | ||||
|         btnSel.style.textShadow = sel ? '1px 1px 0 #b38' : ''; | ||||
|         btnSel.style.boxShadow = sel ? '.15em .15em 0 #502' : ''; | ||||
|     } | ||||
|  | ||||
|     function keyUpHandler(e) { | ||||
|         if (e.ctrlKey || e.altKey || e.metaKey || e.isComposing) | ||||
|             return; | ||||
|  | ||||
|         var k = e.code + ''; | ||||
|  | ||||
|         if (k == "Space") | ||||
|             ev(e); | ||||
|     } | ||||
|  | ||||
|     var passiveSupp = false; | ||||
|     try { | ||||
|         var opts = { | ||||
|             get passive() { | ||||
|                 passiveSupp = true; | ||||
|                 return false; | ||||
|             } | ||||
|         }; | ||||
|         window.addEventListener('test', null, opts); | ||||
|         window.removeEventListener('test', null, opts); | ||||
|     } | ||||
|     catch (ex) { | ||||
|         passiveSupp = false; | ||||
|     } | ||||
|     var passiveEvent = passiveSupp ? { passive: false } : null; | ||||
|     var nonPassiveEvent = passiveSupp ? { passive: true } : null; | ||||
|  | ||||
|     function bindEvents() { | ||||
|         bind(overlay, 'click', overlayClickHandler); | ||||
|         bind(btnPrev, 'click', showPreviousImage); | ||||
|         bind(btnNext, 'click', showNextImage); | ||||
|         bind(btnClose, 'click', hideOverlay); | ||||
|         bind(btnVmode, 'click', tglVmode); | ||||
|         bind(btnHelp, 'click', halp); | ||||
|         bind(btnRotL, 'click', rotl); | ||||
|         bind(btnRotR, 'click', rotr); | ||||
|         bind(btnSel, 'click', tglsel); | ||||
|         bind(slider, 'contextmenu', contextmenuHandler); | ||||
|         bind(overlay, 'touchstart', touchstartHandler, nonPassiveEvent); | ||||
|         bind(overlay, 'touchmove', touchmoveHandler, passiveEvent); | ||||
|         bind(overlay, 'touchend', touchendHandler); | ||||
|         bind(document, 'focus', trapFocusInsideOverlay, true); | ||||
|     } | ||||
|  | ||||
|     function unbindEvents() { | ||||
|         unbind(overlay, 'click', overlayClickHandler); | ||||
|         unbind(btnPrev, 'click', showPreviousImage); | ||||
|         unbind(btnNext, 'click', showNextImage); | ||||
|         unbind(btnClose, 'click', hideOverlay); | ||||
|         unbind(btnVmode, 'click', tglVmode); | ||||
|         unbind(btnHelp, 'click', halp); | ||||
|         unbind(btnRotL, 'click', rotl); | ||||
|         unbind(btnRotR, 'click', rotr); | ||||
|         unbind(btnSel, 'click', tglsel); | ||||
|         unbind(slider, 'contextmenu', contextmenuHandler); | ||||
|         unbind(overlay, 'touchstart', touchstartHandler, nonPassiveEvent); | ||||
|         unbind(overlay, 'touchmove', touchmoveHandler, passiveEvent); | ||||
|         unbind(overlay, 'touchend', touchendHandler); | ||||
|         unbind(document, 'focus', trapFocusInsideOverlay, true); | ||||
|         timer.rm(rotn); | ||||
|     } | ||||
|  | ||||
|     function prepareOverlay(gallery, userOptions) { | ||||
|         if (currentGallery === gallery) | ||||
|             return; | ||||
|  | ||||
|         currentGallery = gallery; | ||||
|         setOptions(userOptions); | ||||
|         slider.innerHTML = ''; | ||||
|         imagesElements.length = 0; | ||||
|  | ||||
|         var imagesFiguresIds = []; | ||||
|         var imagesCaptionsIds = []; | ||||
|         for (var i = 0, fullImage; i < gallery.length; i++) { | ||||
|             fullImage = mknod('div'); | ||||
|             fullImage.className = 'full-image'; | ||||
|             fullImage.id = 'baguette-img-' + i; | ||||
|             imagesElements.push(fullImage); | ||||
|  | ||||
|             imagesFiguresIds.push('bbox-figure-' + i); | ||||
|             imagesCaptionsIds.push('bbox-figcaption-' + i); | ||||
|             slider.appendChild(imagesElements[i]); | ||||
|         } | ||||
|         overlay.setAttribute('aria-labelledby', imagesFiguresIds.join(' ')); | ||||
|         overlay.setAttribute('aria-describedby', imagesCaptionsIds.join(' ')); | ||||
|     } | ||||
|  | ||||
|     function setOptions(newOptions) { | ||||
|         if (!newOptions) | ||||
|             newOptions = {}; | ||||
|  | ||||
|         for (var item in defaults) { | ||||
|             options[item] = defaults[item]; | ||||
|             if (typeof newOptions[item] !== 'undefined') | ||||
|                 options[item] = newOptions[item]; | ||||
|         } | ||||
|         slider.style.transition = (options.animation === 'fadeIn' ? 'opacity .4s ease' : | ||||
|             options.animation === 'slideIn' ? '' : 'none'); | ||||
|  | ||||
|         if (options.buttons === 'auto' && ('ontouchstart' in window || currentGallery.length === 1)) | ||||
|             options.buttons = false; | ||||
|  | ||||
|         btnPrev.style.display = btnNext.style.display = (options.buttons ? '' : 'none'); | ||||
|     } | ||||
|  | ||||
|     function showOverlay(chosenImageIndex) { | ||||
|         if (options.noScrollbars) { | ||||
|             document.documentElement.style.overflowY = 'hidden'; | ||||
|             document.body.style.overflowY = 'scroll'; | ||||
|         } | ||||
|         if (overlay.style.display === 'block') | ||||
|             return; | ||||
|  | ||||
|         bind(document, 'keydown', keyDownHandler); | ||||
|         bind(document, 'keyup', keyUpHandler); | ||||
|         bind(document, 'fullscreenchange', onFSC); | ||||
|         currentIndex = chosenImageIndex; | ||||
|         touch = { | ||||
|             count: 0, | ||||
|             startX: null, | ||||
|             startY: null | ||||
|         }; | ||||
|         loadImage(currentIndex, function () { | ||||
|             preloadNext(currentIndex); | ||||
|             preloadPrev(currentIndex); | ||||
|         }); | ||||
|  | ||||
|         updateOffset(); | ||||
|         overlay.style.display = 'block'; | ||||
|         // Fade in overlay | ||||
|         setTimeout(function () { | ||||
|             overlay.className = 'visible'; | ||||
|             if (options.bodyClass && document.body.classList) | ||||
|                 document.body.classList.add(options.bodyClass); | ||||
|  | ||||
|             if (options.afterShow) | ||||
|                 options.afterShow(); | ||||
|         }, 50); | ||||
|  | ||||
|         if (options.onChange) | ||||
|             options.onChange(currentIndex, imagesElements.length); | ||||
|  | ||||
|         documentLastFocus = document.activeElement; | ||||
|         btnClose.focus(); | ||||
|         isOverlayVisible = true; | ||||
|     } | ||||
|  | ||||
|     function hideOverlay(e) { | ||||
|         ev(e); | ||||
|         playvid(false); | ||||
|         if (options.noScrollbars) { | ||||
|             document.documentElement.style.overflowY = 'auto'; | ||||
|             document.body.style.overflowY = 'auto'; | ||||
|         } | ||||
|         if (overlay.style.display === 'none') | ||||
|             return; | ||||
|  | ||||
|         unbind(document, 'keydown', keyDownHandler); | ||||
|         unbind(document, 'keyup', keyUpHandler); | ||||
|         unbind(document, 'fullscreenchange', onFSC); | ||||
|         // Fade out and hide the overlay | ||||
|         overlay.className = ''; | ||||
|         setTimeout(function () { | ||||
|             overlay.style.display = 'none'; | ||||
|             if (options.bodyClass && document.body.classList) | ||||
|                 document.body.classList.remove(options.bodyClass); | ||||
|  | ||||
|             var h = ebi('bbox-halp'); | ||||
|             if (h) | ||||
|                 h.parentNode.removeChild(h); | ||||
|  | ||||
|             if (options.afterHide) | ||||
|                 options.afterHide(); | ||||
|  | ||||
|             documentLastFocus && documentLastFocus.focus(); | ||||
|             isOverlayVisible = false; | ||||
|         }, 500); | ||||
|     } | ||||
|  | ||||
|     function loadImage(index, callback) { | ||||
|         var imageContainer = imagesElements[index]; | ||||
|         var galleryItem = currentGallery[index]; | ||||
|  | ||||
|         if (typeof imageContainer === 'undefined' || typeof galleryItem === 'undefined') | ||||
|             return;  // out-of-bounds or gallery dirty | ||||
|  | ||||
|         if (imageContainer.querySelector('img, video')) | ||||
|             // was loaded, cb and bail | ||||
|             return callback ? callback() : null; | ||||
|  | ||||
|         // maybe unloaded video | ||||
|         while (imageContainer.firstChild) | ||||
|             imageContainer.removeChild(imageContainer.firstChild); | ||||
|  | ||||
|         var imageElement = galleryItem.imageElement, | ||||
|             imageSrc = imageElement.href, | ||||
|             is_vid = re_v.test(imageSrc), | ||||
|             thumbnailElement = imageElement.querySelector('img, video'), | ||||
|             imageCaption = typeof options.captions === 'function' ? | ||||
|                 options.captions.call(currentGallery, imageElement) : | ||||
|                 imageElement.getAttribute('data-caption') || imageElement.title; | ||||
|  | ||||
|         imageSrc += imageSrc.indexOf('?') < 0 ? '?cache' : '&cache'; | ||||
|  | ||||
|         if (is_vid && index != currentIndex) | ||||
|             return;  // no preload | ||||
|  | ||||
|         var figure = mknod('figure'); | ||||
|         figure.id = 'bbox-figure-' + index; | ||||
|         figure.innerHTML = '<div class="bbox-spinner">' + | ||||
|             '<div class="bbox-double-bounce1"></div>' + | ||||
|             '<div class="bbox-double-bounce2"></div>' + | ||||
|             '</div>'; | ||||
|  | ||||
|         if (options.captions && imageCaption) { | ||||
|             var figcaption = mknod('figcaption'); | ||||
|             figcaption.id = 'bbox-figcaption-' + index; | ||||
|             figcaption.innerHTML = imageCaption; | ||||
|             figure.appendChild(figcaption); | ||||
|         } | ||||
|         imageContainer.appendChild(figure); | ||||
|  | ||||
|         var image = mknod(is_vid ? 'video' : 'img'); | ||||
|         clmod(imageContainer, 'vid', is_vid); | ||||
|  | ||||
|         image.addEventListener(is_vid ? 'loadedmetadata' : 'load', function () { | ||||
|             // Remove loader element | ||||
|             var spinner = QS('#baguette-img-' + index + ' .bbox-spinner'); | ||||
|             figure.removeChild(spinner); | ||||
|             if (!options.async && callback) | ||||
|                 callback(); | ||||
|         }); | ||||
|         image.setAttribute('src', imageSrc); | ||||
|         if (is_vid) { | ||||
|             image.setAttribute('controls', 'controls'); | ||||
|             image.onended = vidEnd; | ||||
|         } | ||||
|         image.alt = thumbnailElement ? thumbnailElement.alt || '' : ''; | ||||
|         if (options.titleTag && imageCaption) | ||||
|             image.title = imageCaption; | ||||
|  | ||||
|         figure.appendChild(image); | ||||
|  | ||||
|         if (options.async && callback) | ||||
|             callback(); | ||||
|     } | ||||
|  | ||||
|     function showNextImage(e) { | ||||
|         ev(e); | ||||
|         return show(currentIndex + 1); | ||||
|     } | ||||
|  | ||||
|     function showPreviousImage(e) { | ||||
|         ev(e); | ||||
|         return show(currentIndex - 1); | ||||
|     } | ||||
|  | ||||
|     function showFirstImage(e) { | ||||
|         if (e) | ||||
|             e.preventDefault(); | ||||
|  | ||||
|         return show(0); | ||||
|     } | ||||
|  | ||||
|     function showLastImage(e) { | ||||
|         if (e) | ||||
|             e.preventDefault(); | ||||
|  | ||||
|         return show(currentGallery.length - 1); | ||||
|     } | ||||
|  | ||||
|     function show(index, gallery) { | ||||
|         if (!isOverlayVisible && index >= 0 && index < gallery.length) { | ||||
|             prepareOverlay(gallery, options); | ||||
|             showOverlay(index); | ||||
|             return true; | ||||
|         } | ||||
|         if (index < 0) { | ||||
|             if (options.animation) | ||||
|                 bounceAnimation('left'); | ||||
|  | ||||
|             return false; | ||||
|         } | ||||
|         if (index >= imagesElements.length) { | ||||
|             if (options.animation) | ||||
|                 bounceAnimation('right'); | ||||
|  | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         var v = vid(); | ||||
|         if (v) { | ||||
|             v.src = ''; | ||||
|             v.load(); | ||||
|             v.parentNode.removeChild(v); | ||||
|         } | ||||
|  | ||||
|         currentIndex = index; | ||||
|         loadImage(currentIndex, function () { | ||||
|             preloadNext(currentIndex); | ||||
|             preloadPrev(currentIndex); | ||||
|         }); | ||||
|         updateOffset(); | ||||
|  | ||||
|         if (options.onChange) | ||||
|             options.onChange(currentIndex, imagesElements.length); | ||||
|  | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     var prev_cw = 0, prev_ch = 0, unrot_timer = null; | ||||
|     function rotn(n) { | ||||
|         var el = vidimg(), | ||||
|             orot = parseInt(el.getAttribute('rot') || 0), | ||||
|             frot = orot + (n || 0) * 90; | ||||
|  | ||||
|         if (!frot && !orot) | ||||
|             return;  // reflow noop | ||||
|  | ||||
|         var co = ebi('bbox-overlay'), | ||||
|             cw = co.clientWidth, | ||||
|             ch = co.clientHeight; | ||||
|  | ||||
|         if (!n && prev_cw === cw && prev_ch === ch) | ||||
|             return;  // reflow noop | ||||
|  | ||||
|         prev_cw = cw; | ||||
|         prev_ch = ch; | ||||
|         var rot = frot, | ||||
|             iw = el.naturalWidth || el.videoWidth, | ||||
|             ih = el.naturalHeight || el.videoHeight, | ||||
|             magic = 4,  // idk, works in enough browsers | ||||
|             dl = el.closest('div').querySelector('figcaption a'), | ||||
|             vw = cw, | ||||
|             vh = ch - dl.offsetHeight + magic, | ||||
|             pmag = Math.min(1, Math.min(vw / ih, vh / iw)), | ||||
|             wmag = Math.min(1, Math.min(vw / iw, vh / ih)); | ||||
|  | ||||
|         while (rot < 0) rot += 360; | ||||
|         while (rot >= 360) rot -= 360; | ||||
|         var q = rot == 90 || rot == 270 ? 1 : 0, | ||||
|             mag = q ? pmag : wmag; | ||||
|  | ||||
|         el.style.cssText = 'max-width:none; max-height:none; position:absolute; display:block; margin:0'; | ||||
|         if (!orot) { | ||||
|             el.style.width = iw * wmag + 'px'; | ||||
|             el.style.height = ih * wmag + 'px'; | ||||
|             el.style.left = (vw - iw * wmag) / 2 + 'px'; | ||||
|             el.style.top = (vh - ih * wmag) / 2 - magic + 'px'; | ||||
|             q = el.offsetHeight; | ||||
|         } | ||||
|         el.style.width = iw * mag + 'px'; | ||||
|         el.style.height = ih * mag + 'px'; | ||||
|         el.style.left = (vw - iw * mag) / 2 + 'px'; | ||||
|         el.style.top = (vh - ih * mag) / 2 - magic + 'px'; | ||||
|         el.style.transform = 'rotate(' + frot + 'deg)'; | ||||
|         el.setAttribute('rot', frot); | ||||
|         timer.add(rotn); | ||||
|         if (!rot) { | ||||
|             clearTimeout(unrot_timer); | ||||
|             unrot_timer = setTimeout(unrot, 300); | ||||
|         } | ||||
|     } | ||||
|     function rotl() { | ||||
|         rotn(-1); | ||||
|     } | ||||
|     function rotr() { | ||||
|         rotn(1); | ||||
|     } | ||||
|     function unrot() { | ||||
|         var el = vidimg(), | ||||
|             orot = el.getAttribute('rot'), | ||||
|             rot = parseInt(orot || 0); | ||||
|  | ||||
|         while (rot < 0) rot += 360; | ||||
|         while (rot >= 360) rot -= 360; | ||||
|         if (rot || orot === null) | ||||
|             return; | ||||
|  | ||||
|         clmod(el, 'nt', 1); | ||||
|         el.removeAttribute('rot'); | ||||
|         el.removeAttribute("style"); | ||||
|         rot = el.offsetHeight; | ||||
|         clmod(el, 'nt'); | ||||
|         timer.rm(rotn); | ||||
|     } | ||||
|  | ||||
|     function vid() { | ||||
|         return imagesElements[currentIndex].querySelector('video'); | ||||
|     } | ||||
|  | ||||
|     function vidimg() { | ||||
|         return imagesElements[currentIndex].querySelector('img, video'); | ||||
|     } | ||||
|  | ||||
|     function playvid(play) { | ||||
|         if (vid()) | ||||
|             vid()[play ? 'play' : 'pause'](); | ||||
|     } | ||||
|  | ||||
|     function playpause() { | ||||
|         var v = vid(); | ||||
|         if (v) | ||||
|             v[v.paused ? "play" : "pause"](); | ||||
|     } | ||||
|  | ||||
|     function relseek(sec) { | ||||
|         if (vid()) | ||||
|             vid().currentTime += sec; | ||||
|     } | ||||
|  | ||||
|     function vidEnd() { | ||||
|         if (this == vid() && vnext) | ||||
|             showNextImage(); | ||||
|     } | ||||
|  | ||||
|     function mp_ctl() { | ||||
|         var v = vid(); | ||||
|         if (!vmute && v && mp.au && !mp.au.paused) { | ||||
|             mp.fade_out(); | ||||
|             resume_mp = true; | ||||
|         } | ||||
|         else if (resume_mp && (vmute || !v) && mp.au && mp.au.paused) { | ||||
|             mp.fade_in(); | ||||
|             resume_mp = false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function bounceAnimation(direction) { | ||||
|         slider.className = 'bounce-from-' + direction; | ||||
|         setTimeout(function () { | ||||
|             slider.className = ''; | ||||
|         }, 400); | ||||
|     } | ||||
|  | ||||
|     function updateOffset() { | ||||
|         var offset = -currentIndex * 100 + '%', | ||||
|             xform = slider.style.perspective !== undefined; | ||||
|  | ||||
|         if (options.animation === 'fadeIn') { | ||||
|             slider.style.opacity = 0; | ||||
|             setTimeout(function () { | ||||
|                 xform ? | ||||
|                     slider.style.transform = 'translate3d(' + offset + ',0,0)' : | ||||
|                     slider.style.left = offset; | ||||
|                 slider.style.opacity = 1; | ||||
|             }, 400); | ||||
|         } else { | ||||
|             xform ? | ||||
|                 slider.style.transform = 'translate3d(' + offset + ',0,0)' : | ||||
|                 slider.style.left = offset; | ||||
|         } | ||||
|         playvid(false); | ||||
|         var v = vid(); | ||||
|         if (v) { | ||||
|             playvid(true); | ||||
|             v.muted = vmute; | ||||
|             v.loop = vloop; | ||||
|         } | ||||
|         selbg(); | ||||
|         mp_ctl(); | ||||
|         setVmode(); | ||||
|  | ||||
|         var el = vidimg(); | ||||
|         if (el.getAttribute('rot')) | ||||
|             timer.add(rotn); | ||||
|         else | ||||
|             timer.rm(rotn); | ||||
|  | ||||
|         var prev = QS('.full-image.vis'); | ||||
|         if (prev) | ||||
|             clmod(prev, 'vis'); | ||||
|  | ||||
|         clmod(el.closest('div'), 'vis', 1); | ||||
|     } | ||||
|  | ||||
|     function preloadNext(index) { | ||||
|         if (index - currentIndex >= options.preload) | ||||
|             return; | ||||
|  | ||||
|         loadImage(index + 1, function () { | ||||
|             preloadNext(index + 1); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     function preloadPrev(index) { | ||||
|         if (currentIndex - index >= options.preload) | ||||
|             return; | ||||
|  | ||||
|         loadImage(index - 1, function () { | ||||
|             preloadPrev(index - 1); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     function bind(element, event, callback, options) { | ||||
|         element.addEventListener(event, callback, options); | ||||
|     } | ||||
|  | ||||
|     function unbind(element, event, callback, options) { | ||||
|         element.removeEventListener(event, callback, options); | ||||
|     } | ||||
|  | ||||
|     function destroyPlugin() { | ||||
|         unbindEvents(); | ||||
|         clearCachedData(); | ||||
|         unbind(document, 'keydown', keyDownHandler); | ||||
|         unbind(document, 'keyup', keyUpHandler); | ||||
|         document.getElementsByTagName('body')[0].removeChild(ebi('bbox-overlay')); | ||||
|         data = {}; | ||||
|         currentGallery = []; | ||||
|         currentIndex = 0; | ||||
|     } | ||||
|  | ||||
|     return { | ||||
|         run: run, | ||||
|         show: show, | ||||
|         showNext: showNextImage, | ||||
|         showPrevious: showPreviousImage, | ||||
|         relseek: relseek, | ||||
|         playpause: playpause, | ||||
|         hide: hideOverlay, | ||||
|         destroy: destroyPlugin | ||||
|     }; | ||||
| })(); | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -2,74 +2,143 @@ | ||||
| <html lang="en"> | ||||
|  | ||||
| <head> | ||||
|     <meta charset="utf-8"> | ||||
|     <title>⇆🎉 {{ title }}</title> | ||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/browser.css{{ ts }}"> | ||||
|     {%- if can_upload %} | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/upload.css{{ ts }}"> | ||||
|     {%- endif %} | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>⇆🎉 {{ title }}</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
| 	<link rel="stylesheet" media="screen" href="/.cpr/ui.css?_={{ ts }}"> | ||||
| 	<link rel="stylesheet" media="screen" href="/.cpr/browser.css?_={{ ts }}"> | ||||
| 	{%- if css %} | ||||
| 	<link rel="stylesheet" media="screen" href="{{ css }}?_={{ ts }}"> | ||||
| 	{%- endif %} | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
|     {%- if can_upload %} | ||||
|     {%- include 'upload.html' %} | ||||
|     {%- endif %} | ||||
|      | ||||
|     <h1 id="path"> | ||||
|         {%- for n in vpnodes %} | ||||
|         <a href="/{{ n[0] }}">{{ n[1] }}</a> | ||||
|         {%- endfor %} | ||||
|     </h1> | ||||
|      | ||||
|     {%- if can_read %} | ||||
|     {%- if prologue %} | ||||
|     <div id="pro" class="logue">{{ prologue }}</div> | ||||
|     {%- endif %} | ||||
| 	<div id="ops"></div> | ||||
|  | ||||
|     <table id="files"> | ||||
|         <thead> | ||||
|             <tr> | ||||
|                 <th></th> | ||||
|                 <th>File Name</th> | ||||
|                 <th sort="int">File Size</th> | ||||
|                 <th>Date</th> | ||||
|             </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
| 	<div id="op_search" class="opview"> | ||||
| 		{%- if have_tags_idx %} | ||||
| 		<div id="srch_form" class="tags"></div> | ||||
| 		{%- else %} | ||||
| 		<div id="srch_form"></div> | ||||
| 		{%- endif %} | ||||
| 		<div id="srch_q"></div> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="op_player" class="opview opbox opwide"></div> | ||||
|  | ||||
| 	<div id="op_bup" class="opview opbox act"> | ||||
| 		<div id="u2err"></div> | ||||
| 		<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||
| 			<input type="hidden" name="act" value="bput" /> | ||||
| 			<input type="file" name="f" multiple><br /> | ||||
| 			<input type="submit" value="start upload"> | ||||
| 		</form> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="op_mkdir" class="opview opbox act"> | ||||
| 		<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||
| 			<input type="hidden" name="act" value="mkdir" /> | ||||
| 			📂<input type="text" name="name" size="30"> | ||||
| 			<input type="submit" value="make directory"> | ||||
| 		</form> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="op_new_md" class="opview opbox"> | ||||
| 		<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||
| 			<input type="hidden" name="act" value="new_md" /> | ||||
| 			📝<input type="text" name="name" size="30"> | ||||
| 			<input type="submit" value="new markdown doc"> | ||||
| 		</form> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="op_msg" class="opview opbox act"> | ||||
| 		<form method="post" enctype="application/x-www-form-urlencoded" accept-charset="utf-8" action="{{ url_suf }}"> | ||||
| 			📟<input type="text" name="msg" size="30"> | ||||
| 			<input type="submit" value="send msg to server log"> | ||||
| 		</form> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="op_unpost" class="opview opbox"></div> | ||||
|  | ||||
| 	<div id="op_up2k" class="opview"></div> | ||||
|  | ||||
| 	<div id="op_cfg" class="opview opbox opwide"></div> | ||||
| 	 | ||||
| 	<h1 id="path"> | ||||
| 		<a href="#" id="entree" tt="show navpane (directory tree sidebar)$NHotkey: B">🌲</a> | ||||
| 		{%- for n in vpnodes %} | ||||
| 		<a href="/{{ n[0] }}">{{ n[1] }}</a> | ||||
| 		{%- endfor %} | ||||
| 	</h1> | ||||
| 	 | ||||
| 	<div id="tree"></div> | ||||
|  | ||||
| <div id="wrap"> | ||||
|  | ||||
| 	<div id="pro" class="logue">{{ logues[0] }}</div> | ||||
|  | ||||
| 	<table id="files"> | ||||
| 		<thead> | ||||
| 			<tr> | ||||
| 				<th name="lead"><span>c</span></th> | ||||
| 				<th name="href"><span>File Name</span></th> | ||||
| 				<th name="sz" sort="int"><span>Size</span></th> | ||||
| 				{%- for k in taglist %} | ||||
| 					{%- if k.startswith('.') %} | ||||
| 				<th name="tags/{{ k }}" sort="int"><span>{{ k[1:] }}</span></th> | ||||
| 					{%- else %} | ||||
| 				<th name="tags/{{ k }}"><span>{{ k[0]|upper }}{{ k[1:] }}</span></th> | ||||
| 					{%- endif %} | ||||
| 				{%- endfor %} | ||||
| 				<th name="ext"><span>T</span></th> | ||||
| 				<th name="ts"><span>Date</span></th> | ||||
| 			</tr> | ||||
| 		</thead> | ||||
| <tbody> | ||||
|  | ||||
| {%- for f in files %} | ||||
| <tr><td>{{ f[0] }}</td><td><a href="{{ f[1] }}">{{ f[2] }}</a></td><td>{{ f[3] }}</td><td>{{ f[4] }}</td></tr> | ||||
| <tr><td>{{ f.lead }}</td><td><a href="{{ f.href }}">{{ f.name|e }}</a></td><td>{{ f.sz }}</td> | ||||
| 	{%- if f.tags is defined %} | ||||
| 		{%- for k in taglist %} | ||||
| <td>{{ f.tags[k] }}</td> | ||||
| 		{%- endfor %} | ||||
| 	{%- endif %} | ||||
| <td>{{ f.ext }}</td><td>{{ f.dt }}</td></tr> | ||||
| {%- endfor %} | ||||
|  | ||||
|         </tbody> | ||||
|     </table> | ||||
|      | ||||
|     {%- if epilogue %} | ||||
|     <div id="epi" class="logue">{{ epilogue }}</div> | ||||
|     {%- endif %} | ||||
|     {%- endif %} | ||||
| 		</tbody> | ||||
| 	</table> | ||||
| 	 | ||||
| 	<div id="epi" class="logue">{{ logues[1] }}</div> | ||||
|  | ||||
|     <h2><a href="?h">control-panel</a></h2> | ||||
| 	<h2><a href="/?h">control-panel</a></h2> | ||||
| 	 | ||||
| 	<a href="#" id="repl">π</a> | ||||
|  | ||||
|     <div id="widget"> | ||||
|         <div id="wtoggle">♫</div> | ||||
|         <div id="widgeti"> | ||||
|             <div id="pctl"><a href="#" id="bprev">⏮</a><a href="#" id="bplay">▶</a><a href="#" id="bnext">⏭</a></div> | ||||
|             <canvas id="pvol" width="288" height="38"></canvas> | ||||
|             <canvas id="barpos"></canvas> | ||||
|             <canvas id="barbuf"></canvas> | ||||
|         </div> | ||||
|     </div> | ||||
|      | ||||
|     {%- if can_read %} | ||||
|     <script src="/.cpr/browser.js{{ ts }}"></script> | ||||
|     {%- endif %} | ||||
|      | ||||
|     {%- if can_upload %} | ||||
|     <script src="/.cpr/up2k.js{{ ts }}"></script> | ||||
|     {%- endif %} | ||||
| </div> | ||||
|  | ||||
| 	{%- if srv_info %} | ||||
| 	<div id="srv_info"><span>{{ srv_info }}</span></div> | ||||
| 	{%- endif %} | ||||
|  | ||||
| 	<div id="widget"></div> | ||||
|  | ||||
| 	<script> | ||||
| 		var acct = "{{ acct }}", | ||||
| 			perms = {{ perms }}, | ||||
| 			def_hcols = {{ def_hcols|tojson }}, | ||||
| 			have_up2k_idx = {{ have_up2k_idx|tojson }}, | ||||
| 			have_tags_idx = {{ have_tags_idx|tojson }}, | ||||
| 			have_mv = {{ have_mv|tojson }}, | ||||
| 			have_del = {{ have_del|tojson }}, | ||||
| 			have_unpost = {{ have_unpost|tojson }}, | ||||
| 			have_zip = {{ have_zip|tojson }}, | ||||
| 			readme = {{ readme|tojson }}; | ||||
| 	</script> | ||||
| 	<script src="/.cpr/util.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/browser.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/up2k.js?_={{ ts }}"></script> | ||||
| </body> | ||||
|  | ||||
| </html> | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										60
									
								
								copyparty/web/browser2.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								copyparty/web/browser2.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|  | ||||
| <head> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>{{ title }}</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
| 	<style> | ||||
| 		html{font-family:sans-serif} | ||||
| 		td{border:1px solid #999;border-width:1px 1px 0 0;padding:0 5px} | ||||
| 		a{display:block} | ||||
| 	</style> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
| 	{%- if srv_info %} | ||||
| 	<p><span>{{ srv_info }}</span></p> | ||||
| 	{%- endif %} | ||||
|  | ||||
| 	{%- if have_b_u %} | ||||
| 	<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||
| 		<input type="hidden" name="act" value="bput" /> | ||||
| 		<input type="file" name="f" multiple /><br /> | ||||
| 		<input type="submit" value="start upload" /> | ||||
| 	</form> | ||||
| 	<br /> | ||||
| 	{%- endif %} | ||||
|  | ||||
| 	{%- if logues[0] %} | ||||
| 	<div>{{ logues[0] }}</div><br /> | ||||
| 	{%- endif %} | ||||
|  | ||||
| 	<table id="files"> | ||||
| 		<thead> | ||||
| 			<tr> | ||||
| 				<th name="lead"><span>c</span></th> | ||||
| 				<th name="href"><span>File Name</span></th> | ||||
| 				<th name="sz" sort="int"><span>Size</span></th> | ||||
| 				<th name="ts"><span>Date</span></th> | ||||
| 			</tr> | ||||
| 		</thead> | ||||
| 		<tbody> | ||||
| <tr><td></td><td><a href="../{{ url_suf }}">parent folder</a></td><td>-</td><td>-</td></tr> | ||||
|  | ||||
| {%- for f in files %} | ||||
| <tr><td>{{ f.lead }}</td><td><a href="{{ f.href }}{{ url_suf }}">{{ f.name|e }}</a></td><td>{{ f.sz }}</td><td>{{ f.dt }}</td></tr> | ||||
| {%- endfor %} | ||||
|  | ||||
| 		</tbody> | ||||
| 	</table> | ||||
| 	 | ||||
| 	{%- if logues[1] %} | ||||
| 	<div>{{ logues[1] }}</div><br /> | ||||
| 	{%- endif %} | ||||
| 	 | ||||
| 	<h2><a href="/{{ url_suf }}{{ url_suf and '&' or '?' }}h">control-panel</a></h2> | ||||
|  | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										61
									
								
								copyparty/web/dbg-audio.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								copyparty/web/dbg-audio.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| var ofun = audio_eq.apply.bind(audio_eq); | ||||
| audio_eq.apply = function () { | ||||
|     var ac1 = mp.ac; | ||||
|     ofun(); | ||||
|     var ac = mp.ac, | ||||
|         w = 2048, | ||||
|         h = 256; | ||||
|  | ||||
|     if (!audio_eq.filters.length) { | ||||
|         audio_eq.ana = null; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     var can = ebi('fft_can'); | ||||
|     if (!can) { | ||||
|         can = mknod('canvas'); | ||||
|         can.setAttribute('id', 'fft_can'); | ||||
|         can.style.cssText = 'position:absolute;left:0;bottom:5em;width:' + w + 'px;height:' + h + 'px;z-index:9001'; | ||||
|         document.body.appendChild(can); | ||||
|         can.width = w; | ||||
|         can.height = h; | ||||
|     } | ||||
|     var cc = can.getContext('2d'); | ||||
|     if (!ac) | ||||
|         return; | ||||
|  | ||||
|     var ana = ac.createAnalyser(); | ||||
|     ana.smoothingTimeConstant = 0; | ||||
|     ana.fftSize = 8192; | ||||
|  | ||||
|     audio_eq.filters[0].connect(ana); | ||||
|     audio_eq.ana = ana; | ||||
|  | ||||
|     var buf = new Uint8Array(ana.frequencyBinCount), | ||||
|         colw = can.width / buf.length; | ||||
|  | ||||
|     cc.fillStyle = '#fc0'; | ||||
|     function draw() { | ||||
|         if (ana == audio_eq.ana) | ||||
|             requestAnimationFrame(draw); | ||||
|  | ||||
|         ana.getByteFrequencyData(buf); | ||||
|  | ||||
|         cc.clearRect(0, 0, can.width, can.height); | ||||
|  | ||||
|         /*var x = 0, w = 1; | ||||
|         for (var a = 0; a < buf.length; a++) { | ||||
|             cc.fillRect(x, h - buf[a], w, h); | ||||
|             x += w; | ||||
|         }*/ | ||||
|         var mul = Math.pow(w, 4) / buf.length; | ||||
|         for (var x = 0; x < w; x++) { | ||||
|             var a = Math.floor(Math.pow(x, 4) / mul), | ||||
|                 v = buf[a]; | ||||
|  | ||||
|             cc.fillRect(x, h - v, 1, v); | ||||
|         } | ||||
|     } | ||||
|     draw(); | ||||
| }; | ||||
| audio_eq.apply(); | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 248 B | 
| @@ -1,13 +1,17 @@ | ||||
| @font-face { | ||||
| 	font-family: 'scp'; | ||||
| 	src: local('Source Code Pro Regular'), local('SourceCodePro-Regular'), url(/.cpr/deps/scp.woff2) format('woff2'); | ||||
| } | ||||
| html, body { | ||||
| 	color: #333; | ||||
| 	background: #eee; | ||||
| 	font-family: sans-serif; | ||||
| 	line-height: 1.5em; | ||||
| } | ||||
| #repl { | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	right: .5em; | ||||
| 	border: none; | ||||
| 	color: inherit; | ||||
| 	background: none; | ||||
| } | ||||
| #mtw { | ||||
| 	display: none; | ||||
| } | ||||
| @@ -15,115 +19,12 @@ html, body { | ||||
| 	margin: 0 auto; | ||||
| 	padding: 0 1.5em; | ||||
| } | ||||
| pre, code, a { | ||||
| 	color: #480; | ||||
| 	background: #f7f7f7; | ||||
| 	border: .07em solid #ddd; | ||||
| 	border-radius: .2em; | ||||
| 	padding: .1em .3em; | ||||
| 	margin: 0 .1em; | ||||
| #toast { | ||||
| 	bottom: auto; | ||||
| 	top: 1.4em; | ||||
| } | ||||
| code { | ||||
| 	font-size: .96em; | ||||
| } | ||||
| pre, code { | ||||
| 	font-family: 'scp', monospace, monospace; | ||||
| 	white-space: pre-wrap; | ||||
| 	word-break: break-all; | ||||
| } | ||||
| pre { | ||||
| 	counter-reset: precode; | ||||
| } | ||||
| pre code { | ||||
| 	counter-increment: precode; | ||||
| 	display: inline-block; | ||||
| 	margin: 0 -.3em; | ||||
| 	padding: .4em .5em; | ||||
| 	border: none; | ||||
| 	border-bottom: 1px solid #cdc; | ||||
| 	min-width: calc(100% - .6em); | ||||
| 	line-height: 1.1em; | ||||
| } | ||||
| pre code:last-child { | ||||
| 	border-bottom: none; | ||||
| } | ||||
| pre code::before { | ||||
| 	content: counter(precode); | ||||
| 	-webkit-user-select: none; | ||||
| 	display: inline-block; | ||||
| 	text-align: right; | ||||
| 	font-size: .75em; | ||||
| 	color: #48a; | ||||
| 	width: 4em; | ||||
| 	padding-right: 1.5em; | ||||
| 	margin-left: -5.5em; | ||||
| } | ||||
| pre code:hover { | ||||
| 	background: #fec; | ||||
| 	color: #360; | ||||
| } | ||||
| h1, h2 { | ||||
| 	line-height: 1.5em; | ||||
| } | ||||
| h1 { | ||||
| 	font-size: 1.7em; | ||||
| 	text-align: center; | ||||
| 	border: 1em solid #777; | ||||
| 	border-width: .05em 0; | ||||
| 	margin: 3em 0; | ||||
| } | ||||
| h2 { | ||||
| 	font-size: 1.5em; | ||||
| 	font-weight: normal; | ||||
| 	background: #f7f7f7; | ||||
| 	border-top: .07em solid #fff; | ||||
| 	border-bottom: .07em solid #bbb; | ||||
| 	border-radius: .5em .5em 0 0; | ||||
| 	padding-left: .4em; | ||||
| 	margin-top: 3em; | ||||
| } | ||||
| h3 { | ||||
| 	border-bottom: .1em solid #999; | ||||
| } | ||||
| h1 a, h3 a, h5 a, | ||||
| h2 a, h4 a, h6 a { | ||||
| 	color: inherit; | ||||
| 	display: block; | ||||
| 	background: none; | ||||
| 	border: none; | ||||
| 	padding: 0; | ||||
| 	margin: 0; | ||||
| } | ||||
| #mp ul, | ||||
| #mp ol { | ||||
| 	border-left: .3em solid #ddd; | ||||
| } | ||||
| #m>ul, | ||||
| #m>ol { | ||||
| 	border-color: #bbb; | ||||
| } | ||||
| #mp ul>li { | ||||
| 	list-style-type: disc; | ||||
| } | ||||
| #mp ul>li, | ||||
| #mp ol>li { | ||||
| 	margin: .7em 0; | ||||
| } | ||||
| p>em, | ||||
| li>em { | ||||
| 	color: #c50; | ||||
| 	padding: .1em; | ||||
| 	border-bottom: .1em solid #bbb; | ||||
| } | ||||
| blockquote { | ||||
| 	font-family: serif; | ||||
| 	background: #f7f7f7; | ||||
| 	border: .07em dashed #ccc; | ||||
| 	padding: 0 2em; | ||||
| 	margin: 1em 0; | ||||
| } | ||||
| small { | ||||
| 	opacity: .8; | ||||
| a { | ||||
| 	text-decoration: none; | ||||
| } | ||||
| #toc { | ||||
| 	margin: 0 1em; | ||||
| @@ -159,7 +60,7 @@ small { | ||||
| 	z-index: 99; | ||||
| 	position: relative; | ||||
| 	display: inline-block; | ||||
| 	font-family: monospace, monospace; | ||||
| 	font-family: 'scp', monospace, monospace; | ||||
| 	font-weight: bold; | ||||
| 	font-size: 1.3em; | ||||
| 	line-height: .1em; | ||||
| @@ -171,14 +72,6 @@ small { | ||||
| 	color: #6b3; | ||||
| 	text-shadow: .02em 0 0 #6b3; | ||||
| } | ||||
| table { | ||||
| 	border-collapse: collapse; | ||||
| 	margin: 1em 0; | ||||
| } | ||||
| th, td { | ||||
| 	padding: .2em .5em; | ||||
| 	border: .12em solid #aaa; | ||||
| } | ||||
| blink { | ||||
| 	animation: blinker .7s cubic-bezier(.9, 0, .1, 1) infinite; | ||||
| } | ||||
| @@ -191,6 +84,36 @@ blink { | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| .mdo pre { | ||||
| 	counter-reset: precode; | ||||
| } | ||||
| .mdo pre code { | ||||
| 	counter-increment: precode; | ||||
| 	display: inline-block; | ||||
| 	border: none; | ||||
| 	border-bottom: 1px solid #cdc; | ||||
| 	min-width: calc(100% - .6em); | ||||
| } | ||||
| .mdo pre code:last-child { | ||||
| 	border-bottom: none; | ||||
| } | ||||
| .mdo pre code::before { | ||||
| 	content: counter(precode); | ||||
| 	-webkit-user-select: none; | ||||
| 	-moz-user-select: none; | ||||
| 	-ms-user-select: none; | ||||
| 	user-select: none; | ||||
| 	display: inline-block; | ||||
| 	text-align: right; | ||||
| 	font-size: .75em; | ||||
| 	color: #48a; | ||||
| 	width: 4em; | ||||
| 	padding-right: 1.5em; | ||||
| 	margin-left: -5.5em; | ||||
| } | ||||
|  | ||||
|  | ||||
| @media screen { | ||||
| 	html, body { | ||||
| 		margin: 0; | ||||
| @@ -207,34 +130,6 @@ blink { | ||||
| 	#mp { | ||||
| 		max-width: 52em; | ||||
| 		margin-bottom: 6em; | ||||
| 		word-break: break-word; | ||||
| 		overflow-wrap: break-word; | ||||
| 		word-wrap: break-word; /*ie*/ | ||||
| 	} | ||||
| 	a { | ||||
| 		color: #fff; | ||||
| 		background: #39b; | ||||
| 		text-decoration: none; | ||||
| 		padding: 0 .3em; | ||||
| 		border: none; | ||||
| 		border-bottom: .07em solid #079; | ||||
| 	} | ||||
| 	h2 { | ||||
| 		color: #fff; | ||||
| 		background: #555; | ||||
| 		margin-top: 2em; | ||||
| 		border-bottom: .22em solid #999; | ||||
| 		border-top: none; | ||||
| 	} | ||||
| 	h1 { | ||||
| 		color: #fff; | ||||
| 		background: #444; | ||||
| 		font-weight: normal; | ||||
| 		border-top: .4em solid #fb0; | ||||
| 		border-bottom: .4em solid #777; | ||||
| 		border-radius: 0 1em 0 1em; | ||||
| 		margin: 3em 0 1em 0; | ||||
| 		padding: .5em 0; | ||||
| 	} | ||||
| 	#mn { | ||||
| 		padding: 1.3em 0 .7em 1em; | ||||
| @@ -287,8 +182,36 @@ blink { | ||||
| 		color: #444; | ||||
| 		background: none; | ||||
| 		text-decoration: underline; | ||||
| 		margin: 0 .1em; | ||||
| 		padding: 0 .3em; | ||||
| 		border: none; | ||||
| 	} | ||||
| 	#mh a:hover { | ||||
| 		color: #000; | ||||
| 		background: #ddd; | ||||
| 	} | ||||
| 	#toolsbox { | ||||
| 		overflow: hidden; | ||||
| 		display: inline-block; | ||||
| 		background: #eee; | ||||
| 		height: 1.5em; | ||||
| 		padding: 0 .2em; | ||||
| 		margin: 0 .2em; | ||||
| 		position: absolute; | ||||
| 	} | ||||
| 	#toolsbox.open { | ||||
| 		height: auto; | ||||
| 		overflow: visible; | ||||
| 		background: #eee; | ||||
| 		box-shadow: 0 .2em .2em #ccc; | ||||
| 		padding-bottom: .2em; | ||||
| 	} | ||||
| 	#toolsbox a { | ||||
| 		display: block; | ||||
| 	} | ||||
| 	#toolsbox a+a { | ||||
| 		text-decoration: none; | ||||
| 	} | ||||
|  | ||||
|  | ||||
|  | ||||
| @@ -309,51 +232,6 @@ blink { | ||||
| 	html.dark #toc li { | ||||
| 		border-width: 0; | ||||
| 	} | ||||
| 	html.dark #mp a { | ||||
| 		background: #057; | ||||
| 	} | ||||
| 	html.dark #mp h1 a, html.dark #mp h4 a, | ||||
| 	html.dark #mp h2 a, html.dark #mp h5 a, | ||||
| 	html.dark #mp h3 a, html.dark #mp h6 a { | ||||
| 		color: inherit; | ||||
| 		background: none; | ||||
| 	} | ||||
| 	html.dark pre, | ||||
| 	html.dark code { | ||||
| 		color: #8c0; | ||||
| 		background: #1a1a1a; | ||||
| 		border: .07em solid #333; | ||||
| 	} | ||||
| 	html.dark #mp ul, | ||||
| 	html.dark #mp ol { | ||||
| 		border-color: #444; | ||||
| 	} | ||||
| 	html.dark #m>ul, | ||||
| 	html.dark #m>ol { | ||||
| 		border-color: #555; | ||||
| 	} | ||||
| 	html.dark p>em, | ||||
| 	html.dark li>em { | ||||
| 		color: #f94; | ||||
| 		border-color: #666; | ||||
| 	} | ||||
| 	html.dark h1 { | ||||
| 		background: #383838; | ||||
| 		border-top: .4em solid #b80; | ||||
| 		border-bottom: .4em solid #4c4c4c; | ||||
| 	} | ||||
| 	html.dark h2 { | ||||
| 		background: #444; | ||||
| 		border-bottom: .22em solid #555; | ||||
| 	} | ||||
| 	html.dark td, | ||||
| 	html.dark th { | ||||
| 		border-color: #444; | ||||
| 	} | ||||
| 	html.dark blockquote { | ||||
| 		background: #282828; | ||||
| 		border: .07em dashed #444; | ||||
| 	} | ||||
| 	html.dark #mn a:not(:last-child)::after { | ||||
| 		border-color: rgba(255,255,255,0.3); | ||||
| 	} | ||||
| @@ -371,6 +249,17 @@ blink { | ||||
| 		color: #ccc; | ||||
| 		background: none; | ||||
| 	} | ||||
| 	html.dark #mh a:hover { | ||||
| 		background: #333; | ||||
| 		color: #fff; | ||||
| 	} | ||||
| 	html.dark #toolsbox { | ||||
| 		background: #222; | ||||
| 	} | ||||
| 	html.dark #toolsbox.open { | ||||
| 		box-shadow: 0 .2em .2em #069; | ||||
| 		border-radius: 0 0 .4em .4em; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @media screen and (min-width: 66em) { | ||||
| @@ -448,12 +337,15 @@ blink { | ||||
| 		mso-footer-margin: .6in; | ||||
| 		mso-paper-source: 0; | ||||
| 	} | ||||
| 	a { | ||||
| 	.mdo a { | ||||
| 		color: #079; | ||||
| 		text-decoration: none; | ||||
| 		border-bottom: .07em solid #4ac; | ||||
| 		padding: 0 .3em; | ||||
| 	} | ||||
| 	#repl { | ||||
| 		display: none; | ||||
| 	} | ||||
| 	#toc>ul { | ||||
| 		border-left: .1em solid #84c4dd; | ||||
| 	} | ||||
| @@ -478,18 +370,20 @@ blink { | ||||
| 	a[ctr]::before { | ||||
| 		content: attr(ctr) '. '; | ||||
| 	} | ||||
| 	h1 { | ||||
| 	.mdo h1 { | ||||
| 		margin: 2em 0; | ||||
| 	} | ||||
| 	h2 { | ||||
| 	.mdo h2 { | ||||
| 		margin: 2em 0 0 0; | ||||
| 	} | ||||
| 	h1, h2, h3 { | ||||
| 	.mdo h1, | ||||
| 	.mdo h2, | ||||
| 	.mdo h3 { | ||||
| 		page-break-inside: avoid; | ||||
| 	} | ||||
| 	h1::after, | ||||
| 	h2::after, | ||||
| 	h3::after { | ||||
| 	.mdo h1::after, | ||||
| 	.mdo h2::after, | ||||
| 	.mdo h3::after { | ||||
| 		content: 'orz'; | ||||
| 		color: transparent; | ||||
| 		display: block; | ||||
| @@ -497,20 +391,20 @@ blink { | ||||
| 		padding: 4em 0 0 0; | ||||
| 		margin: 0 0 -5em 0; | ||||
| 	} | ||||
| 	p { | ||||
| 	.mdo p { | ||||
| 		page-break-inside: avoid; | ||||
| 	} | ||||
| 	table { | ||||
| 	.mdo table { | ||||
| 		page-break-inside: auto; | ||||
| 	} | ||||
| 	tr { | ||||
| 	.mdo tr { | ||||
| 		page-break-inside: avoid; | ||||
| 		page-break-after: auto; | ||||
| 	} | ||||
| 	thead { | ||||
| 	.mdo thead { | ||||
| 		display: table-header-group; | ||||
| 	} | ||||
| 	tfoot { | ||||
| 	.mdo tfoot { | ||||
| 		display: table-footer-group; | ||||
| 	} | ||||
| 	#mp a.vis::after { | ||||
| @@ -518,39 +412,32 @@ blink { | ||||
| 		border-bottom: 1px solid #bbb; | ||||
| 		color: #444; | ||||
| 	} | ||||
| 	blockquote { | ||||
| 	.mdo blockquote { | ||||
| 		border-color: #555; | ||||
| 	} | ||||
| 	code { | ||||
| 	.mdo code { | ||||
| 		border-color: #bbb; | ||||
| 	} | ||||
| 	pre, pre code { | ||||
| 	.mdo pre, | ||||
| 	.mdo pre code { | ||||
| 		border-color: #999; | ||||
| 	} | ||||
| 	pre code::before { | ||||
| 	.mdo pre code::before { | ||||
| 		color: #058; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| 	 | ||||
| 	html.dark a { | ||||
| 	html.dark .mdo a { | ||||
| 		color: #000; | ||||
| 	} | ||||
| 	html.dark pre, | ||||
| 	html.dark code { | ||||
| 	html.dark .mdo pre, | ||||
| 	html.dark .mdo code { | ||||
| 		color: #240; | ||||
| 	} | ||||
| 	html.dark p>em, | ||||
| 	html.dark li>em { | ||||
| 	html.dark .mdo p>em, | ||||
| 	html.dark .mdo li>em, | ||||
| 	html.dark .mdo td>em { | ||||
| 		color: #940; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* | ||||
| *[data-ln]:before { | ||||
| 	content: attr(data-ln); | ||||
| 	font-size: .8em; | ||||
| 	margin: 0 .4em; | ||||
| 	color: #f0c; | ||||
| } | ||||
| */ | ||||
| @@ -1,11 +1,12 @@ | ||||
| <!DOCTYPE html><html><head> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>📝🎉 {{ title }}</title> <!-- 📜 --> | ||||
| 	<title>📝🎉 {{ title }}</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=0.7"> | ||||
| 	<link href="/.cpr/md.css" rel="stylesheet"> | ||||
| 	<link rel="stylesheet" href="/.cpr/ui.css?_={{ ts }}"> | ||||
| 	<link rel="stylesheet" href="/.cpr/md.css?_={{ ts }}"> | ||||
| 	{%- if edit %} | ||||
| 	<link href="/.cpr/md2.css" rel="stylesheet"> | ||||
| 	<link rel="stylesheet" href="/.cpr/md2.css?_={{ ts }}"> | ||||
| 	{%- endif %} | ||||
| </head> | ||||
| <body> | ||||
| @@ -14,19 +15,26 @@ | ||||
| 		<a id="lightswitch" href="#">go dark</a> | ||||
| 		<a id="navtoggle" href="#">hide nav</a> | ||||
| 		{%- if edit %} | ||||
| 			<a id="save" href="?edit">save</a> | ||||
| 			<a id="sbs" href="#">sbs</a> | ||||
| 			<a id="nsbs" href="#">editor</a> | ||||
| 			<a id="help" href="#">help</a> | ||||
| 			<a id="save" href="?edit" tt="Hotkey: ctrl-s">save</a> | ||||
| 			<a id="sbs" href="#" tt="editor and preview side by side">sbs</a> | ||||
| 			<a id="nsbs" href="#" tt="switch between editor and preview$NHotkey: ctrl-e">editor</a> | ||||
| 			<div id="toolsbox"> | ||||
| 				<a id="tools" href="#">tools</a> | ||||
| 				<a id="fmt_table" href="#">prettify table (ctrl-k)</a> | ||||
| 				<a id="iter_uni" href="#">non-ascii: iterate (ctrl-u)</a> | ||||
| 				<a id="mark_uni" href="#">non-ascii: markup</a> | ||||
| 				<a id="cfg_uni" href="#">non-ascii: whitelist</a> | ||||
| 				<a id="help" href="#">help</a> | ||||
| 			</div> | ||||
| 		{%- else %} | ||||
| 			<a href="?edit">edit (basic)</a> | ||||
| 			<a href="?edit2">edit (fancy)</a> | ||||
| 			<a href="?edit" tt="good: higher performance$Ngood: same document width as viewer$Nbad: assumes you know markdown">edit (basic)</a> | ||||
| 			<a href="?edit2" tt="not in-house so probably less buggy">edit (fancy)</a> | ||||
| 			<a href="?raw">view raw</a> | ||||
| 		{%- endif %} | ||||
| 	</div> | ||||
| 	<div id="toc"></div> | ||||
| 	<div id="mtw"> | ||||
| 		<textarea id="mt">{{ md }}</textarea> | ||||
| 		<textarea id="mt" autocomplete="off">{{ md }}</textarea> | ||||
| 	</div> | ||||
| 	<div id="mw"> | ||||
| 		<div id="ml"> | ||||
| @@ -35,17 +43,21 @@ | ||||
| 				if you're still reading this, check that javascript is allowed | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div id="mp"></div> | ||||
| 		<div id="mp" class="mdo"></div> | ||||
| 	</div> | ||||
| 	<a href="#" id="repl">π</a> | ||||
| 	 | ||||
| 	{%- if edit %} | ||||
| 	<div id="helpbox"> | ||||
| 		<textarea> | ||||
| 		<textarea autocomplete="off"> | ||||
|  | ||||
| write markdown (most html is 🙆 too) | ||||
|  | ||||
| ## hotkey list | ||||
| * `Ctrl-S` to save | ||||
| * `Ctrl-E` to toggle mode | ||||
| * `Ctrl-K` to prettyprint a table | ||||
| * `Ctrl-U` to iterate non-ascii chars | ||||
| * `Ctrl-H` / `Ctrl-Shift-H` to create a header | ||||
| * `TAB` / `Shift-TAB` to indent/dedent a selection | ||||
|  | ||||
| @@ -113,35 +125,33 @@ write markdown (most html is 🙆 too) | ||||
| 	 | ||||
| 	<script> | ||||
|  | ||||
| var link_md_as_html = false;  // TODO (does nothing) | ||||
| var last_modified = {{ lastmod }}; | ||||
| var md_opt = { | ||||
| 	link_md_as_html: false, | ||||
| 	allow_plugins: {{ md_plug }}, | ||||
| 	modpoll_freq: {{ md_chk_rate }} | ||||
| }; | ||||
|  | ||||
| (function () { | ||||
|     var btn = document.getElementById("lightswitch"); | ||||
|     var toggle = function (e) { | ||||
| 		if (e) e.preventDefault(); | ||||
|         var dark = !document.documentElement.getAttribute("class"); | ||||
|         document.documentElement.setAttribute("class", dark ? "dark" : ""); | ||||
|         btn.innerHTML = "go " + (dark ? "light" : "dark"); | ||||
|         if (window.localStorage) | ||||
|             localStorage.setItem('darkmode', dark ? 1 : 0); | ||||
|     }; | ||||
|     btn.onclick = toggle; | ||||
|     if (window.localStorage && localStorage.getItem('darkmode') == 1) | ||||
| 		toggle(); | ||||
|     var l = localStorage, | ||||
| 		drk = l.getItem('lightmode') != 1, | ||||
| 		btn = document.getElementById("lightswitch"), | ||||
| 		f = function (e) { | ||||
| if (e) { e.preventDefault(); drk = !drk; } | ||||
| document.documentElement.setAttribute("class", drk? "dark":"light"); | ||||
| btn.innerHTML = "go " + (drk ? "light":"dark"); | ||||
| l.setItem('lightmode', drk? 0:1); | ||||
|     	}; | ||||
| 	 | ||||
| 	btn.onclick = f; | ||||
| 	f(); | ||||
| })(); | ||||
|  | ||||
| if (!String.startsWith) { | ||||
| 	String.prototype.startsWith = function(s, i) { | ||||
| 		i = i>0 ? i|0 : 0; | ||||
| 		return this.substring(i, i + s.length) === s; | ||||
| 	}; | ||||
| } | ||||
|  | ||||
| 	</script> | ||||
| 	<script src="/.cpr/deps/marked.full.js"></script> | ||||
| 	<script src="/.cpr/md.js"></script> | ||||
|     <script src="/.cpr/util.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/deps/marked.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/md.js?_={{ ts }}"></script> | ||||
| 	{%- if edit %} | ||||
| 		<script src="/.cpr/md2.js"></script> | ||||
| 	<script src="/.cpr/md2.js?_={{ ts }}"></script> | ||||
| 	{%- endif %} | ||||
| </body></html> | ||||
|   | ||||
| @@ -1,10 +1,12 @@ | ||||
| var dom_toc = document.getElementById('toc'); | ||||
| var dom_wrap = document.getElementById('mw'); | ||||
| var dom_hbar = document.getElementById('mh'); | ||||
| var dom_nav = document.getElementById('mn'); | ||||
| var dom_pre = document.getElementById('mp'); | ||||
| var dom_src = document.getElementById('mt'); | ||||
| var dom_navtgl = document.getElementById('navtoggle'); | ||||
| "use strict"; | ||||
|  | ||||
| var dom_toc = ebi('toc'); | ||||
| var dom_wrap = ebi('mw'); | ||||
| var dom_hbar = ebi('mh'); | ||||
| var dom_nav = ebi('mn'); | ||||
| var dom_pre = ebi('mp'); | ||||
| var dom_src = ebi('mt'); | ||||
| var dom_navtgl = ebi('navtoggle'); | ||||
|  | ||||
|  | ||||
| // chrome 49 needs this | ||||
| @@ -18,21 +20,8 @@ var dbg = function () { }; | ||||
| // dbg = console.log | ||||
|  | ||||
|  | ||||
| function hesc(txt) { | ||||
|     return txt.replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">"); | ||||
| } | ||||
|  | ||||
|  | ||||
| function cls(dom, name, add) { | ||||
|     var re = new RegExp('(^| )' + name + '( |$)'); | ||||
|     var lst = (dom.getAttribute('class') + '').replace(re, "$1$2").replace(/  /, ""); | ||||
|     dom.setAttribute('class', lst + (add ? ' ' + name : '')); | ||||
| } | ||||
|  | ||||
|  | ||||
| function static(obj) { | ||||
|     return JSON.parse(JSON.stringify(obj)); | ||||
| } | ||||
| // plugins | ||||
| var md_plug = {}; | ||||
|  | ||||
|  | ||||
| // dodge browser issues | ||||
| @@ -40,7 +29,7 @@ function static(obj) { | ||||
|     var ua = navigator.userAgent; | ||||
|     if (ua.indexOf(') Gecko/') !== -1 && /Linux| Mac /.exec(ua)) { | ||||
|         // necessary on ff-68.7 at least | ||||
|         var s = document.createElement('style'); | ||||
|         var s = mknod('style'); | ||||
|         s.innerHTML = '@page { margin: .5in .6in .8in .6in; }'; | ||||
|         console.log(s.innerHTML); | ||||
|         document.head.appendChild(s); | ||||
| @@ -59,7 +48,7 @@ function static(obj) { | ||||
|         if (a > 0) | ||||
|             loc.push(n[a]); | ||||
|  | ||||
|         var dec = hesc(decodeURIComponent(n[a])); | ||||
|         var dec = esc(uricom_dec(n[a])[0]); | ||||
|  | ||||
|         nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>'); | ||||
|     } | ||||
| @@ -67,6 +56,26 @@ function static(obj) { | ||||
| })(); | ||||
|  | ||||
|  | ||||
| // image load handler | ||||
| var img_load = (function () { | ||||
|     var r = {}; | ||||
|     r.callbacks = []; | ||||
|  | ||||
|     function fire() { | ||||
|         for (var a = 0; a < r.callbacks.length; a++) | ||||
|             r.callbacks[a](); | ||||
|     } | ||||
|  | ||||
|     var timeout = null; | ||||
|     r.done = function () { | ||||
|         clearTimeout(timeout); | ||||
|         timeout = setTimeout(fire, 500); | ||||
|     }; | ||||
|  | ||||
|     return r; | ||||
| })(); | ||||
|  | ||||
|  | ||||
| // faster than replacing the entire html (chrome 1.8x, firefox 1.6x) | ||||
| function copydom(src, dst, lv) { | ||||
|     var sc = src.childNodes, | ||||
| @@ -154,13 +163,110 @@ function copydom(src, dst, lv) { | ||||
| } | ||||
|  | ||||
|  | ||||
| function md_plug_err(ex, js) { | ||||
|     var errbox = ebi('md_errbox'); | ||||
|     if (errbox) | ||||
|         errbox.parentNode.removeChild(errbox); | ||||
|  | ||||
|     if (!ex) | ||||
|         return; | ||||
|  | ||||
|     var msg = (ex + '').split('\n')[0]; | ||||
|     var ln = ex.lineNumber; | ||||
|     var o = null; | ||||
|     if (ln) { | ||||
|         msg = "Line " + ln + ", " + msg; | ||||
|         var lns = js.split('\n'); | ||||
|         if (ln < lns.length) { | ||||
|             o = mknod('span'); | ||||
|             o.style.cssText = "color:#ac2;font-size:.9em;font-family:'scp',monospace,monospace;display:block"; | ||||
|             o.textContent = lns[ln - 1]; | ||||
|         } | ||||
|     } | ||||
|     errbox = mknod('div'); | ||||
|     errbox.setAttribute('id', 'md_errbox'); | ||||
|     errbox.style.cssText = 'position:absolute;top:0;left:0;padding:1em .5em;background:#2b2b2b;color:#fc5' | ||||
|     errbox.textContent = msg; | ||||
|     errbox.onclick = function () { | ||||
|         modal.alert('<pre>' + esc(ex.stack) + '</pre>'); | ||||
|     }; | ||||
|     if (o) { | ||||
|         errbox.appendChild(o); | ||||
|         errbox.style.padding = '.25em .5em'; | ||||
|     } | ||||
|     dom_nav.appendChild(errbox); | ||||
|  | ||||
|     try { | ||||
|         console.trace(); | ||||
|     } | ||||
|     catch (ex2) { } | ||||
| } | ||||
|  | ||||
|  | ||||
| function load_plug(md_text, plug_type) { | ||||
|     if (!md_opt.allow_plugins) | ||||
|         return md_text; | ||||
|  | ||||
|     var find = '\n```copyparty_' + plug_type + '\n'; | ||||
|     var ofs = md_text.indexOf(find); | ||||
|     if (ofs === -1) | ||||
|         return md_text; | ||||
|  | ||||
|     var ofs2 = md_text.indexOf('\n```', ofs + 1); | ||||
|     if (ofs2 == -1) | ||||
|         return md_text; | ||||
|  | ||||
|     var js = md_text.slice(ofs + find.length, ofs2 + 1); | ||||
|     var md = md_text.slice(0, ofs + 1) + md_text.slice(ofs2 + 4); | ||||
|  | ||||
|     var old_plug = md_plug[plug_type]; | ||||
|     if (!old_plug || old_plug[1] != js) { | ||||
|         js = 'const x = { ' + js + ' }; x;'; | ||||
|         try { | ||||
|             var x = eval(js); | ||||
|         } | ||||
|         catch (ex) { | ||||
|             md_plug[plug_type] = null; | ||||
|             md_plug_err(ex, js); | ||||
|             return md; | ||||
|         } | ||||
|         if (x['ctor']) { | ||||
|             x['ctor'](); | ||||
|             delete x['ctor']; | ||||
|         } | ||||
|         md_plug[plug_type] = [x, js]; | ||||
|     } | ||||
|  | ||||
|     return md; | ||||
| } | ||||
|  | ||||
|  | ||||
| function convert_markdown(md_text, dest_dom) { | ||||
|     marked.setOptions({ | ||||
|     md_text = md_text.replace(/\r/g, ''); | ||||
|  | ||||
|     md_plug_err(null); | ||||
|     md_text = load_plug(md_text, 'pre'); | ||||
|     md_text = load_plug(md_text, 'post'); | ||||
|  | ||||
|     var marked_opts = { | ||||
|         //headerPrefix: 'h-', | ||||
|         breaks: true, | ||||
|         gfm: true | ||||
|     }); | ||||
|     var md_html = marked(md_text); | ||||
|     }; | ||||
|  | ||||
|     var ext = md_plug['pre']; | ||||
|     if (ext) | ||||
|         Object.assign(marked_opts, ext[0]); | ||||
|  | ||||
|     try { | ||||
|         var md_html = marked(md_text, marked_opts); | ||||
|     } | ||||
|     catch (ex) { | ||||
|         if (ext) | ||||
|             md_plug_err(ex, ext[1]); | ||||
|  | ||||
|         throw ex; | ||||
|     } | ||||
|     var md_dom = new DOMParser().parseFromString(md_html, "text/html").body; | ||||
|  | ||||
|     var nodes = md_dom.getElementsByTagName('a'); | ||||
| @@ -196,7 +302,7 @@ function convert_markdown(md_text, dest_dom) { | ||||
|     } | ||||
|  | ||||
|     // separate <code> for each line in <pre> | ||||
|     var nodes = md_dom.getElementsByTagName('pre'); | ||||
|     nodes = md_dom.getElementsByTagName('pre'); | ||||
|     for (var a = nodes.length - 1; a >= 0; a--) { | ||||
|         var el = nodes[a]; | ||||
|  | ||||
| @@ -209,7 +315,7 @@ function convert_markdown(md_text, dest_dom) { | ||||
|             continue; | ||||
|  | ||||
|         var nline = parseInt(el.getAttribute('data-ln')) + 1; | ||||
|         var lines = el.innerHTML.replace(/\r?\n<\/code>$/i, '</code>').split(/\r?\n/g); | ||||
|         var lines = el.innerHTML.replace(/\n<\/code>$/i, '</code>').split(/\n/g); | ||||
|         for (var b = 0; b < lines.length - 1; b++) | ||||
|             lines[b] += '</code>\n<code data-ln="' + (nline + b) + '">'; | ||||
|  | ||||
| @@ -242,12 +348,33 @@ function convert_markdown(md_text, dest_dom) { | ||||
|         el.innerHTML = '<a href="#' + id + '">' + el.innerHTML + '</a>'; | ||||
|     } | ||||
|  | ||||
|     ext = md_plug['post']; | ||||
|     if (ext && ext[0].render) | ||||
|         try { | ||||
|             ext[0].render(md_dom); | ||||
|         } | ||||
|         catch (ex) { | ||||
|             md_plug_err(ex, ext[1]); | ||||
|         } | ||||
|  | ||||
|     copydom(md_dom, dest_dom, 0); | ||||
|  | ||||
|     var imgs = dest_dom.getElementsByTagName('img'); | ||||
|     for (var a = 0, aa = imgs.length; a < aa; a++) | ||||
|         imgs[a].onload = img_load.done; | ||||
|  | ||||
|     if (ext && ext[0].render2) | ||||
|         try { | ||||
|             ext[0].render2(dest_dom); | ||||
|         } | ||||
|         catch (ex) { | ||||
|             md_plug_err(ex, ext[1]); | ||||
|         } | ||||
| } | ||||
|  | ||||
|  | ||||
| function init_toc() { | ||||
|     var loader = document.getElementById('ml'); | ||||
|     var loader = ebi('ml'); | ||||
|     loader.parentNode.removeChild(loader); | ||||
|  | ||||
|     var anchors = [];  // list of toc entries, complex objects | ||||
| @@ -281,7 +408,12 @@ function init_toc() { | ||||
|  | ||||
|             elm.childNodes[0].setAttribute('ctr', ctr.slice(0, lv).join('.')); | ||||
|  | ||||
|             html.push('<li>' + elm.innerHTML + '</li>'); | ||||
|             var elm2 = elm.cloneNode(true); | ||||
|             elm2.childNodes[0].textContent = elm.textContent; | ||||
|             while (elm2.childNodes.length > 1) | ||||
|                 elm2.removeChild(elm2.childNodes[1]); | ||||
|  | ||||
|             html.push('<li>' + elm2.innerHTML + '</li>'); | ||||
|  | ||||
|             if (anchor != null) | ||||
|                 anchors.push(anchor); | ||||
| @@ -365,6 +497,7 @@ function init_toc() { | ||||
| // "main" :p | ||||
| convert_markdown(dom_src.value, dom_pre); | ||||
| var toc = init_toc(); | ||||
| img_load.callbacks = [toc.refresh]; | ||||
|  | ||||
|  | ||||
| // scroll handler | ||||
| @@ -399,11 +532,12 @@ dom_navtgl.onclick = function () { | ||||
|     dom_navtgl.innerHTML = hidden ? 'show nav' : 'hide nav'; | ||||
|     dom_nav.style.display = hidden ? 'none' : 'block'; | ||||
|  | ||||
|     if (window.localStorage) | ||||
|         localStorage.setItem('hidenav', hidden ? 1 : 0); | ||||
|  | ||||
|     swrite('hidenav', hidden ? 1 : 0); | ||||
|     redraw(); | ||||
| }; | ||||
|  | ||||
| if (window.localStorage && localStorage.getItem('hidenav') == 1) | ||||
| if (sread('hidenav') == 1) | ||||
|     dom_navtgl.onclick(); | ||||
|  | ||||
| if (window['tt']) | ||||
|     tt.init(); | ||||
|   | ||||
| @@ -1,108 +1,110 @@ | ||||
| #toc { | ||||
|     display: none; | ||||
| 	display: none; | ||||
| } | ||||
| #mtw { | ||||
|     display: block; | ||||
|     position: fixed; | ||||
|     left: .5em; | ||||
|     bottom: 0; | ||||
|     width: calc(100% - 56em); | ||||
| 	display: block; | ||||
| 	position: fixed; | ||||
| 	left: .5em; | ||||
| 	bottom: 0; | ||||
| 	width: calc(100% - 56em); | ||||
| } | ||||
| #mw { | ||||
|     left: calc(100% - 55em); | ||||
|     overflow-y: auto; | ||||
|     position: fixed; | ||||
|     bottom: 0; | ||||
| 	left: calc(100% - 55em); | ||||
| 	overflow-y: auto; | ||||
| 	position: fixed; | ||||
| 	bottom: 0; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* single-screen */ | ||||
| #mtw.preview, | ||||
| #mw.editor { | ||||
|     opacity: 0; | ||||
|     z-index: 1; | ||||
| 	opacity: 0; | ||||
| 	z-index: 1; | ||||
| } | ||||
| #mw.preview, | ||||
| #mtw.editor { | ||||
|     z-index: 5; | ||||
| 	z-index: 5; | ||||
| } | ||||
| #mtw.single, | ||||
| #mw.single { | ||||
|     margin: 0; | ||||
|     left: 1em; | ||||
|     left: max(1em, calc((100% - 56em) / 2)); | ||||
| 	margin: 0; | ||||
| 	left: 1em; | ||||
| 	left: max(1em, calc((100% - 56em) / 2)); | ||||
| } | ||||
| #mtw.single { | ||||
|     width: 55em; | ||||
|     width: min(55em, calc(100% - 2em)); | ||||
| 	width: 55em; | ||||
| 	width: min(55em, calc(100% - 2em)); | ||||
| } | ||||
|  | ||||
|  | ||||
| #mp { | ||||
|     position: relative; | ||||
| 	position: relative; | ||||
| } | ||||
| #mt, #mtr { | ||||
|     width: 100%; | ||||
|     height: calc(100% - 1px); | ||||
|     color: #444; | ||||
|     background: #f7f7f7; | ||||
|     border: 1px solid #999; | ||||
|     outline: none; | ||||
|     padding: 0; | ||||
|     margin: 0; | ||||
|     font-family: 'consolas', monospace, monospace; | ||||
|     white-space: pre-wrap; | ||||
|     word-break: break-word; | ||||
|     overflow-wrap: break-word; | ||||
|     word-wrap: break-word; /*ie*/ | ||||
|     overflow-y: scroll; | ||||
|     line-height: 1.3em; | ||||
|     font-size: .9em; | ||||
|     position: relative; | ||||
|     scrollbar-color: #eb0 #f7f7f7; | ||||
| 	width: 100%; | ||||
| 	height: calc(100% - 1px); | ||||
| 	color: #444; | ||||
| 	background: #f7f7f7; | ||||
| 	border: 1px solid #999; | ||||
| 	outline: none; | ||||
| 	padding: 0; | ||||
| 	margin: 0; | ||||
| 	font-family: 'scp', monospace, monospace; | ||||
| 	white-space: pre-wrap; | ||||
| 	word-break: break-word; | ||||
| 	overflow-wrap: break-word; | ||||
| 	word-wrap: break-word; /*ie*/ | ||||
| 	overflow-y: scroll; | ||||
| 	line-height: 1.3em; | ||||
| 	font-size: .9em; | ||||
| 	position: relative; | ||||
| 	scrollbar-color: #eb0 #f7f7f7; | ||||
| } | ||||
| html.dark #mt { | ||||
|     color: #eee; | ||||
|     background: #222; | ||||
|     border: 1px solid #777; | ||||
|     scrollbar-color: #b80 #282828; | ||||
| 	color: #eee; | ||||
| 	background: #222; | ||||
| 	border: 1px solid #777; | ||||
| 	scrollbar-color: #b80 #282828; | ||||
| } | ||||
| #mtr { | ||||
|     position: absolute; | ||||
|     top: 0; | ||||
|     left: 0; | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| } | ||||
| #save.force-save { | ||||
|     color: #400; | ||||
|     background: #f97; | ||||
|     border-radius: .15em; | ||||
| 	color: #400; | ||||
| 	background: #f97; | ||||
| 	border-radius: .15em; | ||||
| } | ||||
| html.dark #save.force-save { | ||||
| 	color: #fca; | ||||
| 	background: #720; | ||||
| } | ||||
| #save.disabled { | ||||
|     opacity: .4; | ||||
| 	opacity: .4; | ||||
| } | ||||
| #helpbox { | ||||
|     display: none; | ||||
|     position: fixed; | ||||
|     background: #f7f7f7; | ||||
|     box-shadow: 0 .5em 2em #777; | ||||
|     border-radius: .4em; | ||||
|     padding: 2em; | ||||
|     top: 4em; | ||||
|     overflow-y: auto; | ||||
|     height: calc(100% - 12em); | ||||
|     left: calc(50% - 15em); | ||||
|     right: 0; | ||||
|     width: 30em; | ||||
|     z-index: 9001; | ||||
| 	background: #f7f7f7; | ||||
| 	border-radius: .4em; | ||||
| 	z-index: 9001; | ||||
| 	display: none; | ||||
| 	position: fixed; | ||||
| 	padding: 2em; | ||||
| 	top: 4em; | ||||
| 	overflow-y: auto; | ||||
| 	box-shadow: 0 .5em 2em #777; | ||||
| 	height: calc(100% - 12em); | ||||
| 	left: calc(50% - 15em); | ||||
| 	right: 0; | ||||
| 	width: 30em; | ||||
| } | ||||
| #helpclose { | ||||
|     display: block; | ||||
| 	display: block; | ||||
| } | ||||
| html.dark #helpbox { | ||||
|     background: #222; | ||||
|     box-shadow: 0 .5em 2em #444; | ||||
|     border: 1px solid #079; | ||||
|     border-width: 1px 0; | ||||
| 	box-shadow: 0 .5em 2em #444; | ||||
| 	background: #222; | ||||
| 	border: 1px solid #079; | ||||
| 	border-width: 1px 0; | ||||
| } | ||||
|  | ||||
| # mt {opacity: .5;top:1px} | ||||
|   | ||||
| @@ -1,16 +1,25 @@ | ||||
| "use strict"; | ||||
|  | ||||
|  | ||||
| // server state | ||||
| var server_md = dom_src.value; | ||||
|  | ||||
|  | ||||
| // the non-ascii whitelist | ||||
| var esc_uni_whitelist = '\\n\\t\\x20-\\x7eÆØÅæøå'; | ||||
| var js_uni_whitelist = eval('\'' + esc_uni_whitelist + '\''); | ||||
|  | ||||
|  | ||||
| // dom nodes | ||||
| var dom_swrap = document.getElementById('mtw'); | ||||
| var dom_sbs = document.getElementById('sbs'); | ||||
| var dom_nsbs = document.getElementById('nsbs'); | ||||
| var dom_swrap = ebi('mtw'); | ||||
| var dom_sbs = ebi('sbs'); | ||||
| var dom_nsbs = ebi('nsbs'); | ||||
| var dom_tbox = ebi('toolsbox'); | ||||
| var dom_ref = (function () { | ||||
|     var d = document.createElement('div'); | ||||
|     var d = mknod('div'); | ||||
|     d.setAttribute('id', 'mtr'); | ||||
|     dom_swrap.appendChild(d); | ||||
|     d = document.getElementById('mtr'); | ||||
|     d = ebi('mtr'); | ||||
|     // hide behind the textarea (offsetTop is not computed if display:none) | ||||
|     dom_src.style.zIndex = '4'; | ||||
|     d.style.zIndex = '3'; | ||||
| @@ -62,7 +71,7 @@ var map_src = []; | ||||
| var map_pre = []; | ||||
| function genmap(dom, oldmap) { | ||||
|     var find = nlines; | ||||
|     while (oldmap && find --> 0) { | ||||
|     while (oldmap && find-- > 0) { | ||||
|         var tmap = genmapq(dom, '*[data-ln="' + find + '"]'); | ||||
|         if (!tmap || !tmap.length) | ||||
|             continue; | ||||
| @@ -85,11 +94,11 @@ var nlines = 0; | ||||
| var draw_md = (function () { | ||||
|     var delay = 1; | ||||
|     function draw_md() { | ||||
|         var t0 = new Date().getTime(); | ||||
|         var t0 = Date.now(); | ||||
|         var src = dom_src.value; | ||||
|         convert_markdown(src, dom_pre); | ||||
|  | ||||
|         var lines = hesc(src).replace(/\r/g, "").split('\n'); | ||||
|         var lines = esc(src).replace(/\r/g, "").split('\n'); | ||||
|         nlines = lines.length; | ||||
|         var html = []; | ||||
|         for (var a = 0; a < lines.length; a++) | ||||
| @@ -99,9 +108,9 @@ var draw_md = (function () { | ||||
|         map_src = genmap(dom_ref, map_src); | ||||
|         map_pre = genmap(dom_pre, map_pre); | ||||
|  | ||||
|         cls(document.getElementById('save'), 'disabled', src == server_md); | ||||
|         clmod(ebi('save'), 'disabled', src == server_md); | ||||
|  | ||||
|         var t1 = new Date().getTime(); | ||||
|         var t1 = Date.now(); | ||||
|         delay = t1 - t0 > 100 ? 25 : 1; | ||||
|     } | ||||
|  | ||||
| @@ -118,6 +127,12 @@ var draw_md = (function () { | ||||
| })(); | ||||
|  | ||||
|  | ||||
| // discard TOC callback, just regen editor scroll map | ||||
| img_load.callbacks = [function () { | ||||
|     map_pre = genmap(dom_pre, map_pre); | ||||
| }]; | ||||
|  | ||||
|  | ||||
| // resize handler | ||||
| redraw = (function () { | ||||
|     function onresize() { | ||||
| @@ -127,7 +142,6 @@ redraw = (function () { | ||||
|         dom_ref.style.width = getComputedStyle(dom_src).offsetWidth + 'px'; | ||||
|         map_src = genmap(dom_ref, map_src); | ||||
|         map_pre = genmap(dom_pre, map_pre); | ||||
|         dbg(document.body.clientWidth + 'x' + document.body.clientHeight); | ||||
|     } | ||||
|     function setsbs() { | ||||
|         dom_wrap.setAttribute('class', ''); | ||||
| @@ -135,7 +149,7 @@ redraw = (function () { | ||||
|         onresize(); | ||||
|     } | ||||
|     function modetoggle() { | ||||
|         mode = dom_nsbs.innerHTML; | ||||
|         var mode = dom_nsbs.innerHTML; | ||||
|         dom_nsbs.innerHTML = mode == 'editor' ? 'preview' : 'editor'; | ||||
|         mode += ' single'; | ||||
|         dom_wrap.setAttribute('class', mode); | ||||
| @@ -164,14 +178,14 @@ redraw = (function () { | ||||
|             dst.scrollTop = 0; | ||||
|             return; | ||||
|         } | ||||
|         if (y + 8 + src.clientHeight > src.scrollHeight) { | ||||
|         if (y + 48 + src.clientHeight > src.scrollHeight) { | ||||
|             dst.scrollTop = dst.scrollHeight - dst.clientHeight; | ||||
|             return; | ||||
|         } | ||||
|         y += src.clientHeight / 2; | ||||
|         var sy1 = -1, sy2 = -1, dy1 = -1, dy2 = -1; | ||||
|         for (var a = 1; a < nlines + 1; a++) { | ||||
|             if (srcmap[a] === null || dstmap[a] === null) | ||||
|             if (srcmap[a] == null || dstmap[a] == null) | ||||
|                 continue; | ||||
|  | ||||
|             if (srcmap[a] > y) { | ||||
| @@ -214,61 +228,155 @@ redraw = (function () { | ||||
| })(); | ||||
|  | ||||
|  | ||||
| // modification checker | ||||
| function Modpoll() { | ||||
|     this.skip_one = true; | ||||
|     this.disabled = false; | ||||
|  | ||||
|     this.periodic = function () { | ||||
|         var that = this; | ||||
|         setTimeout(function () { | ||||
|             that.periodic(); | ||||
|         }, 1000 * md_opt.modpoll_freq); | ||||
|  | ||||
|         var skip = null; | ||||
|  | ||||
|         if (toast.visible) | ||||
|             skip = 'toast'; | ||||
|  | ||||
|         else if (this.skip_one) | ||||
|             skip = 'saved'; | ||||
|  | ||||
|         else if (this.disabled) | ||||
|             skip = 'disabled'; | ||||
|  | ||||
|         if (skip) { | ||||
|             console.log('modpoll skip, ' + skip); | ||||
|             this.skip_one = false; | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         console.log('modpoll...'); | ||||
|         var url = (document.location + '').split('?')[0] + '?raw&_=' + Date.now(); | ||||
|         var xhr = new XMLHttpRequest(); | ||||
|         xhr.modpoll = this; | ||||
|         xhr.open('GET', url, true); | ||||
|         xhr.responseType = 'text'; | ||||
|         xhr.onreadystatechange = this.cb; | ||||
|         xhr.send(); | ||||
|     } | ||||
|  | ||||
|     this.cb = function () { | ||||
|         if (this.modpoll.disabled || this.modpoll.skip_one) { | ||||
|             console.log('modpoll abort'); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (this.readyState != XMLHttpRequest.DONE) | ||||
|             return; | ||||
|  | ||||
|         if (this.status !== 200) { | ||||
|             console.log('modpoll err ' + this.status + ": " + this.responseText); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (!this.responseText) | ||||
|             return; | ||||
|  | ||||
|         var server_ref = server_md.replace(/\r/g, ''); | ||||
|         var server_now = this.responseText.replace(/\r/g, ''); | ||||
|  | ||||
|         if (server_ref != server_now) { | ||||
|             console.log("modpoll diff |" + server_ref.length + "|, |" + server_now.length + "|"); | ||||
|             this.modpoll.disabled = true; | ||||
|             var msg = [ | ||||
|                 "The document has changed on the server.", | ||||
|                 "The changes will NOT be loaded into your editor automatically.", | ||||
|                 "", | ||||
|                 "Press F5 or CTRL-R to refresh the page,", | ||||
|                 "replacing your document with the server copy.", | ||||
|                 "", | ||||
|                 "You can close this message to ignore and contnue." | ||||
|             ]; | ||||
|             return toast.warn(0, msg.join('\n')); | ||||
|         } | ||||
|  | ||||
|         console.log('modpoll eq'); | ||||
|     } | ||||
|  | ||||
|     if (md_opt.modpoll_freq > 0) | ||||
|         this.periodic(); | ||||
|  | ||||
|     return this; | ||||
| } | ||||
| var modpoll = new Modpoll(); | ||||
|  | ||||
|  | ||||
| window.onbeforeunload = function (e) { | ||||
|     if ((ebi("save").getAttribute('class') + '').indexOf('disabled') >= 0) | ||||
|         return; //nice (todo) | ||||
|  | ||||
|     e.preventDefault(); //ff | ||||
|     e.returnValue = ''; //chrome | ||||
| }; | ||||
|  | ||||
|  | ||||
| // save handler | ||||
| function save(e) { | ||||
|     if (e) e.preventDefault(); | ||||
|     var save_btn = document.getElementById("save"), | ||||
|     var save_btn = ebi("save"), | ||||
|         save_cls = save_btn.getAttribute('class') + ''; | ||||
|  | ||||
|     if (save_cls.indexOf('disabled') >= 0) { | ||||
|         alert('there is nothing to save'); | ||||
|         return; | ||||
|     } | ||||
|     if (save_cls.indexOf('disabled') >= 0) | ||||
|         return toast.inf(2, "no changes"); | ||||
|  | ||||
|     var force = (save_cls.indexOf('force-save') >= 0); | ||||
|     if (force && !confirm('confirm that you wish to lose the changes made on the server since you opened this document')) { | ||||
|         alert('ok, aborted'); | ||||
|         return; | ||||
|     function save2() { | ||||
|         var txt = dom_src.value, | ||||
|             fd = new FormData(); | ||||
|  | ||||
|         fd.append("act", "tput"); | ||||
|         fd.append("lastmod", (force ? -1 : last_modified)); | ||||
|         fd.append("body", txt); | ||||
|  | ||||
|         var url = (document.location + '').split('?')[0]; | ||||
|         var xhr = new XMLHttpRequest(); | ||||
|         xhr.open('POST', url, true); | ||||
|         xhr.responseType = 'text'; | ||||
|         xhr.onreadystatechange = save_cb; | ||||
|         xhr.btn = save_btn; | ||||
|         xhr.txt = txt; | ||||
|  | ||||
|         modpoll.skip_one = true;  // skip one iteration while we save | ||||
|         xhr.send(fd); | ||||
|     } | ||||
|  | ||||
|     var txt = dom_src.value; | ||||
|  | ||||
|     var fd = new FormData(); | ||||
|     fd.append("act", "tput"); | ||||
|     fd.append("lastmod", (force ? -1 : last_modified)); | ||||
|     fd.append("body", txt); | ||||
|  | ||||
|     var url = (document.location + '').split('?')[0] + '?raw'; | ||||
|     var xhr = new XMLHttpRequest(); | ||||
|     xhr.open('POST', url, true); | ||||
|     xhr.responseType = 'text'; | ||||
|     xhr.onreadystatechange = save_cb; | ||||
|     xhr.btn = save_btn; | ||||
|     xhr.txt = txt; | ||||
|     xhr.send(fd); | ||||
|     if (!force) | ||||
|         save2(); | ||||
|     else | ||||
|         modal.confirm('confirm that you wish to lose the changes made on the server since you opened this document', save2, function () { | ||||
|             toast.inf(3, 'aborted'); | ||||
|         }); | ||||
| } | ||||
|  | ||||
| function save_cb() { | ||||
|     if (this.readyState != XMLHttpRequest.DONE) | ||||
|         return; | ||||
|  | ||||
|     if (this.status !== 200) { | ||||
|         alert('Error!  The file was NOT saved.\n\n' + this.status + ": " + (this.responseText + '').replace(/^<pre>/, "")); | ||||
|         return; | ||||
|     } | ||||
|     if (this.status !== 200) | ||||
|         return toast.err(0, 'Error!  The file was NOT saved.\n\n' + this.status + ": " + (this.responseText + '').replace(/^<pre>/, "")); | ||||
|  | ||||
|     var r; | ||||
|     try { | ||||
|         r = JSON.parse(this.responseText); | ||||
|     } | ||||
|     catch (ex) { | ||||
|         alert('Failed to parse reply from server:\n\n' + this.responseText); | ||||
|         return; | ||||
|         return toast.err(0, 'Failed to parse reply from server:\n\n' + this.responseText); | ||||
|     } | ||||
|  | ||||
|     if (!r.ok) { | ||||
|         if (!this.btn.classList.contains('force-save')) { | ||||
|             this.btn.classList.add('force-save'); | ||||
|         if (!clgot(this.btn, 'force-save')) { | ||||
|             clmod(this.btn, 'force-save', 1); | ||||
|             var msg = [ | ||||
|                 'This file has been modified since you started editing it!\n', | ||||
|                 'if you really want to overwrite, press save again.\n', | ||||
| @@ -278,65 +386,64 @@ function save_cb() { | ||||
|                 r.lastmod + ' lastmod on the server now,', | ||||
|                 r.now + ' server time now,\n', | ||||
|             ]; | ||||
|             alert(msg.join('\n')); | ||||
|             return toast.err(0, msg.join('\n')); | ||||
|         } | ||||
|         else { | ||||
|             alert('Error! Save failed.  Maybe this JSON explains why:\n\n' + this.responseText); | ||||
|         } | ||||
|         return; | ||||
|         else | ||||
|             return toast.err(0, 'Error! Save failed.  Maybe this JSON explains why:\n\n' + this.responseText); | ||||
|     } | ||||
|  | ||||
|     this.btn.classList.remove('force-save'); | ||||
|     clmod(this.btn, 'force-save'); | ||||
|     //alert('save OK -- wrote ' + r.size + ' bytes.\n\nsha512: ' + r.sha512); | ||||
|  | ||||
|     run_savechk(r.lastmod, this.txt, this.btn, 0); | ||||
| } | ||||
|  | ||||
| function run_savechk(lastmod, txt, btn, ntry) { | ||||
|     // download the saved doc from the server and compare | ||||
|     var url = (document.location + '').split('?')[0] + '?raw'; | ||||
|     var url = (document.location + '').split('?')[0] + '?raw&_=' + Date.now(); | ||||
|     var xhr = new XMLHttpRequest(); | ||||
|     xhr.open('GET', url, true); | ||||
|     xhr.responseType = 'text'; | ||||
|     xhr.onreadystatechange = save_chk; | ||||
|     xhr.btn = this.save_btn; | ||||
|     xhr.txt = this.txt; | ||||
|     xhr.lastmod = r.lastmod; | ||||
|     xhr.onreadystatechange = savechk_cb; | ||||
|     xhr.lastmod = lastmod; | ||||
|     xhr.txt = txt; | ||||
|     xhr.btn = btn; | ||||
|     xhr.ntry = ntry; | ||||
|     xhr.send(); | ||||
| } | ||||
|  | ||||
| function save_chk() { | ||||
| function savechk_cb() { | ||||
|     if (this.readyState != XMLHttpRequest.DONE) | ||||
|         return; | ||||
|  | ||||
|     if (this.status !== 200) { | ||||
|         alert('Error!  The file was NOT saved.\n\n' + this.status + ": " + (this.responseText + '').replace(/^<pre>/, "")); | ||||
|         return; | ||||
|     } | ||||
|     if (this.status !== 200) | ||||
|         return toast.err(0, 'Error!  The file was NOT saved.\n\n' + this.status + ": " + (this.responseText + '').replace(/^<pre>/, "")); | ||||
|  | ||||
|     var doc1 = this.txt.replace(/\r\n/g, "\n"); | ||||
|     var doc2 = this.responseText.replace(/\r\n/g, "\n"); | ||||
|     if (doc1 != doc2) { | ||||
|         alert( | ||||
|         var that = this; | ||||
|         if (that.ntry < 10) { | ||||
|             // qnap funny, try a few more times | ||||
|             setTimeout(function () { | ||||
|                 run_savechk(that.lastmod, that.txt, that.btn, that.ntry + 1) | ||||
|             }, 100); | ||||
|             return; | ||||
|         } | ||||
|         modal.alert( | ||||
|             'Error! The document on the server does not appear to have saved correctly (your editor contents and the server copy is not identical). Place the document on your clipboard for now and check the server logs for hints\n\n' + | ||||
|             'Length: yours=' + doc1.length + ', server=' + doc2.length | ||||
|         ); | ||||
|         alert('yours, ' + doc1.length + ' byte:\n[' + doc1 + ']'); | ||||
|         alert('server, ' + doc2.length + ' byte:\n[' + doc2 + ']'); | ||||
|         modal.alert('yours, ' + doc1.length + ' byte:\n[' + doc1 + ']'); | ||||
|         modal.alert('server, ' + doc2.length + ' byte:\n[' + doc2 + ']'); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     last_modified = this.lastmod; | ||||
|     server_md = this.txt; | ||||
|     draw_md(); | ||||
|  | ||||
|     var ok = document.createElement('div'); | ||||
|     ok.setAttribute('style', 'font-size:6em;font-family:serif;font-weight:bold;color:#cf6;background:#444;border-radius:.3em;padding:.6em 0;position:fixed;top:30%;left:calc(50% - 2em);width:4em;text-align:center;z-index:9001;transition:opacity 0.2s ease-in-out;opacity:1'); | ||||
|     ok.innerHTML = 'OK✔️'; | ||||
|     var parent = document.getElementById('m'); | ||||
|     document.documentElement.appendChild(ok); | ||||
|     setTimeout(function () { | ||||
|         ok.style.opacity = 0; | ||||
|     }, 500); | ||||
|     setTimeout(function () { | ||||
|         ok.parentNode.removeChild(ok); | ||||
|     }, 750); | ||||
|     toast.ok(2, 'save OK' + (this.ntry ? '\nattempt ' + this.ntry : '')); | ||||
|     modpoll.disabled = false; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -409,6 +516,9 @@ function setsel(s) { | ||||
|     dom_src.value = [s.pre, s.sel, s.post].join(''); | ||||
|     dom_src.setSelectionRange(s.car, s.cdr, dom_src.selectionDirection); | ||||
|     dom_src.oninput(); | ||||
|     // support chrome: | ||||
|     dom_src.blur(); | ||||
|     dom_src.focus(); | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -482,7 +592,8 @@ function md_newline() { | ||||
|     var s = linebounds(true), | ||||
|         ln = s.md.substring(s.n1, s.n2), | ||||
|         m1 = /^( *)([0-9]+)(\. +)/.exec(ln), | ||||
|         m2 = /^[ \t>+-]*(\* )?/.exec(ln); | ||||
|         m2 = /^[ \t>+-]*(\* )?/.exec(ln), | ||||
|         drop = dom_src.selectionEnd - dom_src.selectionStart; | ||||
|  | ||||
|     var pre = m2[0]; | ||||
|     if (m1 !== null) | ||||
| @@ -494,7 +605,7 @@ function md_newline() { | ||||
|  | ||||
|     s.pre = s.md.substring(0, s.car) + '\n' + pre; | ||||
|     s.sel = ''; | ||||
|     s.post = s.md.substring(s.car); | ||||
|     s.post = s.md.substring(s.car + drop); | ||||
|     s.car = s.cdr = s.pre.length; | ||||
|     setsel(s); | ||||
|     return false; | ||||
| @@ -504,11 +615,21 @@ function md_newline() { | ||||
| // backspace | ||||
| function md_backspace() { | ||||
|     var s = linebounds(true), | ||||
|         ln = s.md.substring(s.n1, s.n2), | ||||
|         m = /^[ \t>+-]*(\* )?([0-9]+\. +)?/.exec(ln); | ||||
|         o0 = dom_src.selectionStart, | ||||
|         left = s.md.slice(s.n1, o0), | ||||
|         m = /^[ \t>+-]*(\* )?([0-9]+\. +)?/.exec(left); | ||||
|  | ||||
|     // if car is in whitespace area, do nothing | ||||
|     if (/^\s*$/.test(left)) | ||||
|         return true; | ||||
|  | ||||
|     // same if selection | ||||
|     if (o0 != dom_src.selectionEnd) | ||||
|         return true; | ||||
|  | ||||
|     // same if line is all-whitespace or non-markup | ||||
|     var v = m[0].replace(/[^ ]/g, " "); | ||||
|     if (v === m[0] || v.length !== ln.length) | ||||
|     if (v === m[0] || v.length !== left.length) | ||||
|         return true; | ||||
|  | ||||
|     s.pre = s.md.substring(0, s.n1) + v; | ||||
| @@ -520,34 +641,268 @@ function md_backspace() { | ||||
| } | ||||
|  | ||||
|  | ||||
| // paragraph jump | ||||
| function md_p_jump(down) { | ||||
|     var txt = dom_src.value, | ||||
|         ofs = dom_src.selectionStart; | ||||
|  | ||||
|     if (down) { | ||||
|         while (txt[ofs] == '\n' && --ofs > 0); | ||||
|         ofs = txt.indexOf("\n\n", ofs); | ||||
|         if (ofs < 0) | ||||
|             ofs = txt.length - 1; | ||||
|  | ||||
|         while (txt[ofs] == '\n' && ++ofs < txt.length - 1); | ||||
|     } | ||||
|     else { | ||||
|         txt += '\n\n'; | ||||
|         while (ofs > 1 && txt[ofs - 1] == '\n') ofs--; | ||||
|         ofs = Math.max(0, txt.lastIndexOf("\n\n", ofs - 1)); | ||||
|         while (txt[ofs] == '\n' && ++ofs < txt.length - 1); | ||||
|     } | ||||
|  | ||||
|     dom_src.setSelectionRange(ofs, ofs, "none"); | ||||
| } | ||||
|  | ||||
|  | ||||
| function reLastIndexOf(txt, ptn, end) { | ||||
|     var ofs = (typeof end !== 'undefined') ? end : txt.length; | ||||
|     end = ofs; | ||||
|     while (ofs >= 0) { | ||||
|         var sub = txt.slice(ofs, end); | ||||
|         if (ptn.test(sub)) | ||||
|             return ofs; | ||||
|  | ||||
|         ofs--; | ||||
|     } | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
|  | ||||
| // table formatter | ||||
| function fmt_table(e) { | ||||
|     if (e) e.preventDefault(); | ||||
|     //dom_tbox.setAttribute('class', ''); | ||||
|  | ||||
|     var txt = dom_src.value, | ||||
|         ofs = dom_src.selectionStart, | ||||
|         //o0 = txt.lastIndexOf('\n\n', ofs), | ||||
|         //o1 = txt.indexOf('\n\n', ofs); | ||||
|         o0 = reLastIndexOf(txt, /\n\s*\n/m, ofs), | ||||
|         o1 = txt.slice(ofs).search(/\n\s*\n|\n\s*$/m); | ||||
|     // note \s contains \n but its fine | ||||
|  | ||||
|     if (o0 < 0) | ||||
|         o0 = 0; | ||||
|     else { | ||||
|         // seek past the hit | ||||
|         var m = /\n\s*\n/m.exec(txt.slice(o0)); | ||||
|         o0 += m[0].length; | ||||
|     } | ||||
|  | ||||
|     o1 = o1 < 0 ? txt.length : o1 + ofs; | ||||
|  | ||||
|     var err = 'cannot format table due to ', | ||||
|         tab = txt.slice(o0, o1).split(/\s*\n/), | ||||
|         re_ind = /^\s*/, | ||||
|         ind = tab[1].match(re_ind)[0], | ||||
|         r0_ind = tab[0].slice(0, ind.length), | ||||
|         lpipe = tab[1].indexOf('|') < tab[1].indexOf('-'), | ||||
|         rpipe = tab[1].lastIndexOf('|') > tab[1].lastIndexOf('-'), | ||||
|         re_lpipe = lpipe ? /^\s*\|\s*/ : /^\s*/, | ||||
|         re_rpipe = rpipe ? /\s*\|\s*$/ : /\s*$/, | ||||
|         ncols; | ||||
|  | ||||
|     // the second row defines the table, | ||||
|     // need to process that first | ||||
|     var tmp = tab[0]; | ||||
|     tab[0] = tab[1]; | ||||
|     tab[1] = tmp; | ||||
|  | ||||
|     for (var a = 0; a < tab.length; a++) { | ||||
|         var row_name = (a == 1) ? 'header' : 'row#' + (a + 1); | ||||
|  | ||||
|         var ind2 = tab[a].match(re_ind)[0]; | ||||
|         if (ind != ind2 && a != 1)  // the table can be a list entry or something, ignore [0] | ||||
|             return toast.err(7, err + 'indentation mismatch on row#2 and ' + row_name + ',\n' + tab[a]); | ||||
|  | ||||
|         var t = tab[a].slice(ind.length); | ||||
|         t = t.replace(re_lpipe, ""); | ||||
|         t = t.replace(re_rpipe, ""); | ||||
|         tab[a] = t.split(/\s*\|\s*/g); | ||||
|  | ||||
|         if (a == 0) | ||||
|             ncols = tab[a].length; | ||||
|         else if (ncols < tab[a].length) | ||||
|             return toast.err(7, err + 'num.columns(' + row_name + ') exceeding row#2;  ' + ncols + ' < ' + tab[a].length); | ||||
|  | ||||
|         // if row has less columns than row2, fill them in | ||||
|         while (tab[a].length < ncols) | ||||
|             tab[a].push(''); | ||||
|     } | ||||
|  | ||||
|     // aight now swap em back | ||||
|     tmp = tab[0]; | ||||
|     tab[0] = tab[1]; | ||||
|     tab[1] = tmp; | ||||
|  | ||||
|     var re_align = /^ *(:?)-+(:?) *$/; | ||||
|     var align = []; | ||||
|     for (var col = 0; col < tab[1].length; col++) { | ||||
|         var m = tab[1][col].match(re_align); | ||||
|         if (!m) | ||||
|             return toast.err(7, err + 'invalid column specification, row#2, col ' + (col + 1) + ', [' + tab[1][col] + ']'); | ||||
|  | ||||
|         if (m[2]) { | ||||
|             if (m[1]) | ||||
|                 align.push('c'); | ||||
|             else | ||||
|                 align.push('r'); | ||||
|         } | ||||
|         else | ||||
|             align.push('l'); | ||||
|     } | ||||
|  | ||||
|     var pad = []; | ||||
|     var tmax = 0; | ||||
|     for (var col = 0; col < ncols; col++) { | ||||
|         var max = 0; | ||||
|         for (var row = 0; row < tab.length; row++) | ||||
|             if (row != 1) | ||||
|                 max = Math.max(max, tab[row][col].length); | ||||
|  | ||||
|         var s = ''; | ||||
|         for (var n = 0; n < max; n++) | ||||
|             s += ' '; | ||||
|  | ||||
|         pad.push(s); | ||||
|         tmax = Math.max(max, tmax); | ||||
|     } | ||||
|  | ||||
|     var dashes = ''; | ||||
|     for (var a = 0; a < tmax; a++) | ||||
|         dashes += '-'; | ||||
|  | ||||
|     var ret = []; | ||||
|     for (var row = 0; row < tab.length; row++) { | ||||
|         var ln = []; | ||||
|         for (var col = 0; col < tab[row].length; col++) { | ||||
|             var p = pad[col]; | ||||
|             var s = tab[row][col]; | ||||
|  | ||||
|             if (align[col] == 'l') { | ||||
|                 s = (s + p).slice(0, p.length); | ||||
|             } | ||||
|             else if (align[col] == 'r') { | ||||
|                 s = (p + s).slice(-p.length); | ||||
|             } | ||||
|             else { | ||||
|                 var pt = p.length - s.length; | ||||
|                 var pl = p.slice(0, Math.floor(pt / 2)); | ||||
|                 var pr = p.slice(0, pt - pl.length); | ||||
|                 s = pl + s + pr; | ||||
|             } | ||||
|  | ||||
|             if (row == 1) { | ||||
|                 if (align[col] == 'l') | ||||
|                     s = dashes.slice(0, p.length); | ||||
|                 else if (align[col] == 'r') | ||||
|                     s = dashes.slice(0, p.length - 1) + ':'; | ||||
|                 else | ||||
|                     s = ':' + dashes.slice(0, p.length - 2) + ':'; | ||||
|             } | ||||
|             ln.push(s); | ||||
|         } | ||||
|         ret.push(ind + '| ' + ln.join(' | ') + ' |'); | ||||
|     } | ||||
|  | ||||
|     // restore any markup in the row0 gutter | ||||
|     ret[0] = r0_ind + ret[0].slice(ind.length); | ||||
|  | ||||
|     ret = { | ||||
|         "pre": txt.slice(0, o0), | ||||
|         "sel": ret.join('\n'), | ||||
|         "post": txt.slice(o1), | ||||
|         "car": o0, | ||||
|         "cdr": o0 | ||||
|     }; | ||||
|     setsel(ret); | ||||
| } | ||||
|  | ||||
|  | ||||
| // show unicode | ||||
| function mark_uni(e) { | ||||
|     if (e) e.preventDefault(); | ||||
|     dom_tbox.setAttribute('class', ''); | ||||
|  | ||||
|     var txt = dom_src.value, | ||||
|         ptn = new RegExp('([^' + js_uni_whitelist + ']+)', 'g'), | ||||
|         mod = txt.replace(/\r/g, "").replace(ptn, "\u2588\u2770$1\u2771"); | ||||
|  | ||||
|     if (txt == mod) | ||||
|         return toast.inf(5, 'no results;  no modifications were made'); | ||||
|  | ||||
|     dom_src.value = mod; | ||||
| } | ||||
|  | ||||
|  | ||||
| // iterate unicode | ||||
| function iter_uni(e) { | ||||
|     if (e) e.preventDefault(); | ||||
|  | ||||
|     var txt = dom_src.value, | ||||
|         ofs = dom_src.selectionDirection == "forward" ? dom_src.selectionEnd : dom_src.selectionStart, | ||||
|         re = new RegExp('([^' + js_uni_whitelist + ']+)'), | ||||
|         m = re.exec(txt.slice(ofs)); | ||||
|  | ||||
|     if (!m) | ||||
|         return toast.inf(5, 'no more hits from cursor onwards'); | ||||
|  | ||||
|     ofs += m.index; | ||||
|  | ||||
|     dom_src.setSelectionRange(ofs, ofs + m[0].length, "forward"); | ||||
|     dom_src.oninput(); | ||||
|     // support chrome: | ||||
|     dom_src.blur(); | ||||
|     dom_src.focus(); | ||||
| } | ||||
|  | ||||
|  | ||||
| // configure whitelist | ||||
| function cfg_uni(e) { | ||||
|     if (e) e.preventDefault(); | ||||
|  | ||||
|     modal.prompt("unicode whitelist", esc_uni_whitelist, function (reply) { | ||||
|         esc_uni_whitelist = reply; | ||||
|         js_uni_whitelist = eval('\'' + esc_uni_whitelist + '\''); | ||||
|     }, null); | ||||
| } | ||||
|  | ||||
|  | ||||
| // hotkeys / toolbar | ||||
| (function () { | ||||
|     function keydown(ev) { | ||||
|         ev = ev || window.event; | ||||
|         var kc = ev.keyCode || ev.which; | ||||
|         var ctrl = ev.ctrlKey || ev.metaKey; | ||||
|         //console.log(ev.code, kc); | ||||
|         if (ctrl && (ev.code == "KeyS" || kc == 83)) { | ||||
|         var kc = ev.code || ev.keyCode || ev.which; | ||||
|         //console.log(ev.key, ev.code, ev.keyCode, ev.which); | ||||
|         if (ctrl(ev) && (ev.code == "KeyS" || kc == 83)) { | ||||
|             save(); | ||||
|             return false; | ||||
|         } | ||||
|         if (document.activeElement == dom_src) { | ||||
|             if (ev.code == "Tab" || kc == 9) { | ||||
|                 md_indent(ev.shiftKey); | ||||
|                 return false; | ||||
|             } | ||||
|             if (ctrl && (ev.code == "KeyH" || kc == 72)) { | ||||
|         if (ev.code == "Escape" || kc == 27) { | ||||
|             var d = ebi('helpclose'); | ||||
|             if (d) | ||||
|                 d.click(); | ||||
|         } | ||||
|         if (document.activeElement != dom_src) | ||||
|             return true; | ||||
|  | ||||
|         if (ctrl(ev)) { | ||||
|             if (ev.code == "KeyH" || kc == 72) { | ||||
|                 md_header(ev.shiftKey); | ||||
|                 return false; | ||||
|             } | ||||
|             if (!ctrl && (ev.code == "Home" || kc == 36)) { | ||||
|                 md_home(ev.shiftKey); | ||||
|                 return false; | ||||
|             } | ||||
|             if (!ctrl && !ev.shiftKey && (ev.code == "Enter" || kc == 13)) { | ||||
|                 return md_newline(); | ||||
|             } | ||||
|             if (ctrl && (ev.code == "KeyZ" || kc == 90)) { | ||||
|             if (ev.code == "KeyZ" || kc == 90) { | ||||
|                 if (ev.shiftKey) | ||||
|                     action_stack.redo(); | ||||
|                 else | ||||
| @@ -555,23 +910,63 @@ function md_backspace() { | ||||
|  | ||||
|                 return false; | ||||
|             } | ||||
|             if (ctrl && (ev.code == "KeyY" || kc == 89)) { | ||||
|             if (ev.code == "KeyY" || kc == 89) { | ||||
|                 action_stack.redo(); | ||||
|                 return false; | ||||
|             } | ||||
|             if (!ctrl && !ev.shiftKey && kc == 8) { | ||||
|             if (ev.code == "KeyK") { | ||||
|                 fmt_table(); | ||||
|                 return false; | ||||
|             } | ||||
|             if (ev.code == "KeyU") { | ||||
|                 iter_uni(); | ||||
|                 return false; | ||||
|             } | ||||
|             if (ev.code == "KeyE") { | ||||
|                 dom_nsbs.click(); | ||||
|                 return false; | ||||
|             } | ||||
|             var up = ev.code == "ArrowUp" || kc == 38; | ||||
|             var dn = ev.code == "ArrowDown" || kc == 40; | ||||
|             if (up || dn) { | ||||
|                 md_p_jump(dn); | ||||
|                 return false; | ||||
|             } | ||||
|         } | ||||
|         else { | ||||
|             if (ev.code == "Tab" || kc == 9) { | ||||
|                 md_indent(ev.shiftKey); | ||||
|                 return false; | ||||
|             } | ||||
|             if (ev.code == "Home" || kc == 36) { | ||||
|                 md_home(ev.shiftKey); | ||||
|                 return false; | ||||
|             } | ||||
|             if (!ev.shiftKey && (ev.code == "Enter" || kc == 13)) { | ||||
|                 return md_newline(); | ||||
|             } | ||||
|             if (!ev.shiftKey && kc == 8) { | ||||
|                 return md_backspace(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     document.onkeydown = keydown; | ||||
|     document.getElementById('save').onclick = save; | ||||
|     ebi('save').onclick = save; | ||||
| })(); | ||||
|  | ||||
|  | ||||
| document.getElementById('help').onclick = function (e) { | ||||
| ebi('tools').onclick = function (e) { | ||||
|     if (e) e.preventDefault(); | ||||
|     var dom = document.getElementById('helpbox'); | ||||
|     var is_open = dom_tbox.getAttribute('class') != 'open'; | ||||
|     dom_tbox.setAttribute('class', is_open ? 'open' : ''); | ||||
| }; | ||||
|  | ||||
|  | ||||
| ebi('help').onclick = function (e) { | ||||
|     if (e) e.preventDefault(); | ||||
|     dom_tbox.setAttribute('class', ''); | ||||
|  | ||||
|     var dom = ebi('helpbox'); | ||||
|     var dtxt = dom.getElementsByTagName('textarea'); | ||||
|     if (dtxt.length > 0) { | ||||
|         convert_markdown(dtxt[0].value, dom); | ||||
| @@ -579,12 +974,18 @@ document.getElementById('help').onclick = function (e) { | ||||
|     } | ||||
|  | ||||
|     dom.style.display = 'block'; | ||||
|     document.getElementById('helpclose').onclick = function () { | ||||
|     ebi('helpclose').onclick = function () { | ||||
|         dom.style.display = 'none'; | ||||
|     }; | ||||
| }; | ||||
|  | ||||
|  | ||||
| ebi('fmt_table').onclick = fmt_table; | ||||
| ebi('mark_uni').onclick = mark_uni; | ||||
| ebi('iter_uni').onclick = iter_uni; | ||||
| ebi('cfg_uni').onclick = cfg_uni; | ||||
|  | ||||
|  | ||||
| // blame steen | ||||
| action_stack = (function () { | ||||
|     var hist = { | ||||
| @@ -610,7 +1011,7 @@ action_stack = (function () { | ||||
|         var p1 = from.length, | ||||
|             p2 = to.length; | ||||
|  | ||||
|         while (p1 --> 0 && p2 --> 0) | ||||
|         while (p1-- > 0 && p2-- > 0) | ||||
|             if (from[p1] != to[p2]) | ||||
|                 break; | ||||
|  | ||||
| @@ -690,13 +1091,12 @@ action_stack = (function () { | ||||
|         ref = newtxt; | ||||
|         dbg('undos(%d) redos(%d)', hist.un.length, hist.re.length); | ||||
|         if (hist.un.length > 0) | ||||
|             dbg(static(hist.un.slice(-1)[0])); | ||||
|             dbg(jcp(hist.un.slice(-1)[0])); | ||||
|         if (hist.re.length > 0) | ||||
|             dbg(static(hist.re.slice(-1)[0])); | ||||
|             dbg(jcp(hist.re.slice(-1)[0])); | ||||
|     } | ||||
|  | ||||
|     return { | ||||
|         push: push, | ||||
|         undo: undo, | ||||
|         redo: redo, | ||||
|         push: schedule_push, | ||||
| @@ -704,14 +1104,3 @@ action_stack = (function () { | ||||
|         _ref: ref | ||||
|     } | ||||
| })(); | ||||
|  | ||||
| /* | ||||
| document.getElementById('help').onclick = function () { | ||||
|     var c1 = getComputedStyle(dom_src).cssText.split(';'); | ||||
|     var c2 = getComputedStyle(dom_ref).cssText.split(';'); | ||||
|     var max = Math.min(c1.length, c2.length); | ||||
|     for (var a = 0; a < max; a++) | ||||
|         if (c1[a] !== c2[a]) | ||||
|             console.log(c1[a] + '\n' + c2[a]); | ||||
| } | ||||
| */ | ||||
| @@ -7,307 +7,149 @@ html .editor-toolbar>button.active { border-color: rgba(0,0,0,0.4); background: | ||||
| html .editor-toolbar>i.separator { border-left: 1px solid #ccc; } | ||||
| html .editor-toolbar.disabled-for-preview>button:not(.no-disable) { opacity: .35 } | ||||
|  | ||||
|  | ||||
|  | ||||
| html { | ||||
|     line-height: 1.5em; | ||||
| 	line-height: 1.5em; | ||||
| } | ||||
| html, body { | ||||
|     margin: 0; | ||||
|     padding: 0; | ||||
|     min-height: 100%; | ||||
|     font-family: sans-serif; | ||||
|     background: #f7f7f7; | ||||
|     color: #333; | ||||
| 	margin: 0; | ||||
| 	padding: 0; | ||||
| 	min-height: 100%; | ||||
| 	font-family: sans-serif; | ||||
| 	background: #f7f7f7; | ||||
| 	color: #333; | ||||
| } | ||||
| #toast { | ||||
| 	bottom: auto; | ||||
| 	top: 1.4em; | ||||
| } | ||||
| #repl { | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	right: .5em; | ||||
| 	border: none; | ||||
| 	color: inherit; | ||||
| 	background: none; | ||||
| 	text-decoration: none; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| #mn { | ||||
|     font-weight: normal; | ||||
|     margin: 1.3em 0 .7em 1em; | ||||
| 	font-weight: normal; | ||||
| 	margin: 1.3em 0 .7em 1em; | ||||
| } | ||||
| #mn a { | ||||
|     color: #444; | ||||
|     margin: 0 0 0 -.2em; | ||||
|     padding: 0 0 0 .4em; | ||||
|     text-decoration: none; | ||||
|     /* ie: */ | ||||
|     border-bottom: .1em solid #777\9; | ||||
|     margin-right: 1em\9; | ||||
| 	color: #444; | ||||
| 	margin: 0 0 0 -.2em; | ||||
| 	padding: 0 0 0 .4em; | ||||
| 	text-decoration: none; | ||||
| 	/* ie: */ | ||||
| 	border-bottom: .1em solid #777\9; | ||||
| 	margin-right: 1em\9; | ||||
| } | ||||
| #mn a:first-child { | ||||
|     padding-left: .5em; | ||||
| 	padding-left: .5em; | ||||
| } | ||||
| #mn a:last-child { | ||||
|     padding-right: .5em; | ||||
| 	padding-right: .5em; | ||||
| } | ||||
| #mn a:not(:last-child):after { | ||||
|     content: ''; | ||||
|     width: 1.05em; | ||||
|     height: 1.05em; | ||||
|     margin: -.2em .3em -.2em -.4em; | ||||
|     display: inline-block; | ||||
|     border: 1px solid rgba(0,0,0,0.2); | ||||
|     border-width: .2em .2em 0 0; | ||||
|     transform: rotate(45deg); | ||||
| 	content: ''; | ||||
| 	width: 1.05em; | ||||
| 	height: 1.05em; | ||||
| 	margin: -.2em .3em -.2em -.4em; | ||||
| 	display: inline-block; | ||||
| 	border: 1px solid rgba(0,0,0,0.2); | ||||
| 	border-width: .2em .2em 0 0; | ||||
| 	transform: rotate(45deg); | ||||
| } | ||||
| #mn a:hover { | ||||
|     color: #000; | ||||
|     text-decoration: underline; | ||||
| 	color: #000; | ||||
| 	text-decoration: underline; | ||||
| } | ||||
|  | ||||
| html .editor-toolbar>button.disabled { | ||||
|     opacity: .35; | ||||
|     pointer-events: none; | ||||
| 	opacity: .35; | ||||
| 	pointer-events: none; | ||||
| } | ||||
| html .editor-toolbar>button.save.force-save { | ||||
|     background: #f97; | ||||
| } | ||||
|  | ||||
| /* | ||||
| *[data-ln]:before { | ||||
| 	content: attr(data-ln); | ||||
| 	font-size: .8em; | ||||
| 	margin: 0 .4em; | ||||
| 	color: #f0c; | ||||
| } | ||||
| .cm-header { font-size: .4em !important } | ||||
| */ | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| /* copied from md.css for now */ | ||||
| .mdo pre, | ||||
| .mdo code, | ||||
| .mdo a { | ||||
| 	color: #480; | ||||
| 	background: #f7f7f7; | ||||
| 	border: .07em solid #ddd; | ||||
| 	border-radius: .2em; | ||||
| 	padding: .1em .3em; | ||||
| 	margin: 0 .1em; | ||||
| } | ||||
| .mdo code { | ||||
| 	font-size: .96em; | ||||
| } | ||||
| .mdo pre, | ||||
| .mdo code { | ||||
| 	font-family: monospace, monospace; | ||||
| 	white-space: pre-wrap; | ||||
| 	word-break: break-all; | ||||
| } | ||||
| .mdo pre code { | ||||
| 	display: block; | ||||
| 	margin: 0 -.3em; | ||||
| 	padding: .4em .5em; | ||||
| 	line-height: 1.1em; | ||||
| } | ||||
| .mdo a { | ||||
|     color: #fff; | ||||
|     background: #39b; | ||||
|     text-decoration: none; | ||||
|     padding: 0 .3em; | ||||
|     border: none; | ||||
|     border-bottom: .07em solid #079; | ||||
| } | ||||
| .mdo h2 { | ||||
|     color: #fff; | ||||
|     background: #555; | ||||
|     margin-top: 2em; | ||||
|     border-bottom: .22em solid #999; | ||||
|     border-top: none; | ||||
| } | ||||
| .mdo h1 { | ||||
|     color: #fff; | ||||
|     background: #444; | ||||
|     font-weight: normal; | ||||
|     border-top: .4em solid #fb0; | ||||
|     border-bottom: .4em solid #777; | ||||
|     border-radius: 0 1em 0 1em; | ||||
|     margin: 3em 0 1em 0; | ||||
|     padding: .5em 0; | ||||
| } | ||||
| h1, h2 { | ||||
| 	line-height: 1.5em; | ||||
| } | ||||
| h1 { | ||||
| 	font-size: 1.7em; | ||||
| 	text-align: center; | ||||
| 	border: 1em solid #777; | ||||
| 	border-width: .05em 0; | ||||
| 	margin: 3em 0; | ||||
| } | ||||
| h2 { | ||||
| 	font-size: 1.5em; | ||||
| 	font-weight: normal; | ||||
| 	background: #f7f7f7; | ||||
| 	border-top: .07em solid #fff; | ||||
| 	border-bottom: .07em solid #bbb; | ||||
| 	border-radius: .5em .5em 0 0; | ||||
| 	padding-left: .4em; | ||||
| 	margin-top: 3em; | ||||
| } | ||||
| .mdo ul, | ||||
| .mdo ol { | ||||
| 	border-left: .3em solid #ddd; | ||||
| } | ||||
| .mdo>ul, | ||||
| .mdo>ol { | ||||
| 	border-color: #bbb; | ||||
| } | ||||
| .mdo ul>li { | ||||
| 	list-style-type: disc; | ||||
| } | ||||
| .mdo ul>li, | ||||
| .mdo ol>li { | ||||
| 	margin: .7em 0; | ||||
| } | ||||
| p>em, | ||||
| li>em { | ||||
| 	color: #c50; | ||||
| 	padding: .1em; | ||||
| 	border-bottom: .1em solid #bbb; | ||||
| } | ||||
| blockquote { | ||||
| 	font-family: serif; | ||||
| 	background: #f7f7f7; | ||||
| 	border: .07em dashed #ccc; | ||||
| 	padding: 0 2em; | ||||
| 	margin: 1em 0; | ||||
| } | ||||
| small { | ||||
| 	opacity: .8; | ||||
| } | ||||
| table { | ||||
| 	border-collapse: collapse; | ||||
| } | ||||
| td { | ||||
| 	padding: .2em .5em; | ||||
| 	border: .12em solid #aaa; | ||||
| } | ||||
| th { | ||||
| 	border: .12em solid #aaa; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| /* mde support */ | ||||
| .mdo { | ||||
|     padding: 1em; | ||||
|     background: #f7f7f7; | ||||
| } | ||||
| html.dark .mdo { | ||||
|     background: #1c1c1c; | ||||
| 	background: #f97; | ||||
| } | ||||
| .CodeMirror { | ||||
|     background: #f7f7f7; | ||||
| 	background: #f7f7f7; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| /* darkmode */ | ||||
| html.dark .mdo, | ||||
| html.dark .CodeMirror { | ||||
|     border-color: #222; | ||||
| 	border-color: #222; | ||||
| } | ||||
| html.dark, | ||||
| html.dark body, | ||||
| html.dark .CodeMirror { | ||||
|     background: #222; | ||||
|     color: #ccc; | ||||
| 	background: #222; | ||||
| 	color: #ccc; | ||||
| } | ||||
| html.dark .CodeMirror-cursor { | ||||
|     border-color: #fff; | ||||
| 	border-color: #fff; | ||||
| } | ||||
| html.dark .CodeMirror-selected { | ||||
|     box-shadow: 0 0 1px #0cf inset; | ||||
| 	box-shadow: 0 0 1px #0cf inset; | ||||
| } | ||||
| html.dark .CodeMirror-selected, | ||||
| html.dark .CodeMirror-selectedtext { | ||||
|     border-radius: .1em; | ||||
|     background: #246; | ||||
|     color: #fff; | ||||
| } | ||||
| html.dark .mdo a { | ||||
|     background: #057; | ||||
| } | ||||
| html.dark .mdo h1 a, html.dark .mdo h4 a, | ||||
| html.dark .mdo h2 a, html.dark .mdo h5 a, | ||||
| html.dark .mdo h3 a, html.dark .mdo h6 a { | ||||
|     color: inherit; | ||||
|     background: none; | ||||
| } | ||||
| html.dark pre, | ||||
| html.dark code { | ||||
|     color: #8c0; | ||||
|     background: #1a1a1a; | ||||
|     border: .07em solid #333; | ||||
| } | ||||
| html.dark .mdo ul, | ||||
| html.dark .mdo ol { | ||||
|     border-color: #444; | ||||
| } | ||||
| html.dark .mdo>ul, | ||||
| html.dark .mdo>ol { | ||||
|     border-color: #555; | ||||
| } | ||||
| html.dark p>em, | ||||
| html.dark li>em { | ||||
|     color: #f94; | ||||
|     border-color: #666; | ||||
| } | ||||
| html.dark h1 { | ||||
|     background: #383838; | ||||
|     border-top: .4em solid #b80; | ||||
|     border-bottom: .4em solid #4c4c4c; | ||||
| } | ||||
| html.dark h2 { | ||||
|     background: #444; | ||||
|     border-bottom: .22em solid #555; | ||||
| } | ||||
| html.dark td, | ||||
| html.dark th { | ||||
|     border-color: #444; | ||||
| } | ||||
| html.dark blockquote { | ||||
|     background: #282828; | ||||
|     border: .07em dashed #444; | ||||
| 	border-radius: .1em; | ||||
| 	background: #246; | ||||
| 	color: #fff; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| html.dark #mn a { | ||||
|     color: #ccc; | ||||
| 	color: #ccc; | ||||
| } | ||||
| html.dark #mn a:not(:last-child):after { | ||||
|     border-color: rgba(255,255,255,0.3); | ||||
| 	border-color: rgba(255,255,255,0.3); | ||||
| } | ||||
| html.dark .editor-toolbar { | ||||
|     border-color: #2c2c2c; | ||||
|     background: #1c1c1c; | ||||
| 	border-color: #2c2c2c; | ||||
| 	background: #1c1c1c; | ||||
| } | ||||
| html.dark .editor-toolbar>i.separator { | ||||
|     border-left: 1px solid #444; | ||||
|     border-right: 1px solid #111; | ||||
| 	border-left: 1px solid #444; | ||||
| 	border-right: 1px solid #111; | ||||
| } | ||||
| html.dark .editor-toolbar>button { | ||||
|     margin-left: -1px; border: 1px solid rgba(255,255,255,0.1); | ||||
|     color: #aaa; | ||||
| 	margin-left: -1px; border: 1px solid rgba(255,255,255,0.1); | ||||
| 	color: #aaa; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| html.dark .editor-toolbar>button:hover { | ||||
|     color: #333; | ||||
| 	color: #333; | ||||
| } | ||||
| html.dark .editor-toolbar>button.active { | ||||
|     color: #333; | ||||
|     border-color: #ec1; | ||||
|     background: #c90; | ||||
| 	color: #333; | ||||
| 	border-color: #ec1; | ||||
| 	background: #c90; | ||||
| } | ||||
| html.dark .editor-toolbar::after, | ||||
| html.dark .editor-toolbar::before { | ||||
|     background: none; | ||||
| } | ||||
| 	background: none; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| /* ui.css overrides */ | ||||
| .mdo { | ||||
| 	padding: 1em; | ||||
| 	background: #f7f7f7; | ||||
| } | ||||
| html.dark .mdo { | ||||
| 	background: #1c1c1c; | ||||
| } | ||||
|   | ||||
| @@ -3,9 +3,10 @@ | ||||
| 	<title>📝🎉 {{ title }}</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=0.7"> | ||||
| 	<link href="/.cpr/mde.css" rel="stylesheet"> | ||||
| 	<link href="/.cpr/deps/mini-fa.css" rel="stylesheet"> | ||||
| 	<link href="/.cpr/deps/easymde.css" rel="stylesheet"> | ||||
| 	<link rel="stylesheet" href="/.cpr/ui.css?_={{ ts }}"> | ||||
| 	<link rel="stylesheet" href="/.cpr/mde.css?_={{ ts }}"> | ||||
| 	<link rel="stylesheet" href="/.cpr/deps/mini-fa.css?_={{ ts }}"> | ||||
| 	<link rel="stylesheet" href="/.cpr/deps/easymde.css?_={{ ts }}"> | ||||
| </head> | ||||
| <body> | ||||
| 	<div id="mw"> | ||||
| @@ -17,28 +18,33 @@ | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div id="m"> | ||||
| 			<textarea id="mt" style="display:none">{{ md }}</textarea> | ||||
| 			<textarea id="mt" style="display:none" autocomplete="off">{{ md }}</textarea> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<a href="#" id="repl">π</a> | ||||
| 	<script> | ||||
|  | ||||
| var link_md_as_html = false;  // TODO (does nothing) | ||||
| var last_modified = {{ lastmod }}; | ||||
| var md_opt = { | ||||
| 	link_md_as_html: false, | ||||
| 	allow_plugins: {{ md_plug }}, | ||||
| 	modpoll_freq: {{ md_chk_rate }} | ||||
| }; | ||||
|  | ||||
| var lightswitch = (function () { | ||||
| 	var fun = function () { | ||||
| 		var dark = !!!document.documentElement.getAttribute("class"); | ||||
| 		document.documentElement.setAttribute("class", dark ? "dark" : ""); | ||||
| 		if (window.localStorage) | ||||
| 			localStorage.setItem('darkmode', dark ? 1 : 0); | ||||
| 	}; | ||||
| 	if (window.localStorage && localStorage.getItem('darkmode') == 1) | ||||
| 		fun(); | ||||
| 	 | ||||
| 	return fun; | ||||
| 	var l = localStorage, | ||||
| 		drk = l.getItem('lightmode') != 1, | ||||
| 		f = function (e) { | ||||
| if (e) drk = !drk; | ||||
| document.documentElement.setAttribute("class", drk? "dark":"light"); | ||||
| l.setItem('lightmode', drk? 0:1); | ||||
| 		}; | ||||
| 	f(); | ||||
| 	return f; | ||||
| })(); | ||||
|  | ||||
| 	</script> | ||||
| 	<script src="/.cpr/deps/easymde.js"></script> | ||||
| 	<script src="/.cpr/mde.js"></script> | ||||
|     <script src="/.cpr/util.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/deps/easymde.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/mde.js?_={{ ts }}"></script> | ||||
| </body></html> | ||||
|   | ||||
| @@ -1,7 +1,9 @@ | ||||
| var dom_wrap = document.getElementById('mw'); | ||||
| var dom_nav = document.getElementById('mn'); | ||||
| var dom_doc = document.getElementById('m'); | ||||
| var dom_md = document.getElementById('mt'); | ||||
| "use strict"; | ||||
|  | ||||
| var dom_wrap = ebi('mw'); | ||||
| var dom_nav = ebi('mn'); | ||||
| var dom_doc = ebi('m'); | ||||
| var dom_md = ebi('mt'); | ||||
|  | ||||
| (function () { | ||||
|     var n = document.location + ''; | ||||
| @@ -13,7 +15,7 @@ var dom_md = document.getElementById('mt'); | ||||
|         if (a > 0) | ||||
|             loc.push(n[a]); | ||||
|  | ||||
|         var dec = decodeURIComponent(n[a]).replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">"); | ||||
|         var dec = uricom_dec(n[a])[0].replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">"); | ||||
|  | ||||
|         nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>'); | ||||
|     } | ||||
| @@ -63,17 +65,17 @@ var mde = (function () { | ||||
|     mde.codemirror.on("change", function () { | ||||
|         md_changed(mde); | ||||
|     }); | ||||
|     var loader = document.getElementById('ml'); | ||||
|     var loader = ebi('ml'); | ||||
|     loader.parentNode.removeChild(loader); | ||||
|     return mde; | ||||
| })(); | ||||
|  | ||||
| function set_jumpto() { | ||||
|     document.querySelector('.editor-preview-side').onclick = jumpto; | ||||
|     QS('.editor-preview-side').onclick = jumpto; | ||||
| } | ||||
|  | ||||
| function jumpto(ev) { | ||||
|     var tgt = ev.target || ev.srcElement; | ||||
|     var tgt = ev.target; | ||||
|     var ln = null; | ||||
|     while (tgt && !ln) { | ||||
|         ln = tgt.getAttribute('data-ln'); | ||||
| @@ -92,67 +94,63 @@ function md_changed(mde, on_srv) { | ||||
|         window.md_saved = mde.value(); | ||||
|  | ||||
|     var md_now = mde.value(); | ||||
|     var save_btn = document.querySelector('.editor-toolbar button.save'); | ||||
|  | ||||
|     if (md_now == window.md_saved) | ||||
|         save_btn.classList.add('disabled'); | ||||
|     else | ||||
|         save_btn.classList.remove('disabled'); | ||||
|     var save_btn = QS('.editor-toolbar button.save'); | ||||
|  | ||||
|     clmod(save_btn, 'disabled', md_now == window.md_saved); | ||||
|     set_jumpto(); | ||||
| } | ||||
|  | ||||
| function save(mde) { | ||||
|     var save_btn = document.querySelector('.editor-toolbar button.save'); | ||||
|     if (save_btn.classList.contains('disabled')) { | ||||
|         alert('there is nothing to save'); | ||||
|         return; | ||||
|     } | ||||
|     var force = save_btn.classList.contains('force-save'); | ||||
|     if (force && !confirm('confirm that you wish to lose the changes made on the server since you opened this document')) { | ||||
|         alert('ok, aborted'); | ||||
|         return; | ||||
|     var save_btn = QS('.editor-toolbar button.save'); | ||||
|     if (clgot(save_btn, 'disabled')) | ||||
|         return toast.inf(2, 'no changes'); | ||||
|  | ||||
|     var force = clgot(save_btn, 'force-save'); | ||||
|     function save2() { | ||||
|         var txt = mde.value(); | ||||
|  | ||||
|         var fd = new FormData(); | ||||
|         fd.append("act", "tput"); | ||||
|         fd.append("lastmod", (force ? -1 : last_modified)); | ||||
|         fd.append("body", txt); | ||||
|  | ||||
|         var url = (document.location + '').split('?')[0]; | ||||
|         var xhr = new XMLHttpRequest(); | ||||
|         xhr.open('POST', url, true); | ||||
|         xhr.responseType = 'text'; | ||||
|         xhr.onreadystatechange = save_cb; | ||||
|         xhr.btn = save_btn; | ||||
|         xhr.mde = mde; | ||||
|         xhr.txt = txt; | ||||
|         xhr.send(fd); | ||||
|     } | ||||
|  | ||||
|     var txt = mde.value(); | ||||
|  | ||||
|     var fd = new FormData(); | ||||
|     fd.append("act", "tput"); | ||||
|     fd.append("lastmod", (force ? -1 : last_modified)); | ||||
|     fd.append("body", txt); | ||||
|  | ||||
|     var url = (document.location + '').split('?')[0] + '?raw'; | ||||
|     var xhr = new XMLHttpRequest(); | ||||
|     xhr.open('POST', url, true); | ||||
|     xhr.responseType = 'text'; | ||||
|     xhr.onreadystatechange = save_cb; | ||||
|     xhr.btn = save_btn; | ||||
|     xhr.mde = mde; | ||||
|     xhr.txt = txt; | ||||
|     xhr.send(fd); | ||||
|     if (!force) | ||||
|         save2(); | ||||
|     else | ||||
|         modal.confirm('confirm that you wish to lose the changes made on the server since you opened this document', save2, function () { | ||||
|             toast.inf(3, 'aborted'); | ||||
|         }); | ||||
| } | ||||
|  | ||||
| function save_cb() { | ||||
|     if (this.readyState != XMLHttpRequest.DONE) | ||||
|         return; | ||||
|  | ||||
|     if (this.status !== 200) { | ||||
|         alert('Error!  The file was NOT saved.\n\n' + this.status + ": " + (this.responseText + '').replace(/^<pre>/, "")); | ||||
|         return; | ||||
|     } | ||||
|     if (this.status !== 200) | ||||
|         return toast.err(0, 'Error!  The file was NOT saved.\n\n' + this.status + ": " + (this.responseText + '').replace(/^<pre>/, "")); | ||||
|  | ||||
|     var r; | ||||
|     try { | ||||
|         r = JSON.parse(this.responseText); | ||||
|     } | ||||
|     catch (ex) { | ||||
|         alert('Failed to parse reply from server:\n\n' + this.responseText); | ||||
|         return; | ||||
|         return toast.err(0, 'Failed to parse reply from server:\n\n' + this.responseText); | ||||
|     } | ||||
|  | ||||
|     if (!r.ok) { | ||||
|         if (!this.btn.classList.contains('force-save')) { | ||||
|             this.btn.classList.add('force-save'); | ||||
|         if (!clgot(this.btn, 'force-save')) { | ||||
|             clmod(this.btn, 'force-save', 1); | ||||
|             var msg = [ | ||||
|                 'This file has been modified since you started editing it!\n', | ||||
|                 'if you really want to overwrite, press save again.\n', | ||||
| @@ -162,15 +160,13 @@ function save_cb() { | ||||
|                 r.lastmod + ' lastmod on the server now,', | ||||
|                 r.now + ' server time now,\n', | ||||
|             ]; | ||||
|             alert(msg.join('\n')); | ||||
|             return toast.err(0, msg.join('\n')); | ||||
|         } | ||||
|         else { | ||||
|             alert('Error! Save failed.  Maybe this JSON explains why:\n\n' + this.responseText); | ||||
|         } | ||||
|         return; | ||||
|         else | ||||
|             return toast.err(0, 'Error! Save failed.  Maybe this JSON explains why:\n\n' + this.responseText); | ||||
|     } | ||||
|  | ||||
|     this.btn.classList.remove('force-save'); | ||||
|     clmod(this.btn, 'force-save'); | ||||
|     //alert('save OK -- wrote ' + r.size + ' bytes.\n\nsha512: ' + r.sha512); | ||||
|  | ||||
|     // download the saved doc from the server and compare | ||||
| @@ -190,35 +186,23 @@ function save_chk() { | ||||
|     if (this.readyState != XMLHttpRequest.DONE) | ||||
|         return; | ||||
|  | ||||
|     if (this.status !== 200) { | ||||
|         alert('Error!  The file was NOT saved.\n\n' + this.status + ": " + (this.responseText + '').replace(/^<pre>/, "")); | ||||
|         return; | ||||
|     } | ||||
|     if (this.status !== 200) | ||||
|         return toast.err(0, 'Error!  The file was NOT saved.\n\n' + this.status + ": " + (this.responseText + '').replace(/^<pre>/, "")); | ||||
|  | ||||
|     var doc1 = this.txt.replace(/\r\n/g, "\n"); | ||||
|     var doc2 = this.responseText.replace(/\r\n/g, "\n"); | ||||
|     if (doc1 != doc2) { | ||||
|         alert( | ||||
|         modal.alert( | ||||
|             'Error! The document on the server does not appear to have saved correctly (your editor contents and the server copy is not identical). Place the document on your clipboard for now and check the server logs for hints\n\n' + | ||||
|             'Length: yours=' + doc1.length + ', server=' + doc2.length | ||||
|         ); | ||||
|         alert('yours, ' + doc1.length + ' byte:\n[' + doc1 + ']'); | ||||
|         alert('server, ' + doc2.length + ' byte:\n[' + doc2 + ']'); | ||||
|         modal.alert('yours, ' + doc1.length + ' byte:\n[' + doc1 + ']'); | ||||
|         modal.alert('server, ' + doc2.length + ' byte:\n[' + doc2 + ']'); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     last_modified = this.lastmod; | ||||
|     md_changed(this.mde, true); | ||||
|  | ||||
|     var ok = document.createElement('div'); | ||||
|     ok.setAttribute('style', 'font-size:6em;font-family:serif;font-weight:bold;color:#cf6;background:#444;border-radius:.3em;padding:.6em 0;position:fixed;top:30%;left:calc(50% - 2em);width:4em;text-align:center;z-index:9001;transition:opacity 0.2s ease-in-out;opacity:1'); | ||||
|     ok.innerHTML = 'OK✔️'; | ||||
|     var parent = document.getElementById('m'); | ||||
|     document.documentElement.appendChild(ok); | ||||
|     setTimeout(function () { | ||||
|         ok.style.opacity = 0; | ||||
|     }, 500); | ||||
|     setTimeout(function () { | ||||
|         ok.parentNode.removeChild(ok); | ||||
|     }, 750); | ||||
|     toast.ok(2, 'save OK' + (this.ntry ? '\nattempt ' + this.ntry : '')); | ||||
| } | ||||
|   | ||||
| @@ -3,7 +3,7 @@ html,body,tr,th,td,#files,a { | ||||
| 	background: none; | ||||
| 	font-weight: inherit; | ||||
| 	font-size: inherit; | ||||
| 	padding: none; | ||||
| 	padding: 0; | ||||
| 	border: none; | ||||
| } | ||||
| html { | ||||
| @@ -11,21 +11,19 @@ html { | ||||
| 	background: #333; | ||||
| 	font-family: sans-serif; | ||||
| 	text-shadow: 1px 1px 0px #000; | ||||
| 	touch-action: manipulation; | ||||
| } | ||||
| html, body { | ||||
| 	margin: 0; | ||||
| 	padding: 0; | ||||
| } | ||||
| body { | ||||
| 	padding-bottom: 5em; | ||||
| } | ||||
| #box { | ||||
|     padding: .5em 1em; | ||||
|     background: #2c2c2c; | ||||
| 	padding: .5em 1em; | ||||
| 	background: #2c2c2c; | ||||
| } | ||||
| pre { | ||||
| 	font-family: monospace, monospace; | ||||
| } | ||||
| a { | ||||
| 	color: #fc5; | ||||
| } | ||||
| } | ||||
|   | ||||
| @@ -6,7 +6,7 @@ | ||||
|     <title>copyparty</title> | ||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/msg.css"> | ||||
|     <link rel="stylesheet" media="screen" href="/.cpr/msg.css?_={{ ts }}"> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
|   | ||||
| @@ -3,6 +3,9 @@ html, body, #wrap { | ||||
| 	background: #f7f7f7; | ||||
| 	font-family: sans-serif; | ||||
| } | ||||
| html { | ||||
| 	touch-action: manipulation; | ||||
| } | ||||
| #wrap { | ||||
| 	max-width: 40em; | ||||
| 	margin: 2em auto; | ||||
| @@ -26,6 +29,32 @@ a { | ||||
| 	border-radius: .2em; | ||||
| 	padding: .2em .8em; | ||||
| } | ||||
| #repl { | ||||
| 	border: none; | ||||
| 	background: none; | ||||
| 	color: inherit; | ||||
| 	padding: 0; | ||||
| } | ||||
| table { | ||||
| 	border-collapse: collapse; | ||||
| } | ||||
| .vols td, | ||||
| .vols th { | ||||
| 	padding: .3em .6em; | ||||
| 	text-align: left; | ||||
| } | ||||
| .num { | ||||
| 	border-right: 1px solid #bbb; | ||||
| } | ||||
| .num td { | ||||
| 	padding: .1em .7em .1em 0; | ||||
| } | ||||
| .num td:first-child { | ||||
| 	text-align: right; | ||||
| } | ||||
| .btns { | ||||
| 	margin: 1em 0; | ||||
| } | ||||
|  | ||||
|  | ||||
| html.dark, | ||||
| @@ -50,4 +79,7 @@ html.dark input { | ||||
| 	border-radius: .5em; | ||||
| 	padding: .5em .7em; | ||||
| 	margin: 0 .5em 0 0; | ||||
| } | ||||
| } | ||||
| html.dark .num { | ||||
| 	border-color: #777; | ||||
| } | ||||
|   | ||||
| @@ -6,26 +6,57 @@ | ||||
|     <title>copyparty</title> | ||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/splash.css"> | ||||
|     <link rel="stylesheet" media="screen" href="/.cpr/splash.css?_={{ ts }}"> | ||||
|     <link rel="stylesheet" media="screen" href="/.cpr/ui.css?_={{ ts }}"> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
|     <div id="wrap"> | ||||
|         <p>hello {{ this.uname }}</p> | ||||
|  | ||||
|         {%- if avol %} | ||||
|         <h1>admin panel:</h1> | ||||
|         <table><tr><td> <!-- hehehe --> | ||||
|             <table class="num"> | ||||
|                 <tr><td>scanning</td><td>{{ scanning }}</td></tr> | ||||
|                 <tr><td>hash-q</td><td>{{ hashq }}</td></tr> | ||||
|                 <tr><td>tag-q</td><td>{{ tagq }}</td></tr> | ||||
|                 <tr><td>mtp-q</td><td>{{ mtpq }}</td></tr> | ||||
|             </table> | ||||
|         </td><td> | ||||
|             <table class="vols"> | ||||
|                 <thead><tr><th>vol</th><th>action</th><th>status</th></tr></thead> | ||||
|                 <tbody> | ||||
|                     {% for mp in avol %} | ||||
|                     {%- if mp in vstate and vstate[mp] %} | ||||
|                     <tr><td><a href="{{ mp }}{{ url_suf }}">{{ mp }}</a></td><td><a href="{{ mp }}?scan">rescan</a></td><td>{{ vstate[mp] }}</td></tr> | ||||
|                     {%- endif %} | ||||
|                     {% endfor %} | ||||
|                 </tbody> | ||||
|             </table> | ||||
|         </td></tr></table> | ||||
|         <div class="btns"> | ||||
|             <a href="/?stack">dump stack</a> | ||||
|         </div> | ||||
|         {%- endif %} | ||||
|  | ||||
|         {%- if rvol %} | ||||
|         <h1>you can browse these:</h1> | ||||
|         <ul> | ||||
|             {% for mp in rvol %} | ||||
|             <li><a href="/{{ mp }}">/{{ mp }}</a></li> | ||||
|             <li><a href="{{ mp }}{{ url_suf }}">{{ mp }}</a></li> | ||||
|             {% endfor %} | ||||
|         </ul> | ||||
|         {%- endif %} | ||||
|  | ||||
|         {%- if wvol %} | ||||
|         <h1>you can upload to:</h1> | ||||
|         <ul> | ||||
|             {% for mp in wvol %} | ||||
|             <li><a href="/{{ mp }}">/{{ mp }}</a></li> | ||||
|             <li><a href="{{ mp }}{{ url_suf }}">{{ mp }}</a></li> | ||||
|             {% endfor %} | ||||
|         </ul> | ||||
|         {%- endif %} | ||||
|  | ||||
|         <h1>login for more:</h1> | ||||
|         <ul> | ||||
| @@ -36,11 +67,13 @@ | ||||
|             </form> | ||||
|         </ul> | ||||
|     </div> | ||||
| 	<a href="#" id="repl">π</a> | ||||
|     <script> | ||||
|  | ||||
| if (window.localStorage && localStorage.getItem('darkmode') == 1) | ||||
| if (localStorage.getItem('lightmode') != 1) | ||||
|     document.documentElement.setAttribute("class", "dark"); | ||||
|  | ||||
| </script> | ||||
| <script src="/.cpr/util.js?_={{ ts }}"></script> | ||||
| </body> | ||||
| </html> | ||||
| </html> | ||||
|   | ||||
							
								
								
									
										442
									
								
								copyparty/web/ui.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										442
									
								
								copyparty/web/ui.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,442 @@ | ||||
| @font-face { | ||||
| 	font-family: 'scp'; | ||||
| 	src: local('Source Code Pro Regular'), local('SourceCodePro-Regular'), url(/.cpr/deps/scp.woff2) format('woff2'); | ||||
| } | ||||
| html { | ||||
| 	touch-action: manipulation; | ||||
| } | ||||
| #tt, #toast { | ||||
| 	position: fixed; | ||||
| 	max-width: 34em; | ||||
| 	max-width: min(34em, 90%); | ||||
| 	max-width: min(34em, calc(100% - 7em)); | ||||
| 	background: #222; | ||||
| 	border: 0 solid #777; | ||||
| 	box-shadow: 0 .2em .5em #222; | ||||
| 	border-radius: .4em; | ||||
| 	z-index: 9001; | ||||
| } | ||||
| #tt { | ||||
| 	max-width: min(34em, calc(100% - 3.3em)); | ||||
| 	overflow: hidden; | ||||
| 	margin: .7em 0; | ||||
| 	padding: 0 1.3em; | ||||
| 	height: 0; | ||||
| 	opacity: .1; | ||||
| 	transition: opacity 0.14s, height 0.14s, padding 0.14s; | ||||
| } | ||||
| #toast { | ||||
| 	bottom: 5em; | ||||
| 	right: -1em; | ||||
| 	line-height: 1.5em; | ||||
| 	padding: 1em 1.3em; | ||||
| 	margin-left: 3em; | ||||
| 	border-width: .4em 0; | ||||
| 	overflow-wrap: break-word; | ||||
| 	transform: translateX(100%); | ||||
| 	transition: | ||||
| 		transform .4s cubic-bezier(.2, 1.2, .5, 1), | ||||
| 		right .4s cubic-bezier(.2, 1.2, .5, 1); | ||||
| 	text-shadow: 1px 1px 0 #000; | ||||
| 	color: #fff; | ||||
| } | ||||
| #toast a { | ||||
| 	color: inherit; | ||||
| 	text-shadow: inherit; | ||||
| 	background: rgba(0, 0, 0, 0.4); | ||||
| 	border-radius: .3em; | ||||
| 	padding: .2em .3em; | ||||
| } | ||||
| #toast a#toastc { | ||||
| 	display: inline-block; | ||||
| 	position: absolute; | ||||
| 	overflow: hidden; | ||||
| 	left: 0; | ||||
| 	width: 0; | ||||
| 	opacity: 0; | ||||
| 	padding: .3em 0; | ||||
| 	margin: -.3em 0 0 0; | ||||
| 	line-height: 1.3em; | ||||
| 	color: #000; | ||||
| 	border: none; | ||||
| 	outline: none; | ||||
| 	text-shadow: none; | ||||
| 	border-radius: .5em 0 0 .5em; | ||||
| 	transition: left .3s, width .3s, padding .3s, opacity .3s; | ||||
| } | ||||
| #toastb { | ||||
| 	max-height: 70vh; | ||||
| 	overflow-y: auto; | ||||
| } | ||||
| #toast.scroll #toastb { | ||||
| 	overflow-y: scroll; | ||||
| 	margin-right: -1.2em; | ||||
| 	padding-right: .7em; | ||||
| } | ||||
| #toast pre { | ||||
| 	margin: 0; | ||||
| } | ||||
| #toast.vis { | ||||
| 	right: 1.3em; | ||||
| 	transform: unset; | ||||
| } | ||||
| #toast.vis #toastc { | ||||
| 	left: -2em; | ||||
| 	width: .4em; | ||||
| 	padding: .3em .8em; | ||||
| 	opacity: 1; | ||||
| } | ||||
| #toast.inf { | ||||
| 	background: #07a; | ||||
| 	border-color: #0be; | ||||
| } | ||||
| #toast.inf #toastc { | ||||
| 	background: #0be; | ||||
| } | ||||
| #toast.ok { | ||||
| 	background: #380; | ||||
| 	border-color: #8e4; | ||||
| } | ||||
| #toast.ok #toastc { | ||||
| 	background: #8e4; | ||||
| } | ||||
| #toast.warn { | ||||
| 	background: #960; | ||||
| 	border-color: #fc0; | ||||
| } | ||||
| #toast.warn #toastc { | ||||
| 	background: #fc0; | ||||
| } | ||||
| #toast.err { | ||||
| 	background: #900; | ||||
| 	border-color: #d06; | ||||
| } | ||||
| #toast.err #toastc { | ||||
| 	background: #d06; | ||||
| } | ||||
| #tt.b { | ||||
| 	padding: 0 2em; | ||||
| 	border-radius: .5em; | ||||
| 	box-shadow: 0 .2em 1em #000; | ||||
| } | ||||
| #tt.show { | ||||
| 	padding: 1em 1.3em; | ||||
| 	border-width: .4em 0; | ||||
| 	height: auto; | ||||
| 	opacity: 1; | ||||
| } | ||||
| #tt.show.b { | ||||
| 	padding: 1.5em 2em; | ||||
| 	border-width: .5em 0; | ||||
| } | ||||
| #modalc code, | ||||
| #tt code { | ||||
| 	background: #3c3c3c; | ||||
| 	padding: .1em .3em; | ||||
| 	border-top: 1px solid #777; | ||||
| 	border-radius: .3em; | ||||
| 	line-height: 1.7em; | ||||
| } | ||||
| #tt em { | ||||
| 	color: #f6a; | ||||
| } | ||||
| html.light #tt { | ||||
| 	background: #fff; | ||||
| 	border-color: #888 #000 #777 #000; | ||||
| } | ||||
| html.light #tt, | ||||
| html.light #toast { | ||||
| 	box-shadow: 0 .3em 1em rgba(0,0,0,0.4); | ||||
| } | ||||
| #modalc code, | ||||
| html.light #tt code { | ||||
| 	background: #060; | ||||
| 	color: #fff; | ||||
| } | ||||
| html.light #tt em { | ||||
| 	color: #d38; | ||||
| } | ||||
| #modal { | ||||
| 	position: fixed; | ||||
|     overflow: auto; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| 	right: 0; | ||||
| 	bottom: 0; | ||||
| 	width: 100%; | ||||
| 	height: 100%; | ||||
| 	z-index: 9001; | ||||
| 	background: rgba(64,64,64,0.6); | ||||
| } | ||||
| #modal>table { | ||||
|     width: 100%; | ||||
|     height: 100%; | ||||
| } | ||||
| #modal td { | ||||
| 	text-align: center; | ||||
| } | ||||
| #modalc { | ||||
| 	position: relative; | ||||
| 	display: inline-block; | ||||
| 	background: #f7f7f7; | ||||
| 	color: #333; | ||||
| 	text-shadow: none; | ||||
| 	text-align: left; | ||||
| 	margin: 3em; | ||||
| 	padding: 1em 1.1em; | ||||
| 	border-radius: .6em; | ||||
|     box-shadow: 0 .3em 3em rgba(0,0,0,0.5); | ||||
| 	max-width: 50em; | ||||
| 	max-height: 30em; | ||||
| 	overflow: auto; | ||||
| } | ||||
| @media (min-width: 40em) { | ||||
|     #modalc { | ||||
|         min-width: 30em; | ||||
|     } | ||||
| } | ||||
| #modalc li { | ||||
| 	margin: 1em 0; | ||||
| } | ||||
| #modalc h6 { | ||||
| 	font-size: 1.3em; | ||||
| 	border-bottom: 1px solid #999; | ||||
| 	margin: 0; | ||||
| 	padding: .3em; | ||||
| 	text-align: center; | ||||
| } | ||||
| #modalb { | ||||
| 	position: sticky; | ||||
| 	text-align: right; | ||||
| 	padding-top: 1em; | ||||
| 	bottom: 0; | ||||
| 	right: 0; | ||||
| } | ||||
| #modalb a { | ||||
| 	color: #000; | ||||
| 	background: #ccc; | ||||
| 	display: inline-block; | ||||
| 	border-radius: .3em; | ||||
| 	padding: .5em 1em; | ||||
| 	outline: none; | ||||
| 	border: none; | ||||
| } | ||||
| #modalb a:focus, | ||||
| #modalb a:hover { | ||||
| 	background: #06d; | ||||
| 	color: #fff; | ||||
| } | ||||
| #modalb a+a { | ||||
| 	margin-left: .5em; | ||||
| } | ||||
| #modali { | ||||
| 	display: block; | ||||
| 	background: #fff; | ||||
| 	color: #000; | ||||
| 	width: calc(100% - 1.25em); | ||||
|     margin: 1em -.1em 0 -.1em; | ||||
| 	padding: .5em; | ||||
| 	outline: none; | ||||
| 	border: .25em solid #ccc; | ||||
| 	border-radius: .4em; | ||||
| } | ||||
| #modali:focus { | ||||
| 	border-color: #06d; | ||||
| } | ||||
| #repl_pre { | ||||
| 	max-width: 24em; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| .mdo pre, | ||||
| .mdo code, | ||||
| .mdo a { | ||||
| 	color: #480; | ||||
| 	background: #f7f7f7; | ||||
| 	border: .07em solid #ddd; | ||||
| 	border-radius: .2em; | ||||
| 	padding: .1em .3em; | ||||
| 	margin: 0 .1em; | ||||
| } | ||||
| .mdo pre, | ||||
| .mdo code, | ||||
| .mdo tt { | ||||
| 	font-family: 'scp', monospace, monospace; | ||||
| 	white-space: pre-wrap; | ||||
| 	word-break: break-all; | ||||
| } | ||||
| .mdo code { | ||||
| 	font-size: .96em; | ||||
| } | ||||
| .mdo h1, | ||||
| .mdo h2 { | ||||
| 	line-height: 1.5em; | ||||
| } | ||||
| .mdo h1 { | ||||
| 	font-size: 1.7em; | ||||
| 	text-align: center; | ||||
| 	border: 1em solid #777; | ||||
| 	border-width: .05em 0; | ||||
| 	margin: 3em 0; | ||||
| } | ||||
| .mdo h2 { | ||||
| 	font-size: 1.5em; | ||||
| 	font-weight: normal; | ||||
| 	background: #f7f7f7; | ||||
| 	border-top: .07em solid #fff; | ||||
| 	border-bottom: .07em solid #bbb; | ||||
| 	border-radius: .5em .5em 0 0; | ||||
| 	padding-left: .4em; | ||||
| 	margin-top: 3em; | ||||
| } | ||||
| .mdo h3 { | ||||
| 	border-bottom: .1em solid #999; | ||||
| } | ||||
| .mdo h1 a, .mdo h3 a, .mdo h5 a, | ||||
| .mdo h2 a, .mdo h4 a, .mdo h6 a { | ||||
| 	color: inherit; | ||||
| 	display: block; | ||||
| 	background: none; | ||||
| 	border: none; | ||||
| 	padding: 0; | ||||
| 	margin: 0; | ||||
| } | ||||
| .mdo ul, | ||||
| .mdo ol { | ||||
| 	border-left: .3em solid #ddd; | ||||
| } | ||||
| .mdo ul>li, | ||||
| .mdo ol>li { | ||||
| 	margin: .7em 0; | ||||
| 	list-style-type: disc; | ||||
| } | ||||
| .mdo strong { | ||||
| 	color: #000; | ||||
| } | ||||
| .mdo p>em, | ||||
| .mdo li>em, | ||||
| .mdo td>em { | ||||
| 	color: #c50; | ||||
| 	padding: .1em; | ||||
| 	border-bottom: .1em solid #bbb; | ||||
| } | ||||
| .mdo blockquote { | ||||
| 	font-family: serif; | ||||
| 	background: #f7f7f7; | ||||
| 	border: .07em dashed #ccc; | ||||
| 	padding: 0 2em; | ||||
| 	margin: 1em 0; | ||||
| } | ||||
| .mdo small { | ||||
| 	opacity: .8; | ||||
| } | ||||
| .mdo pre code { | ||||
| 	display: block; | ||||
| 	margin: 0 -.3em; | ||||
| 	padding: .4em .5em; | ||||
| 	line-height: 1.1em; | ||||
| } | ||||
| .mdo pre code:hover { | ||||
| 	background: #fec; | ||||
| 	color: #360; | ||||
| } | ||||
| .mdo table { | ||||
| 	border-collapse: collapse; | ||||
| 	margin: 1em 0; | ||||
| } | ||||
| .mdo th, | ||||
| .mdo td { | ||||
| 	padding: .2em .5em; | ||||
| 	border: .12em solid #aaa; | ||||
| } | ||||
|  | ||||
| @media screen { | ||||
| 	.mdo { | ||||
| 		word-break: break-word; | ||||
| 		overflow-wrap: break-word; | ||||
| 		word-wrap: break-word; /*ie*/ | ||||
| 	} | ||||
| 	html.light .mdo a, | ||||
| 	.mdo a { | ||||
| 		color: #fff; | ||||
| 		background: #39b; | ||||
| 		text-decoration: none; | ||||
| 		padding: 0 .3em; | ||||
| 		border: none; | ||||
| 		border-bottom: .07em solid #079; | ||||
| 	} | ||||
| 	.mdo h1 { | ||||
| 		color: #fff; | ||||
| 		background: #444; | ||||
| 		font-weight: normal; | ||||
| 		border-top: .4em solid #fb0; | ||||
| 		border-bottom: .4em solid #777; | ||||
| 		border-radius: 0 1em 0 1em; | ||||
| 		margin: 3em 0 1em 0; | ||||
| 		padding: .5em 0; | ||||
| 	} | ||||
| 	.mdo h2 { | ||||
| 		color: #fff; | ||||
| 		background: #555; | ||||
| 		margin-top: 2em; | ||||
| 		border-bottom: .22em solid #999; | ||||
| 		border-top: none; | ||||
| 	} | ||||
|  | ||||
|  | ||||
|  | ||||
| 	html.dark .mdo a { | ||||
| 		background: #057; | ||||
| 	} | ||||
| 	html.dark .mdo h1 a, html.dark .mdo h4 a, | ||||
| 	html.dark .mdo h2 a, html.dark .mdo h5 a, | ||||
| 	html.dark .mdo h3 a, html.dark .mdo h6 a { | ||||
| 		color: inherit; | ||||
| 		background: none; | ||||
| 	} | ||||
| 	html.dark .mdo pre, | ||||
| 	html.dark .mdo code { | ||||
| 		color: #8c0; | ||||
| 		background: #1a1a1a; | ||||
| 		border: .07em solid #333; | ||||
| 	} | ||||
| 	html.dark .mdo ul, | ||||
| 	html.dark .mdo ol { | ||||
| 		border-color: #444; | ||||
| 	} | ||||
| 	html.dark .mdo strong { | ||||
| 		color: #fff; | ||||
| 	} | ||||
| 	html.dark .mdo p>em, | ||||
| 	html.dark .mdo li>em, | ||||
| 	html.dark .mdo td>em { | ||||
| 		color: #f94; | ||||
| 		border-color: #666; | ||||
| 	} | ||||
| 	html.dark .mdo h1 { | ||||
| 		background: #383838; | ||||
| 		border-top: .4em solid #b80; | ||||
| 		border-bottom: .4em solid #4c4c4c; | ||||
| 	} | ||||
| 	html.dark .mdo h2 { | ||||
| 		background: #444; | ||||
| 		border-bottom: .22em solid #555; | ||||
| 	} | ||||
| 	html.dark .mdo td, | ||||
| 	html.dark .mdo th { | ||||
| 		border-color: #444; | ||||
| 	} | ||||
| 	html.dark .mdo blockquote { | ||||
| 		background: #282828; | ||||
| 		border: .07em dashed #444; | ||||
| 	} | ||||
| } | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,221 +0,0 @@ | ||||
| .opview { | ||||
| 	display: none; | ||||
| } | ||||
| .opview.act { | ||||
| 	display: block; | ||||
| } | ||||
| #ops a { | ||||
| 	color: #fc5; | ||||
| 	font-size: 1.5em; | ||||
| 	padding: 0 .3em; | ||||
| 	margin: 0; | ||||
| 	outline: none; | ||||
| } | ||||
| #ops a.act { | ||||
| 	text-decoration: underline; | ||||
| } | ||||
| /* | ||||
| #ops a+a:after, | ||||
| #ops a:first-child:after { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #01a7e1; | ||||
| 	margin-left: .3em; | ||||
| 	position: relative; | ||||
| } | ||||
| #ops a+a:before { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #ff3f1a; | ||||
| 	margin-right: .3em; | ||||
| 	margin-left: -.3em; | ||||
| } | ||||
| #ops a:last-child:after { | ||||
| 	content: ''; | ||||
| } | ||||
| #ops a.act:before, | ||||
| #ops a.act:after { | ||||
| 	text-decoration: none !important; | ||||
| } | ||||
| */ | ||||
| #ops i { | ||||
| 	font-size: 1.5em; | ||||
| } | ||||
| #ops i:before { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #01a7e1; | ||||
| 	position: relative; | ||||
| } | ||||
| #ops i:after { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #ff3f1a; | ||||
| 	margin-left: -.35em; | ||||
| 	font-size: 1.05em; | ||||
| } | ||||
| #ops, | ||||
| .opbox { | ||||
| 	border: 1px solid #3a3a3a; | ||||
| 	box-shadow: 0 0 1em #222 inset; | ||||
| } | ||||
| #ops { | ||||
| 	display: none; | ||||
| 	background: #333; | ||||
| 	margin: 1.7em 1.5em 0 1.5em; | ||||
| 	padding: .3em .6em; | ||||
| 	border-radius: .3em; | ||||
| 	border-width: .15em 0; | ||||
| } | ||||
| .opbox { | ||||
| 	background: #2d2d2d; | ||||
| 	margin: 1.5em 0 0 0; | ||||
| 	padding: .5em; | ||||
| 	border-radius: 0 1em 1em 0; | ||||
| 	border-width: .15em .3em .3em 0; | ||||
| 	max-width: 40em; | ||||
| } | ||||
| .opbox input { | ||||
| 	margin: .5em; | ||||
| } | ||||
| .opbox input[type=text] { | ||||
| 	color: #fff; | ||||
| 	background: #383838; | ||||
| 	border: none; | ||||
| 	box-shadow: 0 0 .3em #222; | ||||
| 	border-bottom: 1px solid #fc5; | ||||
| 	border-radius: .2em; | ||||
| 	padding: .2em .3em; | ||||
| } | ||||
| #op_up2k { | ||||
| 	padding: 0 1em 1em 1em; | ||||
| } | ||||
| #u2form { | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| } | ||||
| #u2form input { | ||||
| 	background: #444; | ||||
| 	border: 0px solid #444; | ||||
| 	outline: none; | ||||
| } | ||||
| #u2err.err { | ||||
| 	color: #f87; | ||||
| 	padding: .5em; | ||||
| } | ||||
| #u2form { | ||||
| 	width: 2px; | ||||
| 	height: 2px; | ||||
| 	overflow: hidden; | ||||
| } | ||||
| #u2btn { | ||||
| 	color: #eee; | ||||
| 	background: #555; | ||||
| 	background: -moz-linear-gradient(top, #367 0%, #489 50%, #38788a 51%, #367 100%); | ||||
| 	background: -webkit-linear-gradient(top, #367 0%, #489 50%, #38788a 51%, #367 100%); | ||||
| 	background: linear-gradient(to bottom, #367 0%, #489 50%, #38788a 51%, #367 100%); | ||||
| 	filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#489', endColorstr='#38788a', GradientType=0); | ||||
| 	text-decoration: none; | ||||
| 	line-height: 1.5em; | ||||
| 	border: 1px solid #222; | ||||
| 	border-radius: .4em; | ||||
| 	text-align: center; | ||||
| 	font-size: 2em; | ||||
| 	margin: 1em auto; | ||||
| 	padding: 1em 0; | ||||
| 	width: 12em; | ||||
| 	cursor: pointer; | ||||
| 	box-shadow: .4em .4em 0 #111; | ||||
| } | ||||
| #u2notbtn { | ||||
| 	display: none; | ||||
| 	text-align: center; | ||||
| 	background: #333; | ||||
| 	padding-top: 1em; | ||||
| } | ||||
| #u2notbtn * { | ||||
| 	line-height: 1.3em; | ||||
| } | ||||
| #u2tab { | ||||
| 	margin: 3em auto; | ||||
| 	width: calc(100% - 2em); | ||||
| 	max-width: 100em; | ||||
| } | ||||
| #u2tab td { | ||||
| 	border: 1px solid #ccc; | ||||
| 	border-width: 0 0px 1px 0; | ||||
| 	padding: .1em .3em; | ||||
| } | ||||
| #u2tab td:nth-child(2) { | ||||
| 	width: 5em; | ||||
| } | ||||
| #u2tab td:nth-child(3) { | ||||
| 	width: 40%; | ||||
| } | ||||
| #u2tab tr+tr:hover td { | ||||
| 	background: #222; | ||||
| } | ||||
| #u2conf { | ||||
| 	margin: 1em auto; | ||||
| 	width: 26em; | ||||
| } | ||||
| #u2conf * { | ||||
| 	text-align: center; | ||||
| 	line-height: 1em; | ||||
| 	margin: 0; | ||||
| 	padding: 0; | ||||
| 	border: none; | ||||
| 	outline: none; | ||||
| } | ||||
| #u2conf .txtbox { | ||||
| 	width: 4em; | ||||
| 	color: #fff; | ||||
| 	background: #444; | ||||
| 	border: 1px solid #777; | ||||
| 	font-size: 1.2em; | ||||
| 	padding: .15em 0; | ||||
| } | ||||
| #u2conf a { | ||||
| 	color: #fff; | ||||
| 	background: #c38; | ||||
| 	text-decoration: none; | ||||
| 	border-radius: .1em; | ||||
| 	font-size: 1.5em; | ||||
| 	padding: .1em 0; | ||||
| 	margin: 0 -.25em; | ||||
| 	width: 1.5em; | ||||
| 	height: 1em; | ||||
| 	display: inline-block; | ||||
| 	position: relative; | ||||
| 	line-height: 1em; | ||||
| 	bottom: -.08em; | ||||
| } | ||||
| #u2conf input+a { | ||||
| 	background: #d80; | ||||
| } | ||||
| #u2foot { | ||||
| 	color: #fff; | ||||
| 	font-style: italic; | ||||
| } | ||||
| .prog { | ||||
| 	font-family: monospace; | ||||
| } | ||||
| .prog>div { | ||||
| 	display: inline-block; | ||||
| 	position: relative; | ||||
| 	overflow: hidden; | ||||
| 	margin: 0; | ||||
| 	padding: 0; | ||||
| 	height: 1.1em; | ||||
| 	margin-bottom: -.15em; | ||||
| 	box-shadow: -1px -1px 0 inset rgba(255,255,255,0.1); | ||||
| } | ||||
| .prog>div>div { | ||||
| 	width: 0%; | ||||
| 	position: absolute; | ||||
| 	left: 0; | ||||
| 	top: 0; | ||||
| 	bottom: 0; | ||||
| 	background: #0a0; | ||||
| } | ||||
| @@ -1,70 +0,0 @@ | ||||
|     <div id="ops"><a | ||||
|         href="#" data-dest="">---</a><i></i><a | ||||
|         href="#" data-dest="up2k">up2k</a><i></i><a | ||||
|         href="#" data-dest="bup">bup</a><i></i><a | ||||
|         href="#" data-dest="mkdir">mkdir</a><i></i><a | ||||
|         href="#" data-dest="new_md">new.md</a></div> | ||||
|  | ||||
|     <div id="op_bup" class="opview opbox act"> | ||||
|         <div id="u2err"></div> | ||||
|         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}"> | ||||
|             <input type="hidden" name="act" value="bput" /> | ||||
|             <input type="file" name="f" multiple><br /> | ||||
|             <input type="submit" value="start upload"> | ||||
|         </form> | ||||
|     </div> | ||||
|  | ||||
|     <div id="op_mkdir" class="opview opbox act"> | ||||
|         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}"> | ||||
|             <input type="hidden" name="act" value="mkdir" /> | ||||
|             <input type="text" name="name" size="30"> | ||||
|             <input type="submit" value="mkdir"> | ||||
|         </form> | ||||
|     </div> | ||||
|  | ||||
|     <div id="op_new_md" class="opview opbox"> | ||||
|         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}"> | ||||
|             <input type="hidden" name="act" value="new_md" /> | ||||
|             <input type="text" name="name" size="30"> | ||||
|             <input type="submit" value="create doc"> | ||||
|         </form> | ||||
|     </div> | ||||
|  | ||||
|     <div id="op_up2k" class="opview"> | ||||
|         <form id="u2form" method="post" enctype="multipart/form-data" onsubmit="return false;"></form> | ||||
|  | ||||
|             <table id="u2conf"> | ||||
|                 <tr> | ||||
|                     <td>parallel uploads</td> | ||||
|                 </tr> | ||||
|                 <tr> | ||||
|                     <td> | ||||
|                         <a href="#" id="nthread_sub">–</a> | ||||
|                         <input class="txtbox" id="nthread" value="2" /> | ||||
|                         <a href="#" id="nthread_add">+</a> | ||||
|                     </td> | ||||
|                     <td rowspan="2"> | ||||
|                         <input type="checkbox" id="multitask" /> | ||||
|                         <label for="multitask">hash while<br />uploading</label> | ||||
|                     </td> | ||||
|                 </tr> | ||||
|             </table> | ||||
|  | ||||
|             <div id="u2notbtn"></div> | ||||
|  | ||||
|             <div id="u2btn"> | ||||
|                 drop files here<br /> | ||||
|                 (or click me) | ||||
|             </div> | ||||
|  | ||||
|             <table id="u2tab"> | ||||
|                 <tr> | ||||
|                     <td>filename</td> | ||||
|                     <td>status</td> | ||||
|                     <td>progress</td> | ||||
|                 </tr> | ||||
|             </table> | ||||
|  | ||||
|             <p id="u2foot"></p> | ||||
|             <p>( if you don't need lastmod timestamps, resumable uploads or progress bars just use the <a href="#" id="u2nope">basic uploader</a>)</p> | ||||
|     </div> | ||||
							
								
								
									
										1106
									
								
								copyparty/web/util.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1106
									
								
								copyparty/web/util.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										51
									
								
								docs/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								docs/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,51 @@ | ||||
| **NOTE:** there's more stuff (sharex config, service scripts, nginx configs, ...) in [`/contrib/`](/contrib/) | ||||
|  | ||||
|  | ||||
|  | ||||
| # 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-css | ||||
| point `--css-browser` to one of these by URL: | ||||
|  | ||||
| * [`browser.css`](browser.css) changes the background | ||||
| * [`browser-icons.css`](browser-icons.css) adds filetype icons | ||||
|  | ||||
|  | ||||
|  | ||||
| # other stuff | ||||
|  | ||||
| ## [`rclone.md`](rclone.md) | ||||
| * notes on using rclone as a fuse client/server | ||||
|  | ||||
| ## [`example.conf`](example.conf) | ||||
| * example config file for `-c` (supports accounts, volumes, and volume-flags) | ||||
|  | ||||
|  | ||||
|  | ||||
| # junk | ||||
|  | ||||
| alphabetical list of the remaining files | ||||
|  | ||||
| | what | why | | ||||
| | -- | -- | | ||||
| | [biquad.html](biquad.html) | bruteforce calibrator for the audio equalizer since im not that good at maths | | ||||
| | [design.txt](design.txt) | initial brainstorming of the copyparty design, unmaintained, incorrect, sentimental value only | | ||||
| | [hls.html](hls.html) | experimenting with hls playback using `hls.js`, works p well, almost became a thing | | ||||
| | [music-analysis.sh](music-analysis.sh) | testing various bpm/key detection libraries before settling on the ones used in [`/bin/mtag/`](/bin/mtag/) | | ||||
| | [notes.sh](notes.sh) | notepad, just scraps really | | ||||
| | [nuitka.txt](nuitka.txt) | how to build a copyparty exe using nuitka (not maintained) | | ||||
| | [pretend-youre-qnap.patch](pretend-youre-qnap.patch) | simulate a NAS which keeps returning old cached data even though you just modified the file yourself | | ||||
| | [tcp-debug.sh](tcp-debug.sh) | looks like this was to debug stuck tcp connections? | | ||||
| | [unirange.py](unirange.py) | uhh | | ||||
| | [up2k.txt](up2k.txt) | initial ideas for how up2k should work, another unmaintained sentimental-value-only thing | | ||||
							
								
								
									
										113
									
								
								docs/biquad.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								docs/biquad.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | ||||
| <!DOCTYPE html><html><head></head><body><script> | ||||
|  | ||||
| setTimeout(location.reload.bind(location), 700); | ||||
| document.documentElement.scrollLeft = 0; | ||||
|  | ||||
| var cali = (function() { | ||||
|     var ac = new AudioContext(), | ||||
|         fi = ac.createBiquadFilter(), | ||||
|         freqs = new Float32Array(1), | ||||
|         mag = new Float32Array(1), | ||||
|         phase = new Float32Array(1); | ||||
|  | ||||
|     freqs[0] = 14000; | ||||
|     fi.type = 'peaking'; | ||||
|     fi.frequency.value = 18000; | ||||
|     fi.Q.value = 0.8; | ||||
|     fi.gain.value = 1; | ||||
|     fi.getFrequencyResponse(freqs, mag, phase); | ||||
|  | ||||
|     return mag[0];  // 1.0407 good, 1.0563 bad | ||||
| })(), | ||||
|     mp = cali < 1.05; | ||||
|  | ||||
| var can = document.createElement('canvas'), | ||||
|     cc = can.getContext('2d'), | ||||
|     w = 2048, | ||||
|     h = 1024; | ||||
|  | ||||
| w = 2048; | ||||
|  | ||||
| can.width = w; | ||||
| can.height = h; | ||||
| document.body.appendChild(can); | ||||
| can.style.cssText = 'width:' + w + 'px;height:' + h + 'px'; | ||||
|  | ||||
| cc.fillStyle = '#000'; | ||||
| cc.fillRect(0, 0, w, h); | ||||
|  | ||||
| var cfg = [ // hz, q, g | ||||
|     [31.25 * 0.88, 0, 1.4],  // shelf | ||||
|     [31.25 * 1.04, 0.7, 0.96],  // peak | ||||
|     [62.5, 0.7, 1], | ||||
|     [125, 0.8, 1], | ||||
|     [250, 0.9, 1.03], | ||||
|     [500, 0.9, 1.1], | ||||
|     [1000, 0.9, 1.1], | ||||
|     [2000, 0.9, 1.105], | ||||
|     [4000, 0.88, 1.05], | ||||
|     [8000 * 1.006, 0.73, mp ? 1.24 : 1.2], | ||||
|     //[16000 * 1.00, 0.5, 1.75],  // peak.v1 | ||||
|     //[16000 * 1.19, 0, 1.8]  // shelf.v1 | ||||
|     [16000 * 0.89, 0.7, mp ? 1.26 : 1.2],  // peak | ||||
|     [16000 * 1.13, 0.82, mp ? 1.09 : 0.75],  // peak | ||||
|     [16000 * 1.205, 0, mp ? 1.9 : 1.85]  // shelf | ||||
| ]; | ||||
|  | ||||
| var freqs = new Float32Array(22000), | ||||
|     sum = new Float32Array(freqs.length), | ||||
|     ac = new AudioContext(), | ||||
|     step = w / freqs.length, | ||||
|     colors = [ | ||||
|         'rgba(255, 0, 0, 0.7)', | ||||
|         'rgba(0, 224, 0, 0.7)', | ||||
|         'rgba(0, 64, 255, 0.7)' | ||||
|     ]; | ||||
|  | ||||
| var order = []; | ||||
|  | ||||
| for (var a = 0; a < cfg.length; a += 2) | ||||
|     order.push(a); | ||||
|  | ||||
| for (var a = 1; a < cfg.length; a += 2) | ||||
|     order.push(a); | ||||
|  | ||||
| for (var ia = 0; ia < order.length; ia++) { | ||||
|     var a = order[ia], | ||||
|         fi = ac.createBiquadFilter(), | ||||
|         mag = new Float32Array(freqs.length), | ||||
|         phase = new Float32Array(freqs.length); | ||||
|  | ||||
|     for (var b = 0; b < freqs.length; b++) | ||||
|         freqs[b] = b; | ||||
|  | ||||
|     fi.type = a == 0 ? 'lowshelf' : a == cfg.length - 1 ? 'highshelf' : 'peaking'; | ||||
|     fi.frequency.value = cfg[a][0]; | ||||
|     fi.Q.value = cfg[a][1]; | ||||
|     fi.gain.value = 1; | ||||
|  | ||||
|     fi.getFrequencyResponse(freqs, mag, phase); | ||||
|     cc.fillStyle = colors[a % colors.length]; | ||||
|     for (var b = 0; b < sum.length; b++) { | ||||
|         mag[b] -= 1; | ||||
|         sum[b] += mag[b] * cfg[a][2]; | ||||
|         var y = h - (mag[b] * h * 3); | ||||
|         cc.fillRect(b * step, y, step, h - y); | ||||
|         cc.fillRect(b * step - 1, y - 1, 3, 3); | ||||
|     } | ||||
| } | ||||
|  | ||||
| var min = 999999, max = 0; | ||||
| for (var a = 0; a < sum.length; a++) { | ||||
|     min = Math.min(min, sum[a]); | ||||
|     max = Math.max(max, sum[a]); | ||||
| } | ||||
| cc.fillStyle = 'rgba(255,255,255,1)'; | ||||
| for (var a = 0; a < sum.length; a++) { | ||||
|     var v = (sum[a] - min) / (max - min); | ||||
|     cc.fillRect(a * step, 0, step, v * h / 2); | ||||
| } | ||||
|  | ||||
| cc.fillRect(0, 460, w, 1); | ||||
|  | ||||
| </script></body></html> | ||||
							
								
								
									
										71
									
								
								docs/browser-icons.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								docs/browser-icons.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| /* video, alternative 1: | ||||
|    top-left icon, just like the other formats | ||||
| ======================================================================= | ||||
|  | ||||
| #ggrid>a:is( | ||||
| [href$=".mkv"i], | ||||
| [href$=".mp4"i], | ||||
| [href$=".webm"i], | ||||
| ):before { | ||||
|     content: '📺'; | ||||
| } | ||||
| */ | ||||
|  | ||||
|  | ||||
|  | ||||
| /* video, alternative 2: | ||||
|    play-icon in the middle of the thumbnail | ||||
| ======================================================================= | ||||
| */ | ||||
| #ggrid>a:is( | ||||
| [href$=".mkv"i], | ||||
| [href$=".mp4"i], | ||||
| [href$=".webm"i], | ||||
| ) { | ||||
| 	position: relative; | ||||
| 	overflow: hidden; | ||||
| } | ||||
| #ggrid>a:is( | ||||
| [href$=".mkv"i], | ||||
| [href$=".mp4"i], | ||||
| [href$=".webm"i], | ||||
| ):before { | ||||
|     content: '▶'; | ||||
| 	opacity: .8; | ||||
| 	margin: 0; | ||||
| 	padding: 1em .5em 1em .7em; | ||||
| 	border-radius: 9em; | ||||
| 	line-height: 0; | ||||
| 	color: #fff; | ||||
| 	text-shadow: none; | ||||
| 	background: rgba(0, 0, 0, 0.7); | ||||
| 	left: calc(50% - 1em); | ||||
| 	top: calc(50% - 1.4em); | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| /* audio */ | ||||
| #ggrid>a:is( | ||||
| [href$=".mp3"i], | ||||
| [href$=".ogg"i], | ||||
| [href$=".opus"i], | ||||
| [href$=".flac"i], | ||||
| [href$=".m4a"i], | ||||
| [href$=".aac"i], | ||||
| ):before { | ||||
|     content: '🎵'; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| /* image */ | ||||
| #ggrid>a:is( | ||||
| [href$=".jpg"i], | ||||
| [href$=".jpeg"i], | ||||
| [href$=".png"i], | ||||
| [href$=".gif"i], | ||||
| [href$=".webp"i], | ||||
| ):before { | ||||
|     content: '🎨'; | ||||
| } | ||||
							
								
								
									
										29
									
								
								docs/browser.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								docs/browser.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| html { | ||||
|     background: #333 url('/wp/wallhaven-mdjrqy.jpg') center / cover no-repeat fixed; | ||||
| } | ||||
| #files th { | ||||
|     background: rgba(32, 32, 32, 0.9) !important; | ||||
| } | ||||
| #ops, | ||||
| #treeul, | ||||
| #files td { | ||||
|     background: rgba(32, 32, 32, 0.3) !important; | ||||
| } | ||||
|  | ||||
|  | ||||
| html.light { | ||||
|     background: #eee url('/wp/wallhaven-dpxl6l.png') center / cover no-repeat fixed; | ||||
| } | ||||
| html.light #files th { | ||||
|     background: rgba(255, 255, 255, 0.9) !important; | ||||
| } | ||||
| html.light #ops, | ||||
| html.light #treeul, | ||||
| html.light #files td { | ||||
|     background: rgba(248, 248, 248, 0.8) !important; | ||||
| } | ||||
|  | ||||
|  | ||||
| #files * { | ||||
|     background: transparent !important; | ||||
| } | ||||
| @@ -10,19 +10,25 @@ u k:k | ||||
| # share "." (the current directory) | ||||
| # as "/" (the webroot) for the following users: | ||||
| # "r" grants read-access for anyone | ||||
| # "a ed" grants read-write to ed | ||||
| # "rw ed" grants read-write to ed | ||||
| . | ||||
| / | ||||
| r | ||||
| a ed | ||||
| rw ed | ||||
|  | ||||
| # custom permissions for the "priv" folder: | ||||
| # user "k" can see/read the contents | ||||
| # and "ed" gets read-write access | ||||
| # user "k" can only see/read the contents | ||||
| # user "ed" gets read-write access | ||||
| ./priv | ||||
| /priv | ||||
| r k | ||||
| a ed | ||||
| rw ed | ||||
|  | ||||
| # this does the same thing: | ||||
| ./priv | ||||
| /priv | ||||
| r ed k | ||||
| w ed | ||||
|  | ||||
| # share /home/ed/Music/ as /music and let anyone read it | ||||
| # (this will replace any folder called "music" in the webroot) | ||||
| @@ -32,10 +38,14 @@ r | ||||
|  | ||||
| # and a folder where anyone can upload | ||||
| # but nobody can see the contents | ||||
| # and set the e2d flag to enable the uploads database | ||||
| # and set the nodupe flag to reject duplicate uploads | ||||
| /home/ed/inc | ||||
| /dump | ||||
| w | ||||
| c e2d | ||||
| c nodupe | ||||
|  | ||||
| # this entire config file can be replaced with these arguments: | ||||
| # -u ed:123 -u k:k -v .::r:aed -v priv:priv:rk:aed -v /home/ed/Music:music:r -v /home/ed/inc:dump:w | ||||
| # -u ed:123 -u k:k -v .::r:a,ed -v priv:priv:r,k:rw,ed -v /home/ed/Music:music:r -v /home/ed/inc:dump:w:c,e2d:c,nodupe | ||||
| # but note that the config file always wins in case of conflicts | ||||
|   | ||||
							
								
								
									
										51
									
								
								docs/hls.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								docs/hls.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,51 @@ | ||||
| <!DOCTYPE html><html lang="en"><head> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>hls-test</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| </head><body> | ||||
|  | ||||
| <video id="vid" controls></video> | ||||
| <script src="hls.light.js"></script> | ||||
| <script> | ||||
|  | ||||
| var video = document.getElementById('vid'); | ||||
| var hls = new Hls({ | ||||
| 	debug: true, | ||||
| 	autoStartLoad: false | ||||
| }); | ||||
| hls.loadSource('live/v.m3u8'); | ||||
| hls.attachMedia(video); | ||||
| hls.on(Hls.Events.MANIFEST_PARSED, function() { | ||||
| 	hls.startLoad(0); | ||||
| }); | ||||
| hls.on(Hls.Events.MEDIA_ATTACHED, function() { | ||||
| 	video.muted = true; | ||||
| 	video.play(); | ||||
| }); | ||||
|  | ||||
| /* | ||||
| general good news: | ||||
| - doesn't need fixed-length segments; ok to let x264 pick optimal keyframes and slice on those | ||||
| - hls.js polls the m3u8 for new segments, scales the duration accordingly, seeking works great | ||||
| - the sfx will grow by 66 KiB since that's how small hls.js can get, wait thats not good | ||||
|  | ||||
| # vod, creates m3u8 at the end, fixed keyframes, v bad | ||||
| ffmpeg -hide_banner -threads 0 -flags -global_header -i ..\CowboyBebopMovie-OP1.webm -vf scale=1280:-4,format=yuv420p -ac 2 -c:a libopus -b:a 128k -c:v libx264 -preset slow -crf 24 -maxrate:v 5M -bufsize:v 10M -g 120 -keyint_min 120 -sc_threshold 0 -hls_time 4 -hls_playlist_type vod -hls_segment_filename v%05d.ts v.m3u8 | ||||
|  | ||||
| # live, updates m3u8 as it goes, dynamic keyframes, streamable with hls.js | ||||
| ffmpeg -hide_banner -threads 0 -flags -global_header -i ..\..\CowboyBebopMovie-OP1.webm -vf scale=1280:-4,format=yuv420p -ac 2 -c:a libopus -b:a 128k -c:v libx264 -preset slow -crf 24 -maxrate:v 5M -bufsize:v 10M -f segment -segment_list v.m3u8 -segment_format mpegts -segment_list_flags live v%05d.ts | ||||
|  | ||||
| # fmp4 (fragmented mp4), doesn't work with hls.js, gets duratoin 149:07:51 (536871s), probably the tkhd/mdhd 0xffffffff (timebase 8000? ok) | ||||
| ffmpeg -re -hide_banner -threads 0 -flags +cgop -i ..\..\CowboyBebopMovie-OP1.webm -vf scale=1280:-4,format=yuv420p -ac 2 -c:a libopus -b:a 128k -c:v libx264 -preset slow -crf 24 -maxrate:v 5M -bufsize:v 10M -f segment -segment_list v.m3u8 -segment_format fmp4 -segment_list_flags live v%05d.mp4 | ||||
|  | ||||
| # try 2, works, uses tempfiles for m3u8 updates, good, 6% smaller | ||||
| ffmpeg -re -hide_banner -threads 0 -flags +cgop -i ..\..\CowboyBebopMovie-OP1.webm -vf scale=1280:-4,format=yuv420p -ac 2 -c:a libopus -b:a 128k -c:v libx264 -preset slow -crf 24 -maxrate:v 5M -bufsize:v 10M -f hls -hls_segment_type fmp4 -hls_list_size 0 -hls_segment_filename v%05d.mp4 v.m3u8 | ||||
|  | ||||
| more notes | ||||
| - adding -hls_flags single_file makes duration wack during playback (for both fmp4 and ts), ok once finalized and refreshed, gives no size reduction anyways | ||||
| - bebop op has good keyframe spacing for testing hls.js, in particular it hops one seg back and immediately resumes if it hits eof with the explicit hls.startLoad(0); otherwise it jumps into the middle of a seg and becomes art | ||||
| - can probably -c:v copy most of the time, is there a way to check for cgop? todo | ||||
|  | ||||
| */ | ||||
| </script> | ||||
| </body></html> | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user