mirror of
				https://github.com/9001/copyparty.git
				synced 2025-10-24 16:43:55 +00:00 
			
		
		
		
	Compare commits
	
		
			1672 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | eeed2a840c | ||
|  | 4aaa111925 | ||
|  | e31248f018 | ||
|  | 8b4cf022f2 | ||
|  | 4e7455268a | ||
|  | 680f8ae814 | ||
|  | 90555a4cea | ||
|  | 56a62db591 | ||
|  | cf51997680 | ||
|  | f05cc18d61 | ||
|  | 5384c2e0f5 | ||
|  | 9bfbf80a0e | ||
|  | f874d7754f | ||
|  | a669f79480 | ||
|  | 1c3894743a | ||
|  | 75cdf17df4 | ||
|  | de7dd1e60a | ||
|  | 0ee574a718 | ||
|  | faac894706 | ||
|  | dac2fad48e | ||
|  | 77f624b01e | ||
|  | e24ffebfc8 | ||
|  | 70d07d1609 | ||
|  | bfb3303d87 | ||
|  | 660705a436 | ||
|  | 74a3f97671 | ||
|  | b3e35bb494 | ||
|  | 76adac7c72 | ||
|  | 5dc75ebb67 | ||
|  | d686ce12b6 | ||
|  | d3c40a423e | ||
|  | 2fb1e6dab8 | ||
|  | 10430b347f | ||
|  | e0e3f6ac3e | ||
|  | c694cbffdc | ||
|  | bdd0e5d771 | ||
|  | aa98e427f0 | ||
|  | daa6f4c94c | ||
|  | 4a76663fb2 | ||
|  | cebda5028a | ||
|  | 3fa377a580 | ||
|  | a11c1005a8 | ||
|  | 4a6aea9328 | ||
|  | 4ca041e93e | ||
|  | 52a866a405 | ||
|  | 8b6bd0e6ac | ||
|  | 780fc4639a | ||
|  | 3692fc9d83 | ||
|  | c2a0b1b4c6 | ||
|  | 21bbdb5419 | ||
|  | aa1c08962c | ||
|  | 8a5d0399dd | ||
|  | f2cd0b0c4a | ||
|  | c2b66bbe73 | ||
|  | 48b957f1d5 | ||
|  | 3683984c8d | ||
|  | a3431512d8 | ||
|  | d832b787e7 | ||
|  | 6f75b02723 | ||
|  | b8241710bd | ||
|  | d638404b6a | ||
|  | 9362ca3ed9 | ||
|  | d1a03c6d17 | ||
|  | c6c31702c2 | ||
|  | bd2d88c96e | ||
|  | 76b1857e4e | ||
|  | 095bd17d10 | ||
|  | 204bfac3fa | ||
|  | ac49b0ca93 | ||
|  | c5b04f6fef | ||
|  | 5c58fda46d | ||
|  | 062730c70c | ||
|  | cade1990ce | ||
|  | 59b6e61816 | ||
|  | daff7ff158 | ||
|  | 0862860961 | ||
|  | 1cb24045a0 | ||
|  | 622358b172 | ||
|  | 7998884a9d | ||
|  | 51ddecd101 | ||
|  | 7a35ab1d1e | ||
|  | 48564ba52a | ||
|  | 49efffd740 | ||
|  | d6ac224c8f | ||
|  | a772b8c3f2 | ||
|  | b580953dcd | ||
|  | d86653c763 | ||
|  | dded4fca76 | ||
|  | 36365ffa6b | ||
|  | 0f9aeeaa27 | ||
|  | d8ebcd0ef7 | ||
|  | 6e445487b1 | ||
|  | 6605e461c7 | ||
|  | 40ce4e2275 | ||
|  | 8fef9e363e | ||
|  | 4792c2770d | ||
|  | 87bb49da36 | ||
|  | 1c0071d9ce | ||
|  | efded35c2e | ||
|  | 1d74240b9a | ||
|  | 098184ff7b | ||
|  | 4083533916 | ||
|  | feb1acd43a | ||
|  | a9591db734 | ||
|  | 9ebf148cbe | ||
|  | a473e5e19a | ||
|  | 5d3034c231 | ||
|  | c3a895af64 | ||
|  | cea5aecbf2 | ||
|  | 0e61e70670 | ||
|  | 1e333c0939 | ||
|  | 917b6ec03c | ||
|  | fe67c52ead | ||
|  | 909c7bee3e | ||
|  | 27ca54d138 | ||
|  | 2147c3a646 | ||
|  | a99120116f | ||
|  | 802efeaff2 | ||
|  | 9ad3af1ef6 | ||
|  | 715727b811 | ||
|  | c6eaa7b836 | ||
|  | c2fceea2a5 | ||
|  | 190e11f7ea | ||
|  | ad7413a5ff | ||
|  | 903b9e627a | ||
|  | c5c1e96cf8 | ||
|  | 62fbb04c9d | ||
|  | 728dc62d0b | ||
|  | 2dfe1b1c6b | ||
|  | 35d4a1a6af | ||
|  | eb3fa5aa6b | ||
|  | 438384425a | ||
|  | 0b6f102436 | ||
|  | c9b7ec72d8 | ||
|  | 256c7f1789 | ||
|  | 4e5a323c62 | ||
|  | f4a3bbd237 | ||
|  | fe73f2d579 | ||
|  | f79fcc7073 | ||
|  | 4c4b3790c7 | ||
|  | bd60b464bb | ||
|  | 6bce852765 | ||
|  | 3b19a5a59d | ||
|  | f024583011 | ||
|  | 1111baacb2 | ||
|  | 1b9c913efb | ||
|  | 3524c36e1b | ||
|  | cf87cea9f8 | ||
|  | bfa34404b8 | ||
|  | 0aba5f35bf | ||
|  | 663bc0842a | ||
|  | 7d10c96e73 | ||
|  | 6b2720fab0 | ||
|  | e74ad5132a | ||
|  | 1f6f89c1fd | ||
|  | 4d55e60980 | ||
|  | ddaaccd5af | ||
|  | c20b7dac3d | ||
|  | 1f779d5094 | ||
|  | 715401ca8e | ||
|  | e7cd922d8b | ||
|  | 187feee0c1 | ||
|  | 49e962a7dc | ||
|  | 633ff601e5 | ||
|  | 331cf37054 | ||
|  | 23e4b9002f | ||
|  | c0de3c8053 | ||
|  | a82a3b084a | ||
|  | 67c298e66b | ||
|  | c110ccb9ae | ||
|  | 0143380306 | ||
|  | af9000d3c8 | ||
|  | 097d798e5e | ||
|  | 1d9f9f221a | ||
|  | 214a367f48 | ||
|  | 2fb46551a2 | ||
|  | 6bcf330ae0 | ||
|  | 2075a8b18c | ||
|  | 1275ac6c42 | ||
|  | 708f20b7af | ||
|  | a2c0c708e8 | ||
|  | 2f2c65d91e | ||
|  | cd5fcc7ca7 | ||
|  | aa29e7be48 | ||
|  | 93febe34b0 | ||
|  | f086e6d3c1 | ||
|  | 22e51e1c96 | ||
|  | 63a5336f31 | ||
|  | bfc6c53cc5 | ||
|  | 236017f310 | ||
|  | 0a1d9b4dfd | ||
|  | b50d090946 | ||
|  | 00b5db52cf | ||
|  | 24cb30e2c5 | ||
|  | 4549145ab5 | ||
|  | 67b0217754 | ||
|  | ccae9efdf0 | ||
|  | 59d596b222 | ||
|  | 4878eb2c45 | ||
|  | 7755392f57 | ||
|  | dc2ea20959 | ||
|  | 8eaea2bd17 | ||
|  | 58e559918f | ||
|  | f38a3fca5b | ||
|  | 1ea145b384 | ||
|  | 0d9567575a | ||
|  | e82f176289 | ||
|  | d4b51c040e | ||
|  | 125d0efbd8 | ||
|  | 3215afc504 | ||
|  | c73ff3ce1b | ||
|  | f9c159a051 | ||
|  | 2ab1325c90 | ||
|  | 5b0f7ff506 | ||
|  | 9269bc84f2 | ||
|  | 4e8b651e18 | ||
|  | 65b4f79534 | ||
|  | 5dd43dbc45 | ||
|  | 5f73074c7e | ||
|  | f5d6ba27b2 | ||
|  | 73fa70b41f | ||
|  | 2a1cda42e7 | ||
|  | 1bd7e31466 | ||
|  | eb49e1fb4a | ||
|  | 9838c2f0ce | ||
|  | 6041df8370 | ||
|  | 2933dce3ef | ||
|  | dab377d37b | ||
|  | f35e41baf1 | ||
|  | c4083a2942 | ||
|  | 36c20bbe53 | ||
|  | e34634f5af | ||
|  | cba9e5b669 | ||
|  | 1f3c46a6b0 | ||
|  | 799a5ffa47 | ||
|  | b000707c10 | ||
|  | feba4de1d6 | ||
|  | 951fdb27ca | ||
|  | 9697fb3d84 | ||
|  | 2dbed4500a | ||
|  | fd9d0e433d | ||
|  | f096f3ef81 | ||
|  | cc4a063695 | ||
|  | b64cabc3c9 | ||
|  | 3dd460717c | ||
|  | bf658a522b | ||
|  | e9be7e712d | ||
|  | e40cd2a809 | ||
|  | dbabeb9692 | ||
|  | 8dd37d76b0 | ||
|  | fd475aa358 | ||
|  | f0988c0e32 | ||
|  | 0632f09bff | ||
|  | ba599aaca0 | ||
|  | ff05919e89 | ||
|  | 52e63fa101 | ||
|  | 96ceccd12a | ||
|  | 87994fe006 | ||
|  | fa12c81a03 | ||
|  | 344ce63455 | ||
|  | ec4daacf9e | ||
|  | f3e8308718 | ||
|  | 515ac5d941 | ||
|  | 954c7e7e50 | ||
|  | 67ff57f3a3 | ||
|  | c10c70c1e5 | ||
|  | 04592a98d2 | ||
|  | c9c4aac6cf | ||
|  | 8b2c7586ce | ||
|  | 32e22dfe84 | ||
|  | d70b885722 | ||
|  | ac6c4b13f5 | ||
|  | ececdad22d | ||
|  | bf659781b0 | ||
|  | 2c6bb195a4 | ||
|  | c032cd08b3 | ||
|  | 39e7a7a231 | ||
|  | 6e14cd2c39 | ||
|  | aab3baaea7 | ||
|  | b8453c3b4f | ||
|  | 6ce0e2cd5b | ||
|  | 76beaae7f2 | ||
|  | c1a7f9edbe | ||
|  | b5f2fe2f0a | ||
|  | 98a90d49cb | ||
|  | f55e982cb5 | ||
|  | 686c7defeb | ||
|  | 0b1e483c53 | ||
|  | 457d7df129 | ||
|  | ce776a547c | ||
|  | ded0567cbf | ||
|  | c9cac83d09 | ||
|  | 4fbe6b01a8 | ||
|  | ee9585264e | ||
|  | c9ffead7bf | ||
|  | ed69d42005 | ||
|  | 0b47ee306b | ||
|  | e4e63619d4 | ||
|  | f32cca292a | ||
|  | e87ea19ff1 | ||
|  | 0214793740 | ||
|  | fc9dd5d743 | ||
|  | 9e6d5dd2b9 | ||
|  | bdad197e2c | ||
|  | 7e139288a6 | ||
|  | 6e7935abaf | ||
|  | 3ba0cc20f1 | ||
|  | dd28de1796 | ||
|  | 9eecc9e19a | ||
|  | 6530cb6b05 | ||
|  | 41ce613379 | ||
|  | 5e2785caba | ||
|  | d7cc000976 | ||
|  | 50d8ff95ae | ||
|  | b2de1459b6 | ||
|  | f0ffbea0b2 | ||
|  | 199ccca0fe | ||
|  | 1d9b355743 | ||
|  | f0437fbb07 | ||
|  | abc404a5b7 | ||
|  | 04b9e21330 | ||
|  | 1044aa071b | ||
|  | 4c3192c8cc | ||
|  | 689e77a025 | ||
|  | 3bd89403d2 | ||
|  | b4800d9bcb | ||
|  | 05485e8539 | ||
|  | 0e03dc0868 | ||
|  | 352b1ed10a | ||
|  | 0db1244d04 | ||
|  | ece08b8179 | ||
|  | b8945ae233 | ||
|  | dcaf7b0a20 | ||
|  | f982cdc178 | ||
|  | b265e59834 | ||
|  | 4a843a6624 | ||
|  | 241ef5b99d | ||
|  | f39f575a9c | ||
|  | 1521307f1e | ||
|  | dd122111e6 | ||
|  | 00c177fa74 | ||
|  | f6c7e49eb8 | ||
|  | 1a8dc3d18a | ||
|  | 38a163a09a | ||
|  | 8f031246d2 | ||
|  | 8f3d97dde7 | ||
|  | 4acaf24d65 | ||
|  | 9a8dbbbcf8 | ||
|  | a3efc4c726 | ||
|  | 0278bf328f | ||
|  | 17ddd96cc6 | ||
|  | 0e82e79aea | ||
|  | 30f124c061 | ||
|  | e19d90fcfc | ||
|  | 184bbdd23d | ||
|  | 30b50aec95 | ||
|  | c3c3d81db1 | ||
|  | 49b7231283 | ||
|  | edbedcdad3 | ||
|  | e4ae5f74e6 | ||
|  | 2c7ffe08d7 | ||
|  | 3ca46bae46 | ||
|  | 7e82aaf843 | ||
|  | 315bd71adf | ||
|  | 2c612c9aeb | ||
|  | 36aee085f7 | ||
|  | d01bb69a9c | ||
|  | c9b1c48c72 | ||
|  | aea3843cf2 | ||
|  | 131b6f4b9a | ||
|  | 6efb8b735a | ||
|  | 223b7af2ce | ||
|  | e72c2a6982 | ||
|  | dd9b93970e | ||
|  | e4c7cd81a9 | ||
|  | 12b3a62586 | ||
|  | 2da3bdcd47 | ||
|  | c1dccbe0ba | ||
|  | 9629fcde68 | ||
|  | cae436b566 | ||
|  | 01714700ae | ||
|  | 51e6c4852b | ||
|  | b206c5d64e | ||
|  | 62c3272351 | ||
|  | c5d822c70a | ||
|  | 9c09b4061a | ||
|  | c26fb43ced | ||
|  | deb8f20db6 | ||
|  | 50e18ed8ff | ||
|  | 31f3895f40 | ||
|  | 615929268a | ||
|  | b8b15814cf | ||
|  | 7766fffe83 | ||
|  | 2a16c150d1 | ||
|  | 418c2166cc | ||
|  | a4dd44f648 | ||
|  | 5352f7cda7 | ||
|  | 5533b47099 | ||
|  | e9b14464ee | ||
|  | 4e986e5cd1 | ||
|  | 8a59b40c53 | ||
|  | 391caca043 | ||
|  | 171ce348d6 | ||
|  | c2cc729135 | ||
|  | e7e71b76f0 | ||
|  | a2af61cf6f | ||
|  | e111edd5e4 | ||
|  | 3375377371 | ||
|  | 0ced020c67 | ||
|  | c0d7aa9e4a | ||
|  | e5b3d2a312 | ||
|  | 7b4a794981 | ||
|  | 86a859de17 | ||
|  | b3aaa7bd0f | ||
|  | a90586e6a8 | ||
|  | 807f272895 | ||
|  | f050647b43 | ||
|  | 73baebbd16 | ||
|  | f327f698b9 | ||
|  | 8164910fe8 | ||
|  | 3498644055 | ||
|  | d31116b54c | ||
|  | aced110cdf | ||
|  | e9ab6aec77 | ||
|  | 15b261c861 | ||
|  | 970badce66 | ||
|  | 64304a9d65 | ||
|  | d1983553d2 | ||
|  | 6b15df3bcd | ||
|  | 730b1fff71 | ||
|  | c3add751e5 | ||
|  | 9da2dbdc1c | ||
|  | 977f09c470 | ||
|  | 4d0c6a8802 | ||
|  | 5345565037 | ||
|  | be38c27c64 | ||
|  | 82a0401099 | ||
|  | 33bea1b663 | ||
|  | f083acd46d | ||
|  | 5aacd15272 | ||
|  | cb7674b091 | ||
|  | 3899c7ad56 | ||
|  | d2debced09 | ||
|  | b86c0ddc48 | ||
|  | ba36f33bd8 | ||
|  | 49368a10ba | ||
|  | ac1568cacf | ||
|  | 862ca3439d | ||
|  | fdd4f9f2aa | ||
|  | aa2dc49ebe | ||
|  | cc23b7ee74 | ||
|  | f6f9fc5a45 | ||
|  | 26c8589399 | ||
|  | c2469935cb | ||
|  | 5e7c20955e | ||
|  | 967fa38108 | ||
|  | 280fe8e36b | ||
|  | 03ca96ccc3 | ||
|  | b5b8a2c9d5 | ||
|  | 0008832730 | ||
|  | c9b385db4b | ||
|  | c951b66ae0 | ||
|  | de735f3a45 | ||
|  | 19161425f3 | ||
|  | c69e8d5bf4 | ||
|  | 3d3bce2788 | ||
|  | 1cb0dc7f8e | ||
|  | cd5c56e601 | ||
|  | 8c979905e4 | ||
|  | 4d69f15f48 | ||
|  | 083f6572f7 | ||
|  | 4e7dd75266 | ||
|  | 3eb83f449b | ||
|  | d31f69117b | ||
|  | f5f9e3ac97 | ||
|  | 598d6c598c | ||
|  | 744727087a | ||
|  | f93212a665 | ||
|  | 6dade82d2c | ||
|  | 6b737bf1d7 | ||
|  | 94dbd70677 | ||
|  | 527ae0348e | ||
|  | 79629c430a | ||
|  | 908dd61be5 | ||
|  | 88f77b8cca | ||
|  | 1e846657d1 | ||
|  | ce70f62a88 | ||
|  | bca0cdbb62 | ||
|  | 1ee11e04e6 | ||
|  | 6eef44f212 | ||
|  | 8bd94f4a1c | ||
|  | 4bc4701372 | ||
|  | dfd89b503a | ||
|  | 060dc54832 | ||
|  | f7a4ea5793 | ||
|  | 71b478e6e2 | ||
|  | ed8fff8c52 | ||
|  | 95dc78db10 | ||
|  | addeac64c7 | ||
|  | d77ec22007 | ||
|  | 20030c91b7 | ||
|  | 8b366e255c | ||
|  | 6da366fcb0 | ||
|  | 2fa35f851e | ||
|  | e4ca4260bb | ||
|  | b69aace8d8 | ||
|  | 79097bb43c | ||
|  | 806fac1742 | ||
|  | 4f97d7cf8d | ||
|  | 42acc457af | ||
|  | c02920607f | ||
|  | 452885c271 | ||
|  | 5c242a07b6 | ||
|  | 088899d59f | ||
|  | 1faff2a37e | ||
|  | 23c8d3d045 | ||
|  | a033388d2b | ||
|  | 82fe45ac56 | ||
|  | bcb7fcda6b | ||
|  | 726a98100b | ||
|  | 2f021a0c2b | ||
|  | eb05cb6c6e | ||
|  | 7530af95da | ||
|  | 8399e95bda | ||
|  | 3b4dfe326f | ||
|  | 2e787a254e | ||
|  | f888bed1a6 | ||
|  | d865e9f35a | ||
|  | fc7fe70f66 | ||
|  | 5aff39d2b2 | ||
|  | d1be37a04a | ||
|  | b0fd8bf7d4 | ||
|  | b9cf8f3973 | ||
|  | 4588f11613 | ||
|  | 1a618c3c97 | ||
|  | d500a51d97 | ||
|  | 734e9d3874 | ||
|  | bd5cfc2f1b | ||
|  | 89f88ee78c | ||
|  | b2ae14695a | ||
|  | 19d86b44d9 | ||
|  | 85be62e38b | ||
|  | 80f3d90200 | ||
|  | 0249fa6e75 | ||
|  | 2d0696e048 | ||
|  | ff32ec515e | ||
|  | a6935b0293 | ||
|  | 63eb08ba9f | ||
|  | e5b67d2b3a | ||
|  | 9e10af6885 | ||
|  | 42bc9115d2 | ||
|  | 0a569ce413 | ||
|  | 9a16639a61 | ||
|  | 57953c68c6 | ||
|  | 088d08963f | ||
|  | 7bc8196821 | ||
|  | 7715299dd3 | ||
|  | b8ac9b7994 | ||
|  | 98e7d8f728 | ||
|  | e7fd871ffe | ||
|  | 14aab62f32 | ||
|  | cb81fe962c | ||
|  | fc970d2dea | ||
|  | b0e203d1f9 | ||
|  | 37cef05b19 | ||
|  | 5886a42901 | ||
|  | 2fd99f807d | ||
|  | 3d4cbd7d10 | ||
|  | f10d03c238 | ||
|  | f9a66ffb0e | ||
|  | 777a50063d | ||
|  | 0bb9154747 | ||
|  | 30c3f45072 | ||
|  | 0d5ca67f32 | ||
|  | 4a8bf6aebd | ||
|  | b11db090d8 | ||
|  | 189391fccd | ||
|  | 86d4c43909 | ||
|  | 5994f40982 | ||
|  | 076d32dee5 | ||
|  | 16c8e38ecd | ||
|  | eacbcda8e5 | ||
|  | 59be76cd44 | ||
|  | 5bb0e7e8b3 | ||
|  | b78d207121 | ||
|  | 0fcbcdd08c | ||
|  | ed6c683922 | ||
|  | 9fe1edb02b | ||
|  | fb3811a708 | ||
|  | 18f8658eec | ||
|  | 3ead4676b0 | ||
|  | d30001d23d | ||
|  | 06bbf0d656 | ||
|  | 6ddd952e04 | ||
|  | 027ad0c3ee | ||
|  | 3abad2b87b | ||
|  | 32a1c7c5d5 | ||
|  | f06e165bd4 | ||
|  | 1c843b24f7 | ||
|  | 2ace9ed380 | ||
|  | 5f30c0ae03 | ||
|  | ef60adf7e2 | ||
|  | 7354b462e8 | ||
|  | da904d6be8 | ||
|  | c5fbbbbb5c | ||
|  | 5010387d8a | ||
|  | f00c54a7fb | ||
|  | 9f52c169d0 | ||
|  | bf18339404 | ||
|  | 2ad12b074b | ||
|  | a6788ffe8d | ||
|  | 0e884df486 | ||
|  | ef1c55286f | ||
|  | abc0424c26 | ||
|  | 44e5c82e6d | ||
|  | 5849c446ed | ||
|  | 12b7317831 | ||
|  | fe323f59af | ||
|  | a00e56f219 | ||
|  | 1a7852794f | ||
|  | 22b1373a57 | ||
|  | 17d78b1469 | ||
|  | 4d8b32b249 | ||
|  | b65bea2550 | ||
|  | 0b52ccd200 | ||
|  | 3006a07059 | ||
|  | 801dbc7a9a | ||
|  | 4f4e895fb7 | ||
|  | cc57c3b655 | ||
|  | 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 | 
							
								
								
									
										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 | ||||
| ``` | ||||
							
								
								
									
										16
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -5,12 +5,16 @@ __pycache__/ | ||||
| MANIFEST.in | ||||
| MANIFEST | ||||
| copyparty.egg-info/ | ||||
| buildenv/ | ||||
| build/ | ||||
| dist/ | ||||
| sfx/ | ||||
| .venv/ | ||||
|  | ||||
| /buildenv/ | ||||
| /build/ | ||||
| /dist/ | ||||
| /py2/ | ||||
| /sfx/ | ||||
| /unt/ | ||||
| /log/ | ||||
|  | ||||
| # ide | ||||
| *.sublime-workspace | ||||
|  | ||||
| @@ -20,3 +24,7 @@ sfx/ | ||||
| # derived | ||||
| copyparty/web/deps/ | ||||
| srv/ | ||||
|  | ||||
| # state/logs | ||||
| up.*.txt | ||||
| .hist/ | ||||
							
								
								
									
										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:rw,ed:c,dupe", | ||||
|                 "-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) | ||||
							
								
								
									
										38
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										38
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -23,7 +23,6 @@ | ||||
|         "terminal.ansiBrightWhite": "#ffffff", | ||||
|     }, | ||||
|     "python.testing.pytestEnabled": false, | ||||
|     "python.testing.nosetestsEnabled": false, | ||||
|     "python.testing.unittestEnabled": true, | ||||
|     "python.testing.unittestArgs": [ | ||||
|         "-v", | ||||
| @@ -35,26 +34,47 @@ | ||||
|     "python.linting.pylintEnabled": true, | ||||
|     "python.linting.flake8Enabled": true, | ||||
|     "python.linting.banditEnabled": true, | ||||
|     "python.linting.mypyEnabled": true, | ||||
|     "python.linting.mypyArgs": [ | ||||
|         "--ignore-missing-imports", | ||||
|         "--follow-imports=silent", | ||||
|         "--show-column-numbers", | ||||
|         "--strict" | ||||
|     ], | ||||
|     "python.linting.flake8Args": [ | ||||
|         "--max-line-length=120", | ||||
|         "--ignore=E722,F405,E203,W503,W293,E402", | ||||
|         "--ignore=E722,F405,E203,W503,W293,E402,E501,E128", | ||||
|     ], | ||||
|     "python.linting.banditArgs": [ | ||||
|         "--ignore=B104" | ||||
|     ], | ||||
|     "python.linting.pylintArgs": [ | ||||
|         "--disable=missing-module-docstring", | ||||
|         "--disable=missing-class-docstring", | ||||
|         "--disable=missing-function-docstring", | ||||
|         "--disable=wrong-import-position", | ||||
|         "--disable=raise-missing-from", | ||||
|         "--disable=bare-except", | ||||
|         "--disable=invalid-name", | ||||
|         "--disable=line-too-long", | ||||
|         "--disable=consider-using-f-string" | ||||
|     ], | ||||
|     // python3 -m isort --py=27 --profile=black copyparty/ | ||||
|     "python.formatting.provider": "black", | ||||
|     "editor.formatOnSave": true, | ||||
|     "[html]": { | ||||
|         "editor.formatOnSave": false, | ||||
|     }, | ||||
|     "[css]": { | ||||
|         "editor.formatOnSave": false, | ||||
|     }, | ||||
|     "files.associations": { | ||||
|         "*.makefile": "makefile" | ||||
|     }, | ||||
|     "editor.codeActionsOnSaveTimeout": 9001, | ||||
|     "editor.formatOnSaveTimeout": 9001, | ||||
|     // | ||||
|     //  things you may wanna edit: | ||||
|     // | ||||
|     "python.pythonPath": "/usr/bin/python3", | ||||
|     //"python.linting.enabled": true, | ||||
|     "python.formatting.blackArgs": [ | ||||
|         "-t", | ||||
|         "py27" | ||||
|     ], | ||||
|     "python.linting.enabled": true, | ||||
|     "python.pythonPath": "/usr/bin/python3" | ||||
| } | ||||
							
								
								
									
										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 | ||||
| @@ -1,4 +1,17 @@ | ||||
| # copyparty-fuse.py | ||||
| # [`up2k.py`](up2k.py) | ||||
| * command-line up2k client [(webm)](https://ocv.me/stuff/u2cli.webm) | ||||
| * file uploads, file-search, autoresume of aborted/broken uploads | ||||
| * faster than browsers | ||||
| * if something breaks just restart it | ||||
|  | ||||
|  | ||||
| # [`partyjournal.py`](partyjournal.py) | ||||
| produces a chronological list of all uploads by collecting info from up2k databases and the filesystem | ||||
| * outputs a standalone html file | ||||
| * optional mapping from IP-addresses to nicknames | ||||
|  | ||||
|  | ||||
| # [`copyparty-fuse.py`](copyparty-fuse.py) | ||||
| * 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 | ||||
| @@ -10,6 +23,8 @@ filecache is default-on for windows and macos; | ||||
|  | ||||
| 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/) | ||||
| @@ -27,8 +42,42 @@ you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releas | ||||
|  | ||||
|  | ||||
|  | ||||
| # copyparty-fuse🅱️.py | ||||
| # [`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=` | ||||
|   | ||||
							
								
								
									
										1101
									
								
								bin/copyparty-fuse-streaming.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1101
									
								
								bin/copyparty-fuse-streaming.py
									
									
									
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -12,13 +12,20 @@ __url__ = "https://github.com/9001/copyparty/" | ||||
| mount a copyparty server (local or remote) as a filesystem | ||||
|  | ||||
| usage: | ||||
|   python copyparty-fuse.py ./music http://192.168.1.69:3923/ | ||||
|   python copyparty-fuse.py http://192.168.1.69:3923/  ./music | ||||
|  | ||||
| dependencies: | ||||
|   python3 -m pip install --user fusepy | ||||
|   + on Linux: sudo apk add fuse | ||||
|   + on Macos: https://osxfuse.github.io/ | ||||
|   + on Windows: https://github.com/billziss-gh/winfsp/releases/latest | ||||
|  | ||||
| note: | ||||
|   you probably want to run this on windows clients: | ||||
|   https://github.com/9001/copyparty/blob/hovudstraum/contrib/explorer-nothumbs-nofoldertypes.reg | ||||
|  | ||||
| get server cert: | ||||
|   awk '/-BEGIN CERTIFICATE-/ {a=1} a; /-END CERTIFICATE-/{exit}' <(openssl s_client -connect 127.0.0.1:3923 </dev/null 2>/dev/null) >cert.pem | ||||
| """ | ||||
|  | ||||
|  | ||||
| @@ -26,24 +33,35 @@ import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import json | ||||
| import stat | ||||
| import errno | ||||
| import struct | ||||
| import codecs | ||||
| import builtins | ||||
| import platform | ||||
| import argparse | ||||
| import threading | ||||
| import traceback | ||||
| import http.client  # py2: httplib | ||||
| import urllib.parse | ||||
| import calendar | ||||
| from datetime import datetime | ||||
| from urllib.parse import quote_from_bytes as quote | ||||
|  | ||||
|  | ||||
| DEBUG = False  # ctrl-f this to configure logging | ||||
|  | ||||
| from urllib.parse import unquote_to_bytes as unquote | ||||
|  | ||||
| WINDOWS = sys.platform == "win32" | ||||
| MACOS = platform.system() == "Darwin" | ||||
| info = log = dbg = None | ||||
|  | ||||
|  | ||||
| print( | ||||
|     "{} v{} @ {}".format( | ||||
|         platform.python_implementation(), | ||||
|         ".".join([str(x) for x in sys.version_info]), | ||||
|         sys.executable, | ||||
|     ) | ||||
| ) | ||||
|  | ||||
|  | ||||
| try: | ||||
| @@ -54,7 +72,7 @@ except: | ||||
|     elif MACOS: | ||||
|         libfuse = "install https://osxfuse.github.io/" | ||||
|     else: | ||||
|         libfuse = "apt install libfuse\n    modprobe fuse" | ||||
|         libfuse = "apt install libfuse3-3\n    modprobe fuse" | ||||
|  | ||||
|     print( | ||||
|         "\n  could not import fuse; these may help:" | ||||
| @@ -97,13 +115,54 @@ def rice_tid(): | ||||
|  | ||||
|  | ||||
| def fancy_log(msg): | ||||
|     print("{} {}\n".format(rice_tid(), msg), end="") | ||||
|     print("{:10.6f} {} {}\n".format(time.time() % 900, rice_tid(), msg), end="") | ||||
|  | ||||
|  | ||||
| def null_log(msg): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| def hexler(binary): | ||||
|     return binary.replace("\r", "\\r").replace("\n", "\\n") | ||||
|     return " ".join(["{}\033[36m{:02x}\033[0m".format(b, ord(b)) for b in binary]) | ||||
|     return " ".join(map(lambda b: format(ord(b), "02x"), binary)) | ||||
|  | ||||
|  | ||||
| def register_wtf8(): | ||||
|     def wtf8_enc(text): | ||||
|         return str(text).encode("utf-8", "surrogateescape"), len(text) | ||||
|  | ||||
|     def wtf8_dec(binary): | ||||
|         return bytes(binary).decode("utf-8", "surrogateescape"), len(binary) | ||||
|  | ||||
|     def wtf8_search(encoding_name): | ||||
|         return codecs.CodecInfo(wtf8_enc, wtf8_dec, name="wtf-8") | ||||
|  | ||||
|     codecs.register(wtf8_search) | ||||
|  | ||||
|  | ||||
| bad_good = {} | ||||
| good_bad = {} | ||||
|  | ||||
|  | ||||
| def enwin(txt): | ||||
|     return "".join([bad_good.get(x, x) for x in txt]) | ||||
|  | ||||
|     for bad, good in bad_good.items(): | ||||
|         txt = txt.replace(bad, good) | ||||
|  | ||||
|     return txt | ||||
|  | ||||
|  | ||||
| def dewin(txt): | ||||
|     return "".join([good_bad.get(x, x) for x in txt]) | ||||
|  | ||||
|     for bad, good in bad_good.items(): | ||||
|         txt = txt.replace(good, bad) | ||||
|  | ||||
|     return txt | ||||
|  | ||||
|  | ||||
| class RecentLog(object): | ||||
|     def __init__(self): | ||||
|         self.mtx = threading.Lock() | ||||
| @@ -115,7 +174,7 @@ class RecentLog(object): | ||||
|         thr.start() | ||||
|  | ||||
|     def put(self, msg): | ||||
|         msg = "{} {}\n".format(rice_tid(), msg) | ||||
|         msg = "{:10.6f} {} {}\n".format(time.time() % 900, rice_tid(), msg) | ||||
|         if self.f: | ||||
|             fmsg = " ".join([datetime.utcnow().strftime("%H%M%S.%f"), str(msg)]) | ||||
|             self.f.write(fmsg.encode("utf-8")) | ||||
| @@ -138,22 +197,6 @@ class RecentLog(object): | ||||
|             print("".join(q), end="") | ||||
|  | ||||
|  | ||||
| if DEBUG: | ||||
|     # debug=on, | ||||
|     #   windows terminals are slow (cmd.exe, mintty) | ||||
|     #   otoh fancy_log beats RecentLog on linux | ||||
|     logger = RecentLog().put if WINDOWS else fancy_log | ||||
|  | ||||
|     info = logger | ||||
|     log = logger | ||||
|     dbg = logger | ||||
| else: | ||||
|     # debug=off, speed is dontcare | ||||
|     info = fancy_log | ||||
|     log = null_log | ||||
|     dbg = null_log | ||||
|  | ||||
|  | ||||
| # [windows/cmd/cpy3]  python dev\copyparty\bin\copyparty-fuse.py q: http://192.168.1.159:1234/ | ||||
| # [windows/cmd/msys2] C:\msys64\mingw64\bin\python3 dev\copyparty\bin\copyparty-fuse.py q: http://192.168.1.159:1234/ | ||||
| # [windows/mty/msys2] /mingw64/bin/python3 /c/Users/ed/dev/copyparty/bin/copyparty-fuse.py q: http://192.168.1.159:1234/ | ||||
| @@ -183,6 +226,8 @@ def html_dec(txt): | ||||
|         txt.replace("<", "<") | ||||
|         .replace(">", ">") | ||||
|         .replace(""", '"') | ||||
|         .replace("
", "\r") | ||||
|         .replace("
", "\n") | ||||
|         .replace("&", "&") | ||||
|     ) | ||||
|  | ||||
| @@ -195,10 +240,11 @@ class CacheNode(object): | ||||
|  | ||||
|  | ||||
| class Gateway(object): | ||||
|     def __init__(self, base_url): | ||||
|         self.base_url = base_url | ||||
|     def __init__(self, ar): | ||||
|         self.base_url = ar.base_url | ||||
|         self.password = ar.a | ||||
|  | ||||
|         ui = urllib.parse.urlparse(base_url) | ||||
|         ui = urllib.parse.urlparse(self.base_url) | ||||
|         self.web_root = ui.path.strip("/") | ||||
|         try: | ||||
|             self.web_host, self.web_port = ui.netloc.split(":") | ||||
| @@ -208,15 +254,25 @@ class Gateway(object): | ||||
|             if ui.scheme == "http": | ||||
|                 self.web_port = 80 | ||||
|             elif ui.scheme == "https": | ||||
|                 raise Exception("todo") | ||||
|                 self.web_port = 443 | ||||
|             else: | ||||
|                 raise Exception("bad url?") | ||||
|  | ||||
|         self.ssl_context = None | ||||
|         self.use_tls = ui.scheme.lower() == "https" | ||||
|         if self.use_tls: | ||||
|             import ssl | ||||
|  | ||||
|             if ar.td: | ||||
|                 self.ssl_context = ssl._create_unverified_context() | ||||
|             elif ar.te: | ||||
|                 self.ssl_context = ssl.SSLContext(ssl.PROTOCOL_TLS) | ||||
|                 self.ssl_context.load_verify_locations(ar.te) | ||||
|  | ||||
|         self.conns = {} | ||||
|  | ||||
|     def quotep(self, path): | ||||
|         # TODO: mojibake support | ||||
|         path = path.encode("utf-8", "ignore") | ||||
|         path = path.encode("wtf-8") | ||||
|         return quote(path, safe="/") | ||||
|  | ||||
|     def getconn(self, tid=None): | ||||
| @@ -226,7 +282,15 @@ class Gateway(object): | ||||
|         except: | ||||
|             info("new conn [{}] [{}]".format(self.web_host, self.web_port)) | ||||
|  | ||||
|             conn = http.client.HTTPConnection(self.web_host, self.web_port, timeout=260) | ||||
|             args = {} | ||||
|             if not self.use_tls: | ||||
|                 C = http.client.HTTPConnection | ||||
|             else: | ||||
|                 C = http.client.HTTPSConnection | ||||
|                 if self.ssl_context: | ||||
|                     args = {"context": self.ssl_context} | ||||
|  | ||||
|             conn = C(self.web_host, self.web_port, timeout=260, **args) | ||||
|  | ||||
|             self.conns[tid] = conn | ||||
|             return conn | ||||
| @@ -239,45 +303,76 @@ class Gateway(object): | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def sendreq(self, *args, **kwargs): | ||||
|     def sendreq(self, meth, path, headers, **kwargs): | ||||
|         tid = get_tid() | ||||
|         if self.password: | ||||
|             headers["Cookie"] = "=".join(["cppwd", self.password]) | ||||
|  | ||||
|         try: | ||||
|             c = self.getconn(tid) | ||||
|             c.request(*list(args), **kwargs) | ||||
|             c.request(meth, path, headers=headers, **kwargs) | ||||
|             return c.getresponse() | ||||
|         except: | ||||
|             self.closeconn(tid) | ||||
|             dbg("bad conn") | ||||
|  | ||||
|         self.closeconn(tid) | ||||
|         try: | ||||
|             c = self.getconn(tid) | ||||
|             c.request(*list(args), **kwargs) | ||||
|             c.request(meth, path, headers=headers, **kwargs) | ||||
|             return c.getresponse() | ||||
|         except: | ||||
|             info("http connection failed:\n" + traceback.format_exc()) | ||||
|             if self.use_tls and not self.ssl_context: | ||||
|                 import ssl | ||||
|  | ||||
|                 cert = ssl.get_server_certificate((self.web_host, self.web_port)) | ||||
|                 info("server certificate probably not trusted:\n" + cert) | ||||
|  | ||||
|             raise | ||||
|  | ||||
|     def listdir(self, path): | ||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots" | ||||
|         r = self.sendreq("GET", web_path) | ||||
|         if bad_good: | ||||
|             path = dewin(path) | ||||
|  | ||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots&ls" | ||||
|         r = self.sendreq("GET", web_path, {}) | ||||
|         if r.status != 200: | ||||
|             self.closeconn() | ||||
|             raise Exception( | ||||
|             log( | ||||
|                 "http error {} reading dir {} in {}".format( | ||||
|                     r.status, web_path, rice_tid() | ||||
|                 ) | ||||
|             ) | ||||
|             raise FuseOSError(errno.ENOENT) | ||||
|  | ||||
|         ctype = r.getheader("Content-Type", "") | ||||
|         if ctype == "application/json": | ||||
|             parser = self.parse_jls | ||||
|         elif ctype.startswith("text/html"): | ||||
|             parser = self.parse_html | ||||
|         else: | ||||
|             log("listdir on file: {}".format(path)) | ||||
|             raise FuseOSError(errno.ENOENT) | ||||
|  | ||||
|         try: | ||||
|             return self.parse_html(r) | ||||
|             return parser(r) | ||||
|         except: | ||||
|             traceback.print_exc() | ||||
|             info(repr(path) + "\n" + traceback.format_exc()) | ||||
|             raise | ||||
|  | ||||
|     def download_file_range(self, path, ofs1, ofs2): | ||||
|         if bad_good: | ||||
|             path = dewin(path) | ||||
|  | ||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?raw" | ||||
|         hdr_range = "bytes={}-{}".format(ofs1, ofs2 - 1) | ||||
|         info( | ||||
|             "DL {:4.0f}K\033[36m{:>9}-{:<9}\033[0m{}".format( | ||||
|                 (ofs2 - ofs1) / 1024.0, ofs1, ofs2 - 1, path | ||||
|                 (ofs2 - ofs1) / 1024.0, ofs1, ofs2 - 1, hexler(path) | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|         r = self.sendreq("GET", web_path, headers={"Range": hdr_range}) | ||||
|         r = self.sendreq("GET", web_path, {"Range": hdr_range}) | ||||
|         if r.status != http.client.PARTIAL_CONTENT: | ||||
|             self.closeconn() | ||||
|             raise Exception( | ||||
| @@ -288,11 +383,35 @@ class Gateway(object): | ||||
|  | ||||
|         return r.read() | ||||
|  | ||||
|     def parse_jls(self, datasrc): | ||||
|         rsp = b"" | ||||
|         while True: | ||||
|             buf = datasrc.read(1024 * 32) | ||||
|             if not buf: | ||||
|                 break | ||||
|  | ||||
|             rsp += buf | ||||
|  | ||||
|         rsp = json.loads(rsp.decode("utf-8")) | ||||
|         ret = [] | ||||
|         for statfun, nodes in [ | ||||
|             [self.stat_dir, rsp["dirs"]], | ||||
|             [self.stat_file, rsp["files"]], | ||||
|         ]: | ||||
|             for n in nodes: | ||||
|                 fname = unquote(n["href"].split("?")[0]).rstrip(b"/").decode("wtf-8") | ||||
|                 if bad_good: | ||||
|                     fname = enwin(fname) | ||||
|  | ||||
|                 ret.append([fname, statfun(n["ts"], n["sz"]), 0]) | ||||
|  | ||||
|         return ret | ||||
|  | ||||
|     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>$" | ||||
|             r'^<tr><td>(-|DIR|<a [^<]+</a>)</td><td><a[^>]* href="([^"]+)"[^>]*>([^<]+)</a></td><td>([^<]+)</td><td>[^<]+</td><td>([^<]+)</td></tr>$' | ||||
|         ) | ||||
|  | ||||
|         while True: | ||||
| @@ -314,18 +433,23 @@ class Gateway(object): | ||||
|                     # print(line) | ||||
|                     continue | ||||
|  | ||||
|                 ftype, fname, fsize, fdate = m.groups() | ||||
|                 fname = html_dec(fname) | ||||
|                 ftype, furl, fname, fsize, fdate = m.groups() | ||||
|                 fname = furl.rstrip("/").split("/")[-1] | ||||
|                 fname = unquote(fname) | ||||
|                 fname = fname.decode("wtf-8") | ||||
|                 if bad_good: | ||||
|                     fname = enwin(fname) | ||||
|  | ||||
|                 sz = 1 | ||||
|                 ts = 60 * 60 * 24 * 2 | ||||
|                 try: | ||||
|                     sz = int(fsize) | ||||
|                     ts = datetime.strptime(fdate, "%Y-%m-%d %H:%M:%S").timestamp() | ||||
|                     ts = calendar.timegm(time.strptime(fdate, "%Y-%m-%d %H:%M:%S")) | ||||
|                 except: | ||||
|                     info("bad HTML or OS [{}] [{}]".format(fdate, fsize)) | ||||
|                     # python cannot strptime(1959-01-01) on windows | ||||
|  | ||||
|                 if ftype == "-": | ||||
|                 if ftype != "DIR": | ||||
|                     ret.append([fname, self.stat_file(ts, sz), 0]) | ||||
|                 else: | ||||
|                     ret.append([fname, self.stat_dir(ts, sz), 0]) | ||||
| @@ -358,11 +482,11 @@ class Gateway(object): | ||||
|  | ||||
|  | ||||
| class CPPF(Operations): | ||||
|     def __init__(self, base_url, dircache, filecache): | ||||
|         self.gw = Gateway(base_url) | ||||
|     def __init__(self, ar): | ||||
|         self.gw = Gateway(ar) | ||||
|         self.junk_fh_ctr = 3 | ||||
|         self.n_dircache = dircache | ||||
|         self.n_filecache = filecache | ||||
|         self.n_dircache = ar.cd | ||||
|         self.n_filecache = ar.cf | ||||
|  | ||||
|         self.dircache = [] | ||||
|         self.dircache_mtx = threading.Lock() | ||||
| @@ -379,7 +503,11 @@ class CPPF(Operations): | ||||
|                 cache_path, cache1 = cn.tag | ||||
|                 cache2 = cache1 + len(cn.data) | ||||
|                 msg += "\n{:<2} {:>7} {:>10}:{:<9} {}".format( | ||||
|                     n, len(cn.data), cache1, cache2, cache_path | ||||
|                     n, | ||||
|                     len(cn.data), | ||||
|                     cache1, | ||||
|                     cache2, | ||||
|                     cache_path.replace("\r", "\\r").replace("\n", "\\n"), | ||||
|                 ) | ||||
|         return msg | ||||
|  | ||||
| @@ -574,8 +702,18 @@ class CPPF(Operations): | ||||
|  | ||||
|         else: | ||||
|             if get2 - get1 <= 1024 * 1024: | ||||
|                 h_ofs = get1 - 256 * 1024 | ||||
|                 h_end = get2 + 1024 * 1024 | ||||
|                 # unless the request is for the last n bytes of the file, | ||||
|                 # grow the start to cache some stuff around the range | ||||
|                 if get2 < file_sz - 1: | ||||
|                     h_ofs = get1 - 1024 * 256 | ||||
|                 else: | ||||
|                     h_ofs = get1 - 1024 * 32 | ||||
|  | ||||
|                 # likewise grow the end unless start is 0 | ||||
|                 if get1 > 0: | ||||
|                     h_end = get2 + 1024 * 1024 | ||||
|                 else: | ||||
|                     h_end = get2 + 1024 * 64 | ||||
|             else: | ||||
|                 # big enough, doesn't need pads | ||||
|                 h_ofs = get1 | ||||
| @@ -610,7 +748,7 @@ class CPPF(Operations): | ||||
|  | ||||
|     def _readdir(self, path, fh=None): | ||||
|         path = path.strip("/") | ||||
|         log("readdir [{}] [{}]".format(path, fh)) | ||||
|         log("readdir [{}] [{}]".format(hexler(path), fh)) | ||||
|  | ||||
|         ret = self.gw.listdir(path) | ||||
|         if not self.n_dircache: | ||||
| @@ -621,6 +759,7 @@ class CPPF(Operations): | ||||
|             self.dircache.append(cn) | ||||
|             self.clean_dircache() | ||||
|  | ||||
|         # import pprint; pprint.pprint(ret) | ||||
|         return ret | ||||
|  | ||||
|     def readdir(self, path, fh=None): | ||||
| @@ -637,7 +776,11 @@ class CPPF(Operations): | ||||
|         path = path.strip("/") | ||||
|         ofs2 = offset + length | ||||
|         file_sz = self.getattr(path)["st_size"] | ||||
|         log("read {} |{}| {}:{} max {}".format(path, length, offset, ofs2, file_sz)) | ||||
|         log( | ||||
|             "read {} |{}| {}:{} max {}".format( | ||||
|                 hexler(path), length, offset, ofs2, file_sz | ||||
|             ) | ||||
|         ) | ||||
|         if ofs2 > file_sz: | ||||
|             ofs2 = file_sz | ||||
|             log("truncate to |{}| :{}".format(ofs2 - offset, ofs2)) | ||||
| @@ -676,7 +819,9 @@ class CPPF(Operations): | ||||
|         return ret | ||||
|  | ||||
|     def getattr(self, path, fh=None): | ||||
|         log("getattr [{}]".format(path)) | ||||
|         log("getattr [{}]".format(hexler(path))) | ||||
|         if WINDOWS: | ||||
|             path = enwin(path)  # windows occasionally decodes f0xx to xx | ||||
|  | ||||
|         path = path.strip("/") | ||||
|         try: | ||||
| @@ -699,11 +844,24 @@ class CPPF(Operations): | ||||
|             dents = self._readdir(dirpath) | ||||
|  | ||||
|         for cache_name, cache_stat, _ in dents: | ||||
|             # if "qw" in cache_name and "qw" in fname: | ||||
|             #     info( | ||||
|             #         "cmp\n  [{}]\n  [{}]\n\n{}\n".format( | ||||
|             #             hexler(cache_name), | ||||
|             #             hexler(fname), | ||||
|             #             "\n".join(traceback.format_stack()[:-1]), | ||||
|             #         ) | ||||
|             #     ) | ||||
|  | ||||
|             if cache_name == fname: | ||||
|                 # dbg("=" + repr(cache_stat)) | ||||
|                 return cache_stat | ||||
|  | ||||
|         info("=ENOENT ({})".format(path)) | ||||
|         fun = info | ||||
|         if MACOS and path.split("/")[-1].startswith("._"): | ||||
|             fun = dbg | ||||
|  | ||||
|         fun("=ENOENT ({})".format(hexler(path))) | ||||
|         raise FuseOSError(errno.ENOENT) | ||||
|  | ||||
|     access = None | ||||
| @@ -773,24 +931,24 @@ class CPPF(Operations): | ||||
|                 raise FuseOSError(errno.ENOENT) | ||||
|  | ||||
|         def open(self, path, flags): | ||||
|             dbg("open [{}] [{}]".format(path, flags)) | ||||
|             dbg("open [{}] [{}]".format(hexler(path), flags)) | ||||
|             return self._open(path) | ||||
|  | ||||
|         def opendir(self, path): | ||||
|             dbg("opendir [{}]".format(path)) | ||||
|             dbg("opendir [{}]".format(hexler(path))) | ||||
|             return self._open(path) | ||||
|  | ||||
|         def flush(self, path, fh): | ||||
|             dbg("flush [{}] [{}]".format(path, fh)) | ||||
|             dbg("flush [{}] [{}]".format(hexler(path), fh)) | ||||
|  | ||||
|         def release(self, ino, fi): | ||||
|             dbg("release [{}] [{}]".format(ino, fi)) | ||||
|             dbg("release [{}] [{}]".format(hexler(ino), fi)) | ||||
|  | ||||
|         def releasedir(self, ino, fi): | ||||
|             dbg("releasedir [{}] [{}]".format(ino, fi)) | ||||
|             dbg("releasedir [{}] [{}]".format(hexler(ino), fi)) | ||||
|  | ||||
|         def access(self, path, mode): | ||||
|             dbg("access [{}] [{}]".format(path, mode)) | ||||
|             dbg("access [{}] [{}]".format(hexler(path), mode)) | ||||
|             try: | ||||
|                 x = self.getattr(path) | ||||
|                 if x["st_mode"] <= 0: | ||||
| @@ -799,41 +957,90 @@ class CPPF(Operations): | ||||
|                 raise FuseOSError(errno.ENOENT) | ||||
|  | ||||
|  | ||||
| class TheArgparseFormatter( | ||||
|     argparse.RawTextHelpFormatter, argparse.ArgumentDefaultsHelpFormatter | ||||
| ): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     global info, log, dbg | ||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||
|  | ||||
|     # filecache helps for reads that are ~64k or smaller; | ||||
|     #   linux generally does 128k so the cache is a slowdown, | ||||
|     #   windows likes to use 4k and 64k so cache is required, | ||||
|     #   value is numChunks (1~3M each) to keep in the cache | ||||
|     nf = 24 if WINDOWS or MACOS else 0 | ||||
|     nf = 24 | ||||
|  | ||||
|     # dircache is always a boost, | ||||
|     #   only want to disable it for tests etc, | ||||
|     #   value is numSec until an entry goes stale | ||||
|     nd = 1 | ||||
|  | ||||
|     try: | ||||
|         local, remote = sys.argv[1:3] | ||||
|         filecache = nf if len(sys.argv) <= 3 else int(sys.argv[3]) | ||||
|         dircache = nd if len(sys.argv) <= 4 else float(sys.argv[4]) | ||||
|     except: | ||||
|         where = "local directory" | ||||
|         if WINDOWS: | ||||
|             where += " or DRIVE:" | ||||
|     where = "local directory" | ||||
|     if WINDOWS: | ||||
|         where += " or DRIVE:" | ||||
|  | ||||
|         print("need arg 1: " + where) | ||||
|         print("need arg 2: root url") | ||||
|         print("optional 3: num files in filecache ({})".format(nf)) | ||||
|         print("optional 4: num seconds / dircache ({})".format(nd)) | ||||
|         print() | ||||
|         print("example:") | ||||
|         print("  copyparty-fuse.py ./music http://192.168.1.69:3923/music/") | ||||
|         if WINDOWS: | ||||
|             print("  copyparty-fuse.py M: http://192.168.1.69:3923/music/") | ||||
|     ex_pre = "\n  " + os.path.basename(__file__) + "  " | ||||
|     examples = ["http://192.168.1.69:3923/music/  ./music"] | ||||
|     if WINDOWS: | ||||
|         examples.append("http://192.168.1.69:3923/music/  M:") | ||||
|  | ||||
|         return | ||||
|     ap = argparse.ArgumentParser( | ||||
|         formatter_class=TheArgparseFormatter, | ||||
|         epilog="example:" + ex_pre + ex_pre.join(examples), | ||||
|     ) | ||||
|     ap.add_argument( | ||||
|         "-cd", metavar="NUM_SECONDS", type=float, default=nd, help="directory cache" | ||||
|     ) | ||||
|     ap.add_argument( | ||||
|         "-cf", metavar="NUM_BLOCKS", type=int, default=nf, help="file cache" | ||||
|     ) | ||||
|     ap.add_argument("-a", metavar="PASSWORD", help="password") | ||||
|     ap.add_argument("-d", action="store_true", help="enable debug") | ||||
|     ap.add_argument("-te", metavar="PEM_FILE", help="certificate to expect/verify") | ||||
|     ap.add_argument("-td", action="store_true", help="disable certificate check") | ||||
|     ap.add_argument("base_url", type=str, help="remote copyparty URL to mount") | ||||
|     ap.add_argument("local_path", type=str, help=where + " to mount it on") | ||||
|     ar = ap.parse_args() | ||||
|  | ||||
|     if ar.d: | ||||
|         # windows terminals are slow (cmd.exe, mintty) | ||||
|         # otoh fancy_log beats RecentLog on linux | ||||
|         logger = RecentLog().put if WINDOWS else fancy_log | ||||
|  | ||||
|         info = logger | ||||
|         log = logger | ||||
|         dbg = logger | ||||
|     else: | ||||
|         # debug=off, speed is dontcare | ||||
|         info = fancy_log | ||||
|         log = null_log | ||||
|         dbg = null_log | ||||
|  | ||||
|     if ar.a and ar.a.startswith("$"): | ||||
|         fn = ar.a[1:] | ||||
|         log("reading password from file [{}]".format(fn)) | ||||
|         with open(fn, "rb") as f: | ||||
|             ar.a = f.read().decode("utf-8").strip() | ||||
|  | ||||
|     if WINDOWS: | ||||
|         os.system("") | ||||
|         os.system("rem") | ||||
|  | ||||
|         for ch in '<>:"\\|?*': | ||||
|             # microsoft maps illegal characters to f0xx | ||||
|             # (e000 to f8ff is basic-plane private-use) | ||||
|             bad_good[ch] = chr(ord(ch) + 0xF000) | ||||
|  | ||||
|         for n in range(0, 0x100): | ||||
|             # map surrogateescape to another private-use area | ||||
|             bad_good[chr(n + 0xDC00)] = chr(n + 0xF100) | ||||
|  | ||||
|         for k, v in bad_good.items(): | ||||
|             good_bad[v] = k | ||||
|  | ||||
|     register_wtf8() | ||||
|  | ||||
|     try: | ||||
|         with open("/etc/fuse.conf", "rb") as f: | ||||
| @@ -845,7 +1052,7 @@ def main(): | ||||
|     if not MACOS: | ||||
|         args["nonempty"] = True | ||||
|  | ||||
|     FUSE(CPPF(remote, dircache, filecache), local, **args) | ||||
|     FUSE(CPPF(ar), ar.local_path, encoding="wtf-8", **args) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -11,14 +11,18 @@ import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import json | ||||
| import stat | ||||
| import errno | ||||
| import struct | ||||
| import codecs | ||||
| import platform | ||||
| import threading | ||||
| import http.client  # py2: httplib | ||||
| import urllib.parse | ||||
| from datetime import datetime | ||||
| from urllib.parse import quote_from_bytes as quote | ||||
| from urllib.parse import unquote_to_bytes as unquote | ||||
|  | ||||
| try: | ||||
|     import fuse | ||||
| @@ -38,7 +42,7 @@ except: | ||||
| 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 | ||||
|   python ./copyparty-fuseb.py -f -o allow_other,auto_unmount,nonempty,pw=wark,url=http://192.168.1.69:3923 /mnt/nas | ||||
|  | ||||
| dependencies: | ||||
|   sudo apk add fuse-dev python3-dev | ||||
| @@ -50,6 +54,10 @@ fork of copyparty-fuse.py based on fuse-python which | ||||
| """ | ||||
|  | ||||
|  | ||||
| WINDOWS = sys.platform == "win32" | ||||
| MACOS = platform.system() == "Darwin" | ||||
|  | ||||
|  | ||||
| def threadless_log(msg): | ||||
|     print(msg + "\n", end="") | ||||
|  | ||||
| @@ -93,6 +101,41 @@ def html_dec(txt): | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def register_wtf8(): | ||||
|     def wtf8_enc(text): | ||||
|         return str(text).encode("utf-8", "surrogateescape"), len(text) | ||||
|  | ||||
|     def wtf8_dec(binary): | ||||
|         return bytes(binary).decode("utf-8", "surrogateescape"), len(binary) | ||||
|  | ||||
|     def wtf8_search(encoding_name): | ||||
|         return codecs.CodecInfo(wtf8_enc, wtf8_dec, name="wtf-8") | ||||
|  | ||||
|     codecs.register(wtf8_search) | ||||
|  | ||||
|  | ||||
| bad_good = {} | ||||
| good_bad = {} | ||||
|  | ||||
|  | ||||
| def enwin(txt): | ||||
|     return "".join([bad_good.get(x, x) for x in txt]) | ||||
|  | ||||
|     for bad, good in bad_good.items(): | ||||
|         txt = txt.replace(bad, good) | ||||
|  | ||||
|     return txt | ||||
|  | ||||
|  | ||||
| def dewin(txt): | ||||
|     return "".join([good_bad.get(x, x) for x in txt]) | ||||
|  | ||||
|     for bad, good in bad_good.items(): | ||||
|         txt = txt.replace(good, bad) | ||||
|  | ||||
|     return txt | ||||
|  | ||||
|  | ||||
| class CacheNode(object): | ||||
|     def __init__(self, tag, data): | ||||
|         self.tag = tag | ||||
| @@ -115,8 +158,9 @@ class Stat(fuse.Stat): | ||||
|  | ||||
|  | ||||
| class Gateway(object): | ||||
|     def __init__(self, base_url): | ||||
|     def __init__(self, base_url, pw): | ||||
|         self.base_url = base_url | ||||
|         self.pw = pw | ||||
|  | ||||
|         ui = urllib.parse.urlparse(base_url) | ||||
|         self.web_root = ui.path.strip("/") | ||||
| @@ -135,8 +179,7 @@ class Gateway(object): | ||||
|         self.conns = {} | ||||
|  | ||||
|     def quotep(self, path): | ||||
|         # TODO: mojibake support | ||||
|         path = path.encode("utf-8", "ignore") | ||||
|         path = path.encode("wtf-8") | ||||
|         return quote(path, safe="/") | ||||
|  | ||||
|     def getconn(self, tid=None): | ||||
| @@ -159,20 +202,29 @@ class Gateway(object): | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def sendreq(self, *args, **kwargs): | ||||
|     def sendreq(self, *args, **ka): | ||||
|         tid = get_tid() | ||||
|         if self.pw: | ||||
|             ck = "cppwd=" + self.pw | ||||
|             try: | ||||
|                 ka["headers"]["Cookie"] = ck | ||||
|             except: | ||||
|                 ka["headers"] = {"Cookie": ck} | ||||
|         try: | ||||
|             c = self.getconn(tid) | ||||
|             c.request(*list(args), **kwargs) | ||||
|             c.request(*list(args), **ka) | ||||
|             return c.getresponse() | ||||
|         except: | ||||
|             self.closeconn(tid) | ||||
|             c = self.getconn(tid) | ||||
|             c.request(*list(args), **kwargs) | ||||
|             c.request(*list(args), **ka) | ||||
|             return c.getresponse() | ||||
|  | ||||
|     def listdir(self, path): | ||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots" | ||||
|         if bad_good: | ||||
|             path = dewin(path) | ||||
|  | ||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots&ls" | ||||
|         r = self.sendreq("GET", web_path) | ||||
|         if r.status != 200: | ||||
|             self.closeconn() | ||||
| @@ -182,9 +234,12 @@ class Gateway(object): | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|         return self.parse_html(r) | ||||
|         return self.parse_jls(r) | ||||
|  | ||||
|     def download_file_range(self, path, ofs1, ofs2): | ||||
|         if bad_good: | ||||
|             path = dewin(path) | ||||
|  | ||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?raw" | ||||
|         hdr_range = "bytes={}-{}".format(ofs1, ofs2 - 1) | ||||
|         log("downloading {}".format(hdr_range)) | ||||
| @@ -200,40 +255,27 @@ class Gateway(object): | ||||
|  | ||||
|         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>$" | ||||
|         ) | ||||
|  | ||||
|     def parse_jls(self, datasrc): | ||||
|         rsp = b"" | ||||
|         while True: | ||||
|             buf = remainder + datasrc.read(4096) | ||||
|             # print('[{}]'.format(buf.decode('utf-8'))) | ||||
|             buf = datasrc.read(1024 * 32) | ||||
|             if not buf: | ||||
|                 break | ||||
|  | ||||
|             remainder = b"" | ||||
|             endpos = buf.rfind(b"\n") | ||||
|             if endpos >= 0: | ||||
|                 remainder = buf[endpos + 1 :] | ||||
|                 buf = buf[:endpos] | ||||
|             rsp += buf | ||||
|  | ||||
|             lines = buf.decode("utf-8").split("\n") | ||||
|             for line in lines: | ||||
|                 m = ptn.match(line) | ||||
|                 if not m: | ||||
|                     # print(line) | ||||
|                     continue | ||||
|         rsp = json.loads(rsp.decode("utf-8")) | ||||
|         ret = [] | ||||
|         for statfun, nodes in [ | ||||
|             [self.stat_dir, rsp["dirs"]], | ||||
|             [self.stat_file, rsp["files"]], | ||||
|         ]: | ||||
|             for n in nodes: | ||||
|                 fname = unquote(n["href"].split("?")[0]).rstrip(b"/").decode("wtf-8") | ||||
|                 if bad_good: | ||||
|                     fname = enwin(fname) | ||||
|  | ||||
|                 ftype, fname, fsize, fdate = m.groups() | ||||
|                 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]) | ||||
|                 ret.append([fname, statfun(n["ts"], n["sz"]), 0]) | ||||
|  | ||||
|         return ret | ||||
|  | ||||
| @@ -262,6 +304,7 @@ class CPPF(Fuse): | ||||
|         Fuse.__init__(self, *args, **kwargs) | ||||
|  | ||||
|         self.url = None | ||||
|         self.pw = None | ||||
|  | ||||
|         self.dircache = [] | ||||
|         self.dircache_mtx = threading.Lock() | ||||
| @@ -271,7 +314,7 @@ class CPPF(Fuse): | ||||
|  | ||||
|     def init2(self): | ||||
|         # TODO figure out how python-fuse wanted this to go | ||||
|         self.gw = Gateway(self.url)  # .decode('utf-8')) | ||||
|         self.gw = Gateway(self.url, self.pw)  # .decode('utf-8')) | ||||
|         info("up") | ||||
|  | ||||
|     def clean_dircache(self): | ||||
| @@ -536,6 +579,8 @@ class CPPF(Fuse): | ||||
|  | ||||
|     def getattr(self, path): | ||||
|         log("getattr [{}]".format(path)) | ||||
|         if WINDOWS: | ||||
|             path = enwin(path)  # windows occasionally decodes f0xx to xx | ||||
|  | ||||
|         path = path.strip("/") | ||||
|         try: | ||||
| @@ -567,8 +612,26 @@ class CPPF(Fuse): | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||
|     register_wtf8() | ||||
|     if WINDOWS: | ||||
|         os.system("rem") | ||||
|  | ||||
|         for ch in '<>:"\\|?*': | ||||
|             # microsoft maps illegal characters to f0xx | ||||
|             # (e000 to f8ff is basic-plane private-use) | ||||
|             bad_good[ch] = chr(ord(ch) + 0xF000) | ||||
|  | ||||
|         for n in range(0, 0x100): | ||||
|             # map surrogateescape to another private-use area | ||||
|             bad_good[chr(n + 0xDC00)] = chr(n + 0xF100) | ||||
|  | ||||
|         for k, v in bad_good.items(): | ||||
|             good_bad[v] = k | ||||
|  | ||||
|     server = CPPF() | ||||
|     server.parser.add_option(mountopt="url", metavar="BASE_URL", default=None) | ||||
|     server.parser.add_option(mountopt="pw", metavar="PASSWORD", default=None) | ||||
|     server.parse(values=server, errex=1) | ||||
|     if not server.url or not str(server.url).startswith("http"): | ||||
|         print("\nerror:") | ||||
| @@ -576,7 +639,7 @@ def main(): | ||||
|         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" | ||||
|             "  ./copyparty-fuseb.py -f -o allow_other,auto_unmount,nonempty,pw=wark,url=http://192.168.1.69:3923 /mnt/nas" | ||||
|         ) | ||||
|         sys.exit(1) | ||||
|  | ||||
|   | ||||
							
								
								
									
										304
									
								
								bin/dbtool.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										304
									
								
								bin/dbtool.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,304 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import shutil | ||||
| import sqlite3 | ||||
| import argparse | ||||
|  | ||||
| DB_VER1 = 3 | ||||
| DB_VER2 = 5 | ||||
|  | ||||
| BY_PATH = None | ||||
| NC = None | ||||
|  | ||||
|  | ||||
| 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 | ||||
|  | ||||
|         if BY_PATH: | ||||
|             q = "select w from up where rd = ? and fn = ?" | ||||
|             hit = d2.execute(q, (rd, fn)).fetchone() | ||||
|         else: | ||||
|             q = "select w from up where substr(w,1,16) = ? and +w = ?" | ||||
|             hit = d2.execute(q, (w1[:16], w1)).fetchone() | ||||
|  | ||||
|         if not hit: | ||||
|             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 w1s, 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 w, rd, fn from up where substr(w,1,16) = ?" | ||||
|         w1, rd, fn = d1.execute(q, (w1s,)).fetchone() | ||||
|         if rd.split("/", 1)[0] == ".hist": | ||||
|             continue | ||||
|  | ||||
|         if BY_PATH: | ||||
|             q = "select w from up where rd = ? and fn = ?" | ||||
|             w2 = d2.execute(q, (rd, fn)).fetchone() | ||||
|         else: | ||||
|             q = "select w from up where substr(w,1,16) = ? and +w = ?" | ||||
|             w2 = d2.execute(q, (w1s, w1)).fetchone() | ||||
|  | ||||
|         if w2: | ||||
|             w2 = w2[0] | ||||
|  | ||||
|         v2 = None | ||||
|         if w2: | ||||
|             v2 = d2.execute( | ||||
|                 "select v from mt where w = ? and +k = ?", (w2[:16], 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:7} more {k:<7} 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 | ||||
|     ncopy = 0 | ||||
|     nskip = 0 | ||||
|     for w1s, 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 {ncopy} copied, {nskip} skipped\033[0m" | ||||
|             print(m) | ||||
|  | ||||
|         q = "select w, rd, fn from up where substr(w,1,16) = ?" | ||||
|         w1, rd, fn = d1.execute(q, (w1s,)).fetchone() | ||||
|         if rd.split("/", 1)[0] == ".hist": | ||||
|             continue | ||||
|  | ||||
|         if BY_PATH: | ||||
|             q = "select w from up where rd = ? and fn = ?" | ||||
|             w2 = d2.execute(q, (rd, fn)).fetchone() | ||||
|         else: | ||||
|             q = "select w from up where substr(w,1,16) = ? and +w = ?" | ||||
|             w2 = d2.execute(q, (w1s, w1)).fetchone() | ||||
|  | ||||
|         if not w2: | ||||
|             continue | ||||
|  | ||||
|         w2s = w2[0][:16] | ||||
|         hit = d2.execute("select v from mt where w = ? and +k = ?", (w2s, k)).fetchone() | ||||
|         if hit: | ||||
|             hit = hit[0] | ||||
|  | ||||
|         if hit != v: | ||||
|             if NC and hit is not None: | ||||
|                 nskip += 1 | ||||
|                 continue | ||||
|  | ||||
|             ncopy += 1 | ||||
|             if hit is not None: | ||||
|                 d2.execute("delete from mt where w = ? and +k = ?", (w2s, k)) | ||||
|  | ||||
|             d2.execute("insert into mt values (?,?,?)", (w2s, k, v)) | ||||
|             if rm: | ||||
|                 d2.execute("delete from mt where w = ? and +k = 't:mtp'", (w2s,)) | ||||
|  | ||||
|     d2.commit() | ||||
|     print(f"copied {ncopy} {tag} tags over, skipped {nskip}") | ||||
|  | ||||
|  | ||||
| def examples(): | ||||
|     print( | ||||
|         """ | ||||
| # clearing the journal | ||||
| ./dbtool.py up2k.db | ||||
|  | ||||
| # copy tags ".bpm" and "key" from old.db to up2k.db, and remove the mtp flag from matching files (so copyparty won't run any mtps on it) | ||||
| ./dbtool.py -ls up2k.db | ||||
| ./dbtool.py -src old.db up2k.db -cmp | ||||
| ./dbtool.py -src old.v3 up2k.db -rm-mtp-flag -copy key | ||||
| ./dbtool.py -src old.v3 up2k.db -rm-mtp-flag -copy .bpm -vac | ||||
|  | ||||
| """ | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     global NC, BY_PATH | ||||
|     os.system("") | ||||
|     print() | ||||
|  | ||||
|     ap = argparse.ArgumentParser() | ||||
|     ap.add_argument("db", help="database to work on") | ||||
|     ap.add_argument("-h2", action="store_true", help="show examples") | ||||
|     ap.add_argument("-src", metavar="DB", type=str, help="database to copy from") | ||||
|  | ||||
|     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 file as done, so copyparty won't run any mtps on those files", | ||||
|     ) | ||||
|     ap2.add_argument("-vac", action="store_true", help="optimize DB") | ||||
|  | ||||
|     ap2 = ap.add_argument_group("behavior modifiers") | ||||
|     ap2.add_argument( | ||||
|         "-nc", | ||||
|         action="store_true", | ||||
|         help="no-clobber; don't replace/overwrite existing tags", | ||||
|     ) | ||||
|     ap2.add_argument( | ||||
|         "-by-path", | ||||
|         action="store_true", | ||||
|         help="match files based on location rather than warks (content-hash), use this if the databases have different wark salts", | ||||
|     ) | ||||
|  | ||||
|     ar = ap.parse_args() | ||||
|     if ar.h2: | ||||
|         examples() | ||||
|         return | ||||
|  | ||||
|     NC = ar.nc | ||||
|     BY_PATH = ar.by_path | ||||
|  | ||||
|     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() | ||||
							
								
								
									
										53
									
								
								bin/mtag/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								bin/mtag/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | ||||
| standalone programs which take an audio file as argument | ||||
|  | ||||
| **NOTE:** these all require `-e2ts` to be functional, meaning you need to do at least one of these: `apt install ffmpeg` or `pip3 install mutagen` | ||||
|  | ||||
| some of these rely on libraries which are not MIT-compatible | ||||
|  | ||||
| * [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 | ||||
|  | ||||
| these invoke standalone programs which are GPL or similar, so is legally fine for most purposes: | ||||
|  | ||||
| * [media-hash.py](./media-hash.py) generates checksums for audio and video streams; uses FFmpeg (LGPL or GPL) | ||||
| * [image-noexif.py](./image-noexif.py) removes exif tags from images; uses exiftool (GPLv1 or artistic-license) | ||||
|  | ||||
| these do not have any problematic dependencies at all: | ||||
|  | ||||
| * [cksum.py](./cksum.py) computes various checksums | ||||
| * [exe.py](./exe.py) grabs metadata from .exe and .dll files (example for retrieving multiple tags with one parser) | ||||
| * [wget.py](./wget.py) lets you download files by POSTing URLs to copyparty | ||||
|  | ||||
|  | ||||
| # 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 volflags | ||||
|  | ||||
| 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` | ||||
							
								
								
									
										70
									
								
								bin/mtag/audio-bpm.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										70
									
								
								bin/mtag/audio-bpm.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,70 @@ | ||||
| #!/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([ | ||||
|         b"ffmpeg", | ||||
|         b"-nostdin", | ||||
|         b"-hide_banner", | ||||
|         b"-v", b"fatal", | ||||
|         b"-ss", b"13", | ||||
|         b"-y", b"-i", fsenc(sys.argv[1]), | ||||
|         b"-map", b"0:a:0", | ||||
|         b"-ac", b"1", | ||||
|         b"-ar", b"22050", | ||||
|         b"-t", b"300", | ||||
|         b"-f", b"f32le", | ||||
|         fsenc(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() | ||||
							
								
								
									
										55
									
								
								bin/mtag/audio-key.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										55
									
								
								bin/mtag/audio-key.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| #!/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([ | ||||
|         b"ffmpeg", | ||||
|         b"-nostdin", | ||||
|         b"-hide_banner", | ||||
|         b"-v", b"fatal", | ||||
|         b"-y", b"-i", fsenc(sys.argv[1]), | ||||
|         b"-map", b"0:a:0", | ||||
|         b"-t", b"300", | ||||
|         b"-sample_fmt", b"s16", | ||||
|         fsenc(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() | ||||
							
								
								
									
										89
									
								
								bin/mtag/cksum.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										89
									
								
								bin/mtag/cksum.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,89 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| import sys | ||||
| import json | ||||
| import zlib | ||||
| import struct | ||||
| import base64 | ||||
| import hashlib | ||||
|  | ||||
| try: | ||||
|     from copyparty.util import fsenc | ||||
| except: | ||||
|  | ||||
|     def fsenc(p): | ||||
|         return p | ||||
|  | ||||
|  | ||||
| """ | ||||
| calculates various checksums for uploads, | ||||
| usage: -mtp crc32,md5,sha1,sha256b=ad,bin/mtag/cksum.py | ||||
| """ | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     config = "crc32 md5 md5b sha1 sha1b sha256 sha256b sha512/240 sha512b/240" | ||||
|     # b suffix = base64 encoded | ||||
|     # slash = truncate to n bits | ||||
|  | ||||
|     known = { | ||||
|         "md5": hashlib.md5, | ||||
|         "sha1": hashlib.sha1, | ||||
|         "sha256": hashlib.sha256, | ||||
|         "sha512": hashlib.sha512, | ||||
|     } | ||||
|     config = config.split() | ||||
|     hashers = { | ||||
|         k: v() | ||||
|         for k, v in known.items() | ||||
|         if k in [x.split("/")[0].rstrip("b") for x in known] | ||||
|     } | ||||
|     crc32 = 0 if "crc32" in config else None | ||||
|  | ||||
|     with open(fsenc(sys.argv[1]), "rb", 512 * 1024) as f: | ||||
|         while True: | ||||
|             buf = f.read(64 * 1024) | ||||
|             if not buf: | ||||
|                 break | ||||
|  | ||||
|             for x in hashers.values(): | ||||
|                 x.update(buf) | ||||
|  | ||||
|             if crc32 is not None: | ||||
|                 crc32 = zlib.crc32(buf, crc32) | ||||
|  | ||||
|     ret = {} | ||||
|     for s in config: | ||||
|         alg = s.split("/")[0] | ||||
|         b64 = alg.endswith("b") | ||||
|         alg = alg.rstrip("b") | ||||
|         if alg in hashers: | ||||
|             v = hashers[alg].digest() | ||||
|         elif alg == "crc32": | ||||
|             v = crc32 | ||||
|             if v < 0: | ||||
|                 v &= 2 ** 32 - 1 | ||||
|             v = struct.pack(">L", v) | ||||
|         else: | ||||
|             raise Exception("what is {}".format(s)) | ||||
|  | ||||
|         if "/" in s: | ||||
|             v = v[: int(int(s.split("/")[1]) / 8)] | ||||
|  | ||||
|         if b64: | ||||
|             v = base64.b64encode(v).decode("ascii").rstrip("=") | ||||
|         else: | ||||
|             try: | ||||
|                 v = v.hex() | ||||
|             except: | ||||
|                 import binascii | ||||
|  | ||||
|                 v = binascii.hexlify(v) | ||||
|  | ||||
|         ret[s] = v | ||||
|  | ||||
|     print(json.dumps(ret, indent=4)) | ||||
|  | ||||
|  | ||||
| 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]) | ||||
							
								
								
									
										95
									
								
								bin/mtag/image-noexif.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								bin/mtag/image-noexif.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| """ | ||||
| remove exif tags from uploaded images | ||||
|  | ||||
| dependencies: | ||||
|   exiftool | ||||
|  | ||||
| about: | ||||
|   creates a "noexif" subfolder and puts exif-stripped copies of each image there, | ||||
|   the reason for the subfolder is to avoid issues with the up2k.db / deduplication: | ||||
|  | ||||
|   if the original image is modified in-place, then copyparty will keep the original | ||||
|   hash in up2k.db for a while (until the next volume rescan), so if the image is | ||||
|   reuploaded after a rescan then the upload will be renamed and kept as a dupe | ||||
|  | ||||
|   alternatively you could switch the logic around, making a copy of the original | ||||
|   image into a subfolder named "exif" and modify the original in-place, but then | ||||
|   up2k.db will be out of sync until the next rescan, so any additional uploads | ||||
|   of the same image will get symlinked (deduplicated) to the modified copy | ||||
|   instead of the original in "exif" | ||||
|  | ||||
|   or maybe delete the original image after processing, that would kinda work too | ||||
|  | ||||
| example copyparty config to use this: | ||||
|   -v/mnt/nas/pics:pics:rwmd,ed:c,e2ts,mte=+noexif:c,mtp=noexif=ejpg,ejpeg,ad,bin/mtag/image-noexif.py | ||||
|  | ||||
| explained: | ||||
|   for realpath /mnt/nas/pics (served at /pics) with read-write-modify-delete for ed, | ||||
|   enable file analysis on upload (e2ts), | ||||
|   append "noexif" to the list of known tags (mtp), | ||||
|   and use mtp plugin "bin/mtag/image-noexif.py" to provide that tag, | ||||
|   do this on all uploads with the file extension "jpg" or "jpeg", | ||||
|   ad = parse file regardless if FFmpeg thinks it is audio or not | ||||
|  | ||||
| PS: this requires e2ts to be functional, | ||||
|   meaning you need to do at least one of these: | ||||
|    * apt install ffmpeg | ||||
|    * pip3 install mutagen | ||||
|   and your python must have sqlite3 support compiled in | ||||
| """ | ||||
|  | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import filecmp | ||||
| import subprocess as sp | ||||
|  | ||||
| try: | ||||
|     from copyparty.util import fsenc | ||||
| except: | ||||
|  | ||||
|     def fsenc(p): | ||||
|         return p.encode("utf-8") | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     cwd, fn = os.path.split(sys.argv[1]) | ||||
|     if os.path.basename(cwd) == "noexif": | ||||
|         return | ||||
|  | ||||
|     os.chdir(cwd) | ||||
|     f1 = fsenc(fn) | ||||
|     f2 = os.path.join(b"noexif", f1) | ||||
|     cmd = [ | ||||
|         b"exiftool", | ||||
|         b"-exif:all=", | ||||
|         b"-iptc:all=", | ||||
|         b"-xmp:all=", | ||||
|         b"-P", | ||||
|         b"-o", | ||||
|         b"noexif/", | ||||
|         b"--", | ||||
|         f1, | ||||
|     ] | ||||
|     sp.check_output(cmd) | ||||
|     if not os.path.exists(f2): | ||||
|         print("failed") | ||||
|         return | ||||
|  | ||||
|     if filecmp.cmp(f1, f2, shallow=False): | ||||
|         print("clean") | ||||
|     else: | ||||
|         print("exif") | ||||
|  | ||||
|     # lastmod = os.path.getmtime(f1) | ||||
|     # times = (int(time.time()), int(lastmod)) | ||||
|     # os.utime(f2, times) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     try: | ||||
|         main() | ||||
|     except: | ||||
|         pass | ||||
							
								
								
									
										300
									
								
								bin/mtag/install-deps.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										300
									
								
								bin/mtag/install-deps.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,300 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
|  | ||||
| # install dependencies for audio-*.py | ||||
| # | ||||
| # linux/alpine: requires gcc g++ make cmake patchelf {python3,ffmpeg,fftw,libsndfile}-dev py3-{wheel,pip} py3-numpy{,-dev} | ||||
| # linux/debian: requires libav{codec,device,filter,format,resample,util}-dev {libfftw3,python3,libsndfile1}-dev python3-{numpy,pip} vamp-{plugin-sdk,examples} patchelf cmake | ||||
| # 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() { | ||||
| 	rm -rf g | ||||
| 	mkdir g | ||||
| 	cd g | ||||
| 	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 | | ||||
| 	head -n 1 | | ||||
| 	tr -d '\r' | tr '\n' '\0' | | ||||
| 	xargs -0 bash -c 'dl_files "$@"' _ | ||||
| 	mv * ../tgz | ||||
| 	cd .. | ||||
| } | ||||
|  | ||||
|  | ||||
| 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 | | ||||
| 	head -n 1 | | ||||
| 	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} | ||||
| 	 | ||||
| 	[ -e $HOME/pe/keyfinder ] && { | ||||
| 		echo found a keyfinder build in ~/pe, skipping | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	cd "$td" | ||||
| 	github_tarball https://api.github.com/repos/mixxxdj/libkeyfinder/releases/latest | ||||
| 	ls -al | ||||
|  | ||||
| 	tar -xf tgz | ||||
| 	rm tgz | ||||
| 	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 | ||||
|  | ||||
| 	cd "$td" | ||||
| 	echo '#include <vamp-sdk/Plugin.h>' | gcc -x c -c -o /dev/null - || [ -e ~/pe/vamp-sdk ] || { | ||||
| 		printf '\033[33mcould not find the vamp-sdk, building from source\033[0m\n' | ||||
| 		(dl_files yolo https://code.soundsoftware.ac.uk/attachments/download/2588/vamp-plugin-sdk-2.9.0.tar.gz) | ||||
| 		sha512sum -c <( | ||||
| 			echo "7ef7f837d19a08048b059e0da408373a7964ced452b290fae40b85d6d70ca9000bcfb3302cd0b4dc76cf2a848528456f78c1ce1ee0c402228d812bd347b6983b  -" | ||||
| 		) <vamp-plugin-sdk-2.9.0.tar.gz | ||||
| 		tar -xf vamp-plugin-sdk-2.9.0.tar.gz | ||||
| 		rm -- *.tar.gz | ||||
| 		ls -al | ||||
| 		cd vamp-plugin-sdk-* | ||||
| 		./configure --prefix=$HOME/pe/vamp-sdk | ||||
| 		make -j1 install | ||||
| 	} | ||||
|  | ||||
| 	cd "$td" | ||||
| 	have_beatroot || { | ||||
| 		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  | ||||
| 		rm -- *.tar.gz | ||||
| 		cd beatroot-vamp-v1.0 | ||||
| 		[ -e ~/pe/vamp-sdk ] && | ||||
| 			sed -ri 's`^(CFLAGS :=.*)`\1 -I'$HOME'/pe/vamp-sdk/include`' Makefile.linux | ||||
| 		make -f Makefile.linux -j4 LDFLAGS=-L$HOME/pe/vamp-sdk/lib | ||||
| 		# /home/ed/vamp /home/ed/.vamp /usr/local/lib/vamp | ||||
| 		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() { | ||||
| 	cd "$td" | ||||
| 	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.encode("utf-8") | ||||
|  | ||||
|  | ||||
| """ | ||||
| dep: ffmpeg | ||||
| """ | ||||
|  | ||||
|  | ||||
| def det(): | ||||
|     # fmt: off | ||||
|     cmd = [ | ||||
|         b"ffmpeg", | ||||
|         b"-nostdin", | ||||
|         b"-hide_banner", | ||||
|         b"-v", b"fatal", | ||||
|         b"-i", fsenc(sys.argv[1]), | ||||
|         b"-f", b"framemd5", | ||||
|         b"-" | ||||
|     ] | ||||
|     # 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() | ||||
							
								
								
									
										76
									
								
								bin/mtag/rclone-upload.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								bin/mtag/rclone-upload.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import json | ||||
| import os | ||||
| import subprocess as sp | ||||
| import sys | ||||
| import time | ||||
|  | ||||
| try: | ||||
|     from copyparty.util import fsenc | ||||
| except: | ||||
|  | ||||
|     def fsenc(p): | ||||
|         return p.encode("utf-8") | ||||
|  | ||||
|  | ||||
| _ = r""" | ||||
| first checks the tag "vidchk" which must be "ok" to continue, | ||||
| then uploads all files to some cloud storage (RCLONE_REMOTE) | ||||
| and DELETES THE ORIGINAL FILES if rclone returns 0 ("success") | ||||
|  | ||||
| deps: | ||||
|   rclone | ||||
|  | ||||
| usage: | ||||
|   -mtp x2=t43200,ay,p2,bin/mtag/rclone-upload.py | ||||
|  | ||||
| explained: | ||||
| t43200: timeout 12h | ||||
|     ay: only process files which contain audio (including video with audio) | ||||
|     p2: set priority 2 (after vidchk's suggested priority of 1), | ||||
|           so the output of vidchk will be passed in here | ||||
|  | ||||
| complete usage example as vflags along with vidchk: | ||||
|   -vsrv/vidchk:vidchk:r:rw,ed:c,e2dsa,e2ts,mtp=vidchk=t600,p,bin/mtag/vidchk.py:c,mtp=rupload=t43200,ay,p2,bin/mtag/rclone-upload.py:c,mte=+vidchk,rupload | ||||
|  | ||||
| setup: see https://rclone.org/drive/ | ||||
|  | ||||
| if you wanna use this script standalone / separately from copyparty, | ||||
| either set CONDITIONAL_UPLOAD False or provide the following stdin: | ||||
|   {"vidchk":"ok"} | ||||
| """ | ||||
|  | ||||
|  | ||||
| RCLONE_REMOTE = "notmybox" | ||||
| CONDITIONAL_UPLOAD = True | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     if CONDITIONAL_UPLOAD: | ||||
|         fp = sys.argv[1] | ||||
|         zb = sys.stdin.buffer.read() | ||||
|         zs = zb.decode("utf-8", "replace") | ||||
|         md = json.loads(zs) | ||||
|  | ||||
|         chk = md.get("vidchk", None) | ||||
|         if chk != "ok": | ||||
|             print(f"vidchk={chk}", file=sys.stderr) | ||||
|             sys.exit(1) | ||||
|  | ||||
|     dst = f"{RCLONE_REMOTE}:".encode("utf-8") | ||||
|     cmd = [b"rclone", b"copy", b"--", fsenc(fp), dst] | ||||
|  | ||||
|     t0 = time.time() | ||||
|     try: | ||||
|         sp.check_call(cmd) | ||||
|     except: | ||||
|         print("rclone failed", file=sys.stderr) | ||||
|         sys.exit(1) | ||||
|  | ||||
|     print(f"{time.time() - t0:.1f} sec") | ||||
|     os.unlink(fsenc(fp)) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										21
									
								
								bin/mtag/res/twitter-unmute.user.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								bin/mtag/res/twitter-unmute.user.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| // ==UserScript== | ||||
| // @name         twitter-unmute | ||||
| // @namespace    http://ocv.me/ | ||||
| // @version      0.1 | ||||
| // @description  memes | ||||
| // @author       ed <irc.rizon.net> | ||||
| // @match        https://twitter.com/* | ||||
| // @icon         https://www.google.com/s2/favicons?domain=twitter.com | ||||
| // @grant        GM_addStyle | ||||
| // ==/UserScript== | ||||
|  | ||||
| function grunnur() { | ||||
|     setInterval(function () { | ||||
|         //document.querySelector('div[aria-label="Unmute"]').click(); | ||||
|         document.querySelector('video').muted = false; | ||||
|     }, 200); | ||||
| } | ||||
|  | ||||
| var scr = document.createElement('script'); | ||||
| scr.textContent = '(' + grunnur.toString() + ')();'; | ||||
| (document.head || document.getElementsByTagName('head')[0]).appendChild(scr); | ||||
							
								
								
									
										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}") | ||||
							
								
								
									
										139
									
								
								bin/mtag/very-bad-idea.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										139
									
								
								bin/mtag/very-bad-idea.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,139 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| """ | ||||
| use copyparty as a chromecast replacement: | ||||
|   * post a URL and it will open in the default browser | ||||
|   * upload a file and it will open in the default application | ||||
|   * the `key` command simulates keyboard input | ||||
|   * the `x` command executes other xdotool commands | ||||
|   * the `c` command executes arbitrary unix commands | ||||
|  | ||||
| the android app makes it a breeze to post pics and links: | ||||
|   https://github.com/9001/party-up/releases | ||||
|   (iOS devices have to rely on the web-UI) | ||||
|  | ||||
| goes without saying, but this is HELLA DANGEROUS, | ||||
|   GIVES RCE TO ANYONE WHO HAVE UPLOAD PERMISSIONS | ||||
|  | ||||
| example copyparty config to use this: | ||||
|   --urlform save,get -v.::w:c,e2d,e2t,mte=+a1:c,mtp=a1=ad,kn,bin/mtag/very-bad-idea.py | ||||
|  | ||||
| recommended deps: | ||||
|   apt install xdotool libnotify-bin | ||||
|   https://github.com/9001/copyparty/blob/hovudstraum/contrib/plugins/meadup.js | ||||
|  | ||||
| and you probably want `twitter-unmute.user.js` from the res folder | ||||
|  | ||||
|  | ||||
| ----------------------------------------------------------------------- | ||||
| -- startup script: | ||||
| ----------------------------------------------------------------------- | ||||
|  | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
| # create qr code | ||||
| ip=$(ip r | awk '/^default/{print$(NF-2)}'); echo http://$ip:3923/ | qrencode -o - -s 4 >/dev/shm/cpp-qr.png | ||||
| /usr/bin/feh -x /dev/shm/cpp-qr.png & | ||||
|  | ||||
| # reposition and make topmost (with janky raspbian support) | ||||
| ( sleep 0.5 | ||||
| xdotool search --name cpp-qr.png windowactivate --sync windowmove 1780 0 | ||||
| wmctrl -r :ACTIVE: -b toggle,above || true | ||||
|  | ||||
| ps aux | grep -E 'sleep[ ]7\.27' || | ||||
| while true; do | ||||
|   w=$(xdotool getactivewindow) | ||||
|   xdotool search --name cpp-qr.png windowactivate windowraise windowfocus | ||||
|   xdotool windowactivate $w | ||||
|   xdotool windowfocus $w | ||||
|   sleep 7.27 || break | ||||
| done & | ||||
| xeyes  # distraction window to prevent ^w from closing the qr-code | ||||
| ) & | ||||
|  | ||||
| # bail if copyparty is already running | ||||
| ps aux | grep -E '[3] copy[p]arty' && exit 0 | ||||
|  | ||||
| # dumb chrome wrapper to allow autoplay | ||||
| cat >/usr/local/bin/chromium-browser <<'EOF' | ||||
| #!/bin/bash | ||||
| set -e | ||||
| /usr/bin/chromium-browser --autoplay-policy=no-user-gesture-required "$@" | ||||
| EOF | ||||
| chmod 755 /usr/local/bin/chromium-browser | ||||
|  | ||||
| # start the server  (note: replace `-v.::rw:` with `-v.::w:` to disallow retrieving uploaded stuff) | ||||
| cd ~/Downloads; python3 copyparty-sfx.py --urlform save,get -v.::rw:c,e2d,e2t,mte=+a1:c,mtp=a1=ad,kn,very-bad-idea.py | ||||
|  | ||||
| """ | ||||
|  | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import subprocess as sp | ||||
| from urllib.parse import unquote_to_bytes as unquote | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     fp = os.path.abspath(sys.argv[1]) | ||||
|     with open(fp, "rb") as f: | ||||
|         txt = f.read(4096) | ||||
|  | ||||
|     if txt.startswith(b"msg="): | ||||
|         open_post(txt) | ||||
|     else: | ||||
|         open_url(fp) | ||||
|  | ||||
|  | ||||
| def open_post(txt): | ||||
|     txt = unquote(txt.replace(b"+", b" ")).decode("utf-8")[4:] | ||||
|     try: | ||||
|         k, v = txt.split(" ", 1) | ||||
|     except: | ||||
|         open_url(txt) | ||||
|  | ||||
|     if k == "key": | ||||
|         sp.call(["xdotool", "key"] + v.split(" ")) | ||||
|     elif k == "x": | ||||
|         sp.call(["xdotool"] + v.split(" ")) | ||||
|     elif k == "c": | ||||
|         env = os.environ.copy() | ||||
|         while " " in v: | ||||
|             v1, v2 = v.split(" ", 1) | ||||
|             if "=" not in v1: | ||||
|                 break | ||||
|  | ||||
|             ek, ev = v1.split("=", 1) | ||||
|             env[ek] = ev | ||||
|             v = v2 | ||||
|  | ||||
|         sp.call(v.split(" "), env=env) | ||||
|     else: | ||||
|         open_url(txt) | ||||
|  | ||||
|  | ||||
| def open_url(txt): | ||||
|     ext = txt.rsplit(".")[-1].lower() | ||||
|     sp.call(["notify-send", "--", txt]) | ||||
|     if ext not in ["jpg", "jpeg", "png", "gif", "webp"]: | ||||
|         # sp.call(["wmctrl", "-c", ":ACTIVE:"])  # closes the active window correctly | ||||
|         sp.call(["killall", "vlc"]) | ||||
|         sp.call(["killall", "mpv"]) | ||||
|         sp.call(["killall", "feh"]) | ||||
|         time.sleep(0.5) | ||||
|         for _ in range(20): | ||||
|             sp.call(["xdotool", "key", "ctrl+w"])  # closes the open tab correctly | ||||
|     # else: | ||||
|     #    sp.call(["xdotool", "getactivewindow", "windowminimize"])  # minimizes the focused windo | ||||
|  | ||||
|     # close any error messages: | ||||
|     sp.call(["xdotool", "search", "--name", "Error", "windowclose"]) | ||||
|     # sp.call(["xdotool", "key", "ctrl+alt+d"])  # doesnt work at all | ||||
|     # sp.call(["xdotool", "keydown", "--delay", "100", "ctrl+alt+d"]) | ||||
|     # sp.call(["xdotool", "keyup", "ctrl+alt+d"]) | ||||
|     sp.call(["xdg-open", txt]) | ||||
|  | ||||
|  | ||||
| main() | ||||
							
								
								
									
										118
									
								
								bin/mtag/vidchk.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										118
									
								
								bin/mtag/vidchk.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,118 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| import json | ||||
| import re | ||||
| import sys | ||||
| import subprocess as sp | ||||
|  | ||||
| try: | ||||
|     from copyparty.util import fsenc | ||||
| except: | ||||
|  | ||||
|     def fsenc(p): | ||||
|         return p.encode("utf-8") | ||||
|  | ||||
|  | ||||
| _ = r""" | ||||
| inspects video files for errors and such | ||||
| plus stores a bunch of metadata to filename.ff.json | ||||
|  | ||||
| usage: | ||||
|   -mtp vidchk=t600,ay,p,bin/mtag/vidchk.py | ||||
|  | ||||
| explained: | ||||
| t600: timeout 10min | ||||
|   ay: only process files which contain audio (including video with audio) | ||||
|    p: set priority 1 (lowest priority after initial ffprobe/mutagen for base tags), | ||||
|        makes copyparty feed base tags into this script as json | ||||
|  | ||||
| if you wanna use this script standalone / separately from copyparty, | ||||
| provide the video resolution on stdin as json:  {"res":"1920x1080"} | ||||
| """ | ||||
|  | ||||
|  | ||||
| FAST = True  # parse entire file at container level | ||||
| # FAST = False  # fully decode audio and video streams | ||||
|  | ||||
|  | ||||
| # warnings to ignore | ||||
| harmless = re.compile("^Unsupported codec with id ") | ||||
|  | ||||
|  | ||||
| def wfilter(lines): | ||||
|     return [x for x in lines if not harmless.search(x)] | ||||
|  | ||||
|  | ||||
| def errchk(so, se, rc): | ||||
|     if rc: | ||||
|         err = (so + se).decode("utf-8", "replace").split("\n", 1) | ||||
|         err = wfilter(err) or err | ||||
|         return f"ERROR {rc}: {err[0]}" | ||||
|  | ||||
|     if se: | ||||
|         err = se.decode("utf-8", "replace").split("\n", 1) | ||||
|         err = wfilter(err) | ||||
|         if err: | ||||
|             return f"Warning: {err[0]}" | ||||
|  | ||||
|     return None | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     fp = sys.argv[1] | ||||
|     zb = sys.stdin.buffer.read() | ||||
|     zs = zb.decode("utf-8", "replace") | ||||
|     md = json.loads(zs) | ||||
|  | ||||
|     try: | ||||
|         w, h = [int(x) for x in md["res"].split("x")] | ||||
|         if not w + h: | ||||
|             raise Exception() | ||||
|     except: | ||||
|         return "could not determine resolution" | ||||
|  | ||||
|     # grab streams/format metadata + 2 seconds of frames at the start and end | ||||
|     zs = "ffprobe -hide_banner -v warning -of json -show_streams -show_format -show_packets -show_data_hash crc32 -read_intervals %+2,999999%+2" | ||||
|     cmd = zs.encode("ascii").split(b" ") + [fsenc(fp)] | ||||
|     p = sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE) | ||||
|     so, se = p.communicate() | ||||
|  | ||||
|     # spaces to tabs, drops filesize from 69k to 48k | ||||
|     so = b"\n".join( | ||||
|         [ | ||||
|             b"\t" * int((len(x) - len(x.lstrip())) / 4) + x.lstrip() | ||||
|             for x in (so or b"").split(b"\n") | ||||
|         ] | ||||
|     ) | ||||
|     with open(fsenc(f"{fp}.ff.json"), "wb") as f: | ||||
|         f.write(so) | ||||
|  | ||||
|     err = errchk(so, se, p.returncode) | ||||
|     if err: | ||||
|         return err | ||||
|  | ||||
|     if min(w, h) < 1080: | ||||
|         return "resolution too small" | ||||
|  | ||||
|     zs = ( | ||||
|         "ffmpeg -y -hide_banner -nostdin -v warning" | ||||
|         + " -err_detect +crccheck+bitstream+buffer+careful+compliant+aggressive+explode" | ||||
|         " -xerror -i" | ||||
|     ) | ||||
|  | ||||
|     cmd = zs.encode("ascii").split(b" ") + [fsenc(fp)] | ||||
|  | ||||
|     if FAST: | ||||
|         zs = "-c copy -f null -" | ||||
|     else: | ||||
|         zs = "-vcodec rawvideo -acodec pcm_s16le -f null -" | ||||
|  | ||||
|     cmd += zs.encode("ascii").split(b" ") | ||||
|  | ||||
|     p = sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE) | ||||
|     so, se = p.communicate() | ||||
|     return errchk(so, se, p.returncode) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     print(main() or "ok") | ||||
							
								
								
									
										85
									
								
								bin/mtag/wget.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								bin/mtag/wget.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| """ | ||||
| use copyparty as a file downloader by POSTing URLs as | ||||
| application/x-www-form-urlencoded (for example using the | ||||
| message/pager function on the website) | ||||
|  | ||||
| example copyparty config to use this: | ||||
|   --urlform save,get -vsrv/wget:wget:rwmd,ed:c,e2ts,mtp=title=ebin,t300,ad,bin/mtag/wget.py | ||||
|  | ||||
| explained: | ||||
|   for realpath srv/wget (served at /wget) with read-write-modify-delete for ed, | ||||
|   enable file analysis on upload (e2ts), | ||||
|   use mtp plugin "bin/mtag/wget.py" to provide metadata tag "title", | ||||
|   do this on all uploads with the file extension "bin", | ||||
|   t300 = 300 seconds timeout for each dwonload, | ||||
|   ad = parse file regardless if FFmpeg thinks it is audio or not | ||||
|  | ||||
| PS: this requires e2ts to be functional, | ||||
|   meaning you need to do at least one of these: | ||||
|    * apt install ffmpeg | ||||
|    * pip3 install mutagen | ||||
| """ | ||||
|  | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import subprocess as sp | ||||
| from urllib.parse import unquote_to_bytes as unquote | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     fp = os.path.abspath(sys.argv[1]) | ||||
|     fdir = os.path.dirname(fp) | ||||
|     fname = os.path.basename(fp) | ||||
|     if not fname.startswith("put-") or not fname.endswith(".bin"): | ||||
|         raise Exception("not a post file") | ||||
|  | ||||
|     buf = b"" | ||||
|     with open(fp, "rb") as f: | ||||
|         while True: | ||||
|             b = f.read(4096) | ||||
|             buf += b | ||||
|             if len(buf) > 4096: | ||||
|                 raise Exception("too big") | ||||
|  | ||||
|             if not b: | ||||
|                 break | ||||
|  | ||||
|     if not buf: | ||||
|         raise Exception("file is empty") | ||||
|  | ||||
|     buf = unquote(buf.replace(b"+", b" ")) | ||||
|     url = buf.decode("utf-8") | ||||
|  | ||||
|     if not url.startswith("msg="): | ||||
|         raise Exception("does not start with msg=") | ||||
|  | ||||
|     url = url[4:] | ||||
|     if "://" not in url: | ||||
|         url = "https://" + url | ||||
|  | ||||
|     os.chdir(fdir) | ||||
|  | ||||
|     name = url.split("?")[0].split("/")[-1] | ||||
|     tfn = "-- DOWNLOADING " + name | ||||
|     open(tfn, "wb").close() | ||||
|  | ||||
|     cmd = ["wget", "--trust-server-names", "--", url] | ||||
|  | ||||
|     try: | ||||
|         sp.check_call(cmd) | ||||
|  | ||||
|         # OPTIONAL: | ||||
|         #   on success, delete the .bin file which contains the URL | ||||
|         os.unlink(fp) | ||||
|     except: | ||||
|         open("-- FAILED TO DONWLOAD " + name, "wb").close() | ||||
|  | ||||
|     os.unlink(tfn) | ||||
|     print(url) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										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,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 | ||||
							
								
								
									
										177
									
								
								bin/partyjournal.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										177
									
								
								bin/partyjournal.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,177 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| """ | ||||
| partyjournal.py: chronological history of uploads | ||||
| 2021-12-31, v0.1, ed <irc.rizon.net>, MIT-Licensed | ||||
| https://github.com/9001/copyparty/blob/hovudstraum/bin/partyjournal.py | ||||
|  | ||||
| produces a chronological list of all uploads, | ||||
| by collecting info from up2k databases and the filesystem | ||||
|  | ||||
| specify subnet `192.168.1.*` with argument `.=192.168.1.`, | ||||
| affecting all successive mappings | ||||
|  | ||||
| usage: | ||||
|   ./partyjournal.py > partyjournal.html .=192.168.1. cart=125 steen=114 steen=131 sleepy=121 fscarlet=144 ed=101 ed=123 | ||||
|  | ||||
| """ | ||||
|  | ||||
| import sys | ||||
| import base64 | ||||
| import sqlite3 | ||||
| import argparse | ||||
| from datetime import datetime | ||||
| from urllib.parse import quote_from_bytes as quote | ||||
| from urllib.parse import unquote_to_bytes as unquote | ||||
|  | ||||
|  | ||||
| FS_ENCODING = sys.getfilesystemencoding() | ||||
|  | ||||
|  | ||||
| class APF(argparse.ArgumentDefaultsHelpFormatter, argparse.RawDescriptionHelpFormatter): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## snibbed from copyparty | ||||
|  | ||||
|  | ||||
| def s3dec(v): | ||||
|     if not v.startswith("//"): | ||||
|         return v | ||||
|  | ||||
|     v = base64.urlsafe_b64decode(v.encode("ascii")[2:]) | ||||
|     return v.decode(FS_ENCODING, "replace") | ||||
|  | ||||
|  | ||||
| def quotep(txt): | ||||
|     btxt = txt.encode("utf-8", "replace") | ||||
|     quot1 = quote(btxt, safe=b"/") | ||||
|     quot1 = quot1.encode("ascii") | ||||
|     quot2 = quot1.replace(b" ", b"+") | ||||
|     return quot2.decode("utf-8", "replace") | ||||
|  | ||||
|  | ||||
| def html_escape(s, quote=False, crlf=False): | ||||
|     """html.escape but also newlines""" | ||||
|     s = s.replace("&", "&").replace("<", "<").replace(">", ">") | ||||
|     if quote: | ||||
|         s = s.replace('"', """).replace("'", "'") | ||||
|     if crlf: | ||||
|         s = s.replace("\r", "
").replace("\n", "
") | ||||
|  | ||||
|     return s | ||||
|  | ||||
|  | ||||
| ## end snibs | ||||
| ## | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     ap = argparse.ArgumentParser(formatter_class=APF) | ||||
|     ap.add_argument("who", nargs="*") | ||||
|     ar = ap.parse_args() | ||||
|  | ||||
|     imap = {} | ||||
|     subnet = "" | ||||
|     for v in ar.who: | ||||
|         if "=" not in v: | ||||
|             raise Exception("bad who: " + v) | ||||
|  | ||||
|         k, v = v.split("=") | ||||
|         if k == ".": | ||||
|             subnet = v | ||||
|             continue | ||||
|  | ||||
|         imap["{}{}".format(subnet, v)] = k | ||||
|  | ||||
|     print(repr(imap), file=sys.stderr) | ||||
|  | ||||
|     print( | ||||
|         """\ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
| <head><meta charset="utf-8"><style> | ||||
|  | ||||
| html, body { | ||||
|     color: #ccc; | ||||
|     background: #222; | ||||
|     font-family: sans-serif; | ||||
| } | ||||
| a { | ||||
|     color: #fc5; | ||||
| } | ||||
| td, th { | ||||
|     padding: .2em .5em; | ||||
|     border: 1px solid #999; | ||||
|     border-width: 0 1px 1px 0; | ||||
|     white-space: nowrap; | ||||
| } | ||||
| td:nth-child(1), | ||||
| td:nth-child(2), | ||||
| td:nth-child(3) { | ||||
|     font-family: monospace, monospace; | ||||
|     text-align: right; | ||||
| } | ||||
| tr:first-child { | ||||
|     position: sticky; | ||||
|     top: -1px; | ||||
| } | ||||
| th { | ||||
|     background: #222; | ||||
|     text-align: left; | ||||
| } | ||||
|  | ||||
| </style></head><body><table><tr> | ||||
|     <th>wark</th> | ||||
|     <th>time</th> | ||||
|     <th>size</th> | ||||
|     <th>who</th> | ||||
|     <th>link</th> | ||||
| </tr>""" | ||||
|     ) | ||||
|  | ||||
|     db_path = ".hist/up2k.db" | ||||
|     conn = sqlite3.connect(db_path) | ||||
|     q = r"pragma table_info(up)" | ||||
|     inf = conn.execute(q).fetchall() | ||||
|     cols = [x[1] for x in inf] | ||||
|     print("<!-- " + str(cols) + " -->") | ||||
|     # ['w', 'mt', 'sz', 'rd', 'fn', 'ip', 'at'] | ||||
|  | ||||
|     q = r"select * from up order by case when at > 0 then at else mt end" | ||||
|     for w, mt, sz, rd, fn, ip, at in conn.execute(q): | ||||
|         link = "/".join([s3dec(x) for x in [rd, fn] if x]) | ||||
|         if fn.startswith("put-") and sz < 4096: | ||||
|             try: | ||||
|                 with open(link, "rb") as f: | ||||
|                     txt = f.read().decode("utf-8", "replace") | ||||
|             except: | ||||
|                 continue | ||||
|  | ||||
|             if txt.startswith("msg="): | ||||
|                 txt = txt.encode("utf-8", "replace") | ||||
|                 txt = unquote(txt.replace(b"+", b" ")) | ||||
|                 link = txt.decode("utf-8")[4:] | ||||
|  | ||||
|         sz = "{:,}".format(sz) | ||||
|         v = [ | ||||
|             w[:16], | ||||
|             datetime.utcfromtimestamp(at if at > 0 else mt).strftime( | ||||
|                 "%Y-%m-%d %H:%M:%S" | ||||
|             ), | ||||
|             sz, | ||||
|             imap.get(ip, ip), | ||||
|         ] | ||||
|  | ||||
|         row = "<tr>\n  " | ||||
|         row += "\n  ".join(["<td>{}</th>".format(x) for x in v]) | ||||
|         row += '\n  <td><a href="{}">{}</a></td>'.format(link, html_escape(link)) | ||||
|         row += "\n</tr>" | ||||
|         print(row) | ||||
|  | ||||
|     print("</table></body></html>") | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										128
									
								
								bin/prisonparty.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										128
									
								
								bin/prisonparty.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,128 @@ | ||||
| #!/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...]] -- python3 copyparty-sfx.py [...] | ||||
|  | ||||
| example: | ||||
|   ./prisonparty.sh /var/lib/copyparty-jail 1000 1000 /mnt/nas/music -- python3 copyparty-sfx.py -v /mnt/nas/music::rwmd | ||||
|  | ||||
| example for running straight from source (instead of using an sfx): | ||||
|   PYTHONPATH=$PWD ./prisonparty.sh /var/lib/copyparty-jail 1000 1000 /mnt/nas/music -- python3 -um copyparty -v /mnt/nas/music::rwmd | ||||
|  | ||||
| note that if you have python modules installed as --user (such as bpm/key detectors), | ||||
|   you should add /home/foo/.local as a VOLDIR | ||||
|  | ||||
| EOF | ||||
| 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="$(command -v "$pybin")" | ||||
| pyarg= | ||||
| while true; do | ||||
| 	v="$1" | ||||
| 	[ "${v:0:1}" = - ] || break | ||||
| 	pyarg="$pyarg $v" | ||||
| 	shift | ||||
| done | ||||
| cpp="$1"; shift | ||||
| [ -d "$cpp" ] && cppdir="$PWD" || { | ||||
| 	# sfx, not module | ||||
| 	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%/}" | ||||
|  | ||||
|  | ||||
| # bind-mount system directories and volumes | ||||
| printf '%s\n' "${sysdirs[@]}" "${vols[@]}" | sed -r 's`/$``' | LC_ALL=C sort | uniq | | ||||
| 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) | ||||
| 	# echo "v [$v] i1 [$i1] i2 [$i2]" | ||||
| 	[ $i1 = $i2 ] && continue | ||||
| 	 | ||||
| 	mkdir -p "$jail$v" | ||||
| 	mount --bind "$v" "$jail$v" | ||||
| done | ||||
|  | ||||
|  | ||||
| cln() { | ||||
| 	rv=$? | ||||
| 	# cleanup if not in use | ||||
| 	lsof "$jail" | grep -qF "$jail" && | ||||
| 		echo "chroot is in use, will not cleanup" || | ||||
| 	{ | ||||
| 		mount | grep -F " on $jail" | | ||||
| 		awk '{sub(/ type .*/,"");sub(/.* on /,"");print}' | | ||||
| 		LC_ALL=C sort -r  | tee /dev/stderr | tr '\n' '\0' | xargs -r0 umount | ||||
| 	} | ||||
| 	exit $rv | ||||
| } | ||||
| trap cln EXIT | ||||
|  | ||||
|  | ||||
| # create a tmp | ||||
| mkdir -p "$jail/tmp" | ||||
| chmod 777 "$jail/tmp" | ||||
|  | ||||
|  | ||||
| # run copyparty | ||||
| export HOME=$(getent passwd $uid | cut -d: -f6) | ||||
| export USER=$(getent passwd $uid | cut -d: -f1) | ||||
| export LOGNAME="$USER" | ||||
| #echo "pybin [$pybin]" | ||||
| #echo "pyarg [$pyarg]" | ||||
| #echo "cpp [$cpp]" | ||||
| chroot --userspec=$uid:$gid "$jail" "$pybin" $pyarg "$cpp" "$@" & | ||||
| p=$! | ||||
| trap 'kill $p' INT TERM | ||||
| wait | ||||
							
								
								
									
										838
									
								
								bin/up2k.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										838
									
								
								bin/up2k.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,838 @@ | ||||
| #!/usr/bin/env python3 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| """ | ||||
| up2k.py: upload to copyparty | ||||
| 2022-06-16, v0.15, ed <irc.rizon.net>, MIT-Licensed | ||||
| https://github.com/9001/copyparty/blob/hovudstraum/bin/up2k.py | ||||
|  | ||||
| - dependencies: requests | ||||
| - supports python 2.6, 2.7, and 3.3 through 3.11 | ||||
|  | ||||
| - almost zero error-handling | ||||
| - but if something breaks just try again and it'll autoresume | ||||
| """ | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import stat | ||||
| import math | ||||
| import time | ||||
| import atexit | ||||
| import signal | ||||
| import base64 | ||||
| import hashlib | ||||
| import argparse | ||||
| import platform | ||||
| import threading | ||||
| import datetime | ||||
|  | ||||
| import requests | ||||
|  | ||||
|  | ||||
| # from copyparty/__init__.py | ||||
| PY2 = sys.version_info[0] == 2 | ||||
| if PY2: | ||||
|     from Queue import Queue | ||||
|     from urllib import unquote | ||||
|     from urllib import quote | ||||
|  | ||||
|     sys.dont_write_bytecode = True | ||||
|     bytes = str | ||||
| else: | ||||
|     from queue import Queue | ||||
|     from urllib.parse import unquote_to_bytes as unquote | ||||
|     from urllib.parse import quote_from_bytes as quote | ||||
|  | ||||
|     unicode = str | ||||
|  | ||||
| VT100 = platform.system() != "Windows" | ||||
|  | ||||
|  | ||||
| req_ses = requests.Session() | ||||
|  | ||||
|  | ||||
| class File(object): | ||||
|     """an up2k upload task; represents a single file""" | ||||
|  | ||||
|     def __init__(self, top, rel, size, lmod): | ||||
|         self.top = top  # type: bytes | ||||
|         self.rel = rel.replace(b"\\", b"/")  # type: bytes | ||||
|         self.size = size  # type: int | ||||
|         self.lmod = lmod  # type: float | ||||
|  | ||||
|         self.abs = os.path.join(top, rel)  # type: bytes | ||||
|         self.name = self.rel.split(b"/")[-1].decode("utf-8", "replace")  # type: str | ||||
|  | ||||
|         # set by get_hashlist | ||||
|         self.cids = []  # type: list[tuple[str, int, int]]  # [ hash, ofs, sz ] | ||||
|         self.kchunks = {}  # type: dict[str, tuple[int, int]]  # hash: [ ofs, sz ] | ||||
|  | ||||
|         # set by handshake | ||||
|         self.ucids = []  # type: list[str]  # chunks which need to be uploaded | ||||
|         self.wark = None  # type: str | ||||
|         self.url = None  # type: str | ||||
|  | ||||
|         # set by upload | ||||
|         self.up_b = 0  # type: int | ||||
|         self.up_c = 0  # type: int | ||||
|  | ||||
|         # t = "size({}) lmod({}) top({}) rel({}) abs({}) name({})\n" | ||||
|         # eprint(t.format(self.size, self.lmod, self.top, self.rel, self.abs, self.name)) | ||||
|  | ||||
|  | ||||
| class FileSlice(object): | ||||
|     """file-like object providing a fixed window into a file""" | ||||
|  | ||||
|     def __init__(self, file, cid): | ||||
|         # type: (File, str) -> None | ||||
|  | ||||
|         self.car, self.len = file.kchunks[cid] | ||||
|         self.cdr = self.car + self.len | ||||
|         self.ofs = 0  # type: int | ||||
|         self.f = open(file.abs, "rb", 512 * 1024) | ||||
|         self.f.seek(self.car) | ||||
|  | ||||
|         # https://stackoverflow.com/questions/4359495/what-is-exactly-a-file-like-object-in-python | ||||
|         # IOBase, RawIOBase, BufferedIOBase | ||||
|         funs = "close closed __enter__ __exit__ __iter__ isatty __next__ readable seekable writable" | ||||
|         try: | ||||
|             for fun in funs.split(): | ||||
|                 setattr(self, fun, getattr(self.f, fun)) | ||||
|         except: | ||||
|             pass  # py27 probably | ||||
|  | ||||
|     def tell(self): | ||||
|         return self.ofs | ||||
|  | ||||
|     def seek(self, ofs, wh=0): | ||||
|         if wh == 1: | ||||
|             ofs = self.ofs + ofs | ||||
|         elif wh == 2: | ||||
|             ofs = self.len + ofs  # provided ofs is negative | ||||
|  | ||||
|         if ofs < 0: | ||||
|             ofs = 0 | ||||
|         elif ofs >= self.len: | ||||
|             ofs = self.len - 1 | ||||
|  | ||||
|         self.ofs = ofs | ||||
|         self.f.seek(self.car + ofs) | ||||
|  | ||||
|     def read(self, sz): | ||||
|         sz = min(sz, self.len - self.ofs) | ||||
|         ret = self.f.read(sz) | ||||
|         self.ofs += len(ret) | ||||
|         return ret | ||||
|  | ||||
|  | ||||
| _print = print | ||||
|  | ||||
|  | ||||
| def eprint(*a, **ka): | ||||
|     ka["file"] = sys.stderr | ||||
|     ka["end"] = "" | ||||
|     if not PY2: | ||||
|         ka["flush"] = True | ||||
|  | ||||
|     _print(*a, **ka) | ||||
|     if PY2 or not VT100: | ||||
|         sys.stderr.flush() | ||||
|  | ||||
|  | ||||
| def flushing_print(*a, **ka): | ||||
|     _print(*a, **ka) | ||||
|     if "flush" not in ka: | ||||
|         sys.stdout.flush() | ||||
|  | ||||
|  | ||||
| if not VT100: | ||||
|     print = flushing_print | ||||
|  | ||||
|  | ||||
| def termsize(): | ||||
|     env = os.environ | ||||
|  | ||||
|     def ioctl_GWINSZ(fd): | ||||
|         try: | ||||
|             import fcntl, termios, struct | ||||
|  | ||||
|             cr = struct.unpack("hh", fcntl.ioctl(fd, termios.TIOCGWINSZ, "1234")) | ||||
|         except: | ||||
|             return | ||||
|         return cr | ||||
|  | ||||
|     cr = ioctl_GWINSZ(0) or ioctl_GWINSZ(1) or ioctl_GWINSZ(2) | ||||
|     if not cr: | ||||
|         try: | ||||
|             fd = os.open(os.ctermid(), os.O_RDONLY) | ||||
|             cr = ioctl_GWINSZ(fd) | ||||
|             os.close(fd) | ||||
|         except: | ||||
|             pass | ||||
|     if not cr: | ||||
|         try: | ||||
|             cr = (env["LINES"], env["COLUMNS"]) | ||||
|         except: | ||||
|             cr = (25, 80) | ||||
|     return int(cr[1]), int(cr[0]) | ||||
|  | ||||
|  | ||||
| class CTermsize(object): | ||||
|     def __init__(self): | ||||
|         self.ev = False | ||||
|         self.margin = None | ||||
|         self.g = None | ||||
|         self.w, self.h = termsize() | ||||
|  | ||||
|         try: | ||||
|             signal.signal(signal.SIGWINCH, self.ev_sig) | ||||
|         except: | ||||
|             return | ||||
|  | ||||
|         thr = threading.Thread(target=self.worker) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|     def worker(self): | ||||
|         while True: | ||||
|             time.sleep(0.5) | ||||
|             if not self.ev: | ||||
|                 continue | ||||
|  | ||||
|             self.ev = False | ||||
|             self.w, self.h = termsize() | ||||
|  | ||||
|             if self.margin is not None: | ||||
|                 self.scroll_region(self.margin) | ||||
|  | ||||
|     def ev_sig(self, *a, **ka): | ||||
|         self.ev = True | ||||
|  | ||||
|     def scroll_region(self, margin): | ||||
|         self.margin = margin | ||||
|         if margin is None: | ||||
|             self.g = None | ||||
|             eprint("\033[s\033[r\033[u") | ||||
|         else: | ||||
|             self.g = 1 + self.h - margin | ||||
|             t = "{0}\033[{1}A".format("\n" * margin, margin) | ||||
|             eprint("{0}\033[s\033[1;{1}r\033[u".format(t, self.g - 1)) | ||||
|  | ||||
|  | ||||
| ss = CTermsize() | ||||
|  | ||||
|  | ||||
| def _scd(err, top): | ||||
|     """non-recursive listing of directory contents, along with stat() info""" | ||||
|     with os.scandir(top) as dh: | ||||
|         for fh in dh: | ||||
|             abspath = os.path.join(top, fh.name) | ||||
|             try: | ||||
|                 yield [abspath, fh.stat()] | ||||
|             except: | ||||
|                 err.append(abspath) | ||||
|  | ||||
|  | ||||
| def _lsd(err, top): | ||||
|     """non-recursive listing of directory contents, along with stat() info""" | ||||
|     for name in os.listdir(top): | ||||
|         abspath = os.path.join(top, name) | ||||
|         try: | ||||
|             yield [abspath, os.stat(abspath)] | ||||
|         except: | ||||
|             err.append(abspath) | ||||
|  | ||||
|  | ||||
| if hasattr(os, "scandir"): | ||||
|     statdir = _scd | ||||
| else: | ||||
|     statdir = _lsd | ||||
|  | ||||
|  | ||||
| def walkdir(err, top): | ||||
|     """recursive statdir""" | ||||
|     for ap, inf in sorted(statdir(err, top)): | ||||
|         if stat.S_ISDIR(inf.st_mode): | ||||
|             try: | ||||
|                 for x in walkdir(err, ap): | ||||
|                     yield x | ||||
|             except: | ||||
|                 err.append(ap) | ||||
|         else: | ||||
|             yield ap, inf | ||||
|  | ||||
|  | ||||
| def walkdirs(err, tops): | ||||
|     """recursive statdir for a list of tops, yields [top, relpath, stat]""" | ||||
|     sep = "{0}".format(os.sep).encode("ascii") | ||||
|     for top in tops: | ||||
|         if top[-1:] == sep: | ||||
|             stop = top.rstrip(sep) | ||||
|         else: | ||||
|             stop = os.path.dirname(top) | ||||
|  | ||||
|         if os.path.isdir(top): | ||||
|             for ap, inf in walkdir(err, top): | ||||
|                 yield stop, ap[len(stop) :].lstrip(sep), inf | ||||
|         else: | ||||
|             d, n = top.rsplit(sep, 1) | ||||
|             yield d, n, os.stat(top) | ||||
|  | ||||
|  | ||||
| # mostly from copyparty/util.py | ||||
| def quotep(btxt): | ||||
|     quot1 = quote(btxt, safe=b"/") | ||||
|     if not PY2: | ||||
|         quot1 = quot1.encode("ascii") | ||||
|  | ||||
|     return quot1.replace(b" ", b"+") | ||||
|  | ||||
|  | ||||
| # from copyparty/util.py | ||||
| def humansize(sz, terse=False): | ||||
|     """picks a sensible unit for the given extent""" | ||||
|     for unit in ["B", "KiB", "MiB", "GiB", "TiB"]: | ||||
|         if sz < 1024: | ||||
|             break | ||||
|  | ||||
|         sz /= 1024.0 | ||||
|  | ||||
|     ret = " ".join([str(sz)[:4].rstrip("."), unit]) | ||||
|  | ||||
|     if not terse: | ||||
|         return ret | ||||
|  | ||||
|     return ret.replace("iB", "").replace(" ", "") | ||||
|  | ||||
|  | ||||
| # from copyparty/up2k.py | ||||
| def up2k_chunksize(filesize): | ||||
|     """gives The correct chunksize for up2k hashing""" | ||||
|     chunksize = 1024 * 1024 | ||||
|     stepsize = 512 * 1024 | ||||
|     while True: | ||||
|         for mul in [1, 2]: | ||||
|             nchunks = math.ceil(filesize * 1.0 / chunksize) | ||||
|             if nchunks <= 256 or chunksize >= 32 * 1024 * 1024: | ||||
|                 return chunksize | ||||
|  | ||||
|             chunksize += stepsize | ||||
|             stepsize *= mul | ||||
|  | ||||
|  | ||||
| # mostly from copyparty/up2k.py | ||||
| def get_hashlist(file, pcb): | ||||
|     # type: (File, any) -> None | ||||
|     """generates the up2k hashlist from file contents, inserts it into `file`""" | ||||
|  | ||||
|     chunk_sz = up2k_chunksize(file.size) | ||||
|     file_rem = file.size | ||||
|     file_ofs = 0 | ||||
|     ret = [] | ||||
|     with open(file.abs, "rb", 512 * 1024) as f: | ||||
|         while file_rem > 0: | ||||
|             hashobj = hashlib.sha512() | ||||
|             chunk_sz = chunk_rem = min(chunk_sz, file_rem) | ||||
|             while chunk_rem > 0: | ||||
|                 buf = f.read(min(chunk_rem, 64 * 1024)) | ||||
|                 if not buf: | ||||
|                     raise Exception("EOF at " + str(f.tell())) | ||||
|  | ||||
|                 hashobj.update(buf) | ||||
|                 chunk_rem -= len(buf) | ||||
|  | ||||
|             digest = hashobj.digest()[:33] | ||||
|             digest = base64.urlsafe_b64encode(digest).decode("utf-8") | ||||
|  | ||||
|             ret.append([digest, file_ofs, chunk_sz]) | ||||
|             file_ofs += chunk_sz | ||||
|             file_rem -= chunk_sz | ||||
|  | ||||
|             if pcb: | ||||
|                 pcb(file, file_ofs) | ||||
|  | ||||
|     file.cids = ret | ||||
|     file.kchunks = {} | ||||
|     for k, v1, v2 in ret: | ||||
|         file.kchunks[k] = [v1, v2] | ||||
|  | ||||
|  | ||||
| def handshake(req_ses, url, file, pw, search): | ||||
|     # type: (requests.Session, str, File, any, bool) -> list[str] | ||||
|     """ | ||||
|     performs a handshake with the server; reply is: | ||||
|       if search, a list of search results | ||||
|       otherwise, a list of chunks to upload | ||||
|     """ | ||||
|  | ||||
|     req = { | ||||
|         "hash": [x[0] for x in file.cids], | ||||
|         "name": file.name, | ||||
|         "lmod": file.lmod, | ||||
|         "size": file.size, | ||||
|     } | ||||
|     if search: | ||||
|         req["srch"] = 1 | ||||
|  | ||||
|     headers = {"Content-Type": "text/plain"}  # wtf ed | ||||
|     if pw: | ||||
|         headers["Cookie"] = "=".join(["cppwd", pw]) | ||||
|  | ||||
|     if file.url: | ||||
|         url = file.url | ||||
|     elif b"/" in file.rel: | ||||
|         url += quotep(file.rel.rsplit(b"/", 1)[0]).decode("utf-8", "replace") | ||||
|  | ||||
|     while True: | ||||
|         try: | ||||
|             r = req_ses.post(url, headers=headers, json=req) | ||||
|             break | ||||
|         except: | ||||
|             eprint("handshake failed, retrying: {0}\n".format(file.name)) | ||||
|             time.sleep(1) | ||||
|  | ||||
|     try: | ||||
|         r = r.json() | ||||
|     except: | ||||
|         raise Exception(r.text) | ||||
|  | ||||
|     if search: | ||||
|         return r["hits"] | ||||
|  | ||||
|     try: | ||||
|         pre, url = url.split("://") | ||||
|         pre += "://" | ||||
|     except: | ||||
|         pre = "" | ||||
|  | ||||
|     file.url = pre + url.split("/")[0] + r["purl"] | ||||
|     file.name = r["name"] | ||||
|     file.wark = r["wark"] | ||||
|  | ||||
|     return r["hash"], r["sprs"] | ||||
|  | ||||
|  | ||||
| def upload(req_ses, file, cid, pw): | ||||
|     # type: (requests.Session, File, str, any) -> None | ||||
|     """upload one specific chunk, `cid` (a chunk-hash)""" | ||||
|  | ||||
|     headers = { | ||||
|         "X-Up2k-Hash": cid, | ||||
|         "X-Up2k-Wark": file.wark, | ||||
|         "Content-Type": "application/octet-stream", | ||||
|     } | ||||
|     if pw: | ||||
|         headers["Cookie"] = "=".join(["cppwd", pw]) | ||||
|  | ||||
|     f = FileSlice(file, cid) | ||||
|     try: | ||||
|         r = req_ses.post(file.url, headers=headers, data=f) | ||||
|         if not r: | ||||
|             raise Exception(repr(r)) | ||||
|  | ||||
|         _ = r.content | ||||
|     finally: | ||||
|         f.f.close() | ||||
|  | ||||
|  | ||||
| class Daemon(threading.Thread): | ||||
|     def __init__(self, *a, **ka): | ||||
|         threading.Thread.__init__(self, *a, **ka) | ||||
|         self.daemon = True | ||||
|  | ||||
|  | ||||
| class Ctl(object): | ||||
|     """ | ||||
|     this will be the coordinator which runs everything in parallel | ||||
|     (hashing, handshakes, uploads)  but right now it's p dumb | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, ar): | ||||
|         self.ar = ar | ||||
|         ar.files = [ | ||||
|             os.path.abspath(os.path.realpath(x.encode("utf-8"))) | ||||
|             + (x[-1:] if x[-1:] == os.sep else "").encode("utf-8") | ||||
|             for x in ar.files | ||||
|         ] | ||||
|         ar.url = ar.url.rstrip("/") + "/" | ||||
|         if "://" not in ar.url: | ||||
|             ar.url = "http://" + ar.url | ||||
|  | ||||
|         eprint("\nscanning {0} locations\n".format(len(ar.files))) | ||||
|  | ||||
|         nfiles = 0 | ||||
|         nbytes = 0 | ||||
|         err = [] | ||||
|         for _, _, inf in walkdirs(err, ar.files): | ||||
|             nfiles += 1 | ||||
|             nbytes += inf.st_size | ||||
|  | ||||
|         if err: | ||||
|             eprint("\n# failed to access {0} paths:\n".format(len(err))) | ||||
|             for x in err: | ||||
|                 eprint(x.decode("utf-8", "replace") + "\n") | ||||
|  | ||||
|             eprint("^ failed to access those {0} paths ^\n\n".format(len(err))) | ||||
|             if not ar.ok: | ||||
|                 eprint("aborting because --ok is not set\n") | ||||
|                 return | ||||
|  | ||||
|         eprint("found {0} files, {1}\n\n".format(nfiles, humansize(nbytes))) | ||||
|         self.nfiles = nfiles | ||||
|         self.nbytes = nbytes | ||||
|  | ||||
|         if ar.td: | ||||
|             requests.packages.urllib3.disable_warnings() | ||||
|             req_ses.verify = False | ||||
|         if ar.te: | ||||
|             req_ses.verify = ar.te | ||||
|  | ||||
|         self.filegen = walkdirs([], ar.files) | ||||
|         if ar.safe: | ||||
|             self._safe() | ||||
|         else: | ||||
|             self.hash_f = 0 | ||||
|             self.hash_c = 0 | ||||
|             self.hash_b = 0 | ||||
|             self.up_f = 0 | ||||
|             self.up_c = 0 | ||||
|             self.up_b = 0 | ||||
|             self.up_br = 0 | ||||
|             self.hasher_busy = 1 | ||||
|             self.handshaker_busy = 0 | ||||
|             self.uploader_busy = 0 | ||||
|             self.serialized = False | ||||
|  | ||||
|             self.t0 = time.time() | ||||
|             self.t0_up = None | ||||
|             self.spd = None | ||||
|  | ||||
|             self.mutex = threading.Lock() | ||||
|             self.q_handshake = Queue()  # type: Queue[File] | ||||
|             self.q_recheck = Queue()  # type: Queue[File]  # partial upload exists [...] | ||||
|             self.q_upload = Queue()  # type: Queue[tuple[File, str]] | ||||
|  | ||||
|             self.st_hash = [None, "(idle, starting...)"]  # type: tuple[File, int] | ||||
|             self.st_up = [None, "(idle, starting...)"]  # type: tuple[File, int] | ||||
|  | ||||
|             self._fancy() | ||||
|  | ||||
|     def _safe(self): | ||||
|         """minimal basic slow boring fallback codepath""" | ||||
|         search = self.ar.s | ||||
|         for nf, (top, rel, inf) in enumerate(self.filegen): | ||||
|             file = File(top, rel, inf.st_size, inf.st_mtime) | ||||
|             upath = file.abs.decode("utf-8", "replace") | ||||
|  | ||||
|             print("{0} {1}\n  hash...".format(self.nfiles - nf, upath)) | ||||
|             get_hashlist(file, None) | ||||
|  | ||||
|             burl = self.ar.url[:12] + self.ar.url[8:].split("/")[0] + "/" | ||||
|             while True: | ||||
|                 print("  hs...") | ||||
|                 hs, _ = handshake(req_ses, self.ar.url, file, self.ar.a, search) | ||||
|                 if search: | ||||
|                     if hs: | ||||
|                         for hit in hs: | ||||
|                             print("  found: {0}{1}".format(burl, hit["rp"])) | ||||
|                     else: | ||||
|                         print("  NOT found") | ||||
|                     break | ||||
|  | ||||
|                 file.ucids = hs | ||||
|                 if not hs: | ||||
|                     break | ||||
|  | ||||
|                 print("{0} {1}".format(self.nfiles - nf, upath)) | ||||
|                 ncs = len(hs) | ||||
|                 for nc, cid in enumerate(hs): | ||||
|                     print("  {0} up {1}".format(ncs - nc, cid)) | ||||
|                     upload(req_ses, file, cid, self.ar.a) | ||||
|  | ||||
|             print("  ok!") | ||||
|  | ||||
|     def _fancy(self): | ||||
|         if VT100: | ||||
|             atexit.register(self.cleanup_vt100) | ||||
|             ss.scroll_region(3) | ||||
|  | ||||
|         Daemon(target=self.hasher).start() | ||||
|         for _ in range(self.ar.j): | ||||
|             Daemon(target=self.handshaker).start() | ||||
|             Daemon(target=self.uploader).start() | ||||
|  | ||||
|         idles = 0 | ||||
|         while idles < 3: | ||||
|             time.sleep(0.07) | ||||
|             with self.mutex: | ||||
|                 if ( | ||||
|                     self.q_handshake.empty() | ||||
|                     and self.q_upload.empty() | ||||
|                     and not self.hasher_busy | ||||
|                     and not self.handshaker_busy | ||||
|                     and not self.uploader_busy | ||||
|                 ): | ||||
|                     idles += 1 | ||||
|                 else: | ||||
|                     idles = 0 | ||||
|  | ||||
|             if VT100: | ||||
|                 maxlen = ss.w - len(str(self.nfiles)) - 14 | ||||
|                 txt = "\033[s\033[{0}H".format(ss.g) | ||||
|                 for y, k, st, f in [ | ||||
|                     [0, "hash", self.st_hash, self.hash_f], | ||||
|                     [1, "send", self.st_up, self.up_f], | ||||
|                 ]: | ||||
|                     txt += "\033[{0}H{1}:".format(ss.g + y, k) | ||||
|                     file, arg = st | ||||
|                     if not file: | ||||
|                         txt += " {0}\033[K".format(arg) | ||||
|                     else: | ||||
|                         if y: | ||||
|                             p = 100 * file.up_b / file.size | ||||
|                         else: | ||||
|                             p = 100 * arg / file.size | ||||
|  | ||||
|                         name = file.abs.decode("utf-8", "replace")[-maxlen:] | ||||
|                         if "/" in name: | ||||
|                             name = "\033[36m{0}\033[0m/{1}".format(*name.rsplit("/", 1)) | ||||
|  | ||||
|                         t = "{0:6.1f}% {1} {2}\033[K" | ||||
|                         txt += t.format(p, self.nfiles - f, name) | ||||
|  | ||||
|                 txt += "\033[{0}H ".format(ss.g + 2) | ||||
|             else: | ||||
|                 txt = " " | ||||
|  | ||||
|             if not self.up_br: | ||||
|                 spd = self.hash_b / (time.time() - self.t0) | ||||
|                 eta = (self.nbytes - self.hash_b) / (spd + 1) | ||||
|             else: | ||||
|                 spd = self.up_br / (time.time() - self.t0_up) | ||||
|                 spd = self.spd = (self.spd or spd) * 0.9 + spd * 0.1 | ||||
|                 eta = (self.nbytes - self.up_b) / (spd + 1) | ||||
|  | ||||
|             spd = humansize(spd) | ||||
|             eta = str(datetime.timedelta(seconds=int(eta))) | ||||
|             sleft = humansize(self.nbytes - self.up_b) | ||||
|             nleft = self.nfiles - self.up_f | ||||
|             tail = "\033[K\033[u" if VT100 else "\r" | ||||
|  | ||||
|             t = "{0} eta @ {1}/s, {2}, {3}# left".format(eta, spd, sleft, nleft) | ||||
|             eprint(txt + "\033]0;{0}\033\\\r{0}{1}".format(t, tail)) | ||||
|  | ||||
|     def cleanup_vt100(self): | ||||
|         ss.scroll_region(None) | ||||
|         eprint("\033[J\033]0;\033\\") | ||||
|  | ||||
|     def cb_hasher(self, file, ofs): | ||||
|         self.st_hash = [file, ofs] | ||||
|  | ||||
|     def hasher(self): | ||||
|         prd = None | ||||
|         ls = {} | ||||
|         for top, rel, inf in self.filegen: | ||||
|             if self.ar.z: | ||||
|                 rd = os.path.dirname(rel) | ||||
|                 if prd != rd: | ||||
|                     prd = rd | ||||
|                     headers = {} | ||||
|                     if self.ar.a: | ||||
|                         headers["Cookie"] = "=".join(["cppwd", self.ar.a]) | ||||
|  | ||||
|                     ls = {} | ||||
|                     try: | ||||
|                         print("      ls ~{0}".format(rd.decode("utf-8", "replace"))) | ||||
|                         r = req_ses.get( | ||||
|                             self.ar.url.encode("utf-8") + quotep(rd) + b"?ls", | ||||
|                             headers=headers, | ||||
|                         ) | ||||
|                         for f in r.json()["files"]: | ||||
|                             rfn = f["href"].split("?")[0].encode("utf-8", "replace") | ||||
|                             ls[unquote(rfn)] = f | ||||
|                     except: | ||||
|                         print("   mkdir ~{0}".format(rd.decode("utf-8", "replace"))) | ||||
|  | ||||
|                 rf = ls.get(os.path.basename(rel), None) | ||||
|                 if rf and rf["sz"] == inf.st_size and abs(rf["ts"] - inf.st_mtime) <= 1: | ||||
|                     self.nfiles -= 1 | ||||
|                     self.nbytes -= inf.st_size | ||||
|                     continue | ||||
|  | ||||
|             file = File(top, rel, inf.st_size, inf.st_mtime) | ||||
|             while True: | ||||
|                 with self.mutex: | ||||
|                     if ( | ||||
|                         self.hash_b - self.up_b < 1024 * 1024 * 128 | ||||
|                         and self.hash_c - self.up_c < 64 | ||||
|                         and ( | ||||
|                             not self.ar.nh | ||||
|                             or ( | ||||
|                                 self.q_upload.empty() | ||||
|                                 and self.q_handshake.empty() | ||||
|                                 and not self.uploader_busy | ||||
|                             ) | ||||
|                         ) | ||||
|                     ): | ||||
|                         break | ||||
|  | ||||
|                 time.sleep(0.05) | ||||
|  | ||||
|             get_hashlist(file, self.cb_hasher) | ||||
|             with self.mutex: | ||||
|                 self.hash_f += 1 | ||||
|                 self.hash_c += len(file.cids) | ||||
|                 self.hash_b += file.size | ||||
|  | ||||
|             self.q_handshake.put(file) | ||||
|  | ||||
|         self.hasher_busy = 0 | ||||
|         self.st_hash = [None, "(finished)"] | ||||
|  | ||||
|     def handshaker(self): | ||||
|         search = self.ar.s | ||||
|         q = self.q_handshake | ||||
|         burl = self.ar.url[:8] + self.ar.url[8:].split("/")[0] + "/" | ||||
|         while True: | ||||
|             file = q.get() | ||||
|             if not file: | ||||
|                 if q == self.q_handshake: | ||||
|                     q = self.q_recheck | ||||
|                     q.put(None) | ||||
|                     continue | ||||
|  | ||||
|                 self.q_upload.put(None) | ||||
|                 break | ||||
|  | ||||
|             with self.mutex: | ||||
|                 self.handshaker_busy += 1 | ||||
|  | ||||
|             upath = file.abs.decode("utf-8", "replace") | ||||
|  | ||||
|             try: | ||||
|                 hs, sprs = handshake(req_ses, self.ar.url, file, self.ar.a, search) | ||||
|             except Exception as ex: | ||||
|                 if q == self.q_handshake and "<pre>partial upload exists" in str(ex): | ||||
|                     self.q_recheck.put(file) | ||||
|                     hs = [] | ||||
|                 else: | ||||
|                     raise | ||||
|  | ||||
|             if search: | ||||
|                 if hs: | ||||
|                     for hit in hs: | ||||
|                         t = "found: {0}\n  {1}{2}\n" | ||||
|                         print(t.format(upath, burl, hit["rp"]), end="") | ||||
|                 else: | ||||
|                     print("NOT found: {0}\n".format(upath), end="") | ||||
|  | ||||
|                 with self.mutex: | ||||
|                     self.up_f += 1 | ||||
|                     self.up_c += len(file.cids) | ||||
|                     self.up_b += file.size | ||||
|                     self.handshaker_busy -= 1 | ||||
|  | ||||
|                 continue | ||||
|  | ||||
|             with self.mutex: | ||||
|                 if not sprs and not self.serialized: | ||||
|                     t = "server filesystem does not support sparse files; serializing uploads\n" | ||||
|                     eprint(t) | ||||
|                     self.serialized = True | ||||
|                     for _ in range(self.ar.j - 1): | ||||
|                         self.q_upload.put(None) | ||||
|                 if not hs: | ||||
|                     # all chunks done | ||||
|                     self.up_f += 1 | ||||
|                     self.up_c += len(file.cids) - file.up_c | ||||
|                     self.up_b += file.size - file.up_b | ||||
|  | ||||
|                 if hs and file.up_c: | ||||
|                     # some chunks failed | ||||
|                     self.up_c -= len(hs) | ||||
|                     file.up_c -= len(hs) | ||||
|                     for cid in hs: | ||||
|                         sz = file.kchunks[cid][1] | ||||
|                         self.up_b -= sz | ||||
|                         file.up_b -= sz | ||||
|  | ||||
|                 file.ucids = hs | ||||
|                 self.handshaker_busy -= 1 | ||||
|  | ||||
|             if not hs: | ||||
|                 kw = "uploaded" if file.up_b else "   found" | ||||
|                 print("{0} {1}".format(kw, upath)) | ||||
|             for cid in hs: | ||||
|                 self.q_upload.put([file, cid]) | ||||
|  | ||||
|     def uploader(self): | ||||
|         while True: | ||||
|             task = self.q_upload.get() | ||||
|             if not task: | ||||
|                 self.st_up = [None, "(finished)"] | ||||
|                 break | ||||
|  | ||||
|             with self.mutex: | ||||
|                 self.uploader_busy += 1 | ||||
|                 self.t0_up = self.t0_up or time.time() | ||||
|  | ||||
|             file, cid = task | ||||
|             try: | ||||
|                 upload(req_ses, file, cid, self.ar.a) | ||||
|             except: | ||||
|                 eprint("upload failed, retrying: {0} #{1}\n".format(file.name, cid[:8])) | ||||
|                 pass  # handshake will fix it | ||||
|  | ||||
|             with self.mutex: | ||||
|                 sz = file.kchunks[cid][1] | ||||
|                 file.ucids = [x for x in file.ucids if x != cid] | ||||
|                 if not file.ucids: | ||||
|                     self.q_handshake.put(file) | ||||
|  | ||||
|                 self.st_up = [file, cid] | ||||
|                 file.up_b += sz | ||||
|                 self.up_b += sz | ||||
|                 self.up_br += sz | ||||
|                 file.up_c += 1 | ||||
|                 self.up_c += 1 | ||||
|                 self.uploader_busy -= 1 | ||||
|  | ||||
|  | ||||
| class APF(argparse.ArgumentDefaultsHelpFormatter, argparse.RawDescriptionHelpFormatter): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||
|     if not VT100: | ||||
|         os.system("rem")  # enables colors | ||||
|  | ||||
|     # fmt: off | ||||
|     ap = app = argparse.ArgumentParser(formatter_class=APF, epilog=""" | ||||
| NOTE: | ||||
| source file/folder selection uses rsync syntax, meaning that: | ||||
|   "foo" uploads the entire folder to URL/foo/ | ||||
|   "foo/" uploads the CONTENTS of the folder into URL/ | ||||
| """) | ||||
|  | ||||
|     ap.add_argument("url", type=unicode, help="server url, including destination folder") | ||||
|     ap.add_argument("files", type=unicode, nargs="+", help="files and/or folders to process") | ||||
|     ap.add_argument("-a", metavar="PASSWORD", help="password") | ||||
|     ap.add_argument("-s", action="store_true", help="file-search (disables upload)") | ||||
|     ap.add_argument("--ok", action="store_true", help="continue even if some local files are inaccessible") | ||||
|     ap = app.add_argument_group("performance tweaks") | ||||
|     ap.add_argument("-j", type=int, metavar="THREADS", default=4, help="parallel connections") | ||||
|     ap.add_argument("-nh", action="store_true", help="disable hashing while uploading") | ||||
|     ap.add_argument("--safe", action="store_true", help="use simple fallback approach") | ||||
|     ap.add_argument("-z", action="store_true", help="ZOOMIN' (skip uploading files if they exist at the destination with the ~same last-modified timestamp, so same as yolo / turbo with date-chk but even faster)") | ||||
|     ap = app.add_argument_group("tls") | ||||
|     ap.add_argument("-te", metavar="PEM_FILE", help="certificate to expect/verify") | ||||
|     ap.add_argument("-td", action="store_true", help="disable certificate check") | ||||
|     # fmt: on | ||||
|  | ||||
|     Ctl(app.parse_args()) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										24
									
								
								bin/up2k.sh
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						
									
										24
									
								
								bin/up2k.sh
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							| @@ -8,7 +8,7 @@ set -e | ||||
| ## | ||||
| ## config | ||||
|  | ||||
| datalen=$((2*1024*1024*1024)) | ||||
| datalen=$((128*1024*1024)) | ||||
| target=127.0.0.1 | ||||
| posturl=/inc | ||||
| passwd=wark | ||||
| @@ -37,10 +37,10 @@ gendata() { | ||||
| # pipe a chunk, get the base64 checksum | ||||
| gethash() { | ||||
|     printf $( | ||||
|         sha512sum | cut -c-64 | | ||||
|         sha512sum | cut -c-66 | | ||||
|         sed -r 's/ .*//;s/(..)/\\x\1/g' | ||||
|     ) | | ||||
|     base64 -w0 | cut -c-43 | | ||||
|     base64 -w0 | cut -c-44 | | ||||
|     tr '+/' '-_' | ||||
| } | ||||
|  | ||||
| @@ -123,7 +123,7 @@ printf '\033[36m' | ||||
| { | ||||
|     { | ||||
|         cat <<EOF | ||||
| POST $posturl/handshake.php HTTP/1.1 | ||||
| POST $posturl/ HTTP/1.1 | ||||
| Connection: Close | ||||
| Cookie: cppwd=$passwd | ||||
| Content-Type: text/plain;charset=UTF-8 | ||||
| @@ -145,14 +145,16 @@ printf '\033[0m\nwark: %s\n' $wark | ||||
| ## | ||||
| ## wait for signal to continue | ||||
|  | ||||
| w8=/dev/shm/$salt.w8 | ||||
| touch $w8 | ||||
| true || { | ||||
|     w8=/dev/shm/$salt.w8 | ||||
|     touch $w8 | ||||
|  | ||||
| echo "ready;  rm -f $w8" | ||||
|     echo "ready;  rm -f $w8" | ||||
|  | ||||
| while [ -e $w8 ]; do | ||||
|     sleep 0.2 | ||||
| done | ||||
|     while [ -e $w8 ]; do | ||||
|         sleep 0.2 | ||||
|     done | ||||
| } | ||||
|  | ||||
|  | ||||
| ## | ||||
| @@ -175,7 +177,7 @@ while [ $remains -gt 0 ]; do | ||||
|      | ||||
|     { | ||||
|         cat <<EOF | ||||
| POST $posturl/chunkpit.php HTTP/1.1 | ||||
| POST $posturl/ HTTP/1.1 | ||||
| Connection: Keep-Alive | ||||
| Cookie: cppwd=$passwd | ||||
| Content-Type: application/octet-stream | ||||
|   | ||||
							
								
								
									
										46
									
								
								contrib/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								contrib/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| ### [`plugins/`](plugins/) | ||||
| * example extensions | ||||
|  | ||||
| ### [`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) | ||||
|  | ||||
| ### [`media-osd-bgone.ps1`](media-osd-bgone.ps1) | ||||
| * disables the [windows OSD popup](https://user-images.githubusercontent.com/241032/122821375-0e08df80-d2dd-11eb-9fd9-184e8aacf1d0.png) (the thing on the left) which appears every time you hit media hotkeys to adjust volume or change song while playing music with the copyparty web-ui, or most other audio players really | ||||
|  | ||||
| ### [`explorer-nothumbs-nofoldertypes.reg`](explorer-nothumbs-nofoldertypes.reg) | ||||
| * 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 | ||||
| * systemd service at [`systemd/cfssl.service`](systemd/cfssl.service) | ||||
|  | ||||
| # OS integration | ||||
| init-scripts to start copyparty as a service | ||||
| * [`systemd/copyparty.service`](systemd/copyparty.service) runs the sfx normally | ||||
| * [`rc/copyparty`](rc/copyparty) runs sfx normally on freebsd, create a `copyparty` user | ||||
| * [`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) | ||||
							
								
								
									
										73
									
								
								contrib/cfssl.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										73
									
								
								contrib/cfssl.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,73 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
| # ca-name and server-fqdn | ||||
| ca_name="$1" | ||||
| srv_fqdn="$2" | ||||
|  | ||||
| [ -z "$srv_fqdn" ] && { | ||||
| 	echo "need arg 1: ca name" | ||||
| 	echo "need arg 2: server fqdn and/or IPs, comma-separated" | ||||
| 	echo "optional arg 3: if set, write cert into copyparty cfg" | ||||
| 	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_fqdn"}]} | ||||
| EOF | ||||
| 	)| | ||||
| 	cfssl gencert -ca ca.pem -ca-key ca.key \ | ||||
| 		-profile=www -hostname="$srv_fqdn" - | | ||||
| 	cfssljson -bare "$srv_fqdn" | ||||
|  | ||||
| 	mv "$srv_fqdn-key.pem" "$srv_fqdn.key" | ||||
| 	rm "$srv_fqdn.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_fqdn.pem" | ||||
|  | ||||
|  | ||||
| # write cert into copyparty config | ||||
| [ -z "$3" ] || { | ||||
| 	mkdir -p ~/.config/copyparty | ||||
| 	cat "$srv_fqdn".{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> | ||||
							
								
								
									
										104
									
								
								contrib/media-osd-bgone.ps1
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								contrib/media-osd-bgone.ps1
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,104 @@ | ||||
| # media-osd-bgone.ps1: disable media-control OSD on win10do | ||||
| # v1.1, 2021-06-25, ed <irc.rizon.net>, MIT-licensed | ||||
| # https://github.com/9001/copyparty/blob/hovudstraum/contrib/media-osd-bgone.ps1 | ||||
| # | ||||
| # locates the first window that looks like the media OSD and minimizes it; | ||||
| # doing this once after each reboot should do the trick | ||||
| # (adjust the width/height filter if it doesn't work) | ||||
| # | ||||
| # --------------------------------------------------------------------- | ||||
| # | ||||
| # tip: save the following as "media-osd-bgone.bat" next to this script: | ||||
| #   start cmd /c "powershell -command ""set-executionpolicy -scope process bypass; .\media-osd-bgone.ps1"" & ping -n 2 127.1 >nul" | ||||
| # | ||||
| # then create a shortcut to that bat-file and move the shortcut here: | ||||
| #   %appdata%\Microsoft\Windows\Start Menu\Programs\Startup | ||||
| # | ||||
| # and now this will autorun on bootup | ||||
|  | ||||
|  | ||||
| Add-Type -TypeDefinition @" | ||||
| using System; | ||||
| using System.IO; | ||||
| using System.Threading; | ||||
| using System.Diagnostics; | ||||
| using System.Runtime.InteropServices; | ||||
| using System.Windows.Forms; | ||||
|  | ||||
| namespace A { | ||||
|   public class B : Control { | ||||
|  | ||||
|     [DllImport("user32.dll")] | ||||
|     static extern void keybd_event(byte bVk, byte bScan, uint dwFlags, int dwExtraInfo); | ||||
|  | ||||
|     [DllImport("user32.dll", SetLastError = true)] | ||||
|     static extern IntPtr FindWindowEx(IntPtr hwndParent, IntPtr hwndChildAfter, string lpszClass, string lpszWindow); | ||||
|  | ||||
|     [DllImport("user32.dll", SetLastError=true)] | ||||
|     static extern bool GetWindowRect(IntPtr hwnd, out RECT lpRect); | ||||
|  | ||||
|     [DllImport("user32.dll")] | ||||
|     static extern bool ShowWindow(IntPtr hWnd, int nCmdShow); | ||||
|  | ||||
|     [StructLayout(LayoutKind.Sequential)] | ||||
|     public struct RECT { | ||||
|       public int x; | ||||
|       public int y; | ||||
|       public int x2; | ||||
|       public int y2; | ||||
|     } | ||||
|      | ||||
|     bool fa() { | ||||
|       RECT r; | ||||
|       IntPtr it = IntPtr.Zero; | ||||
|       while ((it = FindWindowEx(IntPtr.Zero, it, "NativeHWNDHost", "")) != IntPtr.Zero) { | ||||
|         if (FindWindowEx(it, IntPtr.Zero, "DirectUIHWND", "") == IntPtr.Zero) | ||||
|           continue; | ||||
|          | ||||
|         if (!GetWindowRect(it, out r)) | ||||
|           continue; | ||||
|  | ||||
|         int w = r.x2 - r.x + 1; | ||||
|         int h = r.y2 - r.y + 1; | ||||
|  | ||||
|         Console.WriteLine("[*] hwnd {0:x} @ {1}x{2} sz {3}x{4}", it, r.x, r.y, w, h); | ||||
|         if (h != 141) | ||||
|           continue; | ||||
|          | ||||
|         ShowWindow(it, 6); | ||||
|         Console.WriteLine("[+] poof"); | ||||
|         return true; | ||||
|       } | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|     void fb() { | ||||
|       keybd_event((byte)Keys.VolumeMute, 0, 0, 0); | ||||
|       keybd_event((byte)Keys.VolumeMute, 0, 2, 0); | ||||
|       Thread.Sleep(500); | ||||
|       keybd_event((byte)Keys.VolumeMute, 0, 0, 0); | ||||
|       keybd_event((byte)Keys.VolumeMute, 0, 2, 0); | ||||
|  | ||||
|       while (true) { | ||||
|         if (fa()) { | ||||
|           break; | ||||
|         } | ||||
|         Console.WriteLine("[!] not found"); | ||||
|         Thread.Sleep(1000); | ||||
|       } | ||||
|       this.Invoke((MethodInvoker)delegate { | ||||
|         Application.Exit(); | ||||
|       }); | ||||
|     } | ||||
|  | ||||
|     public void Run() { | ||||
|       Console.WriteLine("[+] hi"); | ||||
|       new Thread(new ThreadStart(fb)).Start(); | ||||
|       Application.Run(); | ||||
|       Console.WriteLine("[+] bye"); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| "@ -ReferencedAssemblies System.Windows.Forms | ||||
|  | ||||
| (New-Object -TypeName A.B).Run() | ||||
							
								
								
									
										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 1; | ||||
| } | ||||
| 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" | ||||
							
								
								
									
										33
									
								
								contrib/plugins/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								contrib/plugins/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| # example resource files | ||||
|  | ||||
| can be provided to copyparty to tweak things | ||||
|  | ||||
|  | ||||
|  | ||||
| ## example `.epilogue.html` | ||||
| save one of these as `.epilogue.html` inside a folder to customize it: | ||||
|  | ||||
| * [`minimal-up2k.html`](minimal-up2k.html) will [simplify the upload ui](https://user-images.githubusercontent.com/241032/118311195-dd6ca380-b4ef-11eb-86f3-75a3ff2e1332.png) | ||||
|  | ||||
|  | ||||
|  | ||||
| ## example browser-js | ||||
| point `--js-browser` to one of these by URL: | ||||
|  | ||||
| * [`minimal-up2k.js`](minimal-up2k.js) is similar to the above `minimal-up2k.html` except it applies globally to all write-only folders | ||||
| * [`up2k-hooks.js`](up2k-hooks.js) lets you specify a ruleset for files to skip uploading | ||||
|   * [`up2k-hook-ytid.js`](up2k-hook-ytid.js) is a more specific example checking youtube-IDs against some API | ||||
|  | ||||
|  | ||||
|  | ||||
| ## example browser-css | ||||
| point `--css-browser` to one of these by URL: | ||||
|  | ||||
| * [`browser-icons.css`](browser-icons.css) adds filetype icons | ||||
|  | ||||
|  | ||||
|  | ||||
| ## meadup.js | ||||
|  | ||||
| * turns copyparty into chromecast just more flexible (and probably way more buggy) | ||||
| * usage: put the js somewhere in the webroot and `--js-browser /memes/meadup.js` | ||||
							
								
								
									
										71
									
								
								contrib/plugins/browser-icons.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								contrib/plugins/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: '🎨'; | ||||
| } | ||||
							
								
								
									
										506
									
								
								contrib/plugins/meadup.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										506
									
								
								contrib/plugins/meadup.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,506 @@ | ||||
| // USAGE: | ||||
| //   place this file somewhere in the webroot and then | ||||
| //   python3 -m copyparty --js-browser /memes/meadup.js | ||||
| // | ||||
| // FEATURES: | ||||
| // * adds an onscreen keyboard for operating a media center remotely, | ||||
| //    relies on https://github.com/9001/copyparty/blob/hovudstraum/bin/mtag/very-bad-idea.py | ||||
| // * adds an interactive anime girl (if you can find the dependencies) | ||||
|  | ||||
| var hambagas = [ | ||||
|     "https://www.youtube.com/watch?v=pFA3KGp4GuU" | ||||
| ]; | ||||
|  | ||||
| // keybaord, | ||||
| //   onscreen keyboard by @steinuil | ||||
| function initKeybaord(BASE_URL, HAMBAGA, consoleLog, consoleError) { | ||||
|     document.querySelector('.keybaord-container').innerHTML = ` | ||||
|       <div class="keybaord-body"> | ||||
|         <div class="keybaord-row keybaord-row-1"> | ||||
|           <div class="keybaord-key" data-keybaord-key="Escape"> | ||||
|             esc | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F1"> | ||||
|             F1 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F2"> | ||||
|             F2 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F3"> | ||||
|             F3 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F4"> | ||||
|             F4 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F5"> | ||||
|             F5 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F6"> | ||||
|             F6 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F7"> | ||||
|             F7 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F8"> | ||||
|             F8 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F9"> | ||||
|             F9 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F10"> | ||||
|             F10 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F11"> | ||||
|             F11 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="F12"> | ||||
|             F12 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="Insert"> | ||||
|             ins | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="Delete"> | ||||
|             del | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="keybaord-row keybaord-row-2"> | ||||
|           <div class="keybaord-key" data-keybaord-key="\`"> | ||||
|             \` | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="1"> | ||||
|             1 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="2"> | ||||
|             2 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="3"> | ||||
|             3 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="4"> | ||||
|             4 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="5"> | ||||
|             5 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="6"> | ||||
|             6 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="7"> | ||||
|             7 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="8"> | ||||
|             8 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="9"> | ||||
|             9 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="0"> | ||||
|             0 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="-"> | ||||
|             - | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="="> | ||||
|             = | ||||
|           </div> | ||||
|           <div class="keybaord-key keybaord-backspace" data-keybaord-key="BackSpace"> | ||||
|             backspace | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="keybaord-row keybaord-row-3"> | ||||
|           <div class="keybaord-key keybaord-tab" data-keybaord-key="Tab"> | ||||
|             tab | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="q"> | ||||
|             q | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="w"> | ||||
|             w | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="e"> | ||||
|             e | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="r"> | ||||
|             r | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="t"> | ||||
|             t | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="y"> | ||||
|             y | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="u"> | ||||
|             u | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="i"> | ||||
|             i | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="o"> | ||||
|             o | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="p"> | ||||
|             p | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="["> | ||||
|             [ | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="]"> | ||||
|             ] | ||||
|           </div> | ||||
|           <div class="keybaord-key keybaord-enter" data-keybaord-key="Return"> | ||||
|             enter | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="keybaord-row keybaord-row-4"> | ||||
|           <div class="keybaord-key keybaord-capslock" data-keybaord-key="HAMBAGA"> | ||||
|             🍔 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="a"> | ||||
|             a | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="s"> | ||||
|             s | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="d"> | ||||
|             d | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="f"> | ||||
|             f | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="g"> | ||||
|             g | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="h"> | ||||
|             h | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="j"> | ||||
|             j | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="k"> | ||||
|             k | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="l"> | ||||
|             l | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key=";"> | ||||
|             ; | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="'"> | ||||
|             ' | ||||
|           </div> | ||||
|           <div class="keybaord-key keybaord-backslash" data-keybaord-key="\\"> | ||||
|             \\ | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="keybaord-row keybaord-row-5"> | ||||
|           <div class="keybaord-key keybaord-lshift" data-keybaord-key="Shift_L"> | ||||
|             shift | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="\\"> | ||||
|             \\ | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="z"> | ||||
|             z | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="x"> | ||||
|             x | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="c"> | ||||
|             c | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="v"> | ||||
|             v | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="b"> | ||||
|             b | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="n"> | ||||
|             n | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="m"> | ||||
|             m | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key=","> | ||||
|             , | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="."> | ||||
|             . | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="/"> | ||||
|             / | ||||
|           </div> | ||||
|           <div class="keybaord-key keybaord-rshift" data-keybaord-key="Shift_R"> | ||||
|             shift | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="keybaord-row keybaord-row-6"> | ||||
|           <div class="keybaord-key keybaord-lctrl" data-keybaord-key="Control_L"> | ||||
|             ctrl | ||||
|           </div> | ||||
|           <div class="keybaord-key keybaord-super" data-keybaord-key="Meta_L"> | ||||
|             win | ||||
|           </div> | ||||
|           <div class="keybaord-key keybaord-alt" data-keybaord-key="Alt_L"> | ||||
|             alt | ||||
|           </div> | ||||
|           <div class="keybaord-key keybaord-spacebar" data-keybaord-key="space"> | ||||
|             space | ||||
|           </div> | ||||
|           <div class="keybaord-key keybaord-altgr" data-keybaord-key="Alt_R"> | ||||
|             altgr | ||||
|           </div> | ||||
|           <div class="keybaord-key keybaord-what" data-keybaord-key="Menu"> | ||||
|             menu | ||||
|           </div> | ||||
|           <div class="keybaord-key keybaord-rctrl" data-keybaord-key="Control_R"> | ||||
|             ctrl | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="keybaord-row"> | ||||
|           <div class="keybaord-key" data-keybaord-key="XF86AudioLowerVolume"> | ||||
|             🔉 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="XF86AudioRaiseVolume"> | ||||
|             🔊 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="Left"> | ||||
|             ⬅️ | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="Down"> | ||||
|             ⬇️ | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="Up"> | ||||
|             ⬆️ | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="Right"> | ||||
|             ➡️ | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="Page_Up"> | ||||
|             PgUp | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="Page_Down"> | ||||
|             PgDn | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="Home"> | ||||
|             🏠 | ||||
|           </div> | ||||
|           <div class="keybaord-key" data-keybaord-key="End"> | ||||
|             End | ||||
|           </div> | ||||
|         </div> | ||||
|       <div> | ||||
|     `; | ||||
|  | ||||
|     function arraySample(array) { | ||||
|         return array[Math.floor(Math.random() * array.length)]; | ||||
|     } | ||||
|  | ||||
|     function sendMessage(msg) { | ||||
|         return fetch(BASE_URL, { | ||||
|             method: "POST", | ||||
|             headers: { | ||||
|                 "Content-Type": "application/x-www-form-urlencoded;charset=UTF-8", | ||||
|             }, | ||||
|             body: "msg=" + encodeURIComponent(msg), | ||||
|         }).then( | ||||
|             (r) => r.text(), // so the response body shows up in network tab | ||||
|             (err) => consoleError(err) | ||||
|         ); | ||||
|     } | ||||
|     const MODIFIER_ON_CLASS = "keybaord-modifier-on"; | ||||
|     const KEY_DATASET = "data-keybaord-key"; | ||||
|     const KEY_CLASS = "keybaord-key"; | ||||
|  | ||||
|     const modifiers = new Set() | ||||
|  | ||||
|     function toggleModifier(button, key) { | ||||
|         button.classList.toggle(MODIFIER_ON_CLASS); | ||||
|         if (modifiers.has(key)) { | ||||
|             modifiers.delete(key); | ||||
|         } else { | ||||
|             modifiers.add(key); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function popModifiers() { | ||||
|         let modifierString = ""; | ||||
|  | ||||
|         modifiers.forEach((mod) => { | ||||
|             document.querySelector("[" + KEY_DATASET + "='" + mod + "']") | ||||
|                 .classList.remove(MODIFIER_ON_CLASS); | ||||
|  | ||||
|             modifierString += mod + "+"; | ||||
|         }); | ||||
|  | ||||
|         modifiers.clear(); | ||||
|  | ||||
|         return modifierString; | ||||
|     } | ||||
|  | ||||
|     Array.from(document.querySelectorAll("." + KEY_CLASS)).forEach((button) => { | ||||
|         const key = button.dataset.keybaordKey; | ||||
|  | ||||
|         button.addEventListener("click", (ev) => { | ||||
|             switch (key) { | ||||
|                 case "HAMBAGA": | ||||
|                     sendMessage(arraySample(HAMBAGA)); | ||||
|                     break; | ||||
|  | ||||
|                 case "Shift_L": | ||||
|                 case "Shift_R": | ||||
|  | ||||
|                 case "Control_L": | ||||
|                 case "Control_R": | ||||
|  | ||||
|                 case "Meta_L": | ||||
|  | ||||
|                 case "Alt_L": | ||||
|                 case "Alt_R": | ||||
|                     toggleModifier(button, key); | ||||
|                     break; | ||||
|  | ||||
|                 default: { | ||||
|                     const keyWithModifiers = popModifiers() + key; | ||||
|  | ||||
|                     consoleLog(keyWithModifiers); | ||||
|  | ||||
|                     sendMessage("key " + keyWithModifiers) | ||||
|                         .then(() => consoleLog(keyWithModifiers + " OK")); | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
|     }); | ||||
| } | ||||
|  | ||||
|  | ||||
| // keybaord integration | ||||
| (function () { | ||||
|     var o = mknod('div'); | ||||
|     clmod(o, 'keybaord-container', 1); | ||||
|     ebi('op_msg').appendChild(o); | ||||
|  | ||||
|     o = mknod('style'); | ||||
|     o.innerHTML = ` | ||||
| .keybaord-body { | ||||
| 	display: flex; | ||||
| 	flex-flow: column nowrap; | ||||
|     margin: .6em 0; | ||||
| } | ||||
|  | ||||
| .keybaord-row { | ||||
| 	display: flex; | ||||
| } | ||||
|  | ||||
| .keybaord-key { | ||||
| 	border: 1px solid rgba(128,128,128,0.2); | ||||
| 	width: 41px; | ||||
| 	height: 40px; | ||||
|  | ||||
| 	display: flex; | ||||
| 	justify-content: center; | ||||
| 	align-items: center; | ||||
| } | ||||
|  | ||||
| .keybaord-key:active { | ||||
| 	background-color: lightgrey; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-modifier-on { | ||||
| 	background-color: lightblue; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-backspace { | ||||
| 	width: 82px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-tab { | ||||
| 	width: 55px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-enter { | ||||
| 	width: 69px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-capslock { | ||||
| 	width: 80px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-backslash { | ||||
| 	width: 88px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-lshift { | ||||
| 	width: 65px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-rshift { | ||||
| 	width: 103px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-lctrl { | ||||
| 	width: 55px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-super { | ||||
| 	width: 55px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-alt { | ||||
| 	width: 55px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-altgr { | ||||
| 	width: 55px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-what { | ||||
| 	width: 55px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-rctrl { | ||||
| 	width: 55px; | ||||
| } | ||||
|  | ||||
| .keybaord-key.keybaord-spacebar { | ||||
| 	width: 302px; | ||||
| } | ||||
| `; | ||||
|     document.head.appendChild(o); | ||||
|  | ||||
|     initKeybaord('/', hambagas, | ||||
|         (msg) => { toast.inf(2, msg.toString()) }, | ||||
|         (msg) => { toast.err(30, msg.toString()) }); | ||||
| })(); | ||||
|  | ||||
|  | ||||
| // live2d (dumb pointless meme) | ||||
| //   dependencies for this part are not tracked in git | ||||
| //   so delete this section if you wanna use this file | ||||
| //   (or supply your own l2d model and js) | ||||
| (function () { | ||||
|     var o = mknod('link'); | ||||
|     o.setAttribute('rel', 'stylesheet'); | ||||
|     o.setAttribute('href', "/bad-memes/pio.css"); | ||||
|     document.head.appendChild(o); | ||||
|  | ||||
|     o = mknod('style'); | ||||
|     o.innerHTML = '.pio-container{text-shadow:none;z-index:1}'; | ||||
|     document.head.appendChild(o); | ||||
|  | ||||
|     o = mknod('div'); | ||||
|     clmod(o, 'pio-container', 1); | ||||
|     o.innerHTML = '<div class="pio-action"></div><canvas id="pio" width="280" height="500"></canvas>'; | ||||
|     document.body.appendChild(o); | ||||
|  | ||||
|     var remaining = 3; | ||||
|     for (var a of ['pio', 'l2d', 'fireworks']) { | ||||
|         import_js(`/bad-memes/${a}.js`, function () { | ||||
|             if (remaining --> 1) | ||||
|                 return; | ||||
|  | ||||
|             o = mknod('script'); | ||||
|             o.innerHTML = 'var pio = new Paul_Pio({"selector":[],"mode":"fixed","hidden":false,"content":{"close":"ok bye"},"model":["/bad-memes/sagiri/model.json"]});'; | ||||
|             document.body.appendChild(o); | ||||
|         }); | ||||
|     } | ||||
| })(); | ||||
							
								
								
									
										37
									
								
								contrib/plugins/minimal-up2k.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								contrib/plugins/minimal-up2k.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| <!-- | ||||
|   save this as .epilogue.html inside a write-only folder to declutter the UI,  makes it look like | ||||
|   https://user-images.githubusercontent.com/241032/118311195-dd6ca380-b4ef-11eb-86f3-75a3ff2e1332.png | ||||
| --> | ||||
|  | ||||
| <style> | ||||
|  | ||||
|     /* make the up2k ui REALLY minimal by hiding a bunch of stuff: */ | ||||
|  | ||||
|     #ops, #tree, #path, #epi+h2,  /* main tabs and navigators (tree/breadcrumbs) */ | ||||
|  | ||||
|     #u2conf tr:first-child>td[rowspan]:not(#u2btn_cw),  /* most of the config options */ | ||||
|  | ||||
|     #srch_dz, #srch_zd,  /* the filesearch dropzone */ | ||||
|  | ||||
|     #u2cards, #u2etaw  /* and the upload progress tabs */ | ||||
|  | ||||
|     {display: none !important}  /* do it! */ | ||||
|  | ||||
|  | ||||
|  | ||||
|     /* add some margins because now it's weird */ | ||||
|     .opview {margin-top: 2.5em} | ||||
|     #op_up2k {margin-top: 6em} | ||||
|  | ||||
|     /* and embiggen the upload button */ | ||||
|     #u2conf #u2btn, #u2btn {padding:1.5em 0} | ||||
|  | ||||
|     /* adjust the button area a bit */ | ||||
|     #u2conf.w, #u2conf.ww {width: 35em !important; margin: 5em auto} | ||||
|  | ||||
|     /* a */ | ||||
|     #op_up2k {min-height: 0} | ||||
|  | ||||
| </style> | ||||
|  | ||||
| <a href="#" onclick="this.parentNode.innerHTML='';">show advanced options</a> | ||||
							
								
								
									
										59
									
								
								contrib/plugins/minimal-up2k.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								contrib/plugins/minimal-up2k.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| /* | ||||
|  | ||||
| makes the up2k ui REALLY minimal by hiding a bunch of stuff | ||||
|  | ||||
| almost the same as minimal-up2k.html except this one...: | ||||
|  | ||||
|  -- applies to every write-only folder when used with --js-browser | ||||
|  | ||||
|  -- only applies if javascript is enabled | ||||
|  | ||||
|  -- doesn't hide the total upload ETA display | ||||
|  | ||||
|  -- looks slightly better | ||||
|  | ||||
| */ | ||||
|  | ||||
| var u2min = ` | ||||
| <style> | ||||
|  | ||||
| #ops, #path, #tree, #files, #epi+div+h2, | ||||
| #u2conf td.c+.c, #u2cards, #srch_dz, #srch_zd { | ||||
|   display: none !important; | ||||
| } | ||||
| #u2conf {margin:5em auto 0 auto !important} | ||||
| #u2conf.ww {width:70em} | ||||
| #u2conf.w {width:50em} | ||||
| #u2conf.w .c, | ||||
| #u2conf.w #u2btn_cw {text-align:left} | ||||
| #u2conf.w #u2btn_cw {width:70%} | ||||
| #u2etaw {margin:3em auto} | ||||
| #u2etaw.w { | ||||
|   text-align: center; | ||||
|   margin: -3.5em auto 5em auto; | ||||
| } | ||||
| #u2etaw.w #u2etas {margin-right:-37em} | ||||
| #u2etaw.w #u2etas.o {margin-top:-2.2em} | ||||
| #u2etaw.ww {margin:-1em auto} | ||||
| #u2etaw.ww #u2etas {padding-left:4em} | ||||
| #u2etas { | ||||
|   background: none !important; | ||||
|   border: none !important; | ||||
| } | ||||
| #wrap {margin-left:2em !important} | ||||
| .logue { | ||||
|   border: none !important; | ||||
|   margin: 2em auto !important; | ||||
| } | ||||
| .logue:before {content:'' !important} | ||||
|  | ||||
| </style> | ||||
|  | ||||
| <a href="#" onclick="this.parentNode.innerHTML='';">show advanced options</a> | ||||
| `; | ||||
|  | ||||
| if (!has(perms, 'read')) { | ||||
|   var e2 = mknod('div'); | ||||
|   e2.innerHTML = u2min; | ||||
|   ebi('wrap').insertBefore(e2, QS('#epi+h2')); | ||||
| } | ||||
							
								
								
									
										209
									
								
								contrib/plugins/up2k-hook-ytid.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										209
									
								
								contrib/plugins/up2k-hook-ytid.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,209 @@ | ||||
| // way more specific example -- | ||||
| // assumes all files dropped into the uploader have a youtube-id somewhere in the filename, | ||||
| // locates the youtube-ids and passes them to an API which returns a list of IDs which should be uploaded | ||||
| // | ||||
| // also tries to find the youtube-id in the embedded metadata | ||||
| // | ||||
| // assumes copyparty is behind nginx as /ytq is a standalone service which must be rproxied in place | ||||
|  | ||||
| function up2k_namefilter(good_files, nil_files, bad_files, hooks) { | ||||
|     var passthru = up2k.uc.fsearch; | ||||
|     if (passthru) | ||||
|         return hooks[0](good_files, nil_files, bad_files, hooks.slice(1)); | ||||
|  | ||||
|     a_up2k_namefilter(good_files, nil_files, bad_files, hooks).then(() => { }); | ||||
| } | ||||
|  | ||||
| function bstrpos(buf, ptn) { | ||||
|     var ofs = 0, | ||||
|         ch0 = ptn[0], | ||||
|         sz = buf.byteLength; | ||||
|  | ||||
|     while (true) { | ||||
|         ofs = buf.indexOf(ch0, ofs); | ||||
|         if (ofs < 0 || ofs >= sz) | ||||
|             return -1; | ||||
|  | ||||
|         for (var a = 1; a < ptn.length; a++) | ||||
|             if (buf[ofs + a] !== ptn[a]) | ||||
|                 break; | ||||
|  | ||||
|         if (a === ptn.length) | ||||
|             return ofs; | ||||
|  | ||||
|         ++ofs; | ||||
|     } | ||||
| } | ||||
|  | ||||
| async function a_up2k_namefilter(good_files, nil_files, bad_files, hooks) { | ||||
|     var t0 = Date.now(), | ||||
|         yt_ids = new Set(), | ||||
|         textdec = new TextDecoder('latin1'), | ||||
|         md_ptn = new TextEncoder().encode('youtube.com/watch?v='), | ||||
|         file_ids = [],  // all IDs found for each good_files | ||||
|         mofs = 0, | ||||
|         mnchk = 0, | ||||
|         mfile = ''; | ||||
|  | ||||
|     for (var a = 0; a < good_files.length; a++) { | ||||
|         var [fobj, name] = good_files[a], | ||||
|             sz = fobj.size, | ||||
|             ids = [], | ||||
|             id_ok = false, | ||||
|             m; | ||||
|  | ||||
|         // all IDs found in this file | ||||
|         file_ids.push(ids); | ||||
|  | ||||
|         // look for ID in filename; reduce the | ||||
|         // metadata-scan intensity if the id looks safe | ||||
|         m = /[\[(-]([\w-]{11})[\])]?\.(?:mp4|webm|mkv)$/i.exec(name); | ||||
|         id_ok = !!m; | ||||
|  | ||||
|         while (true) { | ||||
|             // fuzzy catch-all; | ||||
|             // some ytdl fork did %(title)-%(id).%(ext) ... | ||||
|             m = /(?:^|[^\w])([\w-]{11})(?:$|[^\w-])/.exec(name); | ||||
|             if (!m) | ||||
|                 break; | ||||
|  | ||||
|             name = name.replace(m[1], ''); | ||||
|             yt_ids.add(m[1]); | ||||
|             ids.push(m[1]); | ||||
|         } | ||||
|  | ||||
|         // look for IDs in video metadata, | ||||
|         if (/\.(mp4|webm|mkv)$/i.exec(name)) { | ||||
|             toast.show('inf r', 0, `analyzing file ${a + 1} / ${good_files.length} :\n${name}\n\nhave analysed ${++mnchk} files in ${(Date.now() - t0) / 1000} seconds, ${humantime((good_files.length - (a + 1)) * (((Date.now() - t0) / 1000) / mnchk))} remaining,\n\nbiggest offset so far is ${mofs}, in this file:\n\n${mfile}`); | ||||
|  | ||||
|             // check first and last 128 MiB; | ||||
|             // pWxOroN5WCo.mkv @  6edb98 (6.92M) | ||||
|             // Nf-nN1wF5Xo.mp4 @ 4a98034 (74.6M) | ||||
|             var chunksz = 1024 * 1024 * 2,  // byte | ||||
|                 aspan = id_ok ? 128 : 512;  // MiB | ||||
|  | ||||
|             aspan = parseInt(Math.min(sz / 2, aspan * 1024 * 1024) / chunksz) * chunksz; | ||||
|  | ||||
|             for (var side = 0; side < 2; side++) { | ||||
|                 var ofs = side ? Math.max(0, sz - aspan) : 0, | ||||
|                     nchunks = aspan / chunksz; | ||||
|  | ||||
|                 for (var chunk = 0; chunk < nchunks; chunk++) { | ||||
|                     var bchunk = await fobj.slice(ofs, ofs + chunksz + 16).arrayBuffer(), | ||||
|                         uchunk = new Uint8Array(bchunk, 0, bchunk.byteLength), | ||||
|                         bofs = bstrpos(uchunk, md_ptn), | ||||
|                         absofs = Math.min(ofs + bofs, (sz - ofs) + bofs), | ||||
|                         txt = bofs < 0 ? '' : textdec.decode(uchunk.subarray(bofs)), | ||||
|                         m; | ||||
|  | ||||
|                     //console.log(`side ${ side }, chunk ${ chunk }, ofs ${ ofs }, bchunk ${ bchunk.byteLength }, txt ${ txt.length }`); | ||||
|                     while (true) { | ||||
|                         // mkv/webm have [a-z] immediately after url | ||||
|                         m = /(youtube\.com\/watch\?v=[\w-]{11})/.exec(txt); | ||||
|                         if (!m) | ||||
|                             break; | ||||
|  | ||||
|                         txt = txt.replace(m[1], ''); | ||||
|                         m = m[1].slice(-11); | ||||
|  | ||||
|                         console.log(`found ${m} @${bofs}, ${name} `); | ||||
|                         yt_ids.add(m); | ||||
|                         if (!has(ids, m)) | ||||
|                             ids.push(m); | ||||
|  | ||||
|                         // bail after next iteration | ||||
|                         chunk = nchunks - 1; | ||||
|                         side = 9; | ||||
|  | ||||
|                         if (mofs < absofs) { | ||||
|                             mofs = absofs; | ||||
|                             mfile = name; | ||||
|                         } | ||||
|                     } | ||||
|                     ofs += chunksz; | ||||
|                     if (ofs >= sz) | ||||
|                         break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (false) { | ||||
|         var msg = `finished analysing ${mnchk} files in ${(Date.now() - t0) / 1000} seconds,\n\nbiggest offset was ${mofs} in this file:\n\n${mfile}`, | ||||
|             mfun = function () { toast.ok(0, msg); }; | ||||
|  | ||||
|         mfun(); | ||||
|         setTimeout(mfun, 200); | ||||
|  | ||||
|         return hooks[0]([], [], [], hooks.slice(1)); | ||||
|     } | ||||
|  | ||||
|     toast.inf(5, `running query for ${yt_ids.size} videos...`); | ||||
|  | ||||
|     var xhr = new XHR(); | ||||
|     xhr.open('POST', '/ytq', true); | ||||
|     xhr.setRequestHeader('Content-Type', 'text/plain'); | ||||
|     xhr.onload = xhr.onerror = function () { | ||||
|         if (this.status != 200) | ||||
|             return toast.err(0, `sorry, database query failed ;_;\n\nplease let us know so we can look at it, thx!!\n\nerror ${this.status}: ${(this.response && this.response.err) || this.responseText}`); | ||||
|  | ||||
|         process_id_list(this.responseText); | ||||
|     }; | ||||
|     xhr.send(Array.from(yt_ids).join('\n')); | ||||
|  | ||||
|     function process_id_list(txt) { | ||||
|         var wanted_ids = new Set(txt.trim().split('\n')), | ||||
|             wanted_names = new Set(),  // basenames with a wanted ID | ||||
|             wanted_files = new Set();  // filedrops | ||||
|  | ||||
|         for (var a = 0; a < good_files.length; a++) { | ||||
|             var name = good_files[a][1]; | ||||
|             for (var b = 0; b < file_ids[a].length; b++) | ||||
|                 if (wanted_ids.has(file_ids[a][b])) { | ||||
|                     wanted_files.add(good_files[a]); | ||||
|  | ||||
|                     var m = /(.*)\.(mp4|webm|mkv)$/i.exec(name); | ||||
|                     if (m) | ||||
|                         wanted_names.add(m[1]); | ||||
|  | ||||
|                     break; | ||||
|                 } | ||||
|         } | ||||
|  | ||||
|         // add all files with the same basename as each explicitly wanted file | ||||
|         // (infojson/chatlog/etc when ID was discovered from metadata) | ||||
|         for (var a = 0; a < good_files.length; a++) { | ||||
|             var name = good_files[a][1]; | ||||
|             for (var b = 0; b < 3; b++) { | ||||
|                 name = name.replace(/\.[^\.]+$/, ''); | ||||
|                 if (wanted_names.has(name)) { | ||||
|                     wanted_files.add(good_files[a]); | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         function upload_filtered() { | ||||
|             if (!wanted_files.size) | ||||
|                 return modal.alert('Good news -- turns out we already have all those.\n\nBut thank you for checking in!'); | ||||
|  | ||||
|             hooks[0](Array.from(wanted_files), nil_files, bad_files, hooks.slice(1)); | ||||
|         } | ||||
|  | ||||
|         function upload_all() { | ||||
|             hooks[0](good_files, nil_files, bad_files, hooks.slice(1)); | ||||
|         } | ||||
|  | ||||
|         var n_skip = good_files.length - wanted_files.size, | ||||
|             msg = `you added ${good_files.length} files; ${good_files.length == n_skip ? 'all' : n_skip} of them were skipped --\neither because we already have them,\nor because there is no youtube-ID in your filenames.\n\n<code>OK</code> / <code>Enter</code> = continue uploading just the ${wanted_files.size} files we definitely need\n\n<code>Cancel</code> / <code>ESC</code> = override the filter; upload ALL the files you added`; | ||||
|  | ||||
|         if (!n_skip) | ||||
|             upload_filtered(); | ||||
|         else | ||||
|             modal.confirm(msg, upload_filtered, upload_all); | ||||
|     }; | ||||
| } | ||||
|  | ||||
| up2k_hooks.push(function () { | ||||
|     up2k.gotallfiles.unshift(up2k_namefilter); | ||||
| }); | ||||
							
								
								
									
										45
									
								
								contrib/plugins/up2k-hooks.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								contrib/plugins/up2k-hooks.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| // hooks into up2k | ||||
|  | ||||
| function up2k_namefilter(good_files, nil_files, bad_files, hooks) { | ||||
|     // is called when stuff is dropped into the browser, | ||||
|     // after iterating through the directory tree and discovering all files, | ||||
|     // before the upload confirmation dialogue is shown | ||||
|  | ||||
|     // good_files will successfully upload | ||||
|     // nil_files are empty files and will show an alert in the final hook | ||||
|     // bad_files are unreadable and cannot be uploaded | ||||
|     var file_lists = [good_files, nil_files, bad_files]; | ||||
|  | ||||
|     // build a list of filenames | ||||
|     var filenames = []; | ||||
|     for (var lst of file_lists) | ||||
|         for (var ent of lst) | ||||
|             filenames.push(ent[1]); | ||||
|  | ||||
|     toast.inf(5, "running database query..."); | ||||
|  | ||||
|     // simulate delay while passing the list to some api for checking | ||||
|     setTimeout(function () { | ||||
|  | ||||
|         // only keep webm files as an example | ||||
|         var new_lists = []; | ||||
|         for (var lst of file_lists) { | ||||
|             var keep = []; | ||||
|             new_lists.push(keep); | ||||
|  | ||||
|             for (var ent of lst) | ||||
|                 if (/\.webm$/.test(ent[1])) | ||||
|                     keep.push(ent); | ||||
|         } | ||||
|  | ||||
|         // finally, call the next hook in the chain | ||||
|         [good_files, nil_files, bad_files] = new_lists; | ||||
|         hooks[0](good_files, nil_files, bad_files, hooks.slice(1)); | ||||
|  | ||||
|     }, 1000); | ||||
| } | ||||
|  | ||||
| // register | ||||
| up2k_hooks.push(function () { | ||||
|     up2k.gotallfiles.unshift(up2k_namefilter); | ||||
| }); | ||||
							
								
								
									
										31
									
								
								contrib/rc/copyparty
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								contrib/rc/copyparty
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| #!/bin/sh | ||||
| # | ||||
| # PROVIDE: copyparty | ||||
| # REQUIRE: networking | ||||
| # KEYWORD: | ||||
|  | ||||
| . /etc/rc.subr | ||||
|  | ||||
| name="copyparty" | ||||
| rcvar="copyparty_enable" | ||||
| copyparty_user="copyparty" | ||||
| copyparty_args="-e2dsa -v /storage:/storage:r" # change as you see fit | ||||
| copyparty_command="/usr/local/bin/python3.8 /usr/local/copyparty/copyparty-sfx.py ${copyparty_args}" | ||||
| pidfile="/var/run/copyparty/${name}.pid" | ||||
| command="/usr/sbin/daemon" | ||||
| command_args="-P ${pidfile} -r -f ${copyparty_command}" | ||||
|  | ||||
| stop_postcmd="copyparty_shutdown" | ||||
|  | ||||
| copyparty_shutdown() | ||||
| { | ||||
|         if [ -e "${pidfile}" ]; then | ||||
|                 echo "Stopping supervising daemon." | ||||
|                 kill -s TERM `cat ${pidfile}` | ||||
|         fi | ||||
| } | ||||
|  | ||||
| load_rc_config $name | ||||
| : ${copyparty_enable:=no} | ||||
|  | ||||
| run_rc_command "$1" | ||||
							
								
								
									
										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$" | ||||
| } | ||||
							
								
								
									
										23
									
								
								contrib/systemd/cfssl.service
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								contrib/systemd/cfssl.service
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| # systemd service which generates a new TLS certificate on each boot, | ||||
| # that way the one-year expiry time won't cause any issues -- | ||||
| # just have everyone trust the ca.pem once every 10 years | ||||
| # | ||||
| # assumptions/placeholder values: | ||||
| #  * this script and copyparty runs as user "cpp" | ||||
| #  * copyparty repo is at ~cpp/dev/copyparty | ||||
| #  * CA is named partylan | ||||
| #  * server IPs = 10.1.2.3 and 192.168.123.1 | ||||
| #  * server hostname = party.lan | ||||
|  | ||||
| [Unit] | ||||
| Description=copyparty certificate generator | ||||
| Before=copyparty.service | ||||
|  | ||||
| [Service] | ||||
| User=cpp | ||||
| Type=oneshot | ||||
| SyslogIdentifier=cpp-cert | ||||
| ExecStart=/bin/bash -c 'cd ~/dev/copyparty/contrib && ./cfssl.sh partylan 10.1.2.3,192.168.123.1,party.lan y' | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
							
								
								
									
										61
									
								
								contrib/systemd/copyparty.service
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								contrib/systemd/copyparty.service
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| # this will start `/usr/local/bin/copyparty-sfx.py` | ||||
| # and share '/mnt' with anonymous read+write | ||||
| # | ||||
| # installation: | ||||
| #   cp -pv copyparty.service /etc/systemd/system | ||||
| #   restorecon -vr /etc/systemd/system/copyparty.service | ||||
| #   firewall-cmd --permanent --add-port={80,443,3923}/tcp  # --zone=libvirt | ||||
| #   firewall-cmd --reload | ||||
| #   systemctl daemon-reload && systemctl enable --now copyparty | ||||
| # | ||||
| # you may want to: | ||||
| #   change "User=cpp" and "/home/cpp/" to another user | ||||
| #   remove the nft lines to only listen on port 3923 | ||||
| # and in the ExecStart= line: | ||||
| #   change '/usr/bin/python3' to another interpreter | ||||
| #   change '/mnt::rw' to another location or permission-set | ||||
| #   add '-q' to disable logging on busy servers | ||||
| #   add '-i 127.0.0.1' to only allow local connections | ||||
| #   add '-e2dsa' to enable filesystem scanning + indexing | ||||
| #   add '-e2ts' to enable metadata indexing | ||||
| # | ||||
| # with `Type=notify`, copyparty will signal systemd when it is ready to | ||||
| #   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 | ||||
| # | ||||
| # unless you add -q to disable logging, you may want to remove the | ||||
| #   following line to allow buffering (slightly better performance): | ||||
| #   Environment=PYTHONUNBUFFERED=x | ||||
| # | ||||
| # keep ExecStartPre before ExecStart, at least on rhel8 | ||||
|  | ||||
| [Unit] | ||||
| Description=copyparty file server | ||||
|  | ||||
| [Service] | ||||
| Type=notify | ||||
| SyslogIdentifier=copyparty | ||||
| Environment=PYTHONUNBUFFERED=x | ||||
| ExecReload=/bin/kill -s USR1 $MAINPID | ||||
|  | ||||
| # user to run as + where the TLS certificate is (if any) | ||||
| User=cpp | ||||
| Environment=XDG_CONFIG_HOME=/home/cpp/.config | ||||
|  | ||||
| # setup forwarding from ports 80 and 443 to port 3923 | ||||
| ExecStartPre=+/bin/bash -c 'nft -n -a list table nat | awk "/ to :3923 /{print\$NF}" | xargs -rL1 nft delete rule nat prerouting handle; true' | ||||
| ExecStartPre=+nft add table ip nat | ||||
| ExecStartPre=+nft -- add chain ip nat prerouting { type nat hook prerouting priority -100 \; } | ||||
| ExecStartPre=+nft add rule ip nat prerouting tcp dport 80 redirect to :3923 | ||||
| ExecStartPre=+nft add rule ip nat prerouting tcp dport 443 redirect to :3923 | ||||
|  | ||||
| # stop systemd-tmpfiles-clean.timer from deleting copyparty while it's running | ||||
| ExecStartPre=+/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf' | ||||
|  | ||||
| # copyparty settings | ||||
| ExecStart=/usr/bin/python3 /usr/local/bin/copyparty-sfx.py -e2d -v /mnt::rw | ||||
|  | ||||
| [Install] | ||||
| 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 | ||||
| @@ -1,36 +1,82 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import platform | ||||
| import sys | ||||
| import os | ||||
| import time | ||||
|  | ||||
| try: | ||||
|     from collections.abc import Callable | ||||
|  | ||||
|     from typing import TYPE_CHECKING, Any | ||||
| except: | ||||
|     TYPE_CHECKING = False | ||||
|  | ||||
| PY2 = sys.version_info[0] == 2 | ||||
| if PY2: | ||||
|     sys.dont_write_bytecode = True | ||||
|     unicode = unicode  # noqa: F821  # pylint: disable=undefined-variable,self-assigning-variable | ||||
| else: | ||||
|     unicode = str | ||||
|  | ||||
| WINDOWS = False | ||||
| if platform.system() == "Windows": | ||||
|     WINDOWS = [int(x) for x in platform.version().split(".")] | ||||
| WINDOWS: Any = ( | ||||
|     [int(x) for x in platform.version().split(".")] | ||||
|     if platform.system() == "Windows" | ||||
|     else False | ||||
| ) | ||||
|  | ||||
| VT100 = not WINDOWS or WINDOWS >= [10, 0, 14393] | ||||
| # introduced in anniversary update | ||||
|  | ||||
| ANYWIN = WINDOWS or sys.platform in ["msys", "cygwin"] | ||||
|  | ||||
| MACOS = platform.system() == "Darwin" | ||||
|  | ||||
|  | ||||
| def get_unixdir() -> str: | ||||
|     paths: list[tuple[Callable[..., str], str]] = [ | ||||
|         (os.environ.get, "XDG_CONFIG_HOME"), | ||||
|         (os.path.expanduser, "~/.config"), | ||||
|         (os.environ.get, "TMPDIR"), | ||||
|         (os.environ.get, "TEMP"), | ||||
|         (os.environ.get, "TMP"), | ||||
|         (unicode, "/tmp"), | ||||
|     ] | ||||
|     for chk in [os.listdir, os.mkdir]: | ||||
|         for pf, pa in paths: | ||||
|             try: | ||||
|                 p = pf(pa) | ||||
|                 # print(chk.__name__, p, pa) | ||||
|                 if not p or p.startswith("~"): | ||||
|                     continue | ||||
|  | ||||
|                 p = os.path.normpath(p) | ||||
|                 chk(p)  # type: ignore | ||||
|                 p = os.path.join(p, "copyparty") | ||||
|                 if not os.path.isdir(p): | ||||
|                     os.mkdir(p) | ||||
|  | ||||
|                 return p | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|     raise Exception("could not find a writable path for config") | ||||
|  | ||||
|  | ||||
| class EnvParams(object): | ||||
|     def __init__(self): | ||||
|     def __init__(self) -> None: | ||||
|         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_unixdir() | ||||
|  | ||||
|         self.cfg = self.cfg.replace("\\", "/") | ||||
|         try: | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| #!/usr/bin/env python | ||||
| #!/usr/bin/env python3 | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| @@ -8,21 +8,59 @@ __copyright__ = 2019 | ||||
| __license__ = "MIT" | ||||
| __url__ = "https://github.com/9001/copyparty/" | ||||
|  | ||||
| import os | ||||
| import shutil | ||||
| import argparse | ||||
| import filecmp | ||||
| import locale | ||||
| import argparse | ||||
| import os | ||||
| import re | ||||
| import shutil | ||||
| import sys | ||||
| import threading | ||||
| import time | ||||
| import traceback | ||||
| from textwrap import dedent | ||||
|  | ||||
| from .__init__ import E, WINDOWS, VT100 | ||||
| from .__version__ import S_VERSION, S_BUILD_DT, CODENAME | ||||
| from .__init__ import ANYWIN, PY2, VT100, WINDOWS, E, unicode | ||||
| from .__version__ import CODENAME, S_BUILD_DT, S_VERSION | ||||
| from .authsrv import re_vol | ||||
| from .svchub import SvcHub | ||||
| from .util import py_desc | ||||
| from .util import ( | ||||
|     IMPLICATIONS, | ||||
|     JINJA_VER, | ||||
|     PYFTPD_VER, | ||||
|     SQLITE_VER, | ||||
|     align_tab, | ||||
|     ansi_re, | ||||
|     min_ex, | ||||
|     py_desc, | ||||
|     termsize, | ||||
|     wrap, | ||||
| ) | ||||
|  | ||||
| try: | ||||
|     from types import FrameType | ||||
|  | ||||
|     from typing import Any, Optional | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| try: | ||||
|     HAVE_SSL = True | ||||
|     import ssl | ||||
| except: | ||||
|     HAVE_SSL = False | ||||
|  | ||||
| printed: list[str] = [] | ||||
|  | ||||
|  | ||||
| class RiceFormatter(argparse.HelpFormatter): | ||||
|     def _get_help_string(self, action): | ||||
|     def __init__(self, *args: Any, **kwargs: Any) -> None: | ||||
|         if PY2: | ||||
|             kwargs["width"] = termsize()[0] | ||||
|  | ||||
|         super(RiceFormatter, self).__init__(*args, **kwargs) | ||||
|  | ||||
|     def _get_help_string(self, action: argparse.Action) -> str: | ||||
|         """ | ||||
|         same as ArgumentDefaultsHelpFormatter(HelpFormatter) | ||||
|         except the help += [...] line now has colors | ||||
| @@ -31,20 +69,68 @@ class RiceFormatter(argparse.HelpFormatter): | ||||
|         if not VT100: | ||||
|             fmt = " (default: %(default)s)" | ||||
|  | ||||
|         help = action.help | ||||
|         if "%(default)" not in action.help: | ||||
|         ret = unicode(action.help) | ||||
|         if "%(default)" not in ret: | ||||
|             if action.default is not argparse.SUPPRESS: | ||||
|                 defaulting_nargs = [argparse.OPTIONAL, argparse.ZERO_OR_MORE] | ||||
|                 if action.option_strings or action.nargs in defaulting_nargs: | ||||
|                     help += fmt | ||||
|         return help | ||||
|                     ret += fmt | ||||
|         return ret | ||||
|  | ||||
|     def _fill_text(self, text, width, indent): | ||||
|     def _fill_text(self, text: str, width: int, indent: str) -> str: | ||||
|         """same as RawDescriptionHelpFormatter(HelpFormatter)""" | ||||
|         return "".join(indent + line + "\n" for line in text.splitlines()) | ||||
|  | ||||
|     def __add_whitespace(self, idx: int, iWSpace: int, text: str) -> str: | ||||
|         return (" " * iWSpace) + text if idx else text | ||||
|  | ||||
| def ensure_locale(): | ||||
|     def _split_lines(self, text: str, width: int) -> list[str]: | ||||
|         # https://stackoverflow.com/a/35925919 | ||||
|         textRows = text.splitlines() | ||||
|         ptn = re.compile(r"\s*[0-9\-]{0,}\.?\s*") | ||||
|         for idx, line in enumerate(textRows): | ||||
|             search = ptn.search(line) | ||||
|             if not line.strip(): | ||||
|                 textRows[idx] = " " | ||||
|             elif search: | ||||
|                 lWSpace = search.end() | ||||
|                 lines = [ | ||||
|                     self.__add_whitespace(i, lWSpace, x) | ||||
|                     for i, x in enumerate(wrap(line, width, width - 1)) | ||||
|                 ] | ||||
|                 textRows[idx] = lines | ||||
|  | ||||
|         return [item for sublist in textRows for item in sublist] | ||||
|  | ||||
|  | ||||
| class Dodge11874(RiceFormatter): | ||||
|     def __init__(self, *args: Any, **kwargs: Any) -> None: | ||||
|         kwargs["width"] = 9003 | ||||
|         super(Dodge11874, self).__init__(*args, **kwargs) | ||||
|  | ||||
|  | ||||
| class BasicDodge11874( | ||||
|     argparse.ArgumentDefaultsHelpFormatter, argparse.RawDescriptionHelpFormatter | ||||
| ): | ||||
|     def __init__(self, *args: Any, **kwargs: Any) -> None: | ||||
|         kwargs["width"] = 9003 | ||||
|         super(BasicDodge11874, self).__init__(*args, **kwargs) | ||||
|  | ||||
|  | ||||
| def lprint(*a: Any, **ka: Any) -> None: | ||||
|     txt: str = " ".join(unicode(x) for x in a) + ka.get("end", "\n") | ||||
|     printed.append(txt) | ||||
|     if not VT100: | ||||
|         txt = ansi_re.sub("", txt) | ||||
|  | ||||
|     print(txt, **ka) | ||||
|  | ||||
|  | ||||
| def warn(msg: str) -> None: | ||||
|     lprint("\033[1mwarning:\033[0;33m {}\033[0m\n".format(msg)) | ||||
|  | ||||
|  | ||||
| def ensure_locale() -> None: | ||||
|     for x in [ | ||||
|         "en_US.UTF-8", | ||||
|         "English_United States.UTF8", | ||||
| @@ -52,13 +138,13 @@ def ensure_locale(): | ||||
|     ]: | ||||
|         try: | ||||
|             locale.setlocale(locale.LC_ALL, x) | ||||
|             print("Locale:", x) | ||||
|             lprint("Locale:", x) | ||||
|             break | ||||
|         except: | ||||
|             continue | ||||
|  | ||||
|  | ||||
| def ensure_cert(): | ||||
| def ensure_cert() -> None: | ||||
|     """ | ||||
|     the default cert (and the entire TLS support) is only here to enable the | ||||
|     crypto.subtle javascript API, which is necessary due to the webkit guys | ||||
| @@ -69,11 +155,11 @@ def ensure_cert(): | ||||
|     cert_insec = os.path.join(E.mod, "res/insecure.pem") | ||||
|     cert_cfg = os.path.join(E.cfg, "cert.pem") | ||||
|     if not os.path.exists(cert_cfg): | ||||
|         shutil.copy2(cert_insec, cert_cfg) | ||||
|         shutil.copy(cert_insec, cert_cfg) | ||||
|  | ||||
|     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,64 +170,649 @@ 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: argparse.Namespace) -> None: | ||||
|     def terse_sslver(txt: str) -> str: | ||||
|         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: | ||||
|         avail1 = [terse_sslver(x[6:]) for x in flags] | ||||
|         avail = " ".join(sorted(avail1) + ["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("{0}: {1:8x} ({1})".format(k, num)) | ||||
|  | ||||
|     # think i need that beer now | ||||
|  | ||||
|  | ||||
| def configure_ssl_ciphers(al: argparse.Namespace) -> None: | ||||
|     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 args_from_cfg(cfg_path: str) -> list[str]: | ||||
|     ret: list[str] = [] | ||||
|     skip = False | ||||
|     with open(cfg_path, "rb") as f: | ||||
|         for ln in [x.decode("utf-8").strip() for x in f]: | ||||
|             if not ln: | ||||
|                 skip = False | ||||
|                 continue | ||||
|  | ||||
|             if ln.startswith("#"): | ||||
|                 continue | ||||
|  | ||||
|             if not ln.startswith("-"): | ||||
|                 continue | ||||
|  | ||||
|             if skip: | ||||
|                 continue | ||||
|  | ||||
|             try: | ||||
|                 ret.extend(ln.split(" ", 1)) | ||||
|             except: | ||||
|                 ret.append(ln) | ||||
|  | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def sighandler(sig: Optional[int] = None, frame: Optional[FrameType] = None) -> None: | ||||
|     msg = [""] * 5 | ||||
|     for th in threading.enumerate(): | ||||
|         stk = sys._current_frames()[th.ident]  # type: ignore | ||||
|         msg.append(str(th)) | ||||
|         msg.extend(traceback.format_stack(stk)) | ||||
|  | ||||
|     msg.append("\n") | ||||
|     print("\n".join(msg)) | ||||
|  | ||||
|  | ||||
| def disable_quickedit() -> None: | ||||
|     import atexit | ||||
|     import ctypes | ||||
|     from ctypes import wintypes | ||||
|  | ||||
|     def ecb(ok: bool, fun: Any, args: list[Any]) -> list[Any]: | ||||
|         if not ok: | ||||
|             err: int = ctypes.get_last_error()  # type: ignore | ||||
|             if err: | ||||
|                 raise ctypes.WinError(err)  # type: ignore | ||||
|         return args | ||||
|  | ||||
|     k32 = ctypes.WinDLL(str("kernel32"), use_last_error=True)  # type: ignore | ||||
|     if PY2: | ||||
|         wintypes.LPDWORD = ctypes.POINTER(wintypes.DWORD) | ||||
|  | ||||
|     k32.GetStdHandle.errcheck = ecb | ||||
|     k32.GetConsoleMode.errcheck = ecb | ||||
|     k32.SetConsoleMode.errcheck = ecb | ||||
|     k32.GetConsoleMode.argtypes = (wintypes.HANDLE, wintypes.LPDWORD) | ||||
|     k32.SetConsoleMode.argtypes = (wintypes.HANDLE, wintypes.DWORD) | ||||
|  | ||||
|     def cmode(out: bool, mode: Optional[int] = None) -> int: | ||||
|         h = k32.GetStdHandle(-11 if out else -10) | ||||
|         if mode: | ||||
|             return k32.SetConsoleMode(h, mode)  # type: ignore | ||||
|  | ||||
|         cmode = wintypes.DWORD() | ||||
|         k32.GetConsoleMode(h, ctypes.byref(cmode)) | ||||
|         return cmode.value | ||||
|  | ||||
|     # disable quickedit | ||||
|     mode = orig_in = cmode(False) | ||||
|     quickedit = 0x40 | ||||
|     extended = 0x80 | ||||
|     mask = quickedit + extended | ||||
|     if mode & mask != extended: | ||||
|         atexit.register(cmode, False, orig_in) | ||||
|         cmode(False, mode & ~mask | extended) | ||||
|  | ||||
|     # enable colors in case the os.system("rem") trick ever stops working | ||||
|     if VT100: | ||||
|         mode = orig_out = cmode(True) | ||||
|         if mode & 4 != 4: | ||||
|             atexit.register(cmode, True, orig_out) | ||||
|             cmode(True, mode | 4) | ||||
|  | ||||
|  | ||||
| def run_argparse(argv: list[str], formatter: Any, retry: bool) -> argparse.Namespace: | ||||
|     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( | ||||
|             """ | ||||
|     ) | ||||
|  | ||||
|     try: | ||||
|         fk_salt = unicode(os.path.getmtime(os.path.join(E.cfg, "cert.pem"))) | ||||
|     except: | ||||
|         fk_salt = "hunter2" | ||||
|  | ||||
|     cores = os.cpu_count() if hasattr(os, "cpu_count") else 4 | ||||
|  | ||||
|     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:\033[33mperm\033[0m1:\033[33mperm\033[0m2:\033[33mperm\033[0mN:\033[32mvolflag\033[0m1:\033[32mvolflag\033[0m2:\033[32mvolflag\033[0mN:... | ||||
|                 * "\033[33mperm\033[0m" is "permissions,username1,username2,..." | ||||
|                 * "\033[32mvolflag\033[0m" is config flags to set on this volume | ||||
|  | ||||
|             list of permissions: | ||||
|               "r" (read):   list folder contents, download files | ||||
|               "w" (write):  upload files; need "r" to see the uploads | ||||
|               "m" (move):   move files and folders; need "w" at destination | ||||
|               "d" (delete): permanently delete files and folders | ||||
|               "g" (get):    download files, but cannot see folder contents | ||||
|  | ||||
|             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 | ||||
|  | ||||
|             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=3923, 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)") | ||||
|     ap.add_argument("-nih", action="store_true", help="no info hostname") | ||||
|     ap.add_argument("-nid", action="store_true", help="no info disk-usage") | ||||
|     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[36mdf=1g\033[35m ensure 1 GiB free disk space | ||||
|  | ||||
|             \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[36md2ts\033[35m disables metadata collection for existing files | ||||
|               \033[36md2ds\033[35m disables onboot indexing, overrides -e2ds* | ||||
|               \033[36md2t\033[35m disables metadata collection, overrides -e2t* | ||||
|               \033[36md2v\033[35m disables file verification, overrides -e2v* | ||||
|               \033[36md2d\033[35m disables all database stuff, overrides -e2* | ||||
|               \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[36mnohash=\\.iso$\033[35m skips hashing file contents if path matches *.iso | ||||
|               \033[36mnoidx=\\.iso$\033[35m fully ignores the contents at paths matching *.iso | ||||
|               \033[36mxdev\033[35m do not descend into other filesystems | ||||
|               \033[36mxvol\033[35m skip symlinks leaving the volume root | ||||
|  | ||||
|             \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[0mthumbnails: | ||||
|               \033[36mdthumb\033[35m disables all thumbnails | ||||
|               \033[36mdvthumb\033[35m disables video thumbnails | ||||
|               \033[36mdathumb\033[35m disables audio thumbnails (spectrograms) | ||||
|               \033[36mdithumb\033[35m disables image thumbnails | ||||
|  | ||||
|             \033[0mclient and ux: | ||||
|               \033[36mhtml_head=TXT\033[35m includes TXT in the <head> | ||||
|               \033[36mrobots\033[35m allows indexing by search engines (default) | ||||
|               \033[36mnorobots\033[35m kindly asks search engines to leave | ||||
|  | ||||
|             \033[0mothers: | ||||
|               \033[36mfk=8\033[35m generates per-file accesskeys, | ||||
|                 which will then be required at the "g" permission | ||||
|             \033[0m""" | ||||
|             ), | ||||
|         ], | ||||
|         [ | ||||
|             "urlform", | ||||
|             "how to handle url-form POSTs", | ||||
|             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, 0=all") | ||||
|     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; examples [.::r], [/mnt/nas/music:/music:r:aed]") | ||||
|     ap2.add_argument("-ed", action="store_true", help="enable the ?dots url parameter / client option which allows clients to see dotfiles / hidden files") | ||||
|     ap2.add_argument("-emp", action="store_true", help="enable markdown plugins -- neat but dangerous, big XSS risk") | ||||
|     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-form POSTs; see --help-urlform") | ||||
|     ap2.add_argument("--wintitle", metavar="TXT", type=u, default="cpp @ $pub", help="window title, for example '$ip-10.1.2.' or '$ip-'") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('upload options') | ||||
|     ap2.add_argument("--dotpart", action="store_true", help="dotfile incomplete uploads, hiding them from clients unless -ed") | ||||
|     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.add_argument("--reg-cap", metavar="N", type=int, default=38400, help="max number of uploads to keep in memory when running without -e2d; roughly 1 MiB RAM per 600") | ||||
|     ap2.add_argument("--no-fpool", action="store_true", help="disable file-handle pooling -- instead, repeatedly close and reopen files during upload") | ||||
|     ap2.add_argument("--use-fpool", action="store_true", help="force file-handle pooling, even if copyparty thinks you're better off without -- probably useful on nfs and cow filesystems (zfs, btrfs)") | ||||
|     ap2.add_argument("--hardlink", action="store_true", help="prefer hardlinks instead of symlinks when possible (within same filesystem)") | ||||
|     ap2.add_argument("--never-symlink", action="store_true", help="do not fallback to symlinks when a hardlink cannot be made") | ||||
|     ap2.add_argument("--no-dedup", action="store_true", help="disable symlink/hardlink creation; copy file contents instead") | ||||
|     ap2.add_argument("--df", metavar="GiB", type=float, default=0, help="ensure GiB free disk space by rejecting upload requests") | ||||
|     ap2.add_argument("--sparse", metavar="MiB", type=int, default=4, help="windows-only: minimum size of incoming uploads through up2k before they are made into sparse files") | ||||
|     ap2.add_argument("--turbo", metavar="LVL", type=int, default=0, help="configure turbo-mode in up2k client; 0 = off and warn if enabled, 1 = off, 2 = on, 3 = on and disable datecheck") | ||||
|     ap2.add_argument("--u2sort", metavar="TXT", type=u, default="s", help="upload order; s=smallest-first, n=alphabetical, fs=force-s, fn=force-n -- alphabetical is a bit slower on fiber/LAN but makes it easier to eyeball if everything went fine") | ||||
|  | ||||
|     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.add_argument("--s-wr-sz", metavar="B", type=int, default=256*1024, help="socket write size in bytes") | ||||
|     ap2.add_argument("--s-wr-slp", metavar="SEC", type=float, default=0, help="debug: socket write delay in seconds") | ||||
|     ap2.add_argument("--rsp-slp", metavar="SEC", type=float, default=0, help="debug: response delay in seconds") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('SSL/TLS options') | ||||
|     ap2.add_argument("--http-only", action="store_true", help="disable ssl/tls -- force plaintext") | ||||
|     ap2.add_argument("--https-only", action="store_true", help="disable plaintext -- force tls") | ||||
|     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 for later decryption in wireshark") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('FTP options') | ||||
|     ap2.add_argument("--ftp", metavar="PORT", type=int, help="enable FTP server on PORT, for example 3921") | ||||
|     ap2.add_argument("--ftps", metavar="PORT", type=int, help="enable FTPS server on PORT, for example 3990") | ||||
|     ap2.add_argument("--ftp-dbg", action="store_true", help="enable debug logging") | ||||
|     ap2.add_argument("--ftp-nat", metavar="ADDR", type=u, help="the NAT address to use for passive connections") | ||||
|     ap2.add_argument("--ftp-pr", metavar="P-P", type=u, help="the range of TCP ports to use for passive connections, for example 12000-13000") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('opt-outs') | ||||
|     ap2.add_argument("-nw", action="store_true", help="never write anything to disk (debug/benchmark)") | ||||
|     ap2.add_argument("--keep-qem", action="store_true", help="do not disable quick-edit-mode on windows (it is disabled to avoid accidental text selection which will deadlock copyparty)") | ||||
|     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 -- don't show in UI") | ||||
|     ap2.add_argument("-nid", action="store_true", help="no info disk-usage -- don't show in UI") | ||||
|     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("-s", action="count", default=0, help="increase safety: Disable thumbnails / potentially dangerous software (ffmpeg/pillow/vips), hide partial uploads, avoid crawlers.\n └─Alias of\033[32m --dotpart --no-thumb --no-mtag-ff --no-robots --force-js") | ||||
|     ap2.add_argument("-ss", action="store_true", help="further increase safety: Prevent js-injection, accidental move/delete, broken symlinks, 404 on 403.\n └─Alias of\033[32m -s --no-dot-mv --no-dot-ren --unpost=0 --no-del --no-mv --hardlink --vague-403 -nih") | ||||
|     ap2.add_argument("-sss", action="store_true", help="further increase safety: Enable logging to disk, scan for dangerous symlinks.\n └─Alias of\033[32m -ss -lo=cpp-%%Y-%%m%%d-%%H%%M%%S.txt.xz --ls=**,*,ln,p,r") | ||||
|     ap2.add_argument("--ls", metavar="U[,V[,F]]", type=u, help="do a sanity/safety check of all volumes on startup; arguments USER,VOL,FLAGS; example [**,*,ln,p,r]") | ||||
|     ap2.add_argument("--salt", type=u, default="hunter2", help="up2k file-hash salt; used to generate unpredictable internal identifiers for uploads -- doesn't really matter") | ||||
|     ap2.add_argument("--fk-salt", metavar="SALT", type=u, default=fk_salt, help="per-file accesskey salt; used to generate unpredictable URLs for hidden files -- this one DOES matter") | ||||
|     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.add_argument("--vague-403", action="store_true", help="send 404 instead of 403 (security through ambiguity, very enterprise)") | ||||
|     ap2.add_argument("--force-js", action="store_true", help="don't send folder listings as HTML, force clients to use the embedded json instead -- slight protection against misbehaving search engines which ignore --no-robots") | ||||
|     ap2.add_argument("--no-robots", action="store_true", help="adds http and html headers asking search engines to not index anything") | ||||
|     ap2.add_argument("--logout", metavar="H", type=float, default="8086", help="logout clients after H hours of inactivity (0.0028=10sec, 0.1=6min, 24=day, 168=week, 720=month, 8760=year)") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('yolo options') | ||||
|     ap2.add_argument("--ign-ebind", action="store_true", help="continue running even if it's impossible to listen on some of the requested endpoints") | ||||
|     ap2.add_argument("--ign-ebind-all", action="store_true", help="continue running even if it's impossible to receive connections at all") | ||||
|  | ||||
|     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="debug: print tcp-server msgs") | ||||
|     ap2.add_argument("--log-htp", action="store_true", help="debug: 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-reload", action="store_true", help="disable ?reload=cfg (reload users/volumes/volflags from config file)") | ||||
|     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-athumb", action="store_true", help="disable audio thumbnails (spectrograms)") | ||||
|     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=cores, help="num cpu cores to use for generating thumbnails") | ||||
|     ap2.add_argument("--th-convt", metavar="SEC", type=int, default=60, help="conversion timeout in seconds") | ||||
|     ap2.add_argument("--th-no-crop", action="store_true", help="dynamic height; show full image") | ||||
|     ap2.add_argument("--th-dec", metavar="LIBS", default="vips,pil,ff", help="image decoders, in order of preference") | ||||
|     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 output for video thumbs") | ||||
|     ap2.add_argument("--th-ff-swr", action="store_true", help="use swresample instead of soxr for audio thumbs") | ||||
|     ap2.add_argument("--th-poke", metavar="SEC", type=int, default=300, help="activity labeling cooldown -- avoids doing keepalive pokes (updating the mtime) on thumbnail folders more often than SEC seconds") | ||||
|     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 -- folders which haven't been poked for longer than --th-poke seconds will get deleted every --th-clean seconds") | ||||
|     ap2.add_argument("--th-covers", metavar="N,N", type=u, default="folder.png,folder.jpg,cover.png,cover.jpg", help="folder thumbnails to stat/look for") | ||||
|     # https://pillow.readthedocs.io/en/stable/handbook/image-file-formats.html | ||||
|     # https://github.com/libvips/libvips | ||||
|     # ffmpeg -hide_banner -demuxers | awk '/^ D  /{print$2}' | while IFS= read -r x; do ffmpeg -hide_banner -h demuxer=$x; done | grep -E '^Demuxer |extensions:' | ||||
|     ap2.add_argument("--th-r-pil", metavar="T,T", type=u, default="bmp,dib,gif,icns,ico,jpg,jpeg,jp2,jpx,pcx,png,pbm,pgm,ppm,pnm,sgi,tga,tif,tiff,webp,xbm,dds,xpm,heif,heifs,heic,heics,avif,avifs", help="image formats to decode using pillow") | ||||
|     ap2.add_argument("--th-r-vips", metavar="T,T", type=u, default="jpg,jpeg,jp2,jpx,jxl,tif,tiff,png,webp,heic,avif,fit,fits,fts,exr,svg,hdr,ppm,pgm,pfm,gif,nii", help="image formats to decode using pyvips") | ||||
|     ap2.add_argument("--th-r-ffi", metavar="T,T", type=u, default="apng,avif,avifs,bmp,dds,dib,fit,fits,fts,gif,heic,heics,heif,heifs,icns,ico,jp2,jpeg,jpg,jpx,jxl,pbm,pcx,pfm,pgm,png,pnm,ppm,psd,sgi,tga,tif,tiff,webp,xbm,xpm", help="image formats to decode using ffmpeg") | ||||
|     ap2.add_argument("--th-r-ffv", metavar="T,T", type=u, default="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", help="video formats to decode using ffmpeg") | ||||
|     ap2.add_argument("--th-r-ffa", metavar="T,T", type=u, default="aac,m4a,ogg,opus,flac,alac,mp3,mp2,ac3,dts,wma,ra,wav,aif,aiff,au,alaw,ulaw,mulaw,amr,gsm,ape,tak,tta,wv,mpc", help="audio formats to decode using ffmpeg") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('transcoding options') | ||||
|     ap2.add_argument("--no-acode", action="store_true", help="disable audio transcoding") | ||||
|     ap2.add_argument("--ac-maxage", metavar="SEC", type=int, default=86400, help="delete cached transcode output after SEC seconds") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('general db options') | ||||
|     ap2.add_argument("-e2d", action="store_true", help="enable up2k database, making files searchable + enables upload deduplocation") | ||||
|     ap2.add_argument("-e2ds", action="store_true", help="scan writable folders for new files on startup; sets -e2d") | ||||
|     ap2.add_argument("-e2dsa", action="store_true", help="scans all folders on startup; sets -e2ds") | ||||
|     ap2.add_argument("-e2v", action="store_true", help="verify file integrity; rehash all files and compare with db") | ||||
|     ap2.add_argument("-e2vu", action="store_true", help="on hash mismatch: update the database with the new hash") | ||||
|     ap2.add_argument("-e2vp", action="store_true", help="on hash mismatch: panic and quit copyparty") | ||||
|     ap2.add_argument("--hist", metavar="PATH", type=u, help="where to store volume data (db, thumbs)") | ||||
|     ap2.add_argument("--no-hash", metavar="PTN", type=u, help="regex: disable hashing of matching paths during e2ds folder scans") | ||||
|     ap2.add_argument("--no-idx", metavar="PTN", type=u, help="regex: disable indexing of matching paths during e2ds folder scans") | ||||
|     ap2.add_argument("--xdev", action="store_true", help="do not descend into other filesystems (symlink or bind-mount to another HDD, ...)") | ||||
|     ap2.add_argument("--xvol", action="store_true", help="skip symlinks leaving the volume root") | ||||
|     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("--db-act", metavar="SEC", type=float, default=10, help="defer any scheduled volume reindexing until SEC seconds after last db write (uploads, renames, ...)") | ||||
|     ap2.add_argument("--srch-time", metavar="SEC", type=int, default=30, help="search deadline -- terminate searches running for more than SEC seconds") | ||||
|     ap2.add_argument("--srch-hits", metavar="N", type=int, default=7999, help="max search results to allow clients to fetch; 125 results will be shown initially") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('metadata db options') | ||||
|     ap2.add_argument("-e2t", action="store_true", help="enable metadata indexing; makes it possible to search for artist/title/codec/resolution/...") | ||||
|     ap2.add_argument("-e2ts", action="store_true", help="scan existing files on startup; sets -e2t") | ||||
|     ap2.add_argument("-e2tsr", action="store_true", help="delete all metadata from DB and do a full rescan; sets -e2ts") | ||||
|     ap2.add_argument("--no-mutagen", action="store_true", help="use FFprobe for tags instead; will catch more tags") | ||||
|     ap2.add_argument("--no-mtag-ff", action="store_true", help="never use FFprobe as tag reader; is probably safer") | ||||
|     ap2.add_argument("--mtag-mt", metavar="CORES", type=int, default=cores, help="num cpu cores to use for tag scanning") | ||||
|     ap2.add_argument("--mtag-v", action="store_true", help="verbose tag scanning; print errors from mtp subprocesses and such") | ||||
|     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 program BIN to parse the file") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('ui options') | ||||
|     ap2.add_argument("--lang", metavar="LANG", type=u, default="eng", help="language") | ||||
|     ap2.add_argument("--theme", metavar="NUM", type=int, default=0, help="default theme to use") | ||||
|     ap2.add_argument("--themes", metavar="NUM", type=int, default=8, help="number of themes installed") | ||||
|     ap2.add_argument("--favico", metavar="TXT", type=u, default="c 000 none" if retry else "🎉 000 none", help="favicon text [ foreground [ background ] ], set blank to disable") | ||||
|     ap2.add_argument("--js-browser", metavar="L", type=u, help="URL to additional JS to include") | ||||
|     ap2.add_argument("--css-browser", metavar="L", type=u, help="URL to additional CSS to include") | ||||
|     ap2.add_argument("--html-head", metavar="TXT", type=u, default="", help="text to append to the <head> of all HTML pages") | ||||
|     ap2.add_argument("--textfiles", metavar="CSV", type=u, default="txt,nfo,diz,cue,readme", help="file extensions to present as plaintext") | ||||
|     ap2.add_argument("--txt-max", metavar="KiB", type=int, default=64, help="max size of embedded textfiles on ?doc= (anything bigger will be lazy-loaded by JS)") | ||||
|     ap2.add_argument("--doctitle", metavar="TXT", type=u, default="copyparty", help="title / service-name to show in html documents") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('debug options') | ||||
|     ap2.add_argument("--no-sendfile", action="store_true", help="disable sendfile; instead using a traditional file read loop") | ||||
|     ap2.add_argument("--no-scandir", action="store_true", help="disable scandir; instead using listdir + stat on each file") | ||||
|     ap2.add_argument("--no-fastboot", action="store_true", help="wait for up2k indexing before starting the httpd") | ||||
|     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") | ||||
|     ap2.add_argument("--log-fk", metavar="REGEX", type=u, default="", help="log filekey params for files where path matches REGEX; '.' (a single dot) = all files") | ||||
|     # 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: Optional[list[str]] = None) -> None: | ||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||
|     if WINDOWS: | ||||
|         os.system("rem")  # enables colors | ||||
|  | ||||
|     if argv is None: | ||||
|         argv = sys.argv | ||||
|  | ||||
|     f = '\033[36mcopyparty v{} "\033[35m{}\033[36m" ({})\n{}\033[0;36m\n   sqlite v{} | jinja2 v{} | pyftpd v{}\n\033[0m' | ||||
|     f = f.format( | ||||
|         S_VERSION, | ||||
|         CODENAME, | ||||
|         S_BUILD_DT, | ||||
|         py_desc().replace("[", "\033[1;30m["), | ||||
|         SQLITE_VER, | ||||
|         JINJA_VER, | ||||
|         PYFTPD_VER, | ||||
|     ) | ||||
|     lprint(f) | ||||
|  | ||||
|     ensure_locale() | ||||
|     if HAVE_SSL: | ||||
|         ensure_cert() | ||||
|  | ||||
|     for k, v in zip(argv[1:], argv[2:]): | ||||
|         if k == "-c": | ||||
|             supp = args_from_cfg(v) | ||||
|             argv.extend(supp) | ||||
|  | ||||
|     deprecated: list[tuple[str, str]] = [] | ||||
|     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: | ||||
|         if len(argv) == 1 and (ANYWIN or not os.geteuid()): | ||||
|             argv.extend(["-p80,443,3923", "--ign-ebind"]) | ||||
|     except: | ||||
|         pass | ||||
|  | ||||
|     retry = False | ||||
|     for fmtr in [RiceFormatter, RiceFormatter, Dodge11874, BasicDodge11874]: | ||||
|         try: | ||||
|             al = run_argparse(argv, fmtr, retry) | ||||
|         except SystemExit: | ||||
|             raise | ||||
|         except: | ||||
|             retry = True | ||||
|             lprint("\n[ {} ]:\n{}\n".format(fmtr, min_ex())) | ||||
|  | ||||
|     assert al | ||||
|  | ||||
|     if WINDOWS and not al.keep_qem: | ||||
|         try: | ||||
|             disable_quickedit() | ||||
|         except: | ||||
|             lprint("\nfailed to disable quick-edit-mode:\n" + min_ex() + "\n") | ||||
|  | ||||
|     if not VT100: | ||||
|         al.wintitle = "" | ||||
|  | ||||
|     nstrs: list[str] = [] | ||||
|     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("^[rwmdg]*", "", 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") | ||||
|  | ||||
|     for arg, kname, okays in [["--u2sort", "u2sort", "s n fs fn"]]: | ||||
|         val = unicode(getattr(al, kname)) | ||||
|         if val not in okays.split(): | ||||
|             zs = "argument {} cannot be '{}'; try one of these: {}" | ||||
|             raise Exception(zs.format(arg, val, okays)) | ||||
|  | ||||
|     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, "".join(printed)).run() | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| # coding: utf-8 | ||||
|  | ||||
| VERSION = (0, 5, 1) | ||||
| CODENAME = "fuse jelly" | ||||
| BUILD_DT = (2020, 8, 17) | ||||
| VERSION = (1, 3, 10) | ||||
| CODENAME = "god dag" | ||||
| BUILD_DT = (2022, 8, 4) | ||||
|  | ||||
| S_VERSION = ".".join(map(str, VERSION)) | ||||
| S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) | ||||
|   | ||||
							
								
								
									
										1266
									
								
								copyparty/authsrv.py
									
									
									
									
									
								
							
							
						
						
									
										1266
									
								
								copyparty/authsrv.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										0
									
								
								copyparty/bos/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								copyparty/bos/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										76
									
								
								copyparty/bos/bos.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								copyparty/bos/bos.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
|  | ||||
| from ..util import SYMTIME, fsdec, fsenc | ||||
| from . import path | ||||
|  | ||||
| try: | ||||
|     from typing import Optional | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| _ = (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: str, mode: int) -> None: | ||||
|     return os.chmod(fsenc(p), mode) | ||||
|  | ||||
|  | ||||
| def listdir(p: str = ".") -> list[str]: | ||||
|     return [fsdec(x) for x in os.listdir(fsenc(p))] | ||||
|  | ||||
|  | ||||
| def makedirs(name: str, mode: int = 0o755, exist_ok: bool = True) -> None: | ||||
|     bname = fsenc(name) | ||||
|     try: | ||||
|         os.makedirs(bname, mode) | ||||
|     except: | ||||
|         if not exist_ok or not os.path.isdir(bname): | ||||
|             raise | ||||
|  | ||||
|  | ||||
| def mkdir(p: str, mode: int = 0o755) -> None: | ||||
|     return os.mkdir(fsenc(p), mode) | ||||
|  | ||||
|  | ||||
| def rename(src: str, dst: str) -> None: | ||||
|     return os.rename(fsenc(src), fsenc(dst)) | ||||
|  | ||||
|  | ||||
| def replace(src: str, dst: str) -> None: | ||||
|     return os.replace(fsenc(src), fsenc(dst)) | ||||
|  | ||||
|  | ||||
| def rmdir(p: str) -> None: | ||||
|     return os.rmdir(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def stat(p: str) -> os.stat_result: | ||||
|     return os.stat(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def unlink(p: str) -> None: | ||||
|     return os.unlink(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def utime( | ||||
|     p: str, times: Optional[tuple[float, float]] = None, follow_symlinks: bool = True | ||||
| ) -> None: | ||||
|     if SYMTIME: | ||||
|         return os.utime(fsenc(p), times, follow_symlinks=follow_symlinks) | ||||
|     else: | ||||
|         return os.utime(fsenc(p), times) | ||||
|  | ||||
|  | ||||
| if hasattr(os, "lstat"): | ||||
|  | ||||
|     def lstat(p: str) -> os.stat_result: | ||||
|         return os.lstat(fsenc(p)) | ||||
|  | ||||
| else: | ||||
|     lstat = stat | ||||
							
								
								
									
										45
									
								
								copyparty/bos/path.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								copyparty/bos/path.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
|  | ||||
| from ..util import SYMTIME, fsdec, fsenc | ||||
|  | ||||
|  | ||||
| def abspath(p: str) -> str: | ||||
|     return fsdec(os.path.abspath(fsenc(p))) | ||||
|  | ||||
|  | ||||
| def exists(p: str) -> bool: | ||||
|     return os.path.exists(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def getmtime(p: str, follow_symlinks: bool = True) -> float: | ||||
|     if not follow_symlinks and SYMTIME: | ||||
|         return os.lstat(fsenc(p)).st_mtime | ||||
|     else: | ||||
|         return os.path.getmtime(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def getsize(p: str) -> int: | ||||
|     return os.path.getsize(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def isfile(p: str) -> bool: | ||||
|     return os.path.isfile(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def isdir(p: str) -> bool: | ||||
|     return os.path.isdir(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def islink(p: str) -> bool: | ||||
|     return os.path.islink(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def lexists(p: str) -> bool: | ||||
|     return os.path.lexists(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def realpath(p: str) -> str: | ||||
|     return fsdec(os.path.realpath(fsenc(p))) | ||||
| @@ -1,65 +1,73 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import time | ||||
| import threading | ||||
| import time | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, VT100 | ||||
| from .broker_util import try_exec | ||||
| import queue | ||||
|  | ||||
| from .__init__ import TYPE_CHECKING | ||||
| from .broker_mpw import MpWorker | ||||
| from .broker_util import try_exec | ||||
| from .util import mp | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from .svchub import SvcHub | ||||
|  | ||||
| if PY2 and not WINDOWS: | ||||
|     from multiprocessing.reduction import ForkingPickler | ||||
|     from StringIO import StringIO as MemesIO  # pylint: disable=import-error | ||||
| try: | ||||
|     from typing import Any | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class MProcess(mp.Process): | ||||
|     def __init__( | ||||
|         self, | ||||
|         q_pend: queue.Queue[tuple[int, str, list[Any]]], | ||||
|         q_yield: queue.Queue[tuple[int, str, list[Any]]], | ||||
|         target: Any, | ||||
|         args: Any, | ||||
|     ) -> None: | ||||
|         super(MProcess, self).__init__(target=target, args=args) | ||||
|         self.q_pend = q_pend | ||||
|         self.q_yield = q_yield | ||||
|  | ||||
|  | ||||
| class BrokerMp(object): | ||||
|     """external api; manages MpWorkers""" | ||||
|  | ||||
|     def __init__(self, hub): | ||||
|     def __init__(self, hub: "SvcHub") -> None: | ||||
|         self.hub = hub | ||||
|         self.log = hub.log | ||||
|         self.args = hub.args | ||||
|  | ||||
|         self.procs = [] | ||||
|         self.retpend = {} | ||||
|         self.retpend_mutex = threading.Lock() | ||||
|         self.mutex = threading.Lock() | ||||
|  | ||||
|         cores = self.args.j | ||||
|         if not cores: | ||||
|             cores = mp.cpu_count() | ||||
|         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: queue.Queue[tuple[int, str, list[Any]]] = mp.Queue(1) | ||||
|             q_yield: queue.Queue[tuple[int, str, list[Any]]] = mp.Queue(64) | ||||
|  | ||||
|         self.log("broker", "booting {} subprocesses".format(cores)) | ||||
|         for n in range(cores): | ||||
|             q_pend = mp.Queue(1) | ||||
|             q_yield = mp.Queue(64) | ||||
|             proc = MProcess(q_pend, q_yield, MpWorker, (q_pend, q_yield, self.args, n)) | ||||
|  | ||||
|             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): | ||||
|     def shutdown(self) -> None: | ||||
|         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: | ||||
| @@ -73,7 +81,12 @@ class BrokerMp(object): | ||||
|  | ||||
|             procs.pop() | ||||
|  | ||||
|     def collector(self, proc): | ||||
|     def reload(self) -> None: | ||||
|         self.log("broker", "reloading") | ||||
|         for _, proc in enumerate(self.procs): | ||||
|             proc.q_pend.put((0, "reload", [])) | ||||
|  | ||||
|     def collector(self, proc: MProcess) -> None: | ||||
|         """receive message from hub in other process""" | ||||
|         while True: | ||||
|             msg = proc.q_yield.get() | ||||
| @@ -82,26 +95,9 @@ 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: | ||||
|                     retq = self.retpend.pop(retq_id) | ||||
|  | ||||
|                 retq.put(args) | ||||
|                 raise Exception("invalid broker_mp usage") | ||||
|  | ||||
|             else: | ||||
|                 # new ipc invoking managed service in hub | ||||
| @@ -113,46 +109,20 @@ class BrokerMp(object): | ||||
|                 rv = try_exec(retq_id, obj, *args) | ||||
|  | ||||
|                 if retq_id: | ||||
|                     proc.q_pend.put([retq_id, "retq", rv]) | ||||
|                     proc.q_pend.put((retq_id, "retq", rv)) | ||||
|  | ||||
|     def put(self, want_retval, dest, *args): | ||||
|     def say(self, dest: str, *args: Any) -> None: | ||||
|         """ | ||||
|         send message to non-hub component in other process, | ||||
|         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) | ||||
|   | ||||
| @@ -1,87 +1,99 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import sys | ||||
| import time | ||||
| import argparse | ||||
| import signal | ||||
| import sys | ||||
| import threading | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS | ||||
| from .broker_util import ExceptionalQueue | ||||
| import queue | ||||
|  | ||||
| from .authsrv import AuthSrv | ||||
| from .broker_util import BrokerCli, ExceptionalQueue | ||||
| from .httpsrv import HttpSrv | ||||
| from .util import FAKE_MP | ||||
|  | ||||
| if PY2 and not WINDOWS: | ||||
|     import pickle  # nosec | ||||
| try: | ||||
|     from types import FrameType | ||||
|  | ||||
|     from typing import Any, Optional, Union | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class MpWorker(object): | ||||
| class MpWorker(BrokerCli): | ||||
|     """one single mp instance""" | ||||
|  | ||||
|     def __init__(self, q_pend, q_yield, args, n): | ||||
|     def __init__( | ||||
|         self, | ||||
|         q_pend: queue.Queue[tuple[int, str, list[Any]]], | ||||
|         q_yield: queue.Queue[tuple[int, str, list[Any]]], | ||||
|         args: argparse.Namespace, | ||||
|         n: int, | ||||
|     ) -> None: | ||||
|         super(MpWorker, self).__init__() | ||||
|  | ||||
|         self.q_pend = q_pend | ||||
|         self.q_yield = q_yield | ||||
|         self.args = args | ||||
|         self.n = n | ||||
|  | ||||
|         self.retpend = {} | ||||
|         self.log = self._log_disabled if args.q and not args.lo else self._log_enabled | ||||
|  | ||||
|         self.retpend: dict[int, Any] = {} | ||||
|         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.SIGUSR1]: | ||||
|                 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: Optional[int], frame: Optional[FrameType]) -> None: | ||||
|         # print('k') | ||||
|         pass | ||||
|  | ||||
|     def log(self, src, msg): | ||||
|         self.q_yield.put([0, "log", [src, msg]]) | ||||
|     def _log_enabled(self, src: str, msg: str, c: Union[int, str] = 0) -> None: | ||||
|         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: str, msg: str, c: Union[int, str] = 0) -> None: | ||||
|         pass | ||||
|  | ||||
|     def httpdrop(self, addr): | ||||
|         self.q_yield.put([0, "httpdrop", [addr]]) | ||||
|     def logw(self, msg: str, c: Union[int, str] = 0) -> None: | ||||
|         self.log("mp{}".format(self.n), msg, c) | ||||
|  | ||||
|     def main(self): | ||||
|     def main(self) -> None: | ||||
|         while True: | ||||
|             retq_id, dest, args = self.q_pend.get() | ||||
|  | ||||
|             # 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 | ||||
|             elif dest == "reload": | ||||
|                 self.logw("mpw.asrv reloading") | ||||
|                 self.asrv.reload() | ||||
|                 self.logw("mpw.asrv reloaded") | ||||
|  | ||||
|                 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 | ||||
| @@ -93,28 +105,14 @@ class MpWorker(object): | ||||
|             else: | ||||
|                 raise Exception("what is " + str(dest)) | ||||
|  | ||||
|     def put(self, want_retval, dest, *args): | ||||
|         if want_retval: | ||||
|             retq = ExceptionalQueue(1) | ||||
|             retq_id = id(retq) | ||||
|             with self.retpend_mutex: | ||||
|                 self.retpend[retq_id] = retq | ||||
|         else: | ||||
|             retq = None | ||||
|             retq_id = 0 | ||||
|     def ask(self, dest: str, *args: Any) -> ExceptionalQueue: | ||||
|         retq = ExceptionalQueue(1) | ||||
|         retq_id = id(retq) | ||||
|         with self.retpend_mutex: | ||||
|             self.retpend[retq_id] = retq | ||||
|  | ||||
|         self.q_yield.put([retq_id, dest, args]) | ||||
|         self.q_yield.put((retq_id, dest, list(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]]) | ||||
|     def say(self, dest: str, *args: Any) -> None: | ||||
|         self.q_yield.put((0, dest, list(args))) | ||||
|   | ||||
| @@ -3,49 +3,66 @@ from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import threading | ||||
|  | ||||
| from .__init__ import TYPE_CHECKING | ||||
| from .broker_util import BrokerCli, ExceptionalQueue, try_exec | ||||
| from .httpsrv import HttpSrv | ||||
| from .broker_util import ExceptionalQueue, try_exec | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from .svchub import SvcHub | ||||
|  | ||||
| try: | ||||
|     from typing import Any | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class BrokerThr(object): | ||||
| class BrokerThr(BrokerCli): | ||||
|     """external api; behaves like BrokerMP but using plain threads""" | ||||
|  | ||||
|     def __init__(self, hub): | ||||
|     def __init__(self, hub: "SvcHub") -> None: | ||||
|         super(BrokerThr, self).__init__() | ||||
|  | ||||
|         self.hub = hub | ||||
|         self.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) | ||||
|         self.reload = self.noop | ||||
|  | ||||
|     def shutdown(self): | ||||
|     def shutdown(self) -> None: | ||||
|         # self.log("broker", "shutting down") | ||||
|         self.httpsrv.shutdown() | ||||
|  | ||||
|     def noop(self) -> None: | ||||
|         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) | ||||
|     def ask(self, dest: str, *args: Any) -> ExceptionalQueue: | ||||
|  | ||||
|         else: | ||||
|             # new ipc invoking managed service in hub | ||||
|             obj = self.hub | ||||
|             for node in dest.split("."): | ||||
|                 obj = getattr(obj, node) | ||||
|         # new ipc invoking managed service in hub | ||||
|         obj = self.hub | ||||
|         for node in dest.split("."): | ||||
|             obj = getattr(obj, node) | ||||
|  | ||||
|             # TODO will deadlock if dest performs another ipc | ||||
|             rv = try_exec(want_retval, obj, *args) | ||||
|             if not want_retval: | ||||
|                 return | ||||
|         rv = try_exec(True, obj, *args) | ||||
|  | ||||
|             # pretend we're broker_mp | ||||
|             retq = ExceptionalQueue(1) | ||||
|             retq.put(rv) | ||||
|             return retq | ||||
|         # pretend we're broker_mp | ||||
|         retq = ExceptionalQueue(1) | ||||
|         retq.put(rv) | ||||
|         return retq | ||||
|  | ||||
|     def httpdrop(self, addr): | ||||
|         self.hub.tcpsrv.num_clients.add(-1) | ||||
|     def say(self, dest: str, *args: Any) -> None: | ||||
|         if dest == "listen": | ||||
|             self.httpsrv.listen(args[0], 1) | ||||
|             return | ||||
|  | ||||
|         # new ipc invoking managed service in hub | ||||
|         obj = self.hub | ||||
|         for node in dest.split("."): | ||||
|             obj = getattr(obj, node) | ||||
|  | ||||
|         try_exec(False, obj, *args) | ||||
|   | ||||
| @@ -1,17 +1,30 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
|  | ||||
| import argparse | ||||
| import traceback | ||||
|  | ||||
| from .util import Pebkac, Queue | ||||
| from queue import Queue | ||||
|  | ||||
| from .__init__ import TYPE_CHECKING | ||||
| from .authsrv import AuthSrv | ||||
| from .util import Pebkac | ||||
|  | ||||
| try: | ||||
|     from typing import Any, Optional, Union | ||||
|  | ||||
|     from .util import RootLogger | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from .httpsrv import HttpSrv | ||||
|  | ||||
|  | ||||
| class ExceptionalQueue(Queue, object): | ||||
|     def get(self, block=True, timeout=None): | ||||
|     def get(self, block: bool = True, timeout: Optional[float] = None) -> Any: | ||||
|         rv = super(ExceptionalQueue, self).get(block, timeout) | ||||
|  | ||||
|         # TODO: how expensive is this? | ||||
|         if isinstance(rv, list): | ||||
|             if rv[0] == "exception": | ||||
|                 if rv[1] == "pebkac": | ||||
| @@ -22,7 +35,26 @@ class ExceptionalQueue(Queue, object): | ||||
|         return rv | ||||
|  | ||||
|  | ||||
| def try_exec(want_retval, func, *args): | ||||
| class BrokerCli(object): | ||||
|     """ | ||||
|     helps mypy understand httpsrv.broker but still fails a few levels deeper, | ||||
|     for example resolving httpconn.* in httpcli -- see lines tagged #mypy404 | ||||
|     """ | ||||
|  | ||||
|     def __init__(self) -> None: | ||||
|         self.log: "RootLogger" = None | ||||
|         self.args: argparse.Namespace = None | ||||
|         self.asrv: AuthSrv = None | ||||
|         self.httpsrv: "HttpSrv" = None | ||||
|  | ||||
|     def ask(self, dest: str, *args: Any) -> ExceptionalQueue: | ||||
|         return ExceptionalQueue(1) | ||||
|  | ||||
|     def say(self, dest: str, *args: Any) -> None: | ||||
|         pass | ||||
|  | ||||
|  | ||||
| def try_exec(want_retval: Union[bool, int], func: Any, *args: list[Any]) -> Any: | ||||
|     try: | ||||
|         return func(*args) | ||||
|  | ||||
|   | ||||
							
								
								
									
										154
									
								
								copyparty/fsutil.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										154
									
								
								copyparty/fsutil.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,154 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import re | ||||
| import time | ||||
|  | ||||
| from .__init__ import ANYWIN, MACOS | ||||
| from .authsrv import AXS, VFS | ||||
| from .bos import bos | ||||
| from .util import chkcmd, min_ex | ||||
|  | ||||
| try: | ||||
|     from typing import Optional, Union | ||||
|  | ||||
|     from .util import RootLogger | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class Fstab(object): | ||||
|     def __init__(self, log: "RootLogger"): | ||||
|         self.log_func = log | ||||
|  | ||||
|         self.trusted = False | ||||
|         self.tab: Optional[VFS] = None | ||||
|         self.cache: dict[str, str] = {} | ||||
|         self.age = 0.0 | ||||
|  | ||||
|     def log(self, msg: str, c: Union[int, str] = 0) -> None: | ||||
|         self.log_func("fstab", msg + "\033[K", c) | ||||
|  | ||||
|     def get(self, path: str) -> str: | ||||
|         if len(self.cache) > 9000: | ||||
|             self.age = time.time() | ||||
|             self.tab = None | ||||
|             self.cache = {} | ||||
|  | ||||
|         fs = "ext4" | ||||
|         msg = "failed to determine filesystem at [{}]; assuming {}\n{}" | ||||
|  | ||||
|         if ANYWIN: | ||||
|             fs = "vfat" | ||||
|             try: | ||||
|                 path = self._winpath(path) | ||||
|             except: | ||||
|                 self.log(msg.format(path, fs, min_ex()), 3) | ||||
|                 return fs | ||||
|  | ||||
|         path = path.lstrip("/") | ||||
|         try: | ||||
|             return self.cache[path] | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         try: | ||||
|             fs = self.get_w32(path) if ANYWIN else self.get_unix(path) | ||||
|         except: | ||||
|             self.log(msg.format(path, fs, min_ex()), 3) | ||||
|  | ||||
|         fs = fs.lower() | ||||
|         self.cache[path] = fs | ||||
|         self.log("found {} at {}".format(fs, path)) | ||||
|         return fs | ||||
|  | ||||
|     def _winpath(self, path: str) -> str: | ||||
|         # try to combine volume-label + st_dev (vsn) | ||||
|         path = path.replace("/", "\\") | ||||
|         vid = path.split(":", 1)[0].strip("\\").split("\\", 1)[0] | ||||
|         try: | ||||
|             return "{}*{}".format(vid, bos.stat(path).st_dev) | ||||
|         except: | ||||
|             return vid | ||||
|  | ||||
|     def build_fallback(self) -> None: | ||||
|         self.tab = VFS(self.log_func, "idk", "/", AXS(), {}) | ||||
|         self.trusted = False | ||||
|  | ||||
|     def build_tab(self) -> None: | ||||
|         self.log("building tab") | ||||
|  | ||||
|         sptn = r"^.*? on (.*) type ([^ ]+) \(.*" | ||||
|         if MACOS: | ||||
|             sptn = r"^.*? on (.*) \(([^ ]+), .*" | ||||
|  | ||||
|         ptn = re.compile(sptn) | ||||
|         so, _ = chkcmd(["mount"]) | ||||
|         tab1: list[tuple[str, str]] = [] | ||||
|         for ln in so.split("\n"): | ||||
|             m = ptn.match(ln) | ||||
|             if not m: | ||||
|                 continue | ||||
|  | ||||
|             zs1, zs2 = m.groups() | ||||
|             tab1.append((str(zs1), str(zs2))) | ||||
|  | ||||
|         tab1.sort(key=lambda x: (len(x[0]), x[0])) | ||||
|         path1, fs1 = tab1[0] | ||||
|         tab = VFS(self.log_func, fs1, path1, AXS(), {}) | ||||
|         for path, fs in tab1[1:]: | ||||
|             tab.add(fs, path.lstrip("/")) | ||||
|  | ||||
|         self.tab = tab | ||||
|  | ||||
|     def relabel(self, path: str, nval: str) -> None: | ||||
|         assert self.tab | ||||
|         self.cache = {} | ||||
|         if ANYWIN: | ||||
|             path = self._winpath(path) | ||||
|  | ||||
|         path = path.lstrip("/") | ||||
|         ptn = re.compile(r"^[^\\/]*") | ||||
|         vn, rem = self.tab._find(path) | ||||
|         if not self.trusted: | ||||
|             # no mtab access; have to build as we go | ||||
|             if "/" in rem: | ||||
|                 self.tab.add("idk", os.path.join(vn.vpath, rem.split("/")[0])) | ||||
|             if rem: | ||||
|                 self.tab.add(nval, path) | ||||
|             else: | ||||
|                 vn.realpath = nval | ||||
|  | ||||
|             return | ||||
|  | ||||
|         visit = [vn] | ||||
|         while visit: | ||||
|             vn = visit.pop() | ||||
|             vn.realpath = ptn.sub(nval, vn.realpath) | ||||
|             visit.extend(list(vn.nodes.values())) | ||||
|  | ||||
|     def get_unix(self, path: str) -> str: | ||||
|         if not self.tab: | ||||
|             try: | ||||
|                 self.build_tab() | ||||
|                 self.trusted = True | ||||
|             except: | ||||
|                 # prisonparty or other restrictive environment | ||||
|                 self.log("failed to build tab:\n{}".format(min_ex()), 3) | ||||
|                 self.build_fallback() | ||||
|  | ||||
|         assert self.tab | ||||
|         ret = self.tab._find(path)[0] | ||||
|         if self.trusted or path == ret.vpath: | ||||
|             return ret.realpath.split("/")[0] | ||||
|         else: | ||||
|             return "idk" | ||||
|  | ||||
|     def get_w32(self, path: str) -> str: | ||||
|         if not self.tab: | ||||
|             self.build_fallback() | ||||
|  | ||||
|         assert self.tab | ||||
|         ret = self.tab._find(path)[0] | ||||
|         return ret.realpath | ||||
							
								
								
									
										401
									
								
								copyparty/ftpd.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										401
									
								
								copyparty/ftpd.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,401 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import argparse | ||||
| import logging | ||||
| import os | ||||
| import stat | ||||
| import sys | ||||
| import threading | ||||
| import time | ||||
|  | ||||
| from pyftpdlib.authorizers import AuthenticationFailed, DummyAuthorizer | ||||
| from pyftpdlib.filesystems import AbstractedFS, FilesystemError | ||||
| from pyftpdlib.handlers import FTPHandler | ||||
| from pyftpdlib.log import config_logging | ||||
| from pyftpdlib.servers import FTPServer | ||||
|  | ||||
| from .__init__ import PY2, TYPE_CHECKING, E | ||||
| from .bos import bos | ||||
| from .util import Pebkac, exclude_dotfiles, fsenc | ||||
|  | ||||
| try: | ||||
|     from pyftpdlib.ioloop import IOLoop | ||||
| except ImportError: | ||||
|     p = os.path.join(E.mod, "vend") | ||||
|     print("loading asynchat from " + p) | ||||
|     sys.path.append(p) | ||||
|     from pyftpdlib.ioloop import IOLoop | ||||
|  | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from .svchub import SvcHub | ||||
|  | ||||
| try: | ||||
|     import typing | ||||
|     from typing import Any, Optional | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class FtpAuth(DummyAuthorizer): | ||||
|     def __init__(self, hub: "SvcHub") -> None: | ||||
|         super(FtpAuth, self).__init__() | ||||
|         self.hub = hub | ||||
|  | ||||
|     def validate_authentication( | ||||
|         self, username: str, password: str, handler: Any | ||||
|     ) -> None: | ||||
|         asrv = self.hub.asrv | ||||
|         if username == "anonymous": | ||||
|             password = "" | ||||
|  | ||||
|         uname = "*" | ||||
|         if password: | ||||
|             uname = asrv.iacct.get(password, "") | ||||
|  | ||||
|         handler.username = uname | ||||
|  | ||||
|         if password and not uname: | ||||
|             raise AuthenticationFailed("Authentication failed.") | ||||
|  | ||||
|     def get_home_dir(self, username: str) -> str: | ||||
|         return "/" | ||||
|  | ||||
|     def has_user(self, username: str) -> bool: | ||||
|         asrv = self.hub.asrv | ||||
|         return username in asrv.acct | ||||
|  | ||||
|     def has_perm(self, username: str, perm: int, path: Optional[str] = None) -> bool: | ||||
|         return True  # handled at filesystem layer | ||||
|  | ||||
|     def get_perms(self, username: str) -> str: | ||||
|         return "elradfmwMT" | ||||
|  | ||||
|     def get_msg_login(self, username: str) -> str: | ||||
|         return "sup {}".format(username) | ||||
|  | ||||
|     def get_msg_quit(self, username: str) -> str: | ||||
|         return "cya" | ||||
|  | ||||
|  | ||||
| class FtpFs(AbstractedFS): | ||||
|     def __init__( | ||||
|         self, root: str, cmd_channel: Any | ||||
|     ) -> None:  # pylint: disable=super-init-not-called | ||||
|         self.h = self.cmd_channel = cmd_channel  # type: FTPHandler | ||||
|         self.hub: "SvcHub" = cmd_channel.hub | ||||
|         self.args = cmd_channel.args | ||||
|  | ||||
|         self.uname = self.hub.asrv.iacct.get(cmd_channel.password, "*") | ||||
|  | ||||
|         self.cwd = "/"  # pyftpdlib convention of leading slash | ||||
|         self.root = "/var/lib/empty" | ||||
|  | ||||
|         self.listdirinfo = self.listdir | ||||
|         self.chdir(".") | ||||
|  | ||||
|     def v2a( | ||||
|         self, | ||||
|         vpath: str, | ||||
|         r: bool = False, | ||||
|         w: bool = False, | ||||
|         m: bool = False, | ||||
|         d: bool = False, | ||||
|     ) -> str: | ||||
|         try: | ||||
|             vpath = vpath.replace("\\", "/").lstrip("/") | ||||
|             vfs, rem = self.hub.asrv.vfs.get(vpath, self.uname, r, w, m, d) | ||||
|             if not vfs.realpath: | ||||
|                 raise FilesystemError("no filesystem mounted at this path") | ||||
|  | ||||
|             return os.path.join(vfs.realpath, rem) | ||||
|         except Pebkac as ex: | ||||
|             raise FilesystemError(str(ex)) | ||||
|  | ||||
|     def rv2a( | ||||
|         self, | ||||
|         vpath: str, | ||||
|         r: bool = False, | ||||
|         w: bool = False, | ||||
|         m: bool = False, | ||||
|         d: bool = False, | ||||
|     ) -> str: | ||||
|         return self.v2a(os.path.join(self.cwd, vpath), r, w, m, d) | ||||
|  | ||||
|     def ftp2fs(self, ftppath: str) -> str: | ||||
|         # return self.v2a(ftppath) | ||||
|         return ftppath  # self.cwd must be vpath | ||||
|  | ||||
|     def fs2ftp(self, fspath: str) -> str: | ||||
|         # raise NotImplementedError() | ||||
|         return fspath | ||||
|  | ||||
|     def validpath(self, path: str) -> bool: | ||||
|         if "/.hist/" in path: | ||||
|             if "/up2k." in path or path.endswith("/dir.txt"): | ||||
|                 raise FilesystemError("access to this file is forbidden") | ||||
|  | ||||
|         return True | ||||
|  | ||||
|     def open(self, filename: str, mode: str) -> typing.IO[Any]: | ||||
|         r = "r" in mode | ||||
|         w = "w" in mode or "a" in mode or "+" in mode | ||||
|  | ||||
|         ap = self.rv2a(filename, r, w) | ||||
|         if w and bos.path.exists(ap): | ||||
|             raise FilesystemError("cannot open existing file for writing") | ||||
|  | ||||
|         self.validpath(ap) | ||||
|         return open(fsenc(ap), mode) | ||||
|  | ||||
|     def chdir(self, path: str) -> None: | ||||
|         self.cwd = join(self.cwd, path) | ||||
|         x = self.hub.asrv.vfs.can_access(self.cwd.lstrip("/"), self.h.username) | ||||
|         self.can_read, self.can_write, self.can_move, self.can_delete, self.can_get = x | ||||
|  | ||||
|     def mkdir(self, path: str) -> None: | ||||
|         ap = self.rv2a(path, w=True) | ||||
|         bos.mkdir(ap) | ||||
|  | ||||
|     def listdir(self, path: str) -> list[str]: | ||||
|         vpath = join(self.cwd, path).lstrip("/") | ||||
|         try: | ||||
|             vfs, rem = self.hub.asrv.vfs.get(vpath, self.uname, True, False) | ||||
|  | ||||
|             fsroot, vfs_ls1, vfs_virt = vfs.ls( | ||||
|                 rem, self.uname, not self.args.no_scandir, [[True], [False, True]] | ||||
|             ) | ||||
|             vfs_ls = [x[0] for x in vfs_ls1] | ||||
|             vfs_ls.extend(vfs_virt.keys()) | ||||
|  | ||||
|             if not self.args.ed: | ||||
|                 vfs_ls = exclude_dotfiles(vfs_ls) | ||||
|  | ||||
|             vfs_ls.sort() | ||||
|             return vfs_ls | ||||
|         except: | ||||
|             if vpath: | ||||
|                 # display write-only folders as empty | ||||
|                 return [] | ||||
|  | ||||
|             # return list of volumes | ||||
|             r = {x.split("/")[0]: 1 for x in self.hub.asrv.vfs.all_vols.keys()} | ||||
|             return list(sorted(list(r.keys()))) | ||||
|  | ||||
|     def rmdir(self, path: str) -> None: | ||||
|         ap = self.rv2a(path, d=True) | ||||
|         bos.rmdir(ap) | ||||
|  | ||||
|     def remove(self, path: str) -> None: | ||||
|         if self.args.no_del: | ||||
|             raise FilesystemError("the delete feature is disabled in server config") | ||||
|  | ||||
|         vp = join(self.cwd, path).lstrip("/") | ||||
|         try: | ||||
|             self.hub.up2k.handle_rm(self.uname, self.h.remote_ip, [vp]) | ||||
|         except Exception as ex: | ||||
|             raise FilesystemError(str(ex)) | ||||
|  | ||||
|     def rename(self, src: str, dst: str) -> None: | ||||
|         if not self.can_move: | ||||
|             raise FilesystemError("not allowed for user " + self.h.username) | ||||
|  | ||||
|         if self.args.no_mv: | ||||
|             t = "the rename/move feature is disabled in server config" | ||||
|             raise FilesystemError(t) | ||||
|  | ||||
|         svp = join(self.cwd, src).lstrip("/") | ||||
|         dvp = join(self.cwd, dst).lstrip("/") | ||||
|         try: | ||||
|             self.hub.up2k.handle_mv(self.uname, svp, dvp) | ||||
|         except Exception as ex: | ||||
|             raise FilesystemError(str(ex)) | ||||
|  | ||||
|     def chmod(self, path: str, mode: str) -> None: | ||||
|         pass | ||||
|  | ||||
|     def stat(self, path: str) -> os.stat_result: | ||||
|         try: | ||||
|             ap = self.rv2a(path, r=True) | ||||
|             return bos.stat(ap) | ||||
|         except: | ||||
|             ap = self.rv2a(path) | ||||
|             st = bos.stat(ap) | ||||
|             if not stat.S_ISDIR(st.st_mode): | ||||
|                 raise | ||||
|  | ||||
|             return st | ||||
|  | ||||
|     def utime(self, path: str, timeval: float) -> None: | ||||
|         ap = self.rv2a(path, w=True) | ||||
|         return bos.utime(ap, (timeval, timeval)) | ||||
|  | ||||
|     def lstat(self, path: str) -> os.stat_result: | ||||
|         ap = self.rv2a(path) | ||||
|         return bos.lstat(ap) | ||||
|  | ||||
|     def isfile(self, path: str) -> bool: | ||||
|         st = self.stat(path) | ||||
|         return stat.S_ISREG(st.st_mode) | ||||
|  | ||||
|     def islink(self, path: str) -> bool: | ||||
|         ap = self.rv2a(path) | ||||
|         return bos.path.islink(ap) | ||||
|  | ||||
|     def isdir(self, path: str) -> bool: | ||||
|         try: | ||||
|             st = self.stat(path) | ||||
|             return stat.S_ISDIR(st.st_mode) | ||||
|         except: | ||||
|             return True | ||||
|  | ||||
|     def getsize(self, path: str) -> int: | ||||
|         ap = self.rv2a(path) | ||||
|         return bos.path.getsize(ap) | ||||
|  | ||||
|     def getmtime(self, path: str) -> float: | ||||
|         ap = self.rv2a(path) | ||||
|         return bos.path.getmtime(ap) | ||||
|  | ||||
|     def realpath(self, path: str) -> str: | ||||
|         return path | ||||
|  | ||||
|     def lexists(self, path: str) -> bool: | ||||
|         ap = self.rv2a(path) | ||||
|         return bos.path.lexists(ap) | ||||
|  | ||||
|     def get_user_by_uid(self, uid: int) -> str: | ||||
|         return "root" | ||||
|  | ||||
|     def get_group_by_uid(self, gid: int) -> str: | ||||
|         return "root" | ||||
|  | ||||
|  | ||||
| class FtpHandler(FTPHandler): | ||||
|     abstracted_fs = FtpFs | ||||
|     hub: "SvcHub" = None | ||||
|     args: argparse.Namespace = None | ||||
|  | ||||
|     def __init__(self, conn: Any, server: Any, ioloop: Any = None) -> None: | ||||
|         self.hub: "SvcHub" = FtpHandler.hub | ||||
|         self.args: argparse.Namespace = FtpHandler.args | ||||
|  | ||||
|         if PY2: | ||||
|             FTPHandler.__init__(self, conn, server, ioloop) | ||||
|         else: | ||||
|             super(FtpHandler, self).__init__(conn, server, ioloop) | ||||
|  | ||||
|         # abspath->vpath mapping to resolve log_transfer paths | ||||
|         self.vfs_map: dict[str, str] = {} | ||||
|  | ||||
|     def ftp_STOR(self, file: str, mode: str = "w") -> Any: | ||||
|         # Optional[str] | ||||
|         vp = join(self.fs.cwd, file).lstrip("/") | ||||
|         ap = self.fs.v2a(vp) | ||||
|         self.vfs_map[ap] = vp | ||||
|         # print("ftp_STOR: {} {} => {}".format(vp, mode, ap)) | ||||
|         ret = FTPHandler.ftp_STOR(self, file, mode) | ||||
|         # print("ftp_STOR: {} {} OK".format(vp, mode)) | ||||
|         return ret | ||||
|  | ||||
|     def log_transfer( | ||||
|         self, | ||||
|         cmd: str, | ||||
|         filename: bytes, | ||||
|         receive: bool, | ||||
|         completed: bool, | ||||
|         elapsed: float, | ||||
|         bytes: int, | ||||
|     ) -> Any: | ||||
|         # None | ||||
|         ap = filename.decode("utf-8", "replace") | ||||
|         vp = self.vfs_map.pop(ap, None) | ||||
|         # print("xfer_end: {} => {}".format(ap, vp)) | ||||
|         if vp: | ||||
|             vp, fn = os.path.split(vp) | ||||
|             vfs, rem = self.hub.asrv.vfs.get(vp, self.username, False, True) | ||||
|             vfs, rem = vfs.get_dbv(rem) | ||||
|             self.hub.up2k.hash_file( | ||||
|                 vfs.realpath, | ||||
|                 vfs.flags, | ||||
|                 rem, | ||||
|                 fn, | ||||
|                 self.remote_ip, | ||||
|                 time.time(), | ||||
|             ) | ||||
|  | ||||
|         return FTPHandler.log_transfer( | ||||
|             self, cmd, filename, receive, completed, elapsed, bytes | ||||
|         ) | ||||
|  | ||||
|  | ||||
| try: | ||||
|     from pyftpdlib.handlers import TLS_FTPHandler | ||||
|  | ||||
|     class SftpHandler(FtpHandler, TLS_FTPHandler): | ||||
|         pass | ||||
|  | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class Ftpd(object): | ||||
|     def __init__(self, hub: "SvcHub") -> None: | ||||
|         self.hub = hub | ||||
|         self.args = hub.args | ||||
|  | ||||
|         hs = [] | ||||
|         if self.args.ftp: | ||||
|             hs.append([FtpHandler, self.args.ftp]) | ||||
|         if self.args.ftps: | ||||
|             try: | ||||
|                 h1 = SftpHandler | ||||
|             except: | ||||
|                 t = "\nftps requires pyopenssl;\nplease run the following:\n\n  {} -m pip install --user pyopenssl\n" | ||||
|                 print(t.format(sys.executable)) | ||||
|                 sys.exit(1) | ||||
|  | ||||
|             h1.certfile = os.path.join(E.cfg, "cert.pem") | ||||
|             h1.tls_control_required = True | ||||
|             h1.tls_data_required = True | ||||
|  | ||||
|             hs.append([h1, self.args.ftps]) | ||||
|  | ||||
|         for h_lp in hs: | ||||
|             h2, lp = h_lp | ||||
|             h2.hub = hub | ||||
|             h2.args = hub.args | ||||
|             h2.authorizer = FtpAuth(hub) | ||||
|  | ||||
|             if self.args.ftp_pr: | ||||
|                 p1, p2 = [int(x) for x in self.args.ftp_pr.split("-")] | ||||
|                 if self.args.ftp and self.args.ftps: | ||||
|                     # divide port range in half | ||||
|                     d = int((p2 - p1) / 2) | ||||
|                     if lp == self.args.ftp: | ||||
|                         p2 = p1 + d | ||||
|                     else: | ||||
|                         p1 += d + 1 | ||||
|  | ||||
|                 h2.passive_ports = list(range(p1, p2 + 1)) | ||||
|  | ||||
|             if self.args.ftp_nat: | ||||
|                 h2.masquerade_address = self.args.ftp_nat | ||||
|  | ||||
|         if self.args.ftp_dbg: | ||||
|             config_logging(level=logging.DEBUG) | ||||
|  | ||||
|         ioloop = IOLoop() | ||||
|         for ip in self.args.i: | ||||
|             for h, lp in hs: | ||||
|                 FTPServer((ip, int(lp)), h, ioloop) | ||||
|  | ||||
|         thr = threading.Thread(target=ioloop.loop) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|  | ||||
| def join(p1: str, p2: str) -> str: | ||||
|     w = os.path.join(p1, p2.replace("\\", "/")) | ||||
|     return os.path.normpath(w).replace("\\", "/") | ||||
							
								
								
									
										2435
									
								
								copyparty/httpcli.py
									
									
									
									
									
								
							
							
						
						
									
										2435
									
								
								copyparty/httpcli.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,29 +1,36 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import argparse  # typechk | ||||
| import os | ||||
| import sys | ||||
| import ssl | ||||
| import re | ||||
| import socket | ||||
| import threading  # typechk | ||||
| import time | ||||
|  | ||||
| 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) | ||||
|     HAVE_SSL = True | ||||
|     import ssl | ||||
| except: | ||||
|     HAVE_SSL = False | ||||
|  | ||||
| from .__init__ import E | ||||
| from .util import Unrecv | ||||
| from . import util as Util | ||||
| from .__init__ import TYPE_CHECKING, E | ||||
| from .authsrv import AuthSrv  # typechk | ||||
| from .httpcli import HttpCli | ||||
| from .ico import Ico | ||||
| from .mtag import HAVE_FFMPEG | ||||
| from .th_cli import ThumbCli | ||||
| from .th_srv import HAVE_PIL, HAVE_VIPS | ||||
| from .u2idx import U2idx | ||||
|  | ||||
| try: | ||||
|     from typing import Optional, Pattern, Union | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from .httpsrv import HttpSrv | ||||
|  | ||||
|  | ||||
| class HttpConn(object): | ||||
| @@ -32,70 +39,153 @@ class HttpConn(object): | ||||
|     creates an HttpCli for each request (Connection: Keep-Alive) | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, sck, addr, hsrv): | ||||
|     def __init__( | ||||
|         self, sck: socket.socket, addr: tuple[str, int], hsrv: "HttpSrv" | ||||
|     ) -> None: | ||||
|         self.s = sck | ||||
|         self.sr: Optional[Util._Unrecv] = None | ||||
|         self.addr = addr | ||||
|         self.hsrv = hsrv | ||||
|  | ||||
|         self.args = hsrv.args | ||||
|         self.auth = hsrv.auth | ||||
|         self.mutex: threading.Lock = hsrv.mutex  # mypy404 | ||||
|         self.args: argparse.Namespace = hsrv.args  # mypy404 | ||||
|         self.asrv: AuthSrv = hsrv.asrv  # mypy404 | ||||
|         self.cert_path = hsrv.cert_path | ||||
|         self.u2fh: Util.FHC = hsrv.u2fh  # mypy404 | ||||
|  | ||||
|         self.workload = 0 | ||||
|         self.log_func = hsrv.log | ||||
|         self.log_src = "{} \033[36m{}".format(addr[0], addr[1]).ljust(26) | ||||
|         enth = (HAVE_PIL or HAVE_VIPS or HAVE_FFMPEG) and not self.args.no_thumb | ||||
|         self.thumbcli: Optional[ThumbCli] = ThumbCli(hsrv) if enth else None  # mypy404 | ||||
|         self.ico: Ico = Ico(self.args)  # mypy404 | ||||
|  | ||||
|         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: float = time.time()  # mypy404 | ||||
|         self.stopping = False | ||||
|         self.nreq: int = 0  # mypy404 | ||||
|         self.nbyte: int = 0  # mypy404 | ||||
|         self.u2idx: Optional[U2idx] = None | ||||
|         self.log_func: "Util.RootLogger" = hsrv.log  # mypy404 | ||||
|         self.log_src: str = "httpconn"  # mypy404 | ||||
|         self.lf_url: Optional[Pattern[str]] = ( | ||||
|             re.compile(self.args.lf_url) if self.args.lf_url else None | ||||
|         )  # mypy404 | ||||
|         self.set_rproxy() | ||||
|  | ||||
|     def respath(self, res_name): | ||||
|     def shutdown(self) -> None: | ||||
|         self.stopping = True | ||||
|         try: | ||||
|             self.s.shutdown(socket.SHUT_RDWR) | ||||
|             self.s.close() | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def set_rproxy(self, ip: Optional[str] = None) -> str: | ||||
|         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: str) -> str: | ||||
|         return os.path.join(E.mod, "web", res_name) | ||||
|  | ||||
|     def log(self, msg): | ||||
|         self.log_func(self.log_src, msg) | ||||
|     def log(self, msg: str, c: Union[int, str] = 0) -> None: | ||||
|         self.log_func(self.log_src, msg, c) | ||||
|  | ||||
|     def run(self): | ||||
|     def get_u2idx(self) -> U2idx: | ||||
|         # one u2idx per tcp connection; | ||||
|         # sqlite3 fully parallelizes under python threads | ||||
|         if not self.u2idx: | ||||
|             self.u2idx = U2idx(self) | ||||
|  | ||||
|         return self.u2idx | ||||
|  | ||||
|     def _detect_https(self) -> bool: | ||||
|         method = None | ||||
|         self.sr = None | ||||
|         if self.cert_path: | ||||
|             try: | ||||
|                 method = self.s.recv(4, socket.MSG_PEEK) | ||||
|             except socket.timeout: | ||||
|                 return | ||||
|                 return False | ||||
|             except AttributeError: | ||||
|                 # jython does not support msg_peek; forget about https | ||||
|                 method = self.s.recv(4) | ||||
|                 self.sr = Unrecv(self.s) | ||||
|                 self.sr = Util.Unrecv(self.s, self.log) | ||||
|                 self.sr.buf = method | ||||
|  | ||||
|                 # jython used to do this, they stopped since it's broken | ||||
|                 # but reimplementing sendall is out of scope for now | ||||
|                 if not getattr(self.s, "sendall", None): | ||||
|                     self.s.sendall = self.s.send | ||||
|                     self.s.sendall = self.s.send  # type: ignore | ||||
|  | ||||
|             if len(method) != 4: | ||||
|                 err = "need at least 4 bytes in the first packet; got {}".format( | ||||
|                     len(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: | ||||
|                     self.log(err) | ||||
|  | ||||
|         if method not in [None, b"GET ", b"HEAD", b"POST", b"PUT ", b"OPTI"]: | ||||
|                 self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8")) | ||||
|                 return False | ||||
|  | ||||
|         return method not in [None, b"GET ", b"HEAD", b"POST", b"PUT ", b"OPTI"] | ||||
|  | ||||
|     def run(self) -> None: | ||||
|         self.sr = None | ||||
|         if self.args.https_only: | ||||
|             is_https = True | ||||
|         elif self.args.http_only or not HAVE_SSL: | ||||
|             is_https = False | ||||
|         else: | ||||
|             # raise Exception("asdf") | ||||
|             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())  # type: ignore | ||||
|                 ] | ||||
|                 self.log(" ".join(msg) + "\033[0m") | ||||
|  | ||||
|                 if self.args.ssl_dbg and hasattr(self.s, "shared_ciphers"): | ||||
|                     ciphers = self.s.shared_ciphers() | ||||
|                     assert ciphers | ||||
|                     overlap = [str(y[::-1]) for y in ciphers] | ||||
|                     self.log("TLS cipher overlap:" + "\n".join(overlap)) | ||||
|                     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 +194,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) | ||||
|             self.sr = Util.Unrecv(self.s, self.log) | ||||
|  | ||||
|         while True: | ||||
|         while not self.stopping: | ||||
|             self.nreq += 1 | ||||
|             cli = HttpCli(self) | ||||
|             if not cli.run(): | ||||
|                 return | ||||
|   | ||||
| @@ -1,14 +1,45 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import base64 | ||||
| import math | ||||
| import os | ||||
| import time | ||||
| import socket | ||||
| import sys | ||||
| import threading | ||||
| import time | ||||
|  | ||||
| from .__init__ import E, MACOS | ||||
| import queue | ||||
|  | ||||
| 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 MACOS, TYPE_CHECKING, E | ||||
| from .bos import bos | ||||
| from .httpconn import HttpConn | ||||
| from .authsrv import AuthSrv | ||||
| from .util import FHC, min_ex, spack, start_log_thrs, start_stackmon | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from .broker_util import BrokerCli | ||||
|  | ||||
| try: | ||||
|     from typing import Any, Optional | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class HttpSrv(object): | ||||
| @@ -17,105 +48,319 @@ class HttpSrv(object): | ||||
|     relying on MpSrv for performance (HttpSrv is just plain threads) | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, broker): | ||||
|     def __init__(self, broker: "BrokerCli", nid: Optional[int]) -> None: | ||||
|         self.broker = broker | ||||
|         self.nid = nid | ||||
|         self.args = broker.args | ||||
|         self.log = broker.log | ||||
|         self.asrv = broker.asrv | ||||
|  | ||||
|         self.disconnect_func = None | ||||
|         nsuf = "-n{}-i{:x}".format(nid, os.getpid()) if nid else "" | ||||
|  | ||||
|         self.name = "hsrv" + nsuf | ||||
|         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 = 0.0  # latest worker collect | ||||
|         self.tp_q: Optional[queue.LifoQueue[Any]] = ( | ||||
|             None if self.args.no_htp else queue.LifoQueue() | ||||
|         ) | ||||
|         self.t_periodic: Optional[threading.Thread] = None | ||||
|  | ||||
|         self.u2fh = FHC() | ||||
|         self.srvs: list[socket.socket] = [] | ||||
|         self.ncli = 0  # exact | ||||
|         self.clients: set[HttpConn] = set()  # laggy | ||||
|         self.nclimax = 0 | ||||
|         self.cb_ts = 0.0 | ||||
|         self.cb_v = "" | ||||
|  | ||||
|         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", "cf"] | ||||
|         } | ||||
|         self.prism = os.path.exists(os.path.join(E.mod, "web", "deps", "prism.js.gz")) | ||||
|  | ||||
|         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 | ||||
|             self.cert_path = "" | ||||
|  | ||||
|     def accept(self, sck, addr): | ||||
|         if self.tp_q: | ||||
|             self.start_threads(4) | ||||
|  | ||||
|         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) | ||||
|  | ||||
|         self.th_cfg: dict[str, Any] = {} | ||||
|         t = threading.Thread(target=self.post_init) | ||||
|         t.daemon = True | ||||
|         t.start() | ||||
|  | ||||
|     def post_init(self) -> None: | ||||
|         try: | ||||
|             x = self.broker.ask("thumbsrv.getcfg") | ||||
|             self.th_cfg = x.get() | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def start_threads(self, n: int) -> None: | ||||
|         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: int) -> None: | ||||
|         self.tp_nthr -= n | ||||
|         if self.args.log_htp: | ||||
|             self.log(self.name, "workers -= {} = {}".format(n, self.tp_nthr), 6) | ||||
|  | ||||
|         assert self.tp_q | ||||
|         for _ in range(n): | ||||
|             self.tp_q.put(None) | ||||
|  | ||||
|     def periodic(self) -> None: | ||||
|         while True: | ||||
|             time.sleep(2 if self.tp_ncli or self.ncli else 10) | ||||
|             with self.mutex: | ||||
|                 self.u2fh.clean() | ||||
|                 if self.tp_q: | ||||
|                     self.tp_ncli = max(self.ncli, self.tp_ncli - 2) | ||||
|                     if self.tp_nthr > self.tp_ncli + 8: | ||||
|                         self.stop_threads(4) | ||||
|  | ||||
|                 if not self.ncli and not self.u2fh.cache and self.tp_nthr <= 8: | ||||
|                     self.t_periodic = None | ||||
|                     return | ||||
|  | ||||
|     def listen(self, sck: socket.socket, nlisteners: int) -> None: | ||||
|         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: socket.socket) -> None: | ||||
|         """listens on a shared tcp server""" | ||||
|         ip, port = srv_sck.getsockname() | ||||
|         fno = srv_sck.fileno() | ||||
|         msg = "subscribed @ {}:{}  f{} p{}".format(ip, port, fno, os.getpid()) | ||||
|         self.log(self.name, msg) | ||||
|  | ||||
|         def fun() -> None: | ||||
|             self.broker.say("cb_httpsrv_up") | ||||
|  | ||||
|         threading.Thread(target=fun).start() | ||||
|  | ||||
|         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: | ||||
|                 t = "|{}C-acc2 \033[0;36m{} \033[3{}m{}".format( | ||||
|                     "-" * 3, ip, port % 8, port | ||||
|                 ) | ||||
|                 self.log("%s %s" % addr, t, c="1;30") | ||||
|  | ||||
|             self.accept(sck, addr) | ||||
|  | ||||
|     def accept(self, sck: socket.socket, addr: tuple[str, int]) -> None: | ||||
|         """takes an incoming tcp connection and creates a thread to handle it""" | ||||
|         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: | ||||
|             t = "httpserver threadpool died: tpt {:.2f}, now {:.2f}, nthr {}, ncli {}" | ||||
|             self.log(self.name, t.format(self.tp_time, now, self.tp_nthr, self.ncli), 1) | ||||
|             self.tp_time = 0 | ||||
|             self.tp_q = None | ||||
|  | ||||
|         with self.mutex: | ||||
|             self.ncli += 1 | ||||
|             if not self.t_periodic: | ||||
|                 name = "hsrv-pt" | ||||
|                 if self.nid: | ||||
|                     name += "-{}".format(self.nid) | ||||
|  | ||||
|                 thr = threading.Thread(target=self.periodic, name=name) | ||||
|                 self.t_periodic = thr | ||||
|                 thr.daemon = True | ||||
|                 thr.start() | ||||
|  | ||||
|             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: | ||||
|             t = "looks like the httpserver threadpool died; please make an issue on github and tell me the story of how you pulled that off, thanks and dog bless\n" | ||||
|             self.log(self.name, t, 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) -> None: | ||||
|         assert self.tp_q | ||||
|         while True: | ||||
|             task = self.tp_q.get() | ||||
|             if not task: | ||||
|                 break | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.log("ok bye") | ||||
|             with self.mutex: | ||||
|                 self.tp_time = 0 | ||||
|  | ||||
|     def thr_client(self, sck, addr): | ||||
|             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 Exception as ex: | ||||
|                 if str(ex).startswith("client d/c "): | ||||
|                     self.log(self.name, "thr_client: " + str(ex), 6) | ||||
|                 else: | ||||
|                     self.log(self.name, "thr_client: " + min_ex(), 3) | ||||
|  | ||||
|     def shutdown(self) -> None: | ||||
|         self.stopping = True | ||||
|         for srv in self.srvs: | ||||
|             try: | ||||
|                 srv.close() | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         clients = list(self.clients) | ||||
|         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: socket.socket, addr: tuple[str, int]) -> None: | ||||
|         """thread managing one tcp client""" | ||||
|         sck.settimeout(120) | ||||
|  | ||||
|         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() | ||||
|             self.clients.add(cli) | ||||
|  | ||||
|         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.clients.remove(cli) | ||||
|                     self.ncli -= 1 | ||||
|  | ||||
|                 if self.disconnect_func: | ||||
|                     self.disconnect_func(addr)  # pylint: disable=not-callable | ||||
|     def cachebuster(self) -> str: | ||||
|         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 argparse  # typechk | ||||
| import colorsys | ||||
| import hashlib | ||||
|  | ||||
| from .__init__ import PY2 | ||||
|  | ||||
|  | ||||
| class Ico(object): | ||||
|     def __init__(self, args: argparse.Namespace) -> None: | ||||
|         self.args = args | ||||
|  | ||||
|     def get(self, ext: str, as_thumb: bool) -> tuple[str, bytes]: | ||||
|         """placeholder to make thumbnails not break""" | ||||
|  | ||||
|         zb = hashlib.md5(ext.encode("utf-8")).digest()[:2] | ||||
|         if PY2: | ||||
|             zb = [ord(x) for x in zb] | ||||
|  | ||||
|         c1 = colorsys.hsv_to_rgb(zb[0] / 256.0, 1, 0.3) | ||||
|         c2 = colorsys.hsv_to_rgb(zb[0] / 256.0, 1, 1) | ||||
|         ci = [int(x * 255) for x in list(c1) + list(c2)] | ||||
|         c = "".join(["{:02x}".format(x) for x in ci]) | ||||
|  | ||||
|         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) | ||||
|  | ||||
|         return "image/svg+xml", svg.encode("utf-8") | ||||
							
								
								
									
										550
									
								
								copyparty/mtag.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										550
									
								
								copyparty/mtag.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,550 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import argparse | ||||
| import json | ||||
| import os | ||||
| import shutil | ||||
| import subprocess as sp | ||||
| import sys | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, unicode | ||||
| from .bos import bos | ||||
| from .util import REKOBO_LKEY, fsenc, min_ex, retchk, runcmd, uncyg | ||||
|  | ||||
| try: | ||||
|     from typing import Any, Union | ||||
|  | ||||
|     from .util import RootLogger | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| def have_ff(cmd: str) -> bool: | ||||
|     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: str) -> None: | ||||
|         self.tag, args = cmdline.split("=", 1) | ||||
|         self.tags = self.tag.split(",") | ||||
|  | ||||
|         self.timeout = 30 | ||||
|         self.force = False | ||||
|         self.kill = "t"  # tree; all children recursively | ||||
|         self.audio = "y" | ||||
|         self.pri = 0  # priority; higher = later | ||||
|         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.startswith("k"): | ||||
|                 self.kill = arg[1:]  # [t]ree [m]ain [n]one | ||||
|                 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 | ||||
|  | ||||
|             if arg.startswith("p"): | ||||
|                 self.pri = int(arg[1:] or "1") | ||||
|                 continue | ||||
|  | ||||
|             raise Exception() | ||||
|  | ||||
|  | ||||
| def ffprobe( | ||||
|     abspath: str, timeout: int = 10 | ||||
| ) -> tuple[dict[str, tuple[int, Any]], dict[str, list[Any]]]: | ||||
|     cmd = [ | ||||
|         b"ffprobe", | ||||
|         b"-hide_banner", | ||||
|         b"-show_streams", | ||||
|         b"-show_format", | ||||
|         b"--", | ||||
|         fsenc(abspath), | ||||
|     ] | ||||
|     rc, so, se = runcmd(cmd, timeout=timeout) | ||||
|     retchk(rc, cmd, se) | ||||
|     return parse_ffprobe(so) | ||||
|  | ||||
|  | ||||
| def parse_ffprobe(txt: str) -> tuple[dict[str, tuple[int, Any]], dict[str, list[Any]]]: | ||||
|     """ffprobe -show_format -show_streams""" | ||||
|     streams = [] | ||||
|     fmt = {} | ||||
|     g = {} | ||||
|     for ln in [x.rstrip("\r") for x in txt.split("\n")]: | ||||
|         try: | ||||
|             sk, sv = ln.split("=", 1) | ||||
|             g[sk] = sv | ||||
|             continue | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         if ln == "[STREAM]": | ||||
|             g = {} | ||||
|             streams.append(g) | ||||
|  | ||||
|         if ln == "[FORMAT]": | ||||
|             g = {"codec_type": "format"}  # heh | ||||
|             fmt = g | ||||
|  | ||||
|     streams = [fmt] + streams | ||||
|     ret: dict[str, Any] = {}  # processed | ||||
|     md: dict[str, list[Any]] = {}  # 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: | ||||
|             v1 = strm.get(sk) | ||||
|             if v1 is None: | ||||
|                 continue | ||||
|  | ||||
|             if rk.startswith("."): | ||||
|                 try: | ||||
|                     zf = float(v1) | ||||
|                     v2 = ret.get(rk) | ||||
|                     if v2 is None or zf > v2: | ||||
|                         ret[rk] = zf | ||||
|                 except: | ||||
|                     # sqlite doesnt care but the code below does | ||||
|                     if v1 not in ["N/A"]: | ||||
|                         ret[rk] = v1 | ||||
|             else: | ||||
|                 ret[rk] = v1 | ||||
|  | ||||
|     if ret.get("vc") == "ansi":  # shellscript | ||||
|         return {}, {} | ||||
|  | ||||
|     for strm in streams: | ||||
|         for sk, sv in strm.items(): | ||||
|             if not sk.startswith("TAG:"): | ||||
|                 continue | ||||
|  | ||||
|             sk = sk[4:].strip() | ||||
|             sv = sv.strip() | ||||
|             if sk and sv and sk not in md: | ||||
|                 md[sk] = [sv] | ||||
|  | ||||
|     for sk in [".q", ".vq", ".aq"]: | ||||
|         if sk in ret: | ||||
|             ret[sk] /= 1000  # bit_rate=320000 | ||||
|  | ||||
|     for sk in [".q", ".vq", ".aq", ".resw", ".resh"]: | ||||
|         if sk in ret: | ||||
|             ret[sk] = int(ret[sk]) | ||||
|  | ||||
|     if ".fps" in ret: | ||||
|         fps = ret[".fps"] | ||||
|         if "/" in fps: | ||||
|             fa, fb = fps.split("/") | ||||
|             try: | ||||
|                 fps = int(fa) * 1.0 / int(fb) | ||||
|             except: | ||||
|                 fps = 9001 | ||||
|  | ||||
|         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"]) | ||||
|  | ||||
|     zd = {k: (0, v) for k, v in ret.items()} | ||||
|  | ||||
|     return zd, md | ||||
|  | ||||
|  | ||||
| class MTag(object): | ||||
|     def __init__(self, log_func: "RootLogger", args: argparse.Namespace) -> None: | ||||
|         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  # noqa: F401  # pylint: disable=unused-import,import-outside-toplevel | ||||
|             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: str, c: Union[int, str] = 0) -> None: | ||||
|         self.log_func("mtag", msg, c) | ||||
|  | ||||
|     def normalize_tags( | ||||
|         self, parser_output: dict[str, tuple[int, Any]], md: dict[str, list[Any]] | ||||
|     ) -> dict[str, Union[str, float]]: | ||||
|         for sk, tv in dict(md).items(): | ||||
|             if not tv: | ||||
|                 continue | ||||
|  | ||||
|             sk = sk.lower().split("::")[0].strip() | ||||
|             key_mapping = self.rmap.get(sk) | ||||
|             if not key_mapping: | ||||
|                 continue | ||||
|  | ||||
|             priority, alias = key_mapping | ||||
|             if alias not in parser_output or parser_output[alias][0] > priority: | ||||
|                 parser_output[alias] = (priority, tv[0]) | ||||
|  | ||||
|         # take first value (lowest priority / most preferred) | ||||
|         ret = {sk: unicode(tv[1]).strip() for sk, tv in parser_output.items()} | ||||
|  | ||||
|         # track 3/7 => track 3 | ||||
|         for sk, tv in ret.items(): | ||||
|             if sk[0] == ".": | ||||
|                 sv = str(tv).split("/")[0].strip().lstrip("0") | ||||
|                 ret[sk] = sv 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: str) -> dict[str, Union[str, float]]: | ||||
|         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: str) -> dict[str, Union[str, float]]: | ||||
|         ret: dict[str, tuple[int, Any]] = {} | ||||
|  | ||||
|         if not bos.path.isfile(abspath): | ||||
|             return {} | ||||
|  | ||||
|         import mutagen | ||||
|  | ||||
|         try: | ||||
|             md = mutagen.File(fsenc(abspath), easy=True) | ||||
|             if not md.info.length and not md.info.codec: | ||||
|                 raise Exception() | ||||
|         except: | ||||
|             return self.get_ffprobe(abspath) if self.can_ffprobe else {} | ||||
|  | ||||
|         sz = bos.path.getsize(abspath) | ||||
|         try: | ||||
|             ret[".q"] = (0, int((sz / md.info.length) / 128)) | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         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: str) -> dict[str, Union[str, float]]: | ||||
|         if not bos.path.isfile(abspath): | ||||
|             return {} | ||||
|  | ||||
|         ret, md = ffprobe(abspath) | ||||
|         return self.normalize_tags(ret, md) | ||||
|  | ||||
|     def get_bin( | ||||
|         self, parsers: dict[str, MParser], abspath: str, oth_tags: dict[str, Any] | ||||
|     ) -> dict[str, Any]: | ||||
|         if not bos.path.isfile(abspath): | ||||
|             return {} | ||||
|  | ||||
|         pypath = os.path.abspath(os.path.dirname(os.path.dirname(__file__))) | ||||
|         zsl = [str(pypath)] + [str(x) for x in sys.path if x] | ||||
|         pypath = str(os.pathsep.join(zsl)) | ||||
|         env = os.environ.copy() | ||||
|         env["PYTHONPATH"] = pypath | ||||
|  | ||||
|         ret: dict[str, Any] = {} | ||||
|         for tagname, parser in sorted(parsers.items(), key=lambda x: (x[1].pri, x[0])): | ||||
|             try: | ||||
|                 cmd = [parser.bin, abspath] | ||||
|                 if parser.bin.endswith(".py"): | ||||
|                     cmd = [sys.executable] + cmd | ||||
|  | ||||
|                 args = {"env": env, "timeout": parser.timeout, "kill": parser.kill} | ||||
|  | ||||
|                 if parser.pri: | ||||
|                     zd = oth_tags.copy() | ||||
|                     zd.update(ret) | ||||
|                     args["sin"] = json.dumps(zd).encode("utf-8", "replace") | ||||
|  | ||||
|                 if WINDOWS: | ||||
|                     args["creationflags"] = 0x4000 | ||||
|                 else: | ||||
|                     cmd = ["nice"] + cmd | ||||
|  | ||||
|                 bcmd = [fsenc(x) for x in cmd] | ||||
|                 rc, v, err = runcmd(bcmd, **args)  # type: ignore | ||||
|                 retchk(rc, bcmd, err, self.log, 5, self.args.mtag_v) | ||||
|                 v = v.strip() | ||||
|                 if not v: | ||||
|                     continue | ||||
|  | ||||
|                 if "," not in tagname: | ||||
|                     ret[tagname] = v | ||||
|                 else: | ||||
|                     zj = json.loads(v) | ||||
|                     for tag in tagname.split(","): | ||||
|                         if tag and tag in zj: | ||||
|                             ret[tag] = zj[tag] | ||||
|             except: | ||||
|                 if self.args.mtag_v: | ||||
|                     t = "mtag error: tagname {}, parser {}, file {} => {}" | ||||
|                     self.log(t.format(tagname, parser.bin, abspath, min_ex())) | ||||
|  | ||||
|         return ret | ||||
							
								
								
									
										117
									
								
								copyparty/star.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								copyparty/star.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,117 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import tarfile | ||||
| import threading | ||||
|  | ||||
| from queue import Queue | ||||
|  | ||||
| from .bos import bos | ||||
| from .sutil import StreamArc, errdesc | ||||
| from .util import fsenc, min_ex | ||||
|  | ||||
| try: | ||||
|     from typing import Any, Generator, Optional | ||||
|  | ||||
|     from .util import NamedLogger | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class QFile(object):  # inherit io.StringIO for painful typing | ||||
|     """file-like object which buffers writes into a queue""" | ||||
|  | ||||
|     def __init__(self) -> None: | ||||
|         self.q: Queue[Optional[bytes]] = Queue(64) | ||||
|         self.bq: list[bytes] = [] | ||||
|         self.nq = 0 | ||||
|  | ||||
|     def write(self, buf: Optional[bytes]) -> None: | ||||
|         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(StreamArc): | ||||
|     """construct in-memory tar file from the given path""" | ||||
|  | ||||
|     def __init__( | ||||
|         self, | ||||
|         log: "NamedLogger", | ||||
|         fgen: Generator[dict[str, Any], None, None], | ||||
|         **kwargs: Any | ||||
|     ): | ||||
|         super(StreamTar, self).__init__(log, fgen) | ||||
|  | ||||
|         self.ci = 0 | ||||
|         self.co = 0 | ||||
|         self.qfile = QFile() | ||||
|         self.errf: dict[str, Any] = {} | ||||
|  | ||||
|         # 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)  # type: ignore | ||||
|  | ||||
|         w = threading.Thread(target=self._gen, name="star-gen") | ||||
|         w.daemon = True | ||||
|         w.start() | ||||
|  | ||||
|     def gen(self) -> Generator[Optional[bytes], None, None]: | ||||
|         try: | ||||
|             while True: | ||||
|                 buf = self.qfile.q.get() | ||||
|                 if not buf: | ||||
|                     break | ||||
|  | ||||
|                 self.co += len(buf) | ||||
|                 yield buf | ||||
|  | ||||
|             yield None | ||||
|         finally: | ||||
|             if self.errf: | ||||
|                 bos.unlink(self.errf["ap"]) | ||||
|  | ||||
|     def ser(self, f: dict[str, Any]) -> None: | ||||
|         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 fo: | ||||
|             self.tar.addfile(inf, fo) | ||||
|  | ||||
|     def _gen(self) -> None: | ||||
|         errors = [] | ||||
|         for f in self.fgen: | ||||
|             if "err" in f: | ||||
|                 errors.append((f["vp"], f["err"])) | ||||
|                 continue | ||||
|  | ||||
|             try: | ||||
|                 self.ser(f) | ||||
|             except: | ||||
|                 ex = min_ex(5, True).replace("\n", "\n-- ") | ||||
|                 errors.append((f["vp"], 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. | ||||
| @@ -10,23 +12,28 @@ Original source: misc/python/surrogateescape.py in https://bitbucket.org/haypo/m | ||||
|  | ||||
| # This code is released under the Python license and the BSD 2-clause license | ||||
|  | ||||
| import platform | ||||
| import codecs | ||||
| import platform | ||||
| import sys | ||||
|  | ||||
| PY3 = sys.version_info[0] > 2 | ||||
| WINDOWS = platform.system() == "Windows" | ||||
| FS_ERRORS = "surrogateescape" | ||||
|  | ||||
| try: | ||||
|     from typing import Any | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| def u(text): | ||||
|  | ||||
| def u(text: Any) -> str: | ||||
|     if PY3: | ||||
|         return text | ||||
|     else: | ||||
|         return text.decode("unicode_escape") | ||||
|  | ||||
|  | ||||
| def b(data): | ||||
| def b(data: Any) -> bytes: | ||||
|     if PY3: | ||||
|         return data.encode("latin1") | ||||
|     else: | ||||
| @@ -41,7 +48,7 @@ else: | ||||
|     bytes_chr = chr | ||||
|  | ||||
|  | ||||
| def surrogateescape_handler(exc): | ||||
| def surrogateescape_handler(exc: Any) -> tuple[str, int]: | ||||
|     """ | ||||
|     Pure Python implementation of the PEP 383: the "surrogateescape" error | ||||
|     handler of Python 3. Undecodable bytes will be replaced by a Unicode | ||||
| @@ -72,7 +79,7 @@ class NotASurrogateError(Exception): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| def replace_surrogate_encode(mystring): | ||||
| def replace_surrogate_encode(mystring: str) -> str: | ||||
|     """ | ||||
|     Returns a (unicode) string, not the more logical bytes, because the codecs | ||||
|     register_error functionality expects this. | ||||
| @@ -98,7 +105,7 @@ def replace_surrogate_encode(mystring): | ||||
|     return str().join(decoded) | ||||
|  | ||||
|  | ||||
| def replace_surrogate_decode(mybytes): | ||||
| def replace_surrogate_decode(mybytes: bytes) -> str: | ||||
|     """ | ||||
|     Returns a (unicode) string | ||||
|     """ | ||||
| @@ -119,7 +126,7 @@ def replace_surrogate_decode(mybytes): | ||||
|     return str().join(decoded) | ||||
|  | ||||
|  | ||||
| def encodefilename(fn): | ||||
| def encodefilename(fn: str) -> bytes: | ||||
|     if FS_ENCODING == "ascii": | ||||
|         # ASCII encoder of Python 2 expects that the error handler returns a | ||||
|         # Unicode string encodable to ASCII, whereas our surrogateescape error | ||||
| @@ -159,7 +166,7 @@ def encodefilename(fn): | ||||
|         return fn.encode(FS_ENCODING, FS_ERRORS) | ||||
|  | ||||
|  | ||||
| def decodefilename(fn): | ||||
| def decodefilename(fn: bytes) -> str: | ||||
|     return fn.decode(FS_ENCODING, FS_ERRORS) | ||||
|  | ||||
|  | ||||
| @@ -171,7 +178,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. | ||||
| @@ -179,7 +186,7 @@ if WINDOWS and not PY3: | ||||
| FS_ENCODING = codecs.lookup(FS_ENCODING).name | ||||
|  | ||||
|  | ||||
| def register_surrogateescape(): | ||||
| def register_surrogateescape() -> None: | ||||
|     """ | ||||
|     Registers the surrogateescape error handler on Python 2 (only) | ||||
|     """ | ||||
|   | ||||
							
								
								
									
										48
									
								
								copyparty/sutil.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								copyparty/sutil.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import tempfile | ||||
| from datetime import datetime | ||||
|  | ||||
| from .bos import bos | ||||
|  | ||||
| try: | ||||
|     from typing import Any, Generator, Optional | ||||
|  | ||||
|     from .util import NamedLogger | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class StreamArc(object): | ||||
|     def __init__( | ||||
|         self, | ||||
|         log: "NamedLogger", | ||||
|         fgen: Generator[dict[str, Any], None, None], | ||||
|         **kwargs: Any | ||||
|     ): | ||||
|         self.log = log | ||||
|         self.fgen = fgen | ||||
|  | ||||
|     def gen(self) -> Generator[Optional[bytes], None, None]: | ||||
|         pass | ||||
|  | ||||
|  | ||||
| def errdesc(errors: list[tuple[str, str]]) -> tuple[dict[str, Any], list[str]]: | ||||
|     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 | ||||
| @@ -1,139 +1,505 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import sys | ||||
| import time | ||||
| import threading | ||||
| from datetime import datetime, timedelta | ||||
| import argparse | ||||
| import base64 | ||||
| import calendar | ||||
| import gzip | ||||
| import os | ||||
| import re | ||||
| import shlex | ||||
| import signal | ||||
| import socket | ||||
| import string | ||||
| import sys | ||||
| import threading | ||||
| import time | ||||
| from datetime import datetime, timedelta | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, MACOS, VT100 | ||||
| try: | ||||
|     from types import FrameType | ||||
|  | ||||
|     import typing | ||||
|     from typing import Any, Optional, Union | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| from .__init__ import ANYWIN, MACOS, PY2, VT100, WINDOWS, E, unicode | ||||
| from .authsrv import AuthSrv | ||||
| from .mtag import HAVE_FFMPEG, HAVE_FFPROBE | ||||
| from .tcpsrv import TcpSrv | ||||
| from .th_srv import HAVE_PIL, HAVE_VIPS, HAVE_WEBP, ThumbSrv | ||||
| from .up2k import Up2k | ||||
| from .util import mp | ||||
| from .util import ( | ||||
|     VERSIONS, | ||||
|     alltrace, | ||||
|     ansi_re, | ||||
|     min_ex, | ||||
|     mp, | ||||
|     start_log_thrs, | ||||
|     start_stackmon, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class SvcHub(object): | ||||
|     """ | ||||
|     Hosts all services which cannot be parallelized due to reliance on monolithic resources. | ||||
|     Creates a Broker which does most of the heavy stuff; hosted services can use this to perform work: | ||||
|         hub.broker.put(want_reply, destination, args_list). | ||||
|         hub.broker.<say|ask>(destination, args_list). | ||||
|  | ||||
|     Either BrokerThr (plain threads) or BrokerMP (multiprocessing) is used depending on configuration. | ||||
|     Nothing is returned synchronously; if you want any value returned from the call, | ||||
|     put() can return a queue (if want_reply=True) which has a blocking get() with the response. | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, args): | ||||
|     def __init__(self, args: argparse.Namespace, argv: list[str], printed: str) -> None: | ||||
|         self.args = args | ||||
|         self.argv = argv | ||||
|         self.logf: Optional[typing.TextIO] = None | ||||
|         self.logf_base_fn = "" | ||||
|         self.stop_req = False | ||||
|         self.stopping = False | ||||
|         self.stopped = False | ||||
|         self.reload_req = False | ||||
|         self.reloading = False | ||||
|         self.stop_cond = threading.Condition() | ||||
|         self.nsigs = 3 | ||||
|         self.retcode = 0 | ||||
|         self.httpsrv_up = 0 | ||||
|  | ||||
|         self.ansi_re = re.compile("\033\\[[^m]*m") | ||||
|         self.log_mutex = threading.Lock() | ||||
|         self.next_day = 0 | ||||
|         self.tstack = 0.0 | ||||
|  | ||||
|         if args.sss or args.s >= 3: | ||||
|             args.ss = True | ||||
|             args.lo = args.lo or "cpp-%Y-%m%d-%H%M%S.txt.xz" | ||||
|             args.ls = args.ls or "**,*,ln,p,r" | ||||
|  | ||||
|         if args.ss or args.s >= 2: | ||||
|             args.s = True | ||||
|             args.no_logues = True | ||||
|             args.no_readme = True | ||||
|             args.unpost = 0 | ||||
|             args.no_del = True | ||||
|             args.no_mv = True | ||||
|             args.hardlink = True | ||||
|             args.vague_403 = True | ||||
|             args.nih = True | ||||
|  | ||||
|         if args.s: | ||||
|             args.dotpart = True | ||||
|             args.no_thumb = True | ||||
|             args.no_mtag_ff = True | ||||
|             args.no_robots = True | ||||
|             args.force_js = True | ||||
|  | ||||
|         self.log = self._log_disabled if args.q else self._log_enabled | ||||
|         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) | ||||
|  | ||||
|         if not args.use_fpool and args.j != 1: | ||||
|             args.no_fpool = True | ||||
|             t = "multithreading enabled with -j {}, so disabling fpool -- this can reduce upload performance on some filesystems" | ||||
|             self.log("root", t.format(args.j)) | ||||
|  | ||||
|         if not args.no_fpool and args.j != 1: | ||||
|             t = "WARNING: --use-fpool combined with multithreading is untested and can probably cause undefined behavior" | ||||
|             if ANYWIN: | ||||
|                 t = 'windows cannot do multithreading without --no-fpool, so enabling that -- note that upload performance will suffer if you have microsoft defender "real-time protection" enabled, so you probably want to use -j 1 instead' | ||||
|                 args.no_fpool = True | ||||
|  | ||||
|             self.log("root", t, c=3) | ||||
|  | ||||
|         bri = "zy"[args.theme % 2 :][:1] | ||||
|         ch = "abcdefghijklmnopqrstuvwx"[int(args.theme / 2)] | ||||
|         args.theme = "{0}{1} {0} {1}".format(ch, bri) | ||||
|  | ||||
|         if not args.hardlink and args.never_symlink: | ||||
|             args.no_dedup = True | ||||
|  | ||||
|         if args.log_fk: | ||||
|             args.log_fk = re.compile(args.log_fk) | ||||
|  | ||||
|         # 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) | ||||
|  | ||||
|         decs = {k: 1 for k in self.args.th_dec.split(",")} | ||||
|         if not HAVE_VIPS: | ||||
|             decs.pop("vips", None) | ||||
|         if not HAVE_PIL: | ||||
|             decs.pop("pil", None) | ||||
|         if not HAVE_FFMPEG or not HAVE_FFPROBE: | ||||
|             decs.pop("ff", None) | ||||
|  | ||||
|         self.args.th_dec = list(decs.keys()) | ||||
|         self.thumbsrv = None | ||||
|         if not args.no_thumb: | ||||
|             t = "decoder preference: {}".format(", ".join(self.args.th_dec)) | ||||
|             self.log("thumb", t) | ||||
|  | ||||
|             if "pil" in self.args.th_dec and not HAVE_WEBP: | ||||
|                 msg = "disabling webp thumbnails because either libwebp is not available or your Pillow is too old" | ||||
|                 self.log("thumb", msg, c=3) | ||||
|  | ||||
|             if self.args.th_dec: | ||||
|                 self.thumbsrv = ThumbSrv(self) | ||||
|             else: | ||||
|                 msg = "need either Pillow, pyvips, or FFmpeg to create thumbnails; for example:\n{0}{1} -m pip install --user Pillow\n{0}{1} -m pip install --user pyvips\n{0}apt install ffmpeg" | ||||
|                 msg = msg.format(" " * 37, os.path.basename(sys.executable)) | ||||
|                 self.log("thumb", msg, c=3) | ||||
|  | ||||
|         if not args.no_acode and args.no_thumb: | ||||
|             msg = "setting --no-acode because --no-thumb (sorry)" | ||||
|             self.log("thumb", msg, c=6) | ||||
|             args.no_acode = True | ||||
|  | ||||
|         if not args.no_acode and (not HAVE_FFMPEG or not HAVE_FFPROBE): | ||||
|             msg = "setting --no-acode because either FFmpeg or FFprobe is not available" | ||||
|             self.log("thumb", msg, c=6) | ||||
|             args.no_acode = True | ||||
|  | ||||
|         args.th_poke = min(args.th_poke, args.th_maxage, args.ac_maxage) | ||||
|  | ||||
|         if args.ftp or args.ftps: | ||||
|             from .ftpd import Ftpd | ||||
|  | ||||
|             self.ftpd = Ftpd(self) | ||||
|  | ||||
|         # decide which worker impl to use | ||||
|         if self.check_mp_enable(): | ||||
|             from .broker_mp import BrokerMp as Broker | ||||
|         else: | ||||
|             self.log("root", "cannot efficiently use multiple CPU cores") | ||||
|             from .broker_thr import BrokerThr as Broker | ||||
|             from .broker_thr import BrokerThr as Broker  # type: ignore | ||||
|  | ||||
|         self.broker = Broker(self) | ||||
|  | ||||
|     def run(self): | ||||
|         thr = threading.Thread(target=self.tcpsrv.run) | ||||
|     def thr_httpsrv_up(self) -> None: | ||||
|         time.sleep(1 if self.args.ign_ebind_all else 5) | ||||
|         expected = self.broker.num_workers * self.tcpsrv.nsrv | ||||
|         failed = expected - self.httpsrv_up | ||||
|         if not failed: | ||||
|             return | ||||
|  | ||||
|         if self.args.ign_ebind_all: | ||||
|             if not self.tcpsrv.srv: | ||||
|                 for _ in range(self.broker.num_workers): | ||||
|                     self.broker.say("cb_httpsrv_up") | ||||
|             return | ||||
|  | ||||
|         if self.args.ign_ebind and self.tcpsrv.srv: | ||||
|             return | ||||
|  | ||||
|         t = "{}/{} workers failed to start" | ||||
|         t = t.format(failed, expected) | ||||
|         self.log("root", t, 1) | ||||
|  | ||||
|         self.retcode = 1 | ||||
|         os.kill(os.getpid(), signal.SIGTERM) | ||||
|  | ||||
|     def cb_httpsrv_up(self) -> None: | ||||
|         self.httpsrv_up += 1 | ||||
|         if self.httpsrv_up != self.broker.num_workers: | ||||
|             return | ||||
|  | ||||
|         time.sleep(0.1)  # purely cosmetic dw | ||||
|         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 | ||||
|     def _logname(self) -> str: | ||||
|         dt = datetime.utcnow() | ||||
|         fn = str(self.args.lo) | ||||
|         for fs in "YmdHMS": | ||||
|             fs = "%" + fs | ||||
|             if fs in fn: | ||||
|                 fn = fn.replace(fs, dt.strftime(fs)) | ||||
|  | ||||
|         return fn | ||||
|  | ||||
|     def _setup_logfile(self, printed: str) -> None: | ||||
|         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: | ||||
|             while True: | ||||
|                 time.sleep(9001) | ||||
|             import lzma | ||||
|  | ||||
|         except KeyboardInterrupt: | ||||
|             with self.log_mutex: | ||||
|                 print("OPYTHAT") | ||||
|             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") | ||||
|  | ||||
|         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 | ||||
|         self.logf_base_fn = base_fn | ||||
|         print(msg, end="") | ||||
|  | ||||
|     def run(self) -> None: | ||||
|         self.tcpsrv.run() | ||||
|  | ||||
|         thr = threading.Thread(target=self.thr_httpsrv_up) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|         sigs = [signal.SIGINT, signal.SIGTERM] | ||||
|         if not ANYWIN: | ||||
|             sigs.append(signal.SIGUSR1) | ||||
|  | ||||
|         for sig in sigs: | ||||
|             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() | ||||
|             # cant join; eats signals on win10 | ||||
|             while not self.stopped: | ||||
|                 time.sleep(0.1) | ||||
|         else: | ||||
|             self.stop_thr() | ||||
|  | ||||
|     def reload(self) -> str: | ||||
|         if self.reloading: | ||||
|             return "cannot reload; already in progress" | ||||
|  | ||||
|         self.reloading = True | ||||
|         t = threading.Thread(target=self._reload) | ||||
|         t.daemon = True | ||||
|         t.start() | ||||
|         return "reload initiated" | ||||
|  | ||||
|     def _reload(self) -> None: | ||||
|         self.log("root", "reload scheduled") | ||||
|         with self.up2k.mutex: | ||||
|             self.asrv.reload() | ||||
|             self.up2k.reload() | ||||
|             self.broker.reload() | ||||
|  | ||||
|         self.reloading = False | ||||
|  | ||||
|     def stop_thr(self) -> None: | ||||
|         while not self.stop_req: | ||||
|             with self.stop_cond: | ||||
|                 self.stop_cond.wait(9001) | ||||
|  | ||||
|             if self.reload_req: | ||||
|                 self.reload_req = False | ||||
|                 self.reload() | ||||
|  | ||||
|         self.shutdown() | ||||
|  | ||||
|     def signal_handler(self, sig: int, frame: Optional[FrameType]) -> None: | ||||
|         if self.stopping: | ||||
|             if self.nsigs <= 0: | ||||
|                 try: | ||||
|                     threading.Thread(target=self.pr, args=("OMBO BREAKER",)).start() | ||||
|                     time.sleep(0.1) | ||||
|                 except: | ||||
|                     pass | ||||
|  | ||||
|                 if ANYWIN: | ||||
|                     os.system("taskkill /f /pid {}".format(os.getpid())) | ||||
|                 else: | ||||
|                     os.kill(os.getpid(), signal.SIGKILL) | ||||
|             else: | ||||
|                 self.nsigs -= 1 | ||||
|                 return | ||||
|  | ||||
|         if not ANYWIN and sig == signal.SIGUSR1: | ||||
|             self.reload_req = True | ||||
|         else: | ||||
|             self.stop_req = True | ||||
|  | ||||
|         with self.stop_cond: | ||||
|             self.stop_cond.notify_all() | ||||
|  | ||||
|     def shutdown(self) -> None: | ||||
|         if self.stopping: | ||||
|             return | ||||
|  | ||||
|         # start_log_thrs(print, 0.1, 1) | ||||
|  | ||||
|         self.stopping = True | ||||
|         self.stop_req = True | ||||
|         with self.stop_cond: | ||||
|             self.stop_cond.notify_all() | ||||
|  | ||||
|         ret = 1 | ||||
|         try: | ||||
|             self.pr("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: | ||||
|                         self.pr("waiting for thumbsrv (10sec)...") | ||||
|  | ||||
|             self.pr("nailed it", end="") | ||||
|             ret = self.retcode | ||||
|         except: | ||||
|             self.pr("\033[31m[ error during shutdown ]\n{}\033[0m".format(min_ex())) | ||||
|             raise | ||||
|         finally: | ||||
|             if self.args.wintitle: | ||||
|                 print("\033]0;\033\\", file=sys.stderr, end="") | ||||
|                 sys.stderr.flush() | ||||
|  | ||||
|             self.pr("\033[0m") | ||||
|             if self.logf: | ||||
|                 self.logf.close() | ||||
|  | ||||
|             self.stopped = True | ||||
|             sys.exit(ret) | ||||
|  | ||||
|     def _log_disabled(self, src: str, msg: str, c: Union[int, str] = 0) -> None: | ||||
|         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) -> None: | ||||
|         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: str, msg: str, c: Union[int, str] = 0) -> None: | ||||
|         """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{}\033[0m".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="") | ||||
|  | ||||
|     def check_mp_support(self): | ||||
|             if self.logf: | ||||
|                 self.logf.write(msg) | ||||
|  | ||||
|     def pr(self, *a: Any, **ka: Any) -> None: | ||||
|         with self.log_mutex: | ||||
|             print(*a, **ka) | ||||
|  | ||||
|     def check_mp_support(self) -> str: | ||||
|         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: | ||||
|             x = mp.Queue(1) | ||||
|             x.put(["foo", "bar"]) | ||||
|             x: mp.Queue[tuple[str, str]] = mp.Queue(1) | ||||
|             x.put(("foo", "bar")) | ||||
|             if x.get()[0] != "foo": | ||||
|                 raise Exception() | ||||
|         except: | ||||
|             return "multiprocessing is not supported on your platform;" | ||||
|  | ||||
|         return None | ||||
|         return "" | ||||
|  | ||||
|     def check_mp_enable(self): | ||||
|     def check_mp_enable(self) -> bool: | ||||
|         if self.args.j == 1: | ||||
|             self.log("root", "multiprocessing disabled by argument -j 1;") | ||||
|             return False | ||||
|  | ||||
|         if mp.cpu_count() <= 1: | ||||
|             self.log("svchub", "only one CPU detected; multiprocessing disabled") | ||||
|             return False | ||||
|  | ||||
|         try: | ||||
| @@ -147,5 +513,38 @@ class SvcHub(object): | ||||
|         if not err: | ||||
|             return True | ||||
|         else: | ||||
|             self.log("root", err) | ||||
|             self.log("svchub", err) | ||||
|             self.log("svchub", "cannot efficiently use multiple CPU cores") | ||||
|             return False | ||||
|  | ||||
|     def sd_notify(self) -> None: | ||||
|         try: | ||||
|             zb = os.getenv("NOTIFY_SOCKET") | ||||
|             if not zb: | ||||
|                 return | ||||
|  | ||||
|             addr = unicode(zb) | ||||
|             if addr.startswith("@"): | ||||
|                 addr = "\0" + addr[1:] | ||||
|  | ||||
|             t = "".join(x for x in addr if x in string.printable) | ||||
|             self.log("sd_notify", t) | ||||
|  | ||||
|             sck = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM) | ||||
|             sck.connect(addr) | ||||
|             sck.sendall(b"READY=1") | ||||
|         except: | ||||
|             self.log("sd_notify", min_ex()) | ||||
|  | ||||
|     def log_stacks(self) -> None: | ||||
|         td = time.time() - self.tstack | ||||
|         if td < 300: | ||||
|             self.log("stacks", "cooldown {}".format(td)) | ||||
|             return | ||||
|  | ||||
|         self.tstack = time.time() | ||||
|         zs = "{}\n{}".format(VERSIONS, alltrace()) | ||||
|         zb = zs.encode("utf-8", "replace") | ||||
|         zb = gzip.compress(zb) | ||||
|         zs = base64.b64encode(zb).decode("ascii") | ||||
|         self.log("stacks", zs) | ||||
|   | ||||
							
								
								
									
										315
									
								
								copyparty/szip.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										315
									
								
								copyparty/szip.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,315 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import calendar | ||||
| import time | ||||
| import zlib | ||||
|  | ||||
| from .bos import bos | ||||
| from .sutil import StreamArc, errdesc | ||||
| from .util import min_ex, sanitize_fn, spack, sunpack, yieldfile | ||||
|  | ||||
| try: | ||||
|     from typing import Any, Generator, Optional | ||||
|  | ||||
|     from .util import NamedLogger | ||||
| except: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| def dostime2unix(buf: bytes) -> int: | ||||
|     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 = time.strptime(iso, "%Y-%m-%d %H:%M:%S") | ||||
|     return int(calendar.timegm(dt)) | ||||
|  | ||||
|  | ||||
| def unixtime2dos(ts: int) -> bytes: | ||||
|     tt = time.gmtime(ts + 1) | ||||
|     dy, dm, dd, th, tm, ts = list(tt)[:6] | ||||
|  | ||||
|     bd = ((dy - 1980) << 9) + (dm << 5) + dd | ||||
|     bt = (th << 11) + (tm << 5) + ts // 2 | ||||
|     try: | ||||
|         return spack(b"<HH", bt, bd) | ||||
|     except: | ||||
|         return b"\x00\x00\x21\x00" | ||||
|  | ||||
|  | ||||
| def gen_fdesc(sz: int, crc32: int, z64: bool) -> bytes: | ||||
|     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: Optional[int], | ||||
|     fn: str, | ||||
|     sz: int, | ||||
|     lastmod: int, | ||||
|     utf8: bool, | ||||
|     icrc32: int, | ||||
|     pre_crc: bool, | ||||
| ) -> bytes: | ||||
|     """ | ||||
|     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 icrc32: | ||||
|         crc32 = spack(b"<L", icrc32) | ||||
|     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 (1b compat, 1b os (00 dos, 03 unix)), 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"_") | ||||
|  | ||||
|     # add ntfs (0x24) and/or unix (0x10) extrafields for utc, add z64 if requested | ||||
|     z64_len = len(z64v) * 8 + 4 if z64v else 0 | ||||
|     ret += spack(b"<HH", len(bfn), 0x10 + 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 (spec-ver 1e03) file:644 | ||||
|         # infozip-macos: 0100 0100 0080 (spec-ver 1e03) file:000 | ||||
|         #     win10-zip: 0000 2000 0000 (spec-ver xx00) FILE_ATTRIBUTE_ARCHIVE | ||||
|         ret += b"\x00\x00\x00\x00\xa4\x81"  # unx | ||||
|         # ret += b"\x00\x00\x20\x00\x00\x00"  # fat | ||||
|  | ||||
|         # 4b local-header-ofs | ||||
|         ret += spack(b"<L", min(h_pos, 0xFFFFFFFF)) | ||||
|  | ||||
|     ret += bfn | ||||
|  | ||||
|     # ntfs: type 0a, size 20, rsvd, attr1, len 18, mtime, atime, ctime | ||||
|     # b"\xa3\x2f\x82\x41\x55\x68\xd8\x01"  1652616838.798941100  ~5.861518  132970904387989411  ~58615181 | ||||
|     # nt = int((lastmod + 11644473600) * 10000000) | ||||
|     # ret += spack(b"<HHLHHQQQ", 0xA, 0x20, 0, 1, 0x18, nt, nt, nt) | ||||
|  | ||||
|     # unix: type 0d, size 0c, atime, mtime, uid, gid | ||||
|     ret += spack(b"<HHLLHH", 0xD, 0xC, int(lastmod), int(lastmod), 1000, 1000) | ||||
|  | ||||
|     if z64v: | ||||
|         ret += spack(b"<HH" + b"Q" * len(z64v), 1, len(z64v) * 8, *z64v) | ||||
|  | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def gen_ecdr( | ||||
|     items: list[tuple[str, int, int, int, int]], cdir_pos: int, cdir_end: int | ||||
| ) -> tuple[bytes, bool]: | ||||
|     """ | ||||
|     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: list[tuple[str, int, int, int, int]], cdir_pos: int, cdir_end: int | ||||
| ) -> bytes: | ||||
|     """ | ||||
|     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: int) -> bytes: | ||||
|     """ | ||||
|     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(StreamArc): | ||||
|     def __init__( | ||||
|         self, | ||||
|         log: "NamedLogger", | ||||
|         fgen: Generator[dict[str, Any], None, None], | ||||
|         utf8: bool = False, | ||||
|         pre_crc: bool = False, | ||||
|     ) -> None: | ||||
|         super(StreamZip, self).__init__(log, fgen) | ||||
|  | ||||
|         self.utf8 = utf8 | ||||
|         self.pre_crc = pre_crc | ||||
|  | ||||
|         self.pos = 0 | ||||
|         self.items: list[tuple[str, int, int, int, int]] = [] | ||||
|  | ||||
|     def _ct(self, buf: bytes) -> bytes: | ||||
|         self.pos += len(buf) | ||||
|         return buf | ||||
|  | ||||
|     def ser(self, f: dict[str, Any]) -> Generator[bytes, None, None]: | ||||
|         name = f["vp"] | ||||
|         src = f["ap"] | ||||
|         st = f["st"] | ||||
|  | ||||
|         sz = st.st_size | ||||
|         ts = st.st_mtime | ||||
|  | ||||
|         crc = 0 | ||||
|         if self.pre_crc: | ||||
|             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) | ||||
|  | ||||
|         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) -> Generator[bytes, None, None]: | ||||
|         errors = [] | ||||
|         try: | ||||
|             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 GeneratorExit: | ||||
|                     raise | ||||
|                 except: | ||||
|                     ex = min_ex(5, True).replace("\n", "\n-- ") | ||||
|                     errors.append((f["vp"], 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) | ||||
|         finally: | ||||
|             if errors: | ||||
|                 bos.unlink(errf["ap"]) | ||||
| @@ -1,11 +1,16 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import re | ||||
| import time | ||||
| import socket | ||||
| import sys | ||||
|  | ||||
| from .util import chkcmd, Counter | ||||
| from .__init__ import ANYWIN, MACOS, TYPE_CHECKING, unicode | ||||
| from .util import chkcmd | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from .svchub import SvcHub | ||||
|  | ||||
|  | ||||
| class TcpSrv(object): | ||||
| @@ -14,81 +19,275 @@ class TcpSrv(object): | ||||
|     which then uses the least busy HttpSrv to handle it | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, hub): | ||||
|     def __init__(self, hub: "SvcHub"): | ||||
|         self.hub = hub | ||||
|         self.args = hub.args | ||||
|         self.log = hub.log | ||||
|  | ||||
|         self.num_clients = Counter() | ||||
|         self.stopping = False | ||||
|  | ||||
|         self.srv: list[socket.socket] = [] | ||||
|         self.nsrv = 0 | ||||
|         ok: dict[str, list[int]] = {} | ||||
|         for ip in self.args.i: | ||||
|             ok[ip] = [] | ||||
|             for port in self.args.p: | ||||
|                 self.nsrv += 1 | ||||
|                 try: | ||||
|                     self._listen(ip, port) | ||||
|                     ok[ip].append(port) | ||||
|                 except Exception as ex: | ||||
|                     if self.args.ign_ebind or self.args.ign_ebind_all: | ||||
|                         t = "could not listen on {}:{}: {}" | ||||
|                         self.log("tcpsrv", t.format(ip, port, ex), c=3) | ||||
|                     else: | ||||
|                         raise | ||||
|  | ||||
|         if not self.srv and not self.args.ign_ebind_all: | ||||
|             raise Exception("could not listen on any of the given interfaces") | ||||
|  | ||||
|         if self.nsrv != len(self.srv): | ||||
|             self.log("tcpsrv", "") | ||||
|  | ||||
|         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 = [] | ||||
|         title_tab: dict[str, dict[str, int]] = {} | ||||
|         title_vars = [x[1:] for x in self.args.wintitle.split(" ") if x.startswith("$")] | ||||
|         t = "available @ {}://{}:{}/  (\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): | ||||
|                 if port not in ok.get(ip, ok.get("0.0.0.0", [])): | ||||
|                     continue | ||||
|  | ||||
|         self.srv = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | ||||
|         self.srv.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | ||||
|                 proto = " http" | ||||
|                 if self.args.http_only: | ||||
|                     pass | ||||
|                 elif self.args.https_only or port == 443: | ||||
|                     proto = "https" | ||||
|  | ||||
|                 msgs.append(t.format(proto, ip, port, desc)) | ||||
|  | ||||
|                 if not self.args.wintitle: | ||||
|                     continue | ||||
|  | ||||
|                 if port in [80, 443]: | ||||
|                     ep = ip | ||||
|                 else: | ||||
|                     ep = "{}:{}".format(ip, port) | ||||
|  | ||||
|                 hits = [] | ||||
|                 if "pub" in title_vars and "external" in unicode(desc): | ||||
|                     hits.append(("pub", ep)) | ||||
|  | ||||
|                 if "pub" in title_vars or "all" in title_vars: | ||||
|                     hits.append(("all", ep)) | ||||
|  | ||||
|                 for var in title_vars: | ||||
|                     if var.startswith("ip-") and ep.startswith(var[3:]): | ||||
|                         hits.append((var, ep)) | ||||
|  | ||||
|                 for tk, tv in hits: | ||||
|                     try: | ||||
|                         title_tab[tk][tv] = 1 | ||||
|                     except: | ||||
|                         title_tab[tk] = {tv: 1} | ||||
|  | ||||
|         if msgs: | ||||
|             msgs[-1] += "\n" | ||||
|             for t in msgs: | ||||
|                 self.log("tcpsrv", t) | ||||
|  | ||||
|         if self.args.wintitle: | ||||
|             self._set_wintitle(title_tab) | ||||
|  | ||||
|     def _listen(self, ip: str, port: int) -> None: | ||||
|         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)) | ||||
|             self.srv.append(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 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) | ||||
|  | ||||
|             if ex.errno == 99: | ||||
|                 raise Exception( | ||||
|                     "\033[1;31minterface {} does not exist\033[0m".format(self.args.i) | ||||
|                 ) | ||||
|     def run(self) -> None: | ||||
|         for srv in self.srv: | ||||
|             srv.listen(self.args.nc) | ||||
|             ip, port = srv.getsockname() | ||||
|             fno = srv.fileno() | ||||
|             msg = "listening @ {}:{}  f{} p{}".format(ip, port, fno, os.getpid()) | ||||
|             self.log("tcpsrv", msg) | ||||
|             if self.args.q: | ||||
|                 print(msg) | ||||
|  | ||||
|     def run(self): | ||||
|         self.srv.listen(self.args.nc) | ||||
|             self.hub.broker.say("listen", srv) | ||||
|  | ||||
|         self.log("tcpsrv", "listening @ {0}:{1}".format(self.args.i, self.args.p)) | ||||
|     def shutdown(self) -> None: | ||||
|         self.stopping = True | ||||
|         try: | ||||
|             for srv in self.srv: | ||||
|                 srv.close() | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         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) | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.log("tcpsrv", "ok bye") | ||||
|  | ||||
|     def detect_interfaces(self, listen_ip): | ||||
|         eps = {} | ||||
|  | ||||
|         # get all ips and their interfaces | ||||
|     def ips_linux_ifconfig(self) -> dict[str, str]: | ||||
|         # for termux | ||||
|         try: | ||||
|             ip_addr, _ = chkcmd("ip", "addr") | ||||
|             txt, _ = chkcmd(["ifconfig"]) | ||||
|         except: | ||||
|             ip_addr = None | ||||
|             return {} | ||||
|  | ||||
|         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 | ||||
|         eps: dict[str, str] = {} | ||||
|         dev = None | ||||
|         ip = None | ||||
|         up = None | ||||
|         for ln in (txt + "\n").split("\n"): | ||||
|             if not ln.strip() and dev and ip: | ||||
|                 eps[ip] = dev + ("" if up else ", \033[31mLINK-DOWN") | ||||
|                 dev = ip = up = None | ||||
|                 continue | ||||
|  | ||||
|             if ln == ln.lstrip(): | ||||
|                 dev = re.split(r"[: ]", ln)[0] | ||||
|  | ||||
|             if "UP" in re.split(r"[<>, \t]", ln): | ||||
|                 up = True | ||||
|  | ||||
|             m = re.match(r"^\s+inet\s+([^ ]+)", ln) | ||||
|             if m: | ||||
|                 ip = m.group(1) | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def ips_linux(self) -> dict[str, str]: | ||||
|         try: | ||||
|             txt, _ = chkcmd(["ip", "addr"]) | ||||
|         except: | ||||
|             return self.ips_linux_ifconfig() | ||||
|  | ||||
|         r = re.compile(r"^\s+inet ([^ ]+)/.* (.*)") | ||||
|         ri = re.compile(r"^\s*[0-9]+\s*:.*") | ||||
|         up = False | ||||
|         eps: dict[str, str] = {} | ||||
|         for ln in txt.split("\n"): | ||||
|             if ri.match(ln): | ||||
|                 up = "UP" in re.split("[>,< ]", ln) | ||||
|  | ||||
|             try: | ||||
|                 ip, dev = r.match(ln.rstrip()).groups()  # type: ignore | ||||
|                 eps[ip] = dev + ("" if up else ", \033[31mLINK-DOWN") | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def ips_macos(self) -> dict[str, str]: | ||||
|         eps: dict[str, str] = {} | ||||
|         try: | ||||
|             txt, _ = chkcmd(["ifconfig"]) | ||||
|         except: | ||||
|             return eps | ||||
|  | ||||
|         rdev = re.compile(r"^([^ ]+):") | ||||
|         rip = re.compile(r"^\tinet ([0-9\.]+) ") | ||||
|         dev = "UNKNOWN" | ||||
|         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 = "UNKNOWN" | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def ips_windows_ipconfig(self) -> tuple[dict[str, str], set[str]]: | ||||
|         eps: dict[str, str] = {} | ||||
|         offs: set[str] = set() | ||||
|         try: | ||||
|             txt, _ = chkcmd(["ipconfig"]) | ||||
|         except: | ||||
|             return eps, offs | ||||
|  | ||||
|         rdev = re.compile(r"(^[^ ].*):$") | ||||
|         rip = re.compile(r"^ +IPv?4? [^:]+: *([0-9\.]{7,15})$") | ||||
|         roff = re.compile(r".*: Media disconnected$") | ||||
|         dev = None | ||||
|         for ln in txt.replace("\r", "").split("\n"): | ||||
|             m = rdev.match(ln) | ||||
|             if m: | ||||
|                 if dev and dev not in eps.values(): | ||||
|                     offs.add(dev) | ||||
|  | ||||
|                 dev = m.group(1).split(" adapter ", 1)[-1] | ||||
|  | ||||
|             if dev and roff.match(ln): | ||||
|                 offs.add(dev) | ||||
|                 dev = None | ||||
|  | ||||
|             m = rip.match(ln) | ||||
|             if m and dev: | ||||
|                 eps[m.group(1)] = dev | ||||
|                 dev = None | ||||
|  | ||||
|         if dev and dev not in eps.values(): | ||||
|             offs.add(dev) | ||||
|  | ||||
|         return eps, offs | ||||
|  | ||||
|     def ips_windows_netsh(self) -> dict[str, str]: | ||||
|         eps: dict[str, str] = {} | ||||
|         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 | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def detect_interfaces(self, listen_ips: list[str]) -> dict[str, str]: | ||||
|         if MACOS: | ||||
|             eps = self.ips_macos() | ||||
|         elif ANYWIN: | ||||
|             eps, off = self.ips_windows_ipconfig()  # sees more interfaces + link state | ||||
|             eps.update(self.ips_windows_netsh())  # has better names | ||||
|             for k, v in eps.items(): | ||||
|                 if v in off: | ||||
|                     eps[k] += ", \033[31mLINK-DOWN" | ||||
|         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) | ||||
| @@ -102,7 +301,6 @@ class TcpSrv(object): | ||||
|         ]: | ||||
|             try: | ||||
|                 s.connect((ip, 1)) | ||||
|                 # raise OSError(13, "a") | ||||
|                 default_route = s.getsockname()[0] | ||||
|                 break | ||||
|             except (OSError, socket.error) as ex: | ||||
| @@ -113,11 +311,35 @@ 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 | ||||
|  | ||||
|     def _set_wintitle(self, vs: dict[str, dict[str, int]]) -> None: | ||||
|         vs["all"] = vs.get("all", {"Local-Only": 1}) | ||||
|         vs["pub"] = vs.get("pub", vs["all"]) | ||||
|  | ||||
|         vs2 = {} | ||||
|         for k, eps in vs.items(): | ||||
|             vs2[k] = { | ||||
|                 ep: 1 | ||||
|                 for ep in eps.keys() | ||||
|                 if ":" not in ep or ep.split(":")[0] not in eps | ||||
|             } | ||||
|  | ||||
|         title = "" | ||||
|         vs = vs2 | ||||
|         for p in self.args.wintitle.split(" "): | ||||
|             if p.startswith("$"): | ||||
|                 p = " and ".join(sorted(vs.get(p[1:], {"(None)": 1}).keys())) | ||||
|  | ||||
|             title += "{} ".format(p) | ||||
|  | ||||
|         print("\033]0;{}\033\\".format(title), file=sys.stderr, end="") | ||||
|         sys.stderr.flush() | ||||
|   | ||||
							
								
								
									
										135
									
								
								copyparty/th_cli.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										135
									
								
								copyparty/th_cli.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,135 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
|  | ||||
| from .__init__ import TYPE_CHECKING | ||||
| from .authsrv import VFS | ||||
| from .bos import bos | ||||
| from .th_srv import HAVE_WEBP, thumb_path | ||||
| from .util import Cooldown | ||||
|  | ||||
| try: | ||||
|     from typing import Optional, Union | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from .httpsrv import HttpSrv | ||||
|  | ||||
|  | ||||
| class ThumbCli(object): | ||||
|     def __init__(self, hsrv: "HttpSrv") -> None: | ||||
|         self.broker = hsrv.broker | ||||
|         self.log_func = hsrv.log | ||||
|         self.args = hsrv.args | ||||
|         self.asrv = hsrv.asrv | ||||
|  | ||||
|         # cache on both sides for less broker spam | ||||
|         self.cooldown = Cooldown(self.args.th_poke) | ||||
|  | ||||
|         try: | ||||
|             c = hsrv.th_cfg | ||||
|         except: | ||||
|             c = {k: {} for k in ["thumbable", "pil", "vips", "ffi", "ffv", "ffa"]} | ||||
|  | ||||
|         self.thumbable = c["thumbable"] | ||||
|         self.fmt_pil = c["pil"] | ||||
|         self.fmt_vips = c["vips"] | ||||
|         self.fmt_ffi = c["ffi"] | ||||
|         self.fmt_ffv = c["ffv"] | ||||
|         self.fmt_ffa = c["ffa"] | ||||
|  | ||||
|         # defer args.th_ff_jpg, can change at runtime | ||||
|         d = next((x for x in self.args.th_dec if x in ("vips", "pil")), None) | ||||
|         self.can_webp = HAVE_WEBP or d == "vips" | ||||
|  | ||||
|     def log(self, msg: str, c: Union[int, str] = 0) -> None: | ||||
|         self.log_func("thumbcli", msg, c) | ||||
|  | ||||
|     def get(self, dbv: VFS, rem: str, mtime: float, fmt: str) -> Optional[str]: | ||||
|         ptop = dbv.realpath | ||||
|         ext = rem.rsplit(".")[-1].lower() | ||||
|         if ext not in self.thumbable or "dthumb" in dbv.flags: | ||||
|             return None | ||||
|  | ||||
|         is_vid = ext in self.fmt_ffv | ||||
|         if is_vid and "dvthumb" in dbv.flags: | ||||
|             return None | ||||
|  | ||||
|         want_opus = fmt in ("opus", "caf") | ||||
|         is_au = ext in self.fmt_ffa | ||||
|         if is_au: | ||||
|             if want_opus: | ||||
|                 if self.args.no_acode: | ||||
|                     return None | ||||
|             else: | ||||
|                 if "dathumb" in dbv.flags: | ||||
|                     return None | ||||
|         elif want_opus: | ||||
|             return None | ||||
|  | ||||
|         is_img = not is_vid and not is_au | ||||
|         if is_img and "dithumb" in dbv.flags: | ||||
|             return None | ||||
|  | ||||
|         preferred = self.args.th_dec[0] if self.args.th_dec else "" | ||||
|  | ||||
|         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_img and not self.can_webp) | ||||
|                 or (self.args.th_ff_jpg and (not is_img or preferred == "ff")) | ||||
|             ): | ||||
|                 fmt = "j" | ||||
|  | ||||
|         histpath = self.asrv.vfs.histtab.get(ptop) | ||||
|         if not histpath: | ||||
|             self.log("no histpath for [{}]".format(ptop)) | ||||
|             return None | ||||
|  | ||||
|         tpath = thumb_path(histpath, rem, mtime, fmt) | ||||
|         tpaths = [tpath] | ||||
|         if fmt == "w": | ||||
|             # also check for jpg (maybe webp is unavailable) | ||||
|             tpaths.append(tpath.rsplit(".", 1)[0] + ".jpg") | ||||
|  | ||||
|         ret = None | ||||
|         abort = False | ||||
|         for tp in tpaths: | ||||
|             try: | ||||
|                 st = bos.stat(tp) | ||||
|                 if st.st_size: | ||||
|                     ret = tpath = tp | ||||
|                     fmt = ret.rsplit(".")[1] | ||||
|                 else: | ||||
|                     abort = True | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         if ret: | ||||
|             tdir = os.path.dirname(tpath) | ||||
|             if self.cooldown.poke(tdir): | ||||
|                 self.broker.say("thumbsrv.poke", tdir) | ||||
|  | ||||
|             if want_opus: | ||||
|                 # audio files expire individually | ||||
|                 if self.cooldown.poke(tpath): | ||||
|                     self.broker.say("thumbsrv.poke", tpath) | ||||
|  | ||||
|             return ret | ||||
|  | ||||
|         if abort: | ||||
|             return None | ||||
|  | ||||
|         if not bos.path.getsize(os.path.join(ptop, rem)): | ||||
|             return None | ||||
|  | ||||
|         x = self.broker.ask("thumbsrv.get", ptop, rem, mtime, fmt) | ||||
|         return x.get()  # type: ignore | ||||
							
								
								
									
										630
									
								
								copyparty/th_srv.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										630
									
								
								copyparty/th_srv.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,630 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import base64 | ||||
| import hashlib | ||||
| import os | ||||
| import shutil | ||||
| import subprocess as sp | ||||
| import threading | ||||
| import time | ||||
|  | ||||
| from queue import Queue | ||||
|  | ||||
| from .__init__ import TYPE_CHECKING | ||||
| from .bos import bos | ||||
| from .mtag import HAVE_FFMPEG, HAVE_FFPROBE, ffprobe | ||||
| from .util import BytesIO, Cooldown, fsenc, min_ex, runcmd, statdir, vsplit | ||||
|  | ||||
| try: | ||||
|     from typing import Optional, Union | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from .svchub import SvcHub | ||||
|  | ||||
| HAVE_PIL = False | ||||
| HAVE_HEIF = False | ||||
| HAVE_AVIF = False | ||||
| HAVE_WEBP = False | ||||
|  | ||||
| try: | ||||
|     from PIL import ExifTags, Image, ImageOps | ||||
|  | ||||
|     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  # noqa: F401  # pylint: disable=unused-import | ||||
|  | ||||
|         HAVE_AVIF = True | ||||
|     except: | ||||
|         pass | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| try: | ||||
|     HAVE_VIPS = True | ||||
|     import pyvips | ||||
| except: | ||||
|     HAVE_VIPS = False | ||||
|  | ||||
|  | ||||
| def thumb_path(histpath: str, rem: str, mtime: float, fmt: str) -> str: | ||||
|     # 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] | ||||
|  | ||||
|     if fmt in ("opus", "caf"): | ||||
|         cat = "ac" | ||||
|     else: | ||||
|         fmt = "webp" if fmt == "w" else "jpg" | ||||
|         cat = "th" | ||||
|  | ||||
|     return "{}/{}/{}/{}.{:x}.{}".format(histpath, cat, rd, fn, int(mtime), fmt) | ||||
|  | ||||
|  | ||||
| class ThumbSrv(object): | ||||
|     def __init__(self, hub: "SvcHub") -> None: | ||||
|         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: dict[str, list[threading.Condition]] = {} | ||||
|         self.stopping = False | ||||
|         self.nthr = max(1, self.args.th_mt) | ||||
|  | ||||
|         self.q: Queue[Optional[tuple[str, str]]] = Queue(self.nthr * 4) | ||||
|         for n in range(self.nthr): | ||||
|             thr = threading.Thread( | ||||
|                 target=self.worker, name="thumb-{}-{}".format(n, self.nthr) | ||||
|             ) | ||||
|             thr.daemon = True | ||||
|             thr.start() | ||||
|  | ||||
|         want_ff = not self.args.no_vthumb or not self.args.no_athumb | ||||
|         if want_ff 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 audio/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.cln") | ||||
|             t.daemon = True | ||||
|             t.start() | ||||
|  | ||||
|         self.fmt_pil, self.fmt_vips, self.fmt_ffi, self.fmt_ffv, self.fmt_ffa = [ | ||||
|             set(y.split(",")) | ||||
|             for y in [ | ||||
|                 self.args.th_r_pil, | ||||
|                 self.args.th_r_vips, | ||||
|                 self.args.th_r_ffi, | ||||
|                 self.args.th_r_ffv, | ||||
|                 self.args.th_r_ffa, | ||||
|             ] | ||||
|         ] | ||||
|  | ||||
|         if not HAVE_HEIF: | ||||
|             for f in "heif heifs heic heics".split(" "): | ||||
|                 self.fmt_pil.discard(f) | ||||
|  | ||||
|         if not HAVE_AVIF: | ||||
|             for f in "avif avifs".split(" "): | ||||
|                 self.fmt_pil.discard(f) | ||||
|  | ||||
|         self.thumbable: set[str] = set() | ||||
|  | ||||
|         if "pil" in self.args.th_dec: | ||||
|             self.thumbable |= self.fmt_pil | ||||
|  | ||||
|         if "vips" in self.args.th_dec: | ||||
|             self.thumbable |= self.fmt_vips | ||||
|  | ||||
|         if "ff" in self.args.th_dec: | ||||
|             for zss in [self.fmt_ffi, self.fmt_ffv, self.fmt_ffa]: | ||||
|                 self.thumbable |= zss | ||||
|  | ||||
|     def log(self, msg: str, c: Union[int, str] = 0) -> None: | ||||
|         self.log_func("thumb", msg, c) | ||||
|  | ||||
|     def shutdown(self) -> None: | ||||
|         self.stopping = True | ||||
|         for _ in range(self.nthr): | ||||
|             self.q.put(None) | ||||
|  | ||||
|     def stopped(self) -> bool: | ||||
|         with self.mutex: | ||||
|             return not self.nthr | ||||
|  | ||||
|     def get(self, ptop: str, rem: str, mtime: float, fmt: str) -> Optional[str]: | ||||
|         histpath = self.asrv.vfs.histtab.get(ptop) | ||||
|         if not histpath: | ||||
|             self.log("no histpath for [{}]".format(ptop)) | ||||
|             return None | ||||
|  | ||||
|         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: | ||||
|                 self.poke(tpath) | ||||
|                 return tpath | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         return None | ||||
|  | ||||
|     def getcfg(self) -> dict[str, set[str]]: | ||||
|         return { | ||||
|             "thumbable": self.thumbable, | ||||
|             "pil": self.fmt_pil, | ||||
|             "vips": self.fmt_vips, | ||||
|             "ffi": self.fmt_ffi, | ||||
|             "ffv": self.fmt_ffv, | ||||
|             "ffa": self.fmt_ffa, | ||||
|         } | ||||
|  | ||||
|     def worker(self) -> None: | ||||
|         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): | ||||
|                 for lib in self.args.th_dec: | ||||
|                     if fun: | ||||
|                         break | ||||
|                     elif lib == "pil" and ext in self.fmt_pil: | ||||
|                         fun = self.conv_pil | ||||
|                     elif lib == "vips" and ext in self.fmt_vips: | ||||
|                         fun = self.conv_vips | ||||
|                     elif lib == "ff" and ext in self.fmt_ffi or ext in self.fmt_ffv: | ||||
|                         fun = self.conv_ffmpeg | ||||
|                     elif lib == "ff" and ext in self.fmt_ffa: | ||||
|                         if tpath.endswith(".opus") or tpath.endswith(".caf"): | ||||
|                             fun = self.conv_opus | ||||
|                         else: | ||||
|                             fun = self.conv_spec | ||||
|  | ||||
|             if fun: | ||||
|                 try: | ||||
|                     fun(abspath, tpath) | ||||
|                 except: | ||||
|                     msg = "{} could not create thumbnail of {}\n{}" | ||||
|                     msg = msg.format(fun.__name__, abspath, min_ex()) | ||||
|                     c: Union[str, int] = 1 if "<Signals.SIG" in msg else "1;30" | ||||
|                     self.log(msg, c) | ||||
|                     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: "Image.Image") -> "Image.Image": | ||||
|         # 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: str, tpath: str) -> None: | ||||
|         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_vips(self, abspath: str, tpath: str) -> None: | ||||
|         crops = ["centre", "none"] | ||||
|         if self.args.th_no_crop: | ||||
|             crops = ["none"] | ||||
|  | ||||
|         w, h = self.res | ||||
|         kw = {"height": h, "size": "down", "intent": "relative"} | ||||
|  | ||||
|         for c in crops: | ||||
|             try: | ||||
|                 kw["crop"] = c | ||||
|                 img = pyvips.Image.thumbnail(abspath, w, **kw) | ||||
|                 break | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         img.write_to_file(tpath, Q=40) | ||||
|  | ||||
|     def conv_ffmpeg(self, abspath: str, tpath: str) -> None: | ||||
|         ret, _ = ffprobe(abspath) | ||||
|         if not ret: | ||||
|             return | ||||
|  | ||||
|         ext = abspath.rsplit(".")[-1].lower() | ||||
|         if ext in ["h264", "h265"] or ext in self.fmt_ffi: | ||||
|             seek: list[bytes] = [] | ||||
|         else: | ||||
|             dur = ret[".dur"][1] if ".dur" in ret else 4 | ||||
|             seek = [b"-ss", "{:.0f}".format(dur / 3).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" | ||||
|  | ||||
|         bscale = 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", bscale, | ||||
|             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._run_ff(cmd) | ||||
|  | ||||
|     def _run_ff(self, cmd: list[bytes]) -> None: | ||||
|         # self.log((b" ".join(cmd)).decode("utf-8")) | ||||
|         ret, _, serr = runcmd(cmd, timeout=self.args.th_convt) | ||||
|         if not ret: | ||||
|             return | ||||
|  | ||||
|         c: Union[str, int] = "1;30" | ||||
|         t = "FFmpeg failed (probably a corrupt video file):\n" | ||||
|         if cmd[-1].lower().endswith(b".webp") and ( | ||||
|             "Error selecting an encoder" in serr | ||||
|             or "Automatic encoder selection failed" in serr | ||||
|             or "Default encoder for format webp" in serr | ||||
|             or "Please choose an encoder manually" in serr | ||||
|         ): | ||||
|             self.args.th_ff_jpg = True | ||||
|             t = "FFmpeg failed because it was compiled without libwebp; enabling --th-ff-jpg to force jpeg output:\n" | ||||
|             c = 1 | ||||
|  | ||||
|         if ( | ||||
|             "Requested resampling engine is unavailable" in serr | ||||
|             or "output pad on Parsed_aresample_" in serr | ||||
|         ): | ||||
|             t = "FFmpeg failed because it was compiled without libsox; you must set --th-ff-swr to force swr resampling:\n" | ||||
|             c = 1 | ||||
|  | ||||
|         lines = serr.strip("\n").split("\n") | ||||
|         if len(lines) > 50: | ||||
|             lines = lines[:25] + ["[...]"] + lines[-25:] | ||||
|  | ||||
|         txt = "\n".join(["ff: " + str(x) for x in lines]) | ||||
|         if len(txt) > 5000: | ||||
|             txt = txt[:2500] + "...\nff: [...]\nff: ..." + txt[-2500:] | ||||
|  | ||||
|         self.log(t + txt, c=c) | ||||
|         raise sp.CalledProcessError(ret, (cmd[0], b"...", cmd[-1])) | ||||
|  | ||||
|     def conv_spec(self, abspath: str, tpath: str) -> None: | ||||
|         ret, _ = ffprobe(abspath) | ||||
|         if "ac" not in ret: | ||||
|             raise Exception("not audio") | ||||
|  | ||||
|         fc = "[0:a:0]aresample=48000{},showspectrumpic=s=640x512,crop=780:544:70:50[o]" | ||||
|  | ||||
|         if self.args.th_ff_swr: | ||||
|             fco = ":filter_size=128:cutoff=0.877" | ||||
|         else: | ||||
|             fco = ":resampler=soxr" | ||||
|  | ||||
|         fc = fc.format(fco) | ||||
|  | ||||
|         # fmt: off | ||||
|         cmd = [ | ||||
|             b"ffmpeg", | ||||
|             b"-nostdin", | ||||
|             b"-v", b"error", | ||||
|             b"-hide_banner", | ||||
|             b"-i", fsenc(abspath), | ||||
|             b"-filter_complex", fc.encode("utf-8"), | ||||
|             b"-map", b"[o]" | ||||
|         ] | ||||
|         # 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._run_ff(cmd) | ||||
|  | ||||
|     def conv_opus(self, abspath: str, tpath: str) -> None: | ||||
|         if self.args.no_acode: | ||||
|             raise Exception("disabled in server config") | ||||
|  | ||||
|         ret, _ = ffprobe(abspath) | ||||
|         if "ac" not in ret: | ||||
|             raise Exception("not audio") | ||||
|  | ||||
|         src_opus = abspath.lower().endswith(".opus") or ret["ac"][1] == "opus" | ||||
|         want_caf = tpath.endswith(".caf") | ||||
|         tmp_opus = tpath | ||||
|         if want_caf: | ||||
|             tmp_opus = tpath.rsplit(".", 1)[0] + ".opus" | ||||
|  | ||||
|         if not want_caf or (not src_opus and not bos.path.isfile(tmp_opus)): | ||||
|             # fmt: off | ||||
|             cmd = [ | ||||
|                 b"ffmpeg", | ||||
|                 b"-nostdin", | ||||
|                 b"-v", b"error", | ||||
|                 b"-hide_banner", | ||||
|                 b"-i", fsenc(abspath), | ||||
|                 b"-map_metadata", b"-1", | ||||
|                 b"-map", b"0:a:0", | ||||
|                 b"-c:a", b"libopus", | ||||
|                 b"-b:a", b"128k", | ||||
|                 fsenc(tmp_opus) | ||||
|             ] | ||||
|             # fmt: on | ||||
|             self._run_ff(cmd) | ||||
|  | ||||
|         if want_caf: | ||||
|             # fmt: off | ||||
|             cmd = [ | ||||
|                 b"ffmpeg", | ||||
|                 b"-nostdin", | ||||
|                 b"-v", b"error", | ||||
|                 b"-hide_banner", | ||||
|                 b"-i", fsenc(abspath if src_opus else tmp_opus), | ||||
|                 b"-map_metadata", b"-1", | ||||
|                 b"-map", b"0:a:0", | ||||
|                 b"-c:a", b"copy", | ||||
|                 b"-f", b"caf", | ||||
|                 fsenc(tpath) | ||||
|             ] | ||||
|             # fmt: on | ||||
|             self._run_ff(cmd) | ||||
|  | ||||
|     def poke(self, tdir: str) -> None: | ||||
|         if not self.poke_cd.poke(tdir): | ||||
|             return | ||||
|  | ||||
|         ts = int(time.time()) | ||||
|         try: | ||||
|             for _ in range(4): | ||||
|                 bos.utime(tdir, (ts, ts)) | ||||
|                 tdir = os.path.dirname(tdir) | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def cleaner(self) -> None: | ||||
|         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: str) -> int: | ||||
|         ret = 0 | ||||
|         for cat in ["th", "ac"]: | ||||
|             top = os.path.join(histpath, cat) | ||||
|             if not bos.path.isdir(top): | ||||
|                 continue | ||||
|  | ||||
|             ret += self._clean(cat, top) | ||||
|  | ||||
|         return ret | ||||
|  | ||||
|     def _clean(self, cat: str, thumbpath: str) -> int: | ||||
|         # self.log("cln {}".format(thumbpath)) | ||||
|         exts = ["jpg", "webp"] if cat == "th" else ["opus", "caf"] | ||||
|         maxage = getattr(self.args, cat + "_maxage") | ||||
|         now = time.time() | ||||
|         prev_b64 = None | ||||
|         prev_fp = "" | ||||
|         try: | ||||
|             t1 = statdir(self.log_func, not self.args.no_scandir, False, thumbpath) | ||||
|             ents = sorted(list(t1)) | ||||
|         except: | ||||
|             return 0 | ||||
|  | ||||
|         ndirs = 0 | ||||
|         for f, inf in 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 - inf.st_mtime | ||||
|                 if age > maxage: | ||||
|                     with self.mutex: | ||||
|                         safe = True | ||||
|                         for k in self.busy: | ||||
|                             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(cat, fp) | ||||
|  | ||||
|                 continue | ||||
|  | ||||
|             # thumb file | ||||
|             try: | ||||
|                 b64, ts, ext = f.split(".") | ||||
|                 if len(b64) != 24 or len(ts) != 8 or ext not in exts: | ||||
|                     raise Exception() | ||||
|             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) | ||||
|  | ||||
|             if cat != "th" and inf.st_mtime + maxage < now: | ||||
|                 self.log("rm expired [{}]".format(fp)) | ||||
|                 bos.unlink(fp) | ||||
|  | ||||
|             prev_b64 = b64 | ||||
|             prev_fp = fp | ||||
|  | ||||
|         return ndirs | ||||
							
								
								
									
										379
									
								
								copyparty/u2idx.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										379
									
								
								copyparty/u2idx.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,379 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import calendar | ||||
| import os | ||||
| import re | ||||
| import threading | ||||
| import time | ||||
| from operator import itemgetter | ||||
|  | ||||
| from .__init__ import ANYWIN, TYPE_CHECKING, unicode | ||||
| from .bos import bos | ||||
| from .up2k import up2k_wark_from_hashlist | ||||
| from .util import HAVE_SQLITE3, Pebkac, absreal, gen_filekey, min_ex, quotep, s3dec | ||||
|  | ||||
| if HAVE_SQLITE3: | ||||
|     import sqlite3 | ||||
|  | ||||
| try: | ||||
|     from pathlib import Path | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| try: | ||||
|     from typing import Any, Optional, Union | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from .httpconn import HttpConn | ||||
|  | ||||
|  | ||||
| class U2idx(object): | ||||
|     def __init__(self, conn: "HttpConn") -> None: | ||||
|         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.active_id = "" | ||||
|         self.active_cur: Optional["sqlite3.Cursor"] = None | ||||
|         self.cur: dict[str, "sqlite3.Cursor"] = {} | ||||
|         self.mem_cur = sqlite3.connect(":memory:").cursor() | ||||
|         self.mem_cur.execute(r"create table a (b text)") | ||||
|  | ||||
|         self.p_end = 0.0 | ||||
|         self.p_dur = 0.0 | ||||
|  | ||||
|     def log(self, msg: str, c: Union[int, str] = 0) -> None: | ||||
|         self.log_func("u2idx", msg, c) | ||||
|  | ||||
|     def fsearch( | ||||
|         self, vols: list[tuple[str, str, dict[str, Any]]], body: dict[str, Any] | ||||
|     ) -> list[dict[str, Any]]: | ||||
|         """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 = "substr(w,1,16) = ? and w = ?" | ||||
|         uv: list[Union[str, int]] = [wark[:16], wark] | ||||
|  | ||||
|         try: | ||||
|             return self.run_query(vols, uq, uv, True, False, 99999)[0] | ||||
|         except: | ||||
|             raise Pebkac(500, min_ex()) | ||||
|  | ||||
|     def get_cur(self, ptop: str) -> Optional["sqlite3.Cursor"]: | ||||
|         if not HAVE_SQLITE3: | ||||
|             return None | ||||
|  | ||||
|         cur = self.cur.get(ptop) | ||||
|         if cur: | ||||
|             return cur | ||||
|  | ||||
|         histpath = self.asrv.vfs.histtab.get(ptop) | ||||
|         if not histpath: | ||||
|             self.log("no histpath for [{}]".format(ptop)) | ||||
|             return None | ||||
|  | ||||
|         db_path = os.path.join(histpath, "up2k.db") | ||||
|         if not bos.path.exists(db_path): | ||||
|             return None | ||||
|  | ||||
|         cur = None | ||||
|         if ANYWIN: | ||||
|             uri = "" | ||||
|             try: | ||||
|                 uri = "{}?mode=ro&nolock=1".format(Path(db_path).as_uri()) | ||||
|                 cur = sqlite3.connect(uri, 2, uri=True).cursor() | ||||
|                 self.log("ro: {}".format(db_path)) | ||||
|             except: | ||||
|                 self.log("could not open read-only: {}\n{}".format(uri, min_ex())) | ||||
|  | ||||
|         if not cur: | ||||
|             # on windows, this steals the write-lock from up2k.deferred_init -- | ||||
|             # seen on win 10.0.17763.2686, py 3.10.4, sqlite 3.37.2 | ||||
|             cur = sqlite3.connect(db_path, 2).cursor() | ||||
|             self.log("opened {}".format(db_path)) | ||||
|  | ||||
|         self.cur[ptop] = cur | ||||
|         return cur | ||||
|  | ||||
|     def search( | ||||
|         self, vols: list[tuple[str, str, dict[str, Any]]], uq: str, lim: int | ||||
|     ) -> tuple[list[dict[str, Any]], list[str]]: | ||||
|         """search by query params""" | ||||
|         if not HAVE_SQLITE3: | ||||
|             return [], [] | ||||
|  | ||||
|         q = "" | ||||
|         v: Union[str, int] = "" | ||||
|         va: list[Union[str, int]] = [] | ||||
|         have_up = False  # query has up.* operands | ||||
|         have_mt = False | ||||
|         is_key = True | ||||
|         is_size = False | ||||
|         is_date = False | ||||
|         field_end = ""  # closing parenthesis or whatever | ||||
|         kw_key = ["(", ")", "and ", "or ", "not "] | ||||
|         kw_val = ["==", "=", "!=", ">", ">=", "<", "<=", "like "] | ||||
|         ptn_mt = re.compile(r"^\.?[a-z_-]+$") | ||||
|         ptn_lc = re.compile(r" (mt\.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 | ||||
|  | ||||
|             if uq.startswith('"'): | ||||
|                 v, uq = uq[1:].split('"', 1) | ||||
|                 while v.endswith("\\"): | ||||
|                     v2, uq = uq.split('"', 1) | ||||
|                     v = v[:-1] + '"' + v2 | ||||
|                 uq = uq.strip() | ||||
|             else: | ||||
|                 v, uq = (uq + " ").split(" ", 1) | ||||
|                 v = v.replace('\\"', '"') | ||||
|  | ||||
|             if is_key: | ||||
|                 is_key = False | ||||
|  | ||||
|                 if v == "size": | ||||
|                     v = "up.sz" | ||||
|                     is_size = True | ||||
|                     have_up = True | ||||
|  | ||||
|                 elif v == "date": | ||||
|                     v = "up.mt" | ||||
|                     is_date = True | ||||
|                     have_up = True | ||||
|  | ||||
|                 elif v == "path": | ||||
|                     v = "trim(?||up.rd,'/')" | ||||
|                     va.append("\nrd") | ||||
|                     have_up = True | ||||
|  | ||||
|                 elif v == "name": | ||||
|                     v = "up.fn" | ||||
|                     have_up = True | ||||
|  | ||||
|                 elif v == "tags" or ptn_mt.match(v): | ||||
|                     have_mt = True | ||||
|                     field_end = ") " | ||||
|                     if v == "tags": | ||||
|                         vq = "mt.v" | ||||
|                     else: | ||||
|                         vq = "+mt.k = '{}' and mt.v".format(v) | ||||
|  | ||||
|                     v = "exists(select 1 from mt where mt.w = mtw and " + vq | ||||
|  | ||||
|                 else: | ||||
|                     raise Pebkac(400, "invalid key [" + v + "]") | ||||
|  | ||||
|                 q += v + " " | ||||
|                 continue | ||||
|  | ||||
|             head = "" | ||||
|             tail = "" | ||||
|  | ||||
|             if is_date: | ||||
|                 is_date = False | ||||
|                 v = re.sub(r"[tzTZ, ]+", " ", v).strip() | ||||
|                 for fmt in [ | ||||
|                     "%Y-%m-%d %H:%M:%S", | ||||
|                     "%Y-%m-%d %H:%M", | ||||
|                     "%Y-%m-%d %H", | ||||
|                     "%Y-%m-%d", | ||||
|                     "%Y-%m", | ||||
|                     "%Y", | ||||
|                 ]: | ||||
|                     try: | ||||
|                         v = calendar.timegm(time.strptime(str(v), fmt)) | ||||
|                         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 | ||||
|  | ||||
|             if field_end: | ||||
|                 q += field_end | ||||
|                 field_end = "" | ||||
|  | ||||
|             # lowercase tag searches | ||||
|             m = ptn_lc.search(q) | ||||
|             zs = unicode(v) | ||||
|             if not m or not ptn_lcv.search(zs): | ||||
|                 continue | ||||
|  | ||||
|             va.pop() | ||||
|             va.append(zs.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, q, va, have_up, have_mt, lim) | ||||
|         except Exception as ex: | ||||
|             raise Pebkac(500, repr(ex)) | ||||
|  | ||||
|     def run_query( | ||||
|         self, | ||||
|         vols: list[tuple[str, str, dict[str, Any]]], | ||||
|         uq: str, | ||||
|         uv: list[Union[str, int]], | ||||
|         have_up: bool, | ||||
|         have_mt: bool, | ||||
|         lim: int, | ||||
|     ) -> tuple[list[dict[str, Any]], list[str]]: | ||||
|         done_flag: list[bool] = [] | ||||
|         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: | ||||
|             uq = "select * from up" | ||||
|             uv = [] | ||||
|         elif have_mt: | ||||
|             uq = "select up.*, substr(up.w,1,16) mtw from up where " + uq | ||||
|         else: | ||||
|             uq = "select up.* from up where " + uq | ||||
|  | ||||
|         self.log("qs: {!r} {!r}".format(uq, uv)) | ||||
|  | ||||
|         ret = [] | ||||
|         lim = min(lim, int(self.args.srch_hits)) | ||||
|         taglist = {} | ||||
|         for (vtop, ptop, flags) in vols: | ||||
|             cur = self.get_cur(ptop) | ||||
|             if not cur: | ||||
|                 continue | ||||
|  | ||||
|             self.active_cur = cur | ||||
|  | ||||
|             vuv = [] | ||||
|             for v in uv: | ||||
|                 if v == "\nrd": | ||||
|                     v = vtop + "/" | ||||
|  | ||||
|                 vuv.append(v) | ||||
|  | ||||
|             sret = [] | ||||
|             fk = flags.get("fk") | ||||
|             c = cur.execute(uq, tuple(vuv)) | ||||
|             for hit in c: | ||||
|                 w, ts, sz, rd, fn, ip, at = hit[:7] | ||||
|                 lim -= 1 | ||||
|                 if lim < 0: | ||||
|                     break | ||||
|  | ||||
|                 if rd.startswith("//") or fn.startswith("//"): | ||||
|                     rd, fn = s3dec(rd, fn) | ||||
|  | ||||
|                 if not fk: | ||||
|                     suf = "" | ||||
|                 else: | ||||
|                     try: | ||||
|                         ap = absreal(os.path.join(ptop, rd, fn)) | ||||
|                         inf = bos.stat(ap) | ||||
|                     except: | ||||
|                         continue | ||||
|  | ||||
|                     suf = ( | ||||
|                         "?k=" | ||||
|                         + gen_filekey( | ||||
|                             self.args.fk_salt, ap, sz, 0 if ANYWIN else inf.st_ino | ||||
|                         )[:fk] | ||||
|                     ) | ||||
|  | ||||
|                 rp = quotep("/".join([x for x in [vtop, rd, fn] if x])) + suf | ||||
|                 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 = "" | ||||
|  | ||||
|         # 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"].split("?")[0] != y["rp"].split("?")[0] | ||||
|             ] | ||||
|  | ||||
|         ret.sort(key=itemgetter("rp")) | ||||
|  | ||||
|         return ret, list(taglist.keys()) | ||||
|  | ||||
|     def terminator(self, identifier: str, done_flag: list[bool]) -> None: | ||||
|         for _ in range(self.timeout): | ||||
|             time.sleep(1) | ||||
|             if done_flag: | ||||
|                 return | ||||
|  | ||||
|         if identifier == self.active_id: | ||||
|             assert self.active_cur | ||||
|             self.active_cur.connection.interrupt() | ||||
							
								
								
									
										2945
									
								
								copyparty/up2k.py
									
									
									
									
									
								
							
							
						
						
									
										2945
									
								
								copyparty/up2k.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1671
									
								
								copyparty/util.py
									
									
									
									
									
								
							
							
						
						
									
										1671
									
								
								copyparty/util.py
									
									
									
									
									
								
							
										
											
												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 $< | ||||
							
								
								
									
										1015
									
								
								copyparty/web/baguettebox.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1015
									
								
								copyparty/web/baguettebox.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -2,78 +2,169 @@ | ||||
| <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"> | ||||
| {{ html_head }} | ||||
| 	<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 opbox"></div> | ||||
| 		{%- else %} | ||||
| 		<div id="srch_form" class="opbox"></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> | ||||
| 		<a id="bbsw" href="?b=u"><br />switch to basic browser</a> | ||||
| 	</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" class="i"> | ||||
| 			<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" class="i"> | ||||
| 			<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" class="i"> | ||||
| 			<input type="submit" value="send msg to srv 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">🌲</a> | ||||
| 		{%- for n in vpnodes %} | ||||
| 		<a href="/{{ n[0] }}">{{ n[1] }}</a> | ||||
| 		{%- endfor %} | ||||
| 	</h1> | ||||
| 	 | ||||
| 	<div id="tree"></div> | ||||
|  | ||||
| <div id="wrap"> | ||||
|  | ||||
| 	{%- if doc %} | ||||
| 	<div id="bdoc"><pre>{{ doc|e }}</pre></div> | ||||
| 	{%- else %} | ||||
| 	<div id="bdoc"></div> | ||||
| 	{%- endif %} | ||||
|  | ||||
| 	<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" id="goh">control-panel</a></h2> | ||||
| 	 | ||||
| 	<a href="#" id="repl">π</a> | ||||
|  | ||||
|     {%- if srv_info %} | ||||
|     <div id="srv_info"><span>{{ srv_info }}</span></div> | ||||
|     {%- endif %} | ||||
| </div> | ||||
|  | ||||
|     <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 %} | ||||
| 	{%- if srv_info %} | ||||
| 	<div id="srv_info"><span>{{ srv_info }}</span></div> | ||||
| 	{%- endif %} | ||||
|  | ||||
| 	<div id="widget"></div> | ||||
|  | ||||
| 	<script> | ||||
| 		var acct = "{{ acct }}", | ||||
| 			perms = {{ perms }}, | ||||
| 			themes = {{ themes }}, | ||||
| 			dtheme = "{{ dtheme }}", | ||||
| 			srvinf = "{{ srv_info }}", | ||||
| 			lang = "{{ lang }}", | ||||
| 			dfavico = "{{ favico }}", | ||||
| 			def_hcols = {{ def_hcols|tojson }}, | ||||
| 			have_up2k_idx = {{ have_up2k_idx|tojson }}, | ||||
| 			have_tags_idx = {{ have_tags_idx|tojson }}, | ||||
| 			have_acode = {{ have_acode|tojson }}, | ||||
| 			have_mv = {{ have_mv|tojson }}, | ||||
| 			have_del = {{ have_del|tojson }}, | ||||
| 			have_unpost = {{ have_unpost|tojson }}, | ||||
| 			have_zip = {{ have_zip|tojson }}, | ||||
| 			turbolvl = {{ turbolvl }}, | ||||
| 			u2sort = "{{ u2sort }}", | ||||
| 			have_emp = {{ have_emp|tojson }}, | ||||
| 			txt_ext = "{{ txt_ext }}", | ||||
| 			{% if no_prism %}no_prism = 1,{% endif %} | ||||
| 			readme = {{ readme|tojson }}, | ||||
| 			ls0 = {{ ls0|tojson }}; | ||||
|  | ||||
| 		document.documentElement.className = localStorage.theme || dtheme; | ||||
| 	</script> | ||||
| 	<script src="/.cpr/util.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/baguettebox.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/browser.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/up2k.js?_={{ ts }}"></script> | ||||
| 	{%- if js %} | ||||
| 	<script src="{{ js }}?_={{ ts }}"></script> | ||||
| 	{%- endif %} | ||||
| </body> | ||||
|  | ||||
| </html> | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										63
									
								
								copyparty/web/browser2.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								copyparty/web/browser2.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| <!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"> | ||||
| {{ html_head }} | ||||
| 	<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[1:] if url_suf[:1] == '?' and '?' in f.href else 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> | ||||
							
								
								
									
										27
									
								
								copyparty/web/cf.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								copyparty/web/cf.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|  | ||||
| <head> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>{{ svcname }}</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
| 	<div id="box" style="opacity: 0; font-family: sans-serif"> | ||||
| 		<h3>please press F5 to reload the page</h3> | ||||
| 		<p>sorry for the inconvenience</p> | ||||
| 	</div> | ||||
|  | ||||
| 	<script> | ||||
| 		setTimeout(function() { | ||||
| 			document.getElementById('box').style.opacity = 1; | ||||
| 		}, 500); | ||||
|  | ||||
| 		parent.toast.ok(30, parent.L.cf_ok); | ||||
| 		parent.qsr('#cf_frame'); | ||||
| 	</script> | ||||
| </body> | ||||
|  | ||||
| </html> | ||||
							
								
								
									
										60
									
								
								copyparty/web/dbg-audio.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								copyparty/web/dbg-audio.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| 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', '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(); | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user