mirror of
				https://github.com/9001/copyparty.git
				synced 2025-10-25 00:53:47 +00:00 
			
		
		
		
	Compare commits
	
		
			866 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 24a2f760b7 | ||
|  | 79bbd8fe38 | ||
|  | 35dce1e3e4 | ||
|  | f886fdf913 | ||
|  | 4476f2f0da | ||
|  | 160f161700 | ||
|  | c164fc58a2 | ||
|  | 0c625a4e62 | ||
|  | bf3941cf7a | ||
|  | 3649e8288a | ||
|  | 9a45e26026 | ||
|  | e65f127571 | ||
|  | 3bfc699787 | ||
|  | 955318428a | ||
|  | f6279b356a | ||
|  | 4cc3cdc989 | ||
|  | f9aa20a3ad | ||
|  | 129d33f1a0 | ||
|  | 1ad7a3f378 | ||
|  | b533be8818 | ||
|  | fb729e5166 | ||
|  | d337ecdb20 | ||
|  | 5f1f0a48b0 | ||
|  | e0f1cb94a5 | ||
|  | a362ee2246 | ||
|  | 19f23c686e | ||
|  | 23b20ff4a6 | ||
|  | 72574da834 | ||
|  | d5a79455d1 | ||
|  | 070d4b9da9 | ||
|  | 0ace22fffe | ||
|  | 9e483d7694 | ||
|  | 26458b7a06 | ||
|  | b6a4604952 | ||
|  | af752fbbc2 | ||
|  | 279c9d706a | ||
|  | 806e7b5530 | ||
|  | f3dc6a217b | ||
|  | 7671d791fa | ||
|  | 8cd84608a5 | ||
|  | 980c6fc810 | ||
|  | fb40a484c5 | ||
|  | daa9dedcaa | ||
|  | 0d634345ac | ||
|  | e648252479 | ||
|  | 179d7a9ad8 | ||
|  | 19bc962ad5 | ||
|  | 27cce086c6 | ||
|  | fec0c620d4 | ||
|  | 05a1a31cab | ||
|  | d020527c6f | ||
|  | 4451485664 | ||
|  | a4e1a3738a | ||
|  | 4339dbeb8d | ||
|  | 5b0605774c | ||
|  | e3684e25f8 | ||
|  | 1359213196 | ||
|  | 03efc6a169 | ||
|  | 15b5982211 | ||
|  | 0eb3a5d387 | ||
|  | 7f8777389c | ||
|  | 4eb20f10ad | ||
|  | daa11df558 | ||
|  | 1bb0db30a0 | ||
|  | 02910b0020 | ||
|  | 23b8901c9c | ||
|  | 99f6ed0cd7 | ||
|  | 890c310880 | ||
|  | 0194eeb31f | ||
|  | f9be4c62b1 | ||
|  | 027e8c18f1 | ||
|  | 4a3bb35a95 | ||
|  | 4bfb0d4494 | ||
|  | 7e0ef03a1e | ||
|  | f7dbd95a54 | ||
|  | 515ee2290b | ||
|  | b0c78910bb | ||
|  | f4ca62b664 | ||
|  | 8eb8043a3d | ||
|  | 3e8541362a | ||
|  | 789724e348 | ||
|  | 5125b9532f | ||
|  | ebc9de02b0 | ||
|  | ec788fa491 | ||
|  | 9b5e264574 | ||
|  | 57c297274b | ||
|  | e9bf092317 | ||
|  | d173887324 | ||
|  | 99820d854c | ||
|  | 62df0a0eb2 | ||
|  | 600e9ac947 | ||
|  | 3ca41be2b4 | ||
|  | 5c7debd900 | ||
|  | 7fa5b23ce3 | ||
|  | ff82738aaf | ||
|  | bf5ee9d643 | ||
|  | 72a8593ecd | ||
|  | bc3bbe07d4 | ||
|  | c7cb64bfef | ||
|  | 629f537d06 | ||
|  | 9e988041b8 | ||
|  | f9a8b5c9d7 | ||
|  | b9c3538253 | ||
|  | 2bc0cdf017 | ||
|  | 02a91f60d4 | ||
|  | fae83da197 | ||
|  | 0fe4aa6418 | ||
|  | 21a51bf0dc | ||
|  | bcb353cc30 | ||
|  | 6af4508518 | ||
|  | 6a559bc28a | ||
|  | 0f5026cd20 | ||
|  | a91b80a311 | ||
|  | ec534701c8 | ||
|  | af5169f67f | ||
|  | 18676c5e65 | ||
|  | e2df6fda7b | ||
|  | e9ae9782fe | ||
|  | 016dba4ca9 | ||
|  | 39c7ef305f | ||
|  | 849c1dc848 | ||
|  | 61414014fe | ||
|  | 578a915884 | ||
|  | eacafb8a63 | ||
|  | 4446760f74 | ||
|  | 6da2a083f9 | ||
|  | 8837c8f822 | ||
|  | bac301ed66 | ||
|  | 061db3906d | ||
|  | fd7df5c952 | ||
|  | a270019147 | ||
|  | 55e0209901 | ||
|  | 2b255fbbed | ||
|  | 8a2345a0fb | ||
|  | bfa9f535aa | ||
|  | f757623ad8 | ||
|  | 3c7465e268 | ||
|  | 108665fc4f | ||
|  | ed519c9138 | ||
|  | 2dd2e2c57e | ||
|  | 6c3a976222 | ||
|  | 80cc26bd95 | ||
|  | 970fb84fd8 | ||
|  | 20cbcf6931 | ||
|  | 8fcde2a579 | ||
|  | b32d1f8ad3 | ||
|  | 03513e0cb1 | ||
|  | e041a2b197 | ||
|  | d7d625be2a | ||
|  | 4121266678 | ||
|  | 22971a6be4 | ||
|  | efbf8d7e0d | ||
|  | 397396ea4a | ||
|  | e59b077c21 | ||
|  | 4bc39f3084 | ||
|  | 21c3570786 | ||
|  | 2f85c1fb18 | ||
|  | 1e27a4c2df | ||
|  | 456f575637 | ||
|  | 51546c9e64 | ||
|  | 83b4b70ef4 | ||
|  | a5120d4f6f | ||
|  | c95941e14f | ||
|  | 0dd531149d | ||
|  | 67da1b5219 | ||
|  | 919bd16437 | ||
|  | ecead109ab | ||
|  | 765294c263 | ||
|  | d6b5351207 | ||
|  | a2009bcc6b | ||
|  | 12709a8a0a | ||
|  | c055baefd2 | ||
|  | 56522599b5 | ||
|  | 664f53b75d | ||
|  | 87200d9f10 | ||
|  | 5c3d0b6520 | ||
|  | bd49979f4a | ||
|  | 7e606cdd9f | ||
|  | 8b4b7fa794 | ||
|  | 05345ddf8b | ||
|  | 66adb470ad | ||
|  | e15c8fd146 | ||
|  | 0f09b98a39 | ||
|  | b4d6f4e24d | ||
|  | 3217fa625b | ||
|  | e719ff8a47 | ||
|  | 9fcf528d45 | ||
|  | 1ddbf5a158 | ||
|  | 64bf4574b0 | ||
|  | 5649d26077 | ||
|  | 92f923effe | ||
|  | 0d46d548b9 | ||
|  | 062df3f0c3 | ||
|  | 789fb53b8e | ||
|  | 351db5a18f | ||
|  | aabbd271c8 | ||
|  | aae8e0171e | ||
|  | 45827a2458 | ||
|  | 726030296f | ||
|  | 6659ab3881 | ||
|  | c6a103609e | ||
|  | c6b3f035e5 | ||
|  | 2b0a7e378e | ||
|  | b75ce909c8 | ||
|  | 229c3f5dab | ||
|  | ec73094506 | ||
|  | c7650c9326 | ||
|  | d94c6d4e72 | ||
|  | 3cc8760733 | ||
|  | a2f6973495 | ||
|  | f8648fa651 | ||
|  | 177aa038df | ||
|  | e0a14ec881 | ||
|  | 9366512f2f | ||
|  | ea38b8041a | ||
|  | f1870daf0d | ||
|  | 9722441aad | ||
|  | 9d014087f4 | ||
|  | 83b4038b85 | ||
|  | 1e0a448feb | ||
|  | fb81de3b36 | ||
|  | aa4f352301 | ||
|  | f1a1c2ea45 | ||
|  | 6249bd4163 | ||
|  | 2579dc64ce | ||
|  | 356512270a | ||
|  | bed27f2b43 | ||
|  | 54013d861b | ||
|  | ec100210dc | ||
|  | 3ab1acf32c | ||
|  | 8c28266418 | ||
|  | 7f8b8dcb92 | ||
|  | 6dd39811d4 | ||
|  | 35e2138e3e | ||
|  | 239b4e9fe6 | ||
|  | 2fcd0e7e72 | ||
|  | 357347ce3a | ||
|  | 36dc1107fb | ||
|  | 0a3bbc4b4a | ||
|  | 855b93dcf6 | ||
|  | 89b79ba267 | ||
|  | f5651b7d94 | ||
|  | 1881019ede | ||
|  | caba4e974c | ||
|  | bc3c9613bc | ||
|  | 15a3ee252e | ||
|  | be055961ae | ||
|  | e3031bdeec | ||
|  | 75917b9f7c | ||
|  | 910732e02c | ||
|  | 264b497681 | ||
|  | 372b949622 | ||
|  | 789a602914 | ||
|  | 093e955100 | ||
|  | c32a89bebf | ||
|  | c0bebe9f9f | ||
|  | 57579b2fe5 | ||
|  | 51d14a6b4d | ||
|  | c50f1b64e5 | ||
|  | 98aaab02c5 | ||
|  | 0fc7973d8b | ||
|  | 10362aa02e | ||
|  | 0a8e759fe6 | ||
|  | d70981cdd1 | ||
|  | e08c03b886 | ||
|  | 56086e8984 | ||
|  | 1aa9033022 | ||
|  | 076e103d53 | ||
|  | 38c00ea8fc | ||
|  | 415757af43 | ||
|  | e72ed8c0ed | ||
|  | 32f9c6b5bb | ||
|  | 6251584ef6 | ||
|  | f3e413bc28 | ||
|  | 6f6cc8f3f8 | ||
|  | 8b081e9e69 | ||
|  | c8a510d10e | ||
|  | 6f834f6679 | ||
|  | cf2d6650ac | ||
|  | cd52dea488 | ||
|  | 6ea75df05d | ||
|  | 4846e1e8d6 | ||
|  | fc024f789d | ||
|  | 473e773aea | ||
|  | 48a2e1a353 | ||
|  | 6da63fbd79 | ||
|  | 5bec37fcee | ||
|  | 3fd0ba0a31 | ||
|  | 241a143366 | ||
|  | a537064da7 | ||
|  | f3dfd24c92 | ||
|  | fa0a7f50bb | ||
|  | 44a78a7e21 | ||
|  | 6b75cbf747 | ||
|  | e7b18ab9fe | ||
|  | aa12830015 | ||
|  | f156e00064 | ||
|  | d53c212516 | ||
|  | ca27f8587c | ||
|  | 88ce008e16 | ||
|  | 081d2cc5d7 | ||
|  | 60ac68d000 | ||
|  | fbe656957d | ||
|  | 5534c78c17 | ||
|  | a45a53fdce | ||
|  | 972a56e738 | ||
|  | 5e03b3ca38 | ||
|  | 1078d933b4 | ||
|  | d6bf300d80 | ||
|  | a359d64d44 | ||
|  | 22396e8c33 | ||
|  | 5ded5a4516 | ||
|  | 79c7639aaf | ||
|  | 5bbf875385 | ||
|  | 5e159432af | ||
|  | 1d6ae409f6 | ||
|  | 9d729d3d1a | ||
|  | 4dd5d4e1b7 | ||
|  | acd8149479 | ||
|  | b97a1088fa | ||
|  | b77bed3324 | ||
|  | a2b7c85a1f | ||
|  | b28533f850 | ||
|  | bd8c7e538a | ||
|  | 89e48cff24 | ||
|  | ae90a7b7b6 | ||
|  | 6fc1be04da | ||
|  | 0061d29534 | ||
|  | a891f34a93 | ||
|  | d6a1e62a95 | ||
|  | cda36ea8b4 | ||
|  | 909a76434a | ||
|  | 39348ef659 | ||
|  | 99d30edef3 | ||
|  | b63ab15bf9 | ||
|  | 485cb4495c | ||
|  | df018eb1f2 | ||
|  | 49aa47a9b8 | ||
|  | 7d20eb202a | ||
|  | c533da9129 | ||
|  | 5cba31a814 | ||
|  | 1d824cb26c | ||
|  | 83b903d60e | ||
|  | 9c8ccabe8e | ||
|  | b1f2c4e70d | ||
|  | 273ca0c8da | ||
|  | d6f516b34f | ||
|  | 83127858ca | ||
|  | d89329757e | ||
|  | 49ffec5320 | ||
|  | 2eaae2b66a | ||
|  | ea4441e25c | ||
|  | e5f34042f9 | ||
|  | 271096874a | ||
|  | 8efd780a72 | ||
|  | 41bcf7308d | ||
|  | d102bb3199 | ||
|  | d0bed95415 | ||
|  | 2528729971 | ||
|  | 292c18b3d0 | ||
|  | 0be7c5e2d8 | ||
|  | eb5aaddba4 | ||
|  | d8fd82bcb5 | ||
|  | 97be495861 | ||
|  | 8b53c159fc | ||
|  | 81e281f703 | ||
|  | 3948214050 | ||
|  | c5e9a643e7 | ||
|  | d25881d5c3 | ||
|  | 38d8d9733f | ||
|  | 118ebf668d | ||
|  | a86f09fa46 | ||
|  | dd4fb35c8f | ||
|  | 621eb4cf95 | ||
|  | deea66ad0b | ||
|  | bf99445377 | ||
|  | 7b54a63396 | ||
|  | 0fcb015f9a | ||
|  | 0a22b1ffb6 | ||
|  | 68cecc52ab | ||
|  | 53657ccfff | ||
|  | 96223fda01 | ||
|  | 374ff3433e | ||
|  | 5d63949e98 | ||
|  | 6b065d507d | ||
|  | e79997498a | ||
|  | f7ee02ec35 | ||
|  | 69dc433e1c | ||
|  | c880cd848c | ||
|  | 5752b6db48 | ||
|  | b36f905eab | ||
|  | 483dd527c6 | ||
|  | e55678e28f | ||
|  | 3f4a8b9d6f | ||
|  | 02a856ecb4 | ||
|  | 4dff726310 | ||
|  | cbc449036f | ||
|  | 8f53152220 | ||
|  | bbb1e165d6 | ||
|  | fed8d94885 | ||
|  | 58040cc0ed | ||
|  | 03d692db66 | ||
|  | 903f8e8453 | ||
|  | 405ae1308e | ||
|  | 8a0f583d71 | ||
|  | b6d7017491 | ||
|  | 0f0217d203 | ||
|  | a203e33347 | ||
|  | 3b8f697dd4 | ||
|  | 78ba16f722 | ||
|  | 0fcfe79994 | ||
|  | c0e6df4b63 | ||
|  | 322abdcb43 | ||
|  | 31100787ce | ||
|  | c57d721be4 | ||
|  | 3b5a03e977 | ||
|  | ed807ee43e | ||
|  | 073c130ae6 | ||
|  | 8810e0be13 | ||
|  | f93016ab85 | ||
|  | b19cf260c2 | ||
|  | db03e1e7eb | ||
|  | e0d975e36a | ||
|  | cfeb15259f | ||
|  | 3b3f8fc8fb | ||
|  | 88bd2c084c | ||
|  | bd367389b0 | ||
|  | 58ba71a76f | ||
|  | d03e34d55d | ||
|  | 24f239a46c | ||
|  | 2c0826f85a | ||
|  | c061461d01 | ||
|  | e7982a04fe | ||
|  | 33b91a7513 | ||
|  | 9bb1323e44 | ||
|  | e62bb807a5 | ||
|  | 3fc0d2cc4a | ||
|  | 0c786b0766 | ||
|  | 68c7528911 | ||
|  | 26e18ae800 | ||
|  | c30dc0b546 | ||
|  | f94aa46a11 | ||
|  | 403261a293 | ||
|  | c7d9cbb11f | ||
|  | 57e1c53cbb | ||
|  | 0754b553dd | ||
|  | 50661d941b | ||
|  | c5db7c1a0c | ||
|  | 2cef5365f7 | ||
|  | fbc4e94007 | ||
|  | 037ed5a2ad | ||
|  | 69dfa55705 | ||
|  | a79a5c4e3e | ||
|  | 7e80eabfe6 | ||
|  | 375b72770d | ||
|  | e2dd683def | ||
|  | 9eba50c6e4 | ||
|  | 5a579dba52 | ||
|  | e86c719575 | ||
|  | 0e87f35547 | ||
|  | b6d3d791a5 | ||
|  | c9c3302664 | ||
|  | c3e4d65b80 | ||
|  | 27a03510c5 | ||
|  | ed7727f7cb | ||
|  | 127ec10c0d | ||
|  | 5a9c0ad225 | ||
|  | 7e8daf650e | ||
|  | 0cf737b4ce | ||
|  | 74635e0113 | ||
|  | e5c4f49901 | ||
|  | e4654ee7f1 | ||
|  | e5d05c05ed | ||
|  | 73c4f99687 | ||
|  | 28c12ef3bf | ||
|  | eed82dbb54 | ||
|  | 2c4b4ab928 | ||
|  | 505a8fc6f6 | ||
|  | e4801d9b06 | ||
|  | 04f1b2cf3a | ||
|  | c06d928bb5 | ||
|  | ab09927e7b | ||
|  | 779437db67 | ||
|  | 28cbdb652e | ||
|  | 2b2415a7d8 | ||
|  | 746a8208aa | ||
|  | a2a041a98a | ||
|  | 10b436e449 | ||
|  | 4d62b34786 | ||
|  | 0546210687 | ||
|  | f8c11faada | ||
|  | 16d6e9be1f | ||
|  | aff8185f2e | ||
|  | 217d15fe81 | ||
|  | 171e93c201 | ||
|  | acc1d2e9e3 | ||
|  | 49c2f37154 | ||
|  | 69e54497aa | ||
|  | 9aa1885669 | ||
|  | 4418508513 | ||
|  | e897df3b34 | ||
|  | 8cd97ab0e7 | ||
|  | bf4949353d | ||
|  | 98a944f7cc | ||
|  | 7c10f81c92 | ||
|  | 126ecc55c3 | ||
|  | 1034a51bd2 | ||
|  | a2657887cc | ||
|  | c14b17bfaf | ||
|  | 59ebc795e7 | ||
|  | 8e128d917e | ||
|  | ea762b05e0 | ||
|  | db374b19f1 | ||
|  | ab3839ef36 | ||
|  | 9886c442f2 | ||
|  | c8d1926d52 | ||
|  | a6bd699e52 | ||
|  | 12143f2702 | ||
|  | 480705dee9 | ||
|  | 781d5094f4 | ||
|  | 5615cb94cd | ||
|  | 302302a2ac | ||
|  | 9761b4e3e9 | ||
|  | 0cf6924dca | ||
|  | 5fd81e9f90 | ||
|  | 52bf6f892b | ||
|  | f3cce232a4 | ||
|  | 53d3c8b28e | ||
|  | 83fec3cca7 | ||
|  | 3cefc99b7d | ||
|  | 3a38dcbc05 | ||
|  | 7ff08bce57 | ||
|  | fd490af434 | ||
|  | 1195b8f17e | ||
|  | 28dce13776 | ||
|  | 431f20177a | ||
|  | 87aff54d9d | ||
|  | f50462de82 | ||
|  | 9bda8c7eb6 | ||
|  | e83c63d239 | ||
|  | b38533b0cc | ||
|  | 5ccca3fbd5 | ||
|  | 9e850fc3ab | ||
|  | ffbfcd7e00 | ||
|  | 5ea7590748 | ||
|  | 290c3bc2bb | ||
|  | b12131e91c | ||
|  | 3b354447b0 | ||
|  | d09ec6feaa | ||
|  | 21405c3fda | ||
|  | 13e5c96cab | ||
|  | 426687b75e | ||
|  | c8f59fb978 | ||
|  | 871dde79a9 | ||
|  | e14d81bc6f | ||
|  | 514d046d1f | ||
|  | 4ed9528d36 | ||
|  | 625560e642 | ||
|  | 73ebd917d1 | ||
|  | cd3e0afad2 | ||
|  | d8d1f94a86 | ||
|  | 00dfd8cfd1 | ||
|  | 273de6db31 | ||
|  | c6c0eeb0ff | ||
|  | e70c74a3b5 | ||
|  | f7d939eeab | ||
|  | e815c091b9 | ||
|  | 963529b7cf | ||
|  | 638a52374d | ||
|  | d9d42b7aa2 | ||
|  | ec7e5f36a2 | ||
|  | 56110883ea | ||
|  | 7f8d7d6006 | ||
|  | 49e4fb7e12 | ||
|  | 8dbbea473f | ||
|  | 3d375d5114 | ||
|  | f3eae67d97 | ||
|  | 40c1b19235 | ||
|  | ccaf0ab159 | ||
|  | d07f147423 | ||
|  | f5cb9f92b9 | ||
|  | f991f74983 | ||
|  | 6b3295059e | ||
|  | b18a07ae6b | ||
|  | 8ab03dabda | ||
|  | 5e760e35dc | ||
|  | afbfa04514 | ||
|  | 7aace470c5 | ||
|  | b4acb24f6a | ||
|  | bcee8a4934 | ||
|  | 36b0718542 | ||
|  | 9a92bca45d | ||
|  | b07445a363 | ||
|  | a62ec0c27e | ||
|  | 57e3a2d382 | ||
|  | b61022b374 | ||
|  | a3e2b2ec87 | ||
|  | a83d3f8801 | ||
|  | 90c5f2b9d2 | ||
|  | 4885653c07 | ||
|  | 21e1cd87ca | ||
|  | 81f82e8e9f | ||
|  | c0e31851da | ||
|  | 6599c3eced | ||
|  | 5d6c61a861 | ||
|  | 1a5c66edd3 | ||
|  | deae9fe95a | ||
|  | abd65c6334 | ||
|  | 8137a99904 | ||
|  | 6f6f9c1f74 | ||
|  | 7b575f716f | ||
|  | 6ba6ea3572 | ||
|  | 9a22ad5ea3 | ||
|  | beaab9778e | ||
|  | f327bdb6b4 | ||
|  | ae180e0f5f | ||
|  | e3f1d19756 | ||
|  | 93c2bd6ef6 | ||
|  | 4d0e5ff6db | ||
|  | 0893f06919 | ||
|  | 46b6abde3f | ||
|  | 0696610dee | ||
|  | edf0d3684c | ||
|  | 7af159f5f6 | ||
|  | 7f2cb6764a | ||
|  | 96495a9bf1 | ||
|  | b2fafec5fc | ||
|  | 0850b8ae2b | ||
|  | 8a68a96c57 | ||
|  | d3aae8ed6a | ||
|  | c62ebadda8 | ||
|  | ffcee6d390 | ||
|  | de32838346 | ||
|  | b9a4e47ea2 | ||
|  | 57d994422d | ||
|  | 6ecd745323 | ||
|  | bd769f5bdb | ||
|  | 2381692aba | ||
|  | 24fdada0a0 | ||
|  | bb5169710a | ||
|  | 9cde2352f3 | ||
|  | 482dd7a938 | ||
|  | bddcc69438 | ||
|  | 19d4540630 | ||
|  | 4f5f6c81f5 | ||
|  | 7e4c1238ba | ||
|  | f7196ac773 | ||
|  | 7a7c832000 | ||
|  | 2b4ccdbebb | ||
|  | 0d16b49489 | ||
|  | 768405b691 | ||
|  | da01413b7b | ||
|  | 914e22c53e | ||
|  | 43a23bf733 | ||
|  | 92bb00c6d2 | ||
|  | b0b97a2648 | ||
|  | 2c452fe323 | ||
|  | ad73d0c77d | ||
|  | 7f9bf1c78c | ||
|  | 61a6bc3a65 | ||
|  | 46e10b0e9f | ||
|  | 8441206e26 | ||
|  | 9fdc5ee748 | ||
|  | 00ff133387 | ||
|  | 96164cb934 | ||
|  | 82fb21ae69 | ||
|  | 89d4a2b4c4 | ||
|  | fc0c7ff374 | ||
|  | 5148c4f2e9 | ||
|  | c3b59f7bcf | ||
|  | 61e148202b | ||
|  | 8a4e0739bc | ||
|  | f75c5f2fe5 | ||
|  | 81d5859588 | ||
|  | 721886bb7a | ||
|  | b23c272820 | ||
|  | cd02bfea7a | ||
|  | 6774bd88f9 | ||
|  | 1046a4f376 | ||
|  | 8081f9ddfd | ||
|  | fa656577d1 | ||
|  | b14b86990f | ||
|  | 2a6dd7b512 | ||
|  | feebdee88b | ||
|  | 99d9277f5d | ||
|  | 9af64d6156 | ||
|  | 5e3775c1af | ||
|  | 2d2e8a3da7 | ||
|  | b2a560b76f | ||
|  | 39397a489d | ||
|  | ff593a0904 | ||
|  | f12789cf44 | ||
|  | 4f8cf2fc87 | ||
|  | fda98730ac | ||
|  | 06c6ddffb6 | ||
|  | d29f0c066c | ||
|  | c9e4de3346 | ||
|  | ca0b97f72d | ||
|  | b38f20b408 | ||
|  | 05b1dbaf56 | ||
|  | b8481e32ba | ||
|  | 9c03c65e07 | ||
|  | d8ed006b9b | ||
|  | 63c0623a5e | ||
|  | fd84506db0 | ||
|  | d8bcb44e44 | ||
|  | 56a26b0916 | ||
|  | efcf1d6b90 | ||
|  | 9f578bfec6 | ||
|  | 1f170d7d28 | ||
|  | 5ae14cf9be | ||
|  | aaf9d53be9 | ||
|  | 75c73f7ba7 | ||
|  | b6dba8beee | ||
|  | 94521cdc1a | ||
|  | 3365b1c355 | ||
|  | 6c957c4923 | ||
|  | 833997f04c | ||
|  | 68d51e4037 | ||
|  | ce274d2011 | ||
|  | 280778ed43 | ||
|  | 0f558ecbbf | ||
|  | 58f9e05d93 | ||
|  | 1ec981aea7 | ||
|  | 2a90286a7c | ||
|  | 12d25d09b2 | ||
|  | a039fae1a4 | ||
|  | 322b9abadc | ||
|  | 0aaf954cea | ||
|  | c2d22aa3d1 | ||
|  | 6934c75bba | ||
|  | c58cf78f86 | ||
|  | 7f0de790ab | ||
|  | d4bb4e3a73 | ||
|  | d25612d038 | ||
|  | 116b2351b0 | ||
|  | 69b83dfdc4 | ||
|  | 3b1839c2ce | ||
|  | 13742ebdf8 | ||
|  | 634657bea1 | ||
|  | 46e70d50b7 | ||
|  | d64e9b85a7 | ||
|  | fb853edbe3 | ||
|  | cc076c1be1 | ||
|  | 98cc9a6755 | ||
|  | 7bd2b9c23a | ||
|  | de724a1ff3 | ||
|  | 2163055dae | ||
|  | 93ed0fc10b | ||
|  | 0d98cefd40 | ||
|  | d58988a033 | ||
|  | 2acfab1e3f | ||
|  | b915dfe9a6 | ||
|  | 25bd5a823e | ||
|  | 1c35de4716 | ||
|  | 4c00435a0a | ||
|  | 844e3079a8 | ||
|  | 4778cb5b2c | ||
|  | ec5d60b919 | ||
|  | e1f4b960e8 | ||
|  | 669e46da54 | ||
|  | ba94cc5df7 | ||
|  | d08245c3df | ||
|  | 5c18d12cbf | ||
|  | 580a42dec7 | ||
|  | 29286e159b | ||
|  | 19bcf90e9f | ||
|  | dae9c00742 | ||
|  | 35324ceb7c | ||
|  | 5aadd47199 | ||
|  | 7d9057cc62 | ||
|  | c4b322b883 | ||
|  | 19b09c898a | ||
|  | eafe2098b6 | ||
|  | 2bc6a20d71 | ||
|  | 8b502a7235 | ||
|  | 37567844af | ||
|  | 2f6c4e0e34 | ||
|  | 1c7cc4cb2b | ||
|  | f83db3648e | ||
|  | b164aa00d4 | ||
|  | a2d866d0c2 | ||
|  | 2dfe4ac4c6 | ||
|  | db65d05cb5 | ||
|  | 300c0194c7 | ||
|  | 37a0d2b087 | ||
|  | a4959300ea | ||
|  | 223657e5f8 | ||
|  | 0c53de6767 | ||
|  | 9c309b1498 | ||
|  | 1aa1b34c80 | ||
|  | 755a2ee023 | ||
|  | 69d3359e47 | ||
|  | a90c49b8fb | ||
|  | b1222edb27 | ||
|  | b967a92f69 | ||
|  | 90a5cb5e59 | ||
|  | 7aba9cb76b | ||
|  | f550a8171d | ||
|  | 82e568d4c9 | ||
|  | 7b2a4a3d59 | ||
|  | 0265455cd1 | ||
|  | afafc886a4 | ||
|  | 8a959f6ac4 | ||
|  | 1c3aa0d2c5 | ||
|  | 79b7d3316a | ||
|  | fa7768583a | ||
|  | faf49f6c15 | ||
|  | 765af31b83 | ||
|  | b6a3c52d67 | ||
|  | b025c2f660 | ||
|  | e559a7c878 | ||
|  | 5c8855aafd | ||
|  | b5fc537b89 | ||
|  | 14899d3a7c | ||
|  | 0ea7881652 | ||
|  | ec29b59d1e | ||
|  | 9405597c15 | ||
|  | 82441978c6 | ||
|  | e0e6291bdb | ||
|  | b2b083fd0a | ||
|  | f8a51b68e7 | ||
|  | e0a19108e5 | ||
|  | 770ea68ca8 | ||
|  | ce36c52baf | ||
|  | a7da1dd233 | ||
|  | 678ef296b4 | ||
|  | 9e5627d805 | ||
|  | 5958ee4439 | ||
|  | 7127e57f0e | ||
|  | ee9c6dc8aa | ||
|  | 92779b3f48 | ||
|  | 2f1baf17d4 | ||
|  | 583da3d4a9 | ||
|  | bf9ff78bcc | ||
|  | 2cb07792cc | ||
|  | 47bc8bb466 | ||
|  | 94ad1f5732 | ||
|  | 09557fbe83 | ||
|  | 1c0f44fa4e | ||
|  | fc4d59d2d7 | ||
|  | 12345fbacc | ||
|  | 2e33c8d222 | ||
|  | db5f07f164 | ||
|  | e050e69a43 | ||
|  | 27cb1d4fc7 | ||
|  | 5d6a740947 | ||
|  | da3f68c363 | ||
|  | d7d1c3685c | ||
|  | dab3407beb | ||
|  | 592987a54a | ||
|  | 8dca8326f7 | ||
|  | 633481fae3 | ||
|  | e7b99e6fb7 | ||
|  | 2a6a3aedd0 | ||
|  | 866c74c841 | ||
|  | dad92bde26 | ||
|  | a994e034f7 | ||
|  | 2801c04f2e | ||
|  | 316e3abfab | ||
|  | c15ecb6c8e | ||
|  | ee96005026 | ||
|  | 5b55d05a20 | ||
|  | 2f09c62c4e | ||
|  | 1cc8b873d4 | ||
|  | 15d5859750 | 
							
								
								
									
										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 | ||||
|   | ||||
							
								
								
									
										10
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -11,14 +11,12 @@ dist/ | ||||
| sfx/ | ||||
| .venv/ | ||||
|  | ||||
| # sublime | ||||
| # ide | ||||
| *.sublime-workspace | ||||
|  | ||||
| # winmerge | ||||
| *.bak | ||||
|  | ||||
| # other licenses | ||||
| contrib/ | ||||
|  | ||||
| # deps | ||||
| copyparty/web/deps | ||||
| # derived | ||||
| copyparty/web/deps/ | ||||
| srv/ | ||||
|   | ||||
							
								
								
									
										30
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										30
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @@ -9,15 +9,25 @@ | ||||
|             "console": "integratedTerminal", | ||||
|             "cwd": "${workspaceFolder}", | ||||
|             "args": [ | ||||
|                 "-j", | ||||
|                 "0", | ||||
|                 //"-nw", | ||||
|                 "-a", | ||||
|                 "ed:wark", | ||||
|                 "-v", | ||||
|                 "srv::r:aed" | ||||
|                 "-ed", | ||||
|                 "-emp", | ||||
|                 "-e2dsa", | ||||
|                 "-e2ts", | ||||
|                 "-mtp", | ||||
|                 ".bpm=f,bin/mtag/audio-bpm.py", | ||||
|                 "-aed:wark", | ||||
|                 "-vsrv::r:aed:cnodupe", | ||||
|                 "-vdist:dist:r" | ||||
|             ] | ||||
|         }, | ||||
|         { | ||||
|             "name": "No debug", | ||||
|             "preLaunchTask": "no_dbg", | ||||
|             "type": "python", | ||||
|             //"request": "attach", "port": 42069 | ||||
|             // fork: nc -l 42069 </dev/null | ||||
|         }, | ||||
|         { | ||||
|             "name": "Run active unit test", | ||||
|             "type": "python", | ||||
| @@ -30,5 +40,13 @@ | ||||
|                 "${file}" | ||||
|             ] | ||||
|         }, | ||||
|         { | ||||
|             "name": "Python: Current File", | ||||
|             "type": "python", | ||||
|             "request": "launch", | ||||
|             "program": "${file}", | ||||
|             "console": "integratedTerminal", | ||||
|             "justMyCode": false | ||||
|         }, | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										45
									
								
								.vscode/launch.py
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								.vscode/launch.py
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| # takes arguments from launch.json | ||||
| # is used by no_dbg in tasks.json | ||||
| # launches 10x faster than mspython debugpy | ||||
| # and is stoppable with ^C | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
|  | ||||
| print(sys.executable) | ||||
|  | ||||
| import shlex | ||||
| import jstyleson | ||||
| import subprocess as sp | ||||
|  | ||||
|  | ||||
| with open(".vscode/launch.json", "r", encoding="utf-8") as f: | ||||
|     tj = f.read() | ||||
|  | ||||
| oj = jstyleson.loads(tj) | ||||
| argv = oj["configurations"][0]["args"] | ||||
|  | ||||
| try: | ||||
|     sargv = " ".join([shlex.quote(x) for x in argv]) | ||||
|     print(sys.executable + " -m copyparty " + sargv + "\n") | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| argv = [os.path.expanduser(x) if x.startswith("~") else x for x in argv] | ||||
|  | ||||
| if re.search(" -j ?[0-9]", " ".join(argv)): | ||||
|     argv = [sys.executable, "-m", "copyparty"] + argv | ||||
|     sp.check_call(argv) | ||||
| else: | ||||
|     sys.path.insert(0, os.getcwd()) | ||||
|     from copyparty.__main__ import main as copyparty | ||||
|  | ||||
|     try: | ||||
|         copyparty(["a"] + argv) | ||||
|     except SystemExit as ex: | ||||
|         if ex.code: | ||||
|             raise | ||||
|  | ||||
| print("\n\033[32mokke\033[0m") | ||||
| sys.exit(1) | ||||
							
								
								
									
										14
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -37,7 +37,7 @@ | ||||
|     "python.linting.banditEnabled": true, | ||||
|     "python.linting.flake8Args": [ | ||||
|         "--max-line-length=120", | ||||
|         "--ignore=E722,F405,E203,W503,W293", | ||||
|         "--ignore=E722,F405,E203,W503,W293,E402", | ||||
|     ], | ||||
|     "python.linting.banditArgs": [ | ||||
|         "--ignore=B104" | ||||
| @@ -50,11 +50,9 @@ | ||||
|     "files.associations": { | ||||
|         "*.makefile": "makefile" | ||||
|     }, | ||||
|     "editor.codeActionsOnSaveTimeout": 9001, | ||||
|     "editor.formatOnSaveTimeout": 9001, | ||||
|     // | ||||
|     //  things you may wanna edit: | ||||
|     // | ||||
|     "python.pythonPath": ".venv/bin/python", | ||||
|     //"python.linting.enabled": true, | ||||
|     "python.formatting.blackArgs": [ | ||||
|         "-t", | ||||
|         "py27" | ||||
|     ], | ||||
|     "python.linting.enabled": true, | ||||
| } | ||||
							
								
								
									
										18
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| { | ||||
|     "version": "2.0.0", | ||||
|     "tasks": [ | ||||
|         { | ||||
|             "label": "pre", | ||||
|             "command": "true;rm -rf inc/* inc/.hist/;mkdir -p inc;", | ||||
|             "type": "shell" | ||||
|         }, | ||||
|         { | ||||
|             "label": "no_dbg", | ||||
|             "type": "shell", | ||||
|             "command": "${config:python.pythonPath}", | ||||
|             "args": [ | ||||
|                 ".vscode/launch.py" | ||||
|             ] | ||||
|         } | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										722
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										722
									
								
								README.md
									
									
									
									
									
								
							| @@ -8,93 +8,704 @@ | ||||
|  | ||||
| turn your phone or raspi into a portable file server with resumable uploads/downloads using IE6 or any other browser | ||||
|  | ||||
| * server runs on anything with `py2.7` or `py3.2+` | ||||
| * *resumable* uploads need `firefox 12+` / `chrome 6+` / `safari 6+` / `IE 10+` | ||||
| * server runs on anything with `py2.7` or `py3.3+` | ||||
| * browse/upload with IE4 / netscape4.0 on win3.11 (heh) | ||||
| * *resumable* uploads need `firefox 34+` / `chrome 41+` / `safari 7+` for full speed | ||||
| * code standard: `black` | ||||
|  | ||||
| 📷 **screenshots:** [browser](#the-browser) // [upload](#uploading) // [thumbnails](#thumbnails) // [md-viewer](#markdown-viewer) // [search](#searching) // [fsearch](#file-search) // [zip-DL](#zip-downloads) // [ie4](#browser-support) | ||||
|  | ||||
|  | ||||
| ## breaking changes \o/ | ||||
|  | ||||
| this is the readme for v0.12 which has a different expression for volume permissions (`-v`); see [the v0.11.x readme](https://github.com/9001/copyparty/tree/15b59822112dda56cee576df30f331252fc62628#readme) for stuff regarding the [current stable release](https://github.com/9001/copyparty/releases/tag/v0.11.47) | ||||
|  | ||||
|  | ||||
| ## readme toc | ||||
|  | ||||
| * top | ||||
|     * [quickstart](#quickstart) | ||||
|         * [on debian](#on-debian) | ||||
|     * [notes](#notes) | ||||
|     * [status](#status) | ||||
|     * [testimonials](#testimonials) | ||||
| * [bugs](#bugs) | ||||
|     * [general bugs](#general-bugs) | ||||
|     * [not my bugs](#not-my-bugs) | ||||
| * [the browser](#the-browser) | ||||
|     * [tabs](#tabs) | ||||
|     * [hotkeys](#hotkeys) | ||||
|     * [navpane](#navpane) | ||||
|     * [thumbnails](#thumbnails) | ||||
|     * [zip downloads](#zip-downloads) | ||||
|     * [uploading](#uploading) | ||||
|         * [file-search](#file-search) | ||||
|     * [file manager](#file-manager) | ||||
|     * [markdown viewer](#markdown-viewer) | ||||
|     * [other tricks](#other-tricks) | ||||
| * [searching](#searching) | ||||
|     * [search configuration](#search-configuration) | ||||
|     * [database location](#database-location) | ||||
|     * [metadata from audio files](#metadata-from-audio-files) | ||||
|     * [file parser plugins](#file-parser-plugins) | ||||
|     * [complete examples](#complete-examples) | ||||
| * [browser support](#browser-support) | ||||
| * [client examples](#client-examples) | ||||
| * [up2k](#up2k) | ||||
| * [performance](#performance) | ||||
| * [dependencies](#dependencies) | ||||
|     * [optional dependencies](#optional-dependencies) | ||||
|     * [install recommended deps](#install-recommended-deps) | ||||
|     * [optional gpl stuff](#optional-gpl-stuff) | ||||
| * [sfx](#sfx) | ||||
|     * [sfx repack](#sfx-repack) | ||||
| * [install on android](#install-on-android) | ||||
| * [building](#building) | ||||
|     * [dev env setup](#dev-env-setup) | ||||
|     * [just the sfx](#just-the-sfx) | ||||
|     * [complete release](#complete-release) | ||||
| * [todo](#todo) | ||||
|     * [discarded ideas](#discarded-ideas) | ||||
|  | ||||
|  | ||||
| ## quickstart | ||||
|  | ||||
| download [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) and you're all set! | ||||
|  | ||||
| running the sfx without arguments (for example doubleclicking it on Windows) will give everyone full access to the current folder; see `-h` for help if you want [accounts and volumes](#accounts-and-volumes) etc | ||||
|  | ||||
| some recommended options: | ||||
| * `-e2dsa` enables general file indexing, see [search configuration](#search-configuration) | ||||
| * `-e2ts` enables audio metadata indexing (needs either FFprobe or Mutagen), see [optional dependencies](#optional-dependencies) | ||||
| * `-v /mnt/music:/music:r:rw,foo -a foo:bar` shares `/mnt/music` as `/music`, `r`eadable by anyone, and read-write for user `foo`, password `bar` | ||||
|   * replace `:r:rw,foo` with `:r,foo` to only make the folder readable by `foo` and nobody else | ||||
|   * see [accounts and volumes](#accounts-and-volumes) for the syntax and other access levels (`r`ead, `w`rite, `m`ove, `d`elete) | ||||
| * `--ls '**,*,ln,p,r'` to crash on startup if any of the volumes contain a symlink which point outside the volume, as that could give users unintended access | ||||
|  | ||||
| you may also want these, especially on servers: | ||||
| * [contrib/systemd/copyparty.service](contrib/systemd/copyparty.service) to run copyparty as a systemd service | ||||
| * [contrib/nginx/copyparty.conf](contrib/nginx/copyparty.conf) to reverse-proxy behind nginx (for better https) | ||||
|  | ||||
|  | ||||
| ### on debian | ||||
|  | ||||
| recommended steps to enable audio metadata and thumbnails (from images and videos): | ||||
|  | ||||
| * as root, run the following:   | ||||
|   `apt install python3 python3-pip python3-dev ffmpeg` | ||||
|  | ||||
| * then, as the user which will be running copyparty (so hopefully not root), run this:   | ||||
|   `python3 -m pip install --user -U Pillow pillow-avif-plugin` | ||||
|  | ||||
| (skipped `pyheif-pillow-opener` because apparently debian is too old to build it) | ||||
|  | ||||
|  | ||||
| ## notes | ||||
|  | ||||
| general: | ||||
| * paper-printing is affected by dark/light-mode! use lightmode for color, darkmode for grayscale | ||||
|   * because no browsers currently implement the media-query to do this properly orz | ||||
|  | ||||
| browser-specific: | ||||
| * iPhone/iPad: use Firefox to download files | ||||
| * Android-Chrome: set max "parallel uploads" for 200% upload speed (android bug) | ||||
| * Android-Firefox: takes a while to select files (in order to avoid the above android-chrome issue) | ||||
| * Desktop-Firefox: may use gigabytes of RAM if your connection is great and your files are massive | ||||
| * Android-Chrome: increase "parallel uploads" for higher speed (android bug) | ||||
| * Android-Firefox: takes a while to select files (their fix for ☝️) | ||||
| * Desktop-Firefox: ~~may use gigabytes of RAM if your files are massive~~ *seems to be OK now* | ||||
| * Desktop-Firefox: may stop you from deleting folders you've uploaded until you visit `about:memory` and click `Minimize memory usage` | ||||
|  | ||||
|  | ||||
| ## status | ||||
|  | ||||
| * [x] sanic multipart parser | ||||
| * [x] load balancer (multiprocessing) | ||||
| * [x] upload (plain multipart, ie6 support) | ||||
| * [x] upload (js, resumable, multithreaded) | ||||
| * [x] download | ||||
| * [x] browser | ||||
| * [x] media player | ||||
| * [ ] thumbnails | ||||
| * [ ] download as zip | ||||
| * [x] volumes | ||||
| * [x] accounts | ||||
| * [x] markdown viewer | ||||
| * [x] markdown editor | ||||
| summary: all planned features work! now please enjoy the bloatening | ||||
|  | ||||
| summary: it works! you can use it! (but technically not even close to beta) | ||||
| * backend stuff | ||||
|   * ☑ sanic multipart parser | ||||
|   * ☑ multiprocessing (actual multithreading) | ||||
|   * ☑ volumes (mountpoints) | ||||
|   * ☑ [accounts](#accounts-and-volumes) | ||||
| * upload | ||||
|   * ☑ basic: plain multipart, ie6 support | ||||
|   * ☑ up2k: js, resumable, multithreaded | ||||
|   * ☑ stash: simple PUT filedropper | ||||
|   * ☑ unpost: undo/delete accidental uploads | ||||
|   * ☑ symlink/discard existing files (content-matching) | ||||
| * download | ||||
|   * ☑ single files in browser | ||||
|   * ☑ folders as zip / tar files | ||||
|   * ☑ FUSE client (read-only) | ||||
| * browser | ||||
|   * ☑ navpane (directory tree sidebar) | ||||
|   * ☑ audio player (with OS media controls) | ||||
|   * ☑ thumbnails | ||||
|     * ☑ ...of images using Pillow | ||||
|     * ☑ ...of videos using FFmpeg | ||||
|     * ☑ cache eviction (max-age; maybe max-size eventually) | ||||
|   * ☑ image gallery with webm player | ||||
|   * ☑ SPA (browse while uploading) | ||||
|     * if you use the navpane to navigate, not folders in the file list | ||||
| * server indexing | ||||
|   * ☑ locate files by contents | ||||
|   * ☑ search by name/path/date/size | ||||
|   * ☑ search by ID3-tags etc. | ||||
| * markdown | ||||
|   * ☑ viewer | ||||
|   * ☑ editor (sure why not) | ||||
|  | ||||
|  | ||||
| ## testimonials | ||||
|  | ||||
| small collection of user feedback | ||||
|  | ||||
| `good enough`, `surprisingly correct`, `certified good software`, `just works`, `why` | ||||
|  | ||||
|  | ||||
| # bugs | ||||
|  | ||||
| * Windows: python 3.7 and older cannot read tags with FFprobe, so use Mutagen or upgrade | ||||
| * Windows: python 2.7 cannot index non-ascii filenames with `-e2d` | ||||
| * Windows: python 2.7 cannot handle filenames with mojibake | ||||
| * `--th-ff-jpg` may fix video thumbnails on some FFmpeg versions | ||||
|  | ||||
| ## general bugs | ||||
|  | ||||
| * all volumes must exist / be available on startup; up2k (mtp especially) gets funky otherwise | ||||
| * cannot mount something at `/d1/d2/d3` unless `d2` exists inside `d1` | ||||
| * probably more, pls let me know | ||||
|  | ||||
| ## not my bugs | ||||
|  | ||||
| * Windows: folders cannot be accessed if the name ends with `.` | ||||
|   * python or windows bug | ||||
|  | ||||
| * Windows: msys2-python 3.8.6 occasionally throws `RuntimeError: release unlocked lock` when leaving a scoped mutex in up2k | ||||
|   * this is an msys2 bug, the regular windows edition of python is fine | ||||
|  | ||||
| * VirtualBox: sqlite throws `Disk I/O Error` when running in a VM and the up2k database is in a vboxsf | ||||
|   * use `--hist` or the `hist` volflag (`-v [...]:chist=/tmp/foo`) to place the db inside the vm instead | ||||
|  | ||||
|  | ||||
| # accounts and volumes | ||||
|  | ||||
| * `-a usr:pwd` adds account `usr` with password `pwd` | ||||
| * `-v .::r` adds current-folder `.` as the webroot, `r`eadable by anyone | ||||
|   * the syntax is `-v src:dst:perm:perm:...` so local-path, url-path, and one or more permissions to set | ||||
|   * when granting permissions to an account, the names are comma-separated: `-v .::r,usr1,usr2:rw,usr3,usr4` | ||||
|  | ||||
| permissions: | ||||
| * `r` (read): browse folder contents, download files, download as zip/tar | ||||
| * `w` (write): upload files, move files *into* folder | ||||
| * `m` (move): move files/folders *from* folder | ||||
| * `d` (delete): delete files/folders | ||||
|  | ||||
| example: | ||||
| * add accounts named u1, u2, u3 with passwords p1, p2, p3: `-a u1:p1 -a u2:p2 -a u3:p3` | ||||
| * make folder `/srv` the root of the filesystem, read-only by anyone: `-v /srv::r` | ||||
| * make folder `/mnt/music` available at `/music`, read-only for u1 and u2, read-write for u3: `-v /mnt/music:music:r,u1,u2:rw,u3` | ||||
|   * unauthorized users accessing the webroot can see that the `music` folder exists, but cannot open it | ||||
| * make folder `/mnt/incoming` available at `/inc`, write-only for u1, read-move for u2: `-v /mnt/incoming:inc:w,u1:rm,u2` | ||||
|   * unauthorized users accessing the webroot can see that the `inc` folder exists, but cannot open it | ||||
|   * `u1` can open the `inc` folder, but cannot see the contents, only upload new files to it | ||||
|   * `u2` can browse it and move files *from* `/inc` into any folder where `u2` has write-access | ||||
|  | ||||
|  | ||||
| # the browser | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## tabs | ||||
|  | ||||
| * `[🔎]` search by size, date, path/name, mp3-tags ... see [searching](#searching) | ||||
| * `[🧯]` unpost: undo/delete accidental uploads | ||||
| * `[🚀]` and `[🎈]` are the uploaders, see [uploading](#uploading) | ||||
| * `[📂]` mkdir: create directories | ||||
| * `[📝]` new-md: create a new markdown document | ||||
| * `[📟]` send-msg: either to server-log or into textfiles if `--urlform save` | ||||
| * `[🎺]` audio-player config options | ||||
| * `[⚙️]` general client config options | ||||
|  | ||||
|  | ||||
| ## hotkeys | ||||
|  | ||||
| the browser has the following hotkeys (assumes qwerty, ignores actual layout) | ||||
| * `B` toggle breadcrumbs / navpane | ||||
| * `I/K` prev/next folder | ||||
| * `M` parent folder (or unexpand current) | ||||
| * `G` toggle list / grid view | ||||
| * `T` toggle thumbnails / icons | ||||
| * `ctrl-X` cut selected files/folders | ||||
| * `ctrl-V` paste | ||||
| * `F2` rename selected file/folder | ||||
| * when a file/folder is selected (in not-grid-view): | ||||
|   * `Up/Down` move cursor | ||||
|   * shift+`Up/Down` select and move cursor | ||||
|   * ctrl+`Up/Down` move cursor and scroll viewport | ||||
|   * `Space` toggle file selection | ||||
|   * `Ctrl-A` toggle select all | ||||
| * when playing audio: | ||||
|   * `J/L` prev/next song | ||||
|   * `U/O` skip 10sec back/forward | ||||
|   * `0..9` jump to 0%..90% | ||||
|   * `P` play/pause (also starts playing the folder) | ||||
| * when viewing images / playing videos: | ||||
|   * `J/L, Left/Right` prev/next file | ||||
|   * `Home/End` first/last file | ||||
|   * `Esc` close viewer | ||||
|   * videos: | ||||
|     * `U/O` skip 10sec back/forward | ||||
|     * `P/K/Space` play/pause | ||||
|     * `F` fullscreen | ||||
|     * `C` continue playing next video | ||||
|     * `R` loop | ||||
|     * `M` mute | ||||
| * when the navpane is open: | ||||
|   * `A/D` adjust tree width | ||||
| * in the grid view: | ||||
|   * `S` toggle multiselect | ||||
|   * shift+`A/D` zoom | ||||
| * in the markdown editor: | ||||
|   * `^s` save | ||||
|   * `^h` header | ||||
|   * `^k` autoformat table | ||||
|   * `^u` jump to next unicode character | ||||
|   * `^e` toggle editor / preview | ||||
|   * `^up, ^down` jump paragraphs | ||||
|  | ||||
|  | ||||
| ## navpane | ||||
|  | ||||
| by default there's a breadcrumbs path; you can replace this with a navpane (tree-browser sidebar thing) by clicking the `🌲` or pressing the `B` hotkey | ||||
|  | ||||
| click `[-]` and `[+]` (or hotkeys `A`/`D`) to adjust the size, and the `[a]` toggles if the tree should widen dynamically as you go deeper or stay fixed-size | ||||
|  | ||||
|  | ||||
| ## thumbnails | ||||
|  | ||||
|  | ||||
|  | ||||
| it does static images with Pillow and uses FFmpeg for video files, so you may want to `--no-thumb` or maybe just `--no-vthumb` depending on how dangerous your users are | ||||
|  | ||||
| images with the following names (see `--th-covers`) become the thumbnail of the folder they're in: `folder.png`, `folder.jpg`, `cover.png`, `cover.jpg` | ||||
|  | ||||
| in the grid/thumbnail view, if the audio player panel is open, songs will start playing when clicked | ||||
|  | ||||
|  | ||||
| ## zip downloads | ||||
|  | ||||
| the `zip` link next to folders can produce various types of zip/tar files using these alternatives in the browser settings tab: | ||||
|  | ||||
| | name | url-suffix | description | | ||||
| |--|--|--| | ||||
| | `tar` | `?tar` | plain gnutar, works great with `curl \| tar -xv` | | ||||
| | `zip` | `?zip=utf8` | works everywhere, glitchy filenames on win7 and older | | ||||
| | `zip_dos` | `?zip` | traditional cp437 (no unicode) to fix glitchy filenames | | ||||
| | `zip_crc` | `?zip=crc` | cp437 with crc32 computed early for truly ancient software | | ||||
|  | ||||
| * hidden files (dotfiles) are excluded unless `-ed` | ||||
|   * `up2k.db` and `dir.txt` is always excluded | ||||
| * `zip_crc` will take longer to download since the server has to read each file twice | ||||
|   * this is only to support MS-DOS PKZIP v2.04g (october 1993) and older | ||||
|     * how are you accessing copyparty actually | ||||
|  | ||||
| you can also zip a selection of files or folders by clicking them in the browser, that brings up a selection editor and zip button in the bottom right | ||||
|  | ||||
|  | ||||
|  | ||||
| ## uploading | ||||
|  | ||||
| two upload methods are available in the html client: | ||||
| * `[🎈] bup`, the basic uploader, supports almost every browser since netscape 4.0 | ||||
| * `[🚀] up2k`, the fancy one | ||||
|  | ||||
| you can undo/delete uploads using `[🧯] unpost` if the server is running with `-e2d` | ||||
|  | ||||
| up2k has several advantages: | ||||
| * you can drop folders into the browser (files are added recursively) | ||||
| * files are processed in chunks, and each chunk is checksummed | ||||
|   * uploads autoresume if they are interrupted by network issues | ||||
|   * uploads resume if you reboot your browser or pc, just upload the same files again | ||||
|   * server detects any corruption; the client reuploads affected chunks | ||||
|   * the client doesn't upload anything that already exists on the server | ||||
| * much higher speeds than ftp/scp/tarpipe on some internet connections (mainly american ones) thanks to parallel connections | ||||
| * the last-modified timestamp of the file is preserved | ||||
|  | ||||
| see [up2k](#up2k) for details on how it works | ||||
|  | ||||
|  | ||||
|  | ||||
| **protip:** you can avoid scaring away users with [docs/minimal-up2k.html](docs/minimal-up2k.html) which makes it look [much simpler](https://user-images.githubusercontent.com/241032/118311195-dd6ca380-b4ef-11eb-86f3-75a3ff2e1332.png) | ||||
|  | ||||
| the up2k UI is the epitome of polished inutitive experiences: | ||||
| * "parallel uploads" specifies how many chunks to upload at the same time | ||||
| * `[🏃]` analysis of other files should continue while one is uploading | ||||
| * `[💭]` ask for confirmation before files are added to the list | ||||
| * `[💤]` sync uploading between other copyparty tabs so only one is active | ||||
| * `[🔎]` switch between upload and file-search mode | ||||
|  | ||||
| and then theres the tabs below it, | ||||
| * `[ok]` is uploads which completed successfully | ||||
| * `[ng]` is the uploads which failed / got rejected (already exists, ...) | ||||
| * `[done]` shows a combined list of `[ok]` and `[ng]`, chronological order | ||||
| * `[busy]` files which are currently hashing, pending-upload, or uploading | ||||
|   * plus up to 3 entries each from `[done]` and `[que]` for context | ||||
| * `[que]` is all the files that are still queued | ||||
|  | ||||
| ### file-search | ||||
|  | ||||
|  | ||||
|  | ||||
| in the `[🚀 up2k]` tab, after toggling the `[🔎]` switch green, any files/folders you drop onto the dropzone will be hashed on the client-side. Each hash is sent to the server which checks if that file exists somewhere already | ||||
|  | ||||
| files go into `[ok]` if they exist (and you get a link to where it is), otherwise they land in `[ng]` | ||||
| * the main reason filesearch is combined with the uploader is cause the code was too spaghetti to separate it out somewhere else, this is no longer the case but now i've warmed up to the idea too much | ||||
|  | ||||
| adding the same file multiple times is blocked, so if you first search for a file and then decide to upload it, you have to click the `[cleanup]` button to discard `[done]` files (or just refresh the page) | ||||
|  | ||||
| note that since up2k has to read the file twice, `[🎈 bup]` can be up to 2x faster in extreme cases (if your internet connection is faster than the read-speed of your HDD) | ||||
|  | ||||
| up2k has saved a few uploads from becoming corrupted in-transfer already; caught an android phone on wifi redhanded in wireshark with a bitflip, however bup with https would *probably* have noticed as well (thanks to tls also functioning as an integrity check) | ||||
|  | ||||
|  | ||||
| ## file manager | ||||
|  | ||||
| if you have the required permissions, you can cut/paste, rename, and delete files/folders | ||||
|  | ||||
| you can move files across browser tabs (cut in one tab, paste in another) | ||||
|  | ||||
|  | ||||
| ## markdown viewer | ||||
|  | ||||
|  | ||||
|  | ||||
| * the document preview has a max-width which is the same as an A4 paper when printed | ||||
|  | ||||
|  | ||||
| ## other tricks | ||||
|  | ||||
| * you can link a particular timestamp in an audio file by adding it to the URL, such as `&20` / `&20s` / `&1m20` / `&t=1:20` after the `.../#af-c8960dab` | ||||
|  | ||||
| * if you are using media hotkeys to switch songs and are getting tired of seeing the OSD popup which Windows doesn't let you disable, consider https://ocv.me/dev/?media-osd-bgone.ps1 | ||||
|  | ||||
|  | ||||
| # searching | ||||
|  | ||||
|  | ||||
|  | ||||
| when started with `-e2dsa` copyparty will scan/index all your files. This avoids duplicates on upload, and also makes the volumes searchable through the web-ui: | ||||
| * make search queries by `size`/`date`/`directory-path`/`filename`, or... | ||||
| * drag/drop a local file to see if the same contents exist somewhere on the server, see [file-search](#file-search) | ||||
|  | ||||
| path/name queries are space-separated, AND'ed together, and words are negated with a `-` prefix, so for example: | ||||
| * path: `shibayan -bossa` finds all files where one of the folders contain `shibayan` but filters out any results where `bossa` exists somewhere in the path | ||||
| * name: `demetori styx` gives you [good stuff](https://www.youtube.com/watch?v=zGh0g14ZJ8I&list=PL3A147BD151EE5218&index=9) | ||||
|  | ||||
| add `-e2ts` to also scan/index tags from music files: | ||||
|  | ||||
|  | ||||
| ## search configuration | ||||
|  | ||||
| searching relies on two databases, the up2k filetree (`-e2d`) and the metadata tags (`-e2t`). Configuration can be done through arguments, volume flags, or a mix of both. | ||||
|  | ||||
| through arguments: | ||||
| * `-e2d` enables file indexing on upload | ||||
| * `-e2ds` scans writable folders for new files on startup | ||||
| * `-e2dsa` scans all mounted volumes (including readonly ones) | ||||
| * `-e2t` enables metadata indexing on upload | ||||
| * `-e2ts` scans for tags in all files that don't have tags yet | ||||
| * `-e2tsr` deletes all existing tags, does a full reindex | ||||
|  | ||||
| the same arguments can be set as volume flags, in addition to `d2d` and `d2t` for disabling: | ||||
| * `-v ~/music::r:ce2dsa:ce2tsr` does a full reindex of everything on startup | ||||
| * `-v ~/music::r:cd2d` disables **all** indexing, even if any `-e2*` are on | ||||
| * `-v ~/music::r:cd2t` disables all `-e2t*` (tags), does not affect `-e2d*` | ||||
|  | ||||
| note: | ||||
| * `e2tsr` is probably always overkill, since `e2ds`/`e2dsa` would pick up any file modifications and `e2ts` would then reindex those, unless there is a new copyparty version with new parsers and the release note says otherwise | ||||
| * the rescan button in the admin panel has no effect unless the volume has `-e2ds` or higher | ||||
|  | ||||
| you can choose to only index filename/path/size/last-modified (and not the hash of the file contents) by setting `--no-hash` or the volume-flag `cdhash`, this has the following consequences: | ||||
| * initial indexing is way faster, especially when the volume is on a network disk | ||||
| * makes it impossible to [file-search](#file-search) | ||||
| * if someone uploads the same file contents, the upload will not be detected as a dupe, so it will not get symlinked or rejected | ||||
|  | ||||
| if you set `--no-hash`, you can enable hashing for specific volumes using flag `cehash` | ||||
|  | ||||
|  | ||||
| ## database location | ||||
|  | ||||
| copyparty creates a subfolder named `.hist` inside each volume where it stores the database, thumbnails, and some other stuff | ||||
|  | ||||
| this can instead be kept in a single place using the `--hist` argument, or the `hist=` volume flag, or a mix of both: | ||||
| * `--hist ~/.cache/copyparty -v ~/music::r:chist=-` sets `~/.cache/copyparty` as the default place to put volume info, but `~/music` gets the regular `.hist` subfolder (`-` restores default behavior) | ||||
|  | ||||
| note: | ||||
| * markdown edits are always stored in a local `.hist` subdirectory | ||||
| * on windows the volflag path is cyglike, so `/c/temp` means `C:\temp` but use regular paths for `--hist` | ||||
|   * you can use cygpaths for volumes too, `-v C:\Users::r` and `-v /c/users::r` both work | ||||
|  | ||||
|  | ||||
| ## metadata from audio files | ||||
|  | ||||
| `-mte` decides which tags to index and display in the browser (and also the display order), this can be changed per-volume: | ||||
| * `-v ~/music::r:cmte=title,artist` indexes and displays *title* followed by *artist* | ||||
|  | ||||
| if you add/remove a tag from `mte` you will need to run with `-e2tsr` once to rebuild the database, otherwise only new files will be affected | ||||
|  | ||||
| `-mtm` can be used to add or redefine a metadata mapping, say you have media files with `foo` and `bar` tags and you want them to display as `qux` in the browser (preferring `foo` if both are present), then do `-mtm qux=foo,bar` and now you can `-mte artist,title,qux` | ||||
|  | ||||
| tags that start with a `.` such as `.bpm` and `.dur`(ation) indicate numeric value | ||||
|  | ||||
| see the beautiful mess of a dictionary in [mtag.py](https://github.com/9001/copyparty/blob/master/copyparty/mtag.py) for the default mappings (should cover mp3,opus,flac,m4a,wav,aif,) | ||||
|  | ||||
| `--no-mutagen` disables Mutagen and uses FFprobe instead, which... | ||||
| * is about 20x slower than Mutagen | ||||
| * catches a few tags that Mutagen doesn't | ||||
|   * melodic key, video resolution, framerate, pixfmt | ||||
| * avoids pulling any GPL code into copyparty | ||||
| * more importantly runs FFprobe on incoming files which is bad if your FFmpeg has a cve | ||||
|  | ||||
|  | ||||
| ## file parser plugins | ||||
|  | ||||
| copyparty can invoke external programs to collect additional metadata for files using `mtp` (either as argument or volume flag), there is a default timeout of 30sec | ||||
|  | ||||
| * `-mtp .bpm=~/bin/audio-bpm.py` will execute `~/bin/audio-bpm.py` with the audio file as argument 1 to provide the `.bpm` tag, if that does not exist in the audio metadata | ||||
| * `-mtp key=f,t5,~/bin/audio-key.py` uses `~/bin/audio-key.py` to get the `key` tag, replacing any existing metadata tag (`f,`), aborting if it takes longer than 5sec (`t5,`) | ||||
| * `-v ~/music::r:cmtp=.bpm=~/bin/audio-bpm.py:cmtp=key=f,t5,~/bin/audio-key.py` both as a per-volume config wow this is getting ugly | ||||
|  | ||||
| *but wait, there's more!* `-mtp` can be used for non-audio files as well using the `a` flag: `ay` only do audio files, `an` only do non-audio files, or `ad` do all files (d as in dontcare)  | ||||
|  | ||||
| * `-mtp ext=an,~/bin/file-ext.py` runs `~/bin/file-ext.py` to get the `ext` tag only if file is not audio (`an`) | ||||
| * `-mtp arch,built,ver,orig=an,eexe,edll,~/bin/exe.py` runs `~/bin/exe.py` to get properties about windows-binaries only if file is not audio (`an`) and file extension is exe or dll | ||||
|  | ||||
|  | ||||
| ## complete examples | ||||
|  | ||||
| * read-only music server with bpm and key scanning   | ||||
|   `python copyparty-sfx.py -v /mnt/nas/music:/music:r -e2dsa -e2ts -mtp .bpm=f,audio-bpm.py -mtp key=f,audio-key.py` | ||||
|  | ||||
|  | ||||
| # browser support | ||||
|  | ||||
|  | ||||
|  | ||||
| `ie` = internet-explorer, `ff` = firefox, `c` = chrome, `iOS` = iPhone/iPad, `Andr` = Android | ||||
|  | ||||
| | feature         | ie6 | ie9 | ie10 | ie11 | ff 52 | c 49 | iOS | Andr | | ||||
| | --------------- | --- | --- | ---- | ---- | ----- | ---- | --- | ---- | | ||||
| | browse files    | yep | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | basic uploader  | yep | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | make directory  | yep | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | send message    | yep | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | set sort order  |  -  | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | zip selection   |  -  | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | navpane         |  -  |  -  | `*1` | yep  | yep   | yep  | yep | yep  | | ||||
| | up2k            |  -  |  -  | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | markdown editor |  -  |  -  | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | markdown viewer |  -  |  -  | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | play mp3/m4a    |  -  | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | play ogg/opus   |  -  |  -  |  -   |  -   | yep   | yep  | `*2` | yep | | ||||
| | thumbnail view  |  -  |  -  | -    | -    | yep   | yep  | yep | yep  | | ||||
| | image viewer    |  -  |  -  | -    | -    | yep   | yep  | yep | yep  | | ||||
| | **= feature =** | ie6 | ie9 | ie10 | ie11 | ff 52 | c 49 | iOS | Andr | | ||||
|  | ||||
| * internet explorer 6 to 8 behave the same | ||||
| * firefox 52 and chrome 49 are the last winxp versions | ||||
| * `*1` only public folders (login session is dropped) and no history / back-button | ||||
| * `*2` using a wasm decoder which can sometimes get stuck and consumes a bit more power | ||||
|  | ||||
| quick summary of more eccentric web-browsers trying to view a directory index: | ||||
|  | ||||
| | browser | will it blend | | ||||
| | ------- | ------------- | | ||||
| | **safari** (14.0.3/macos) | is chrome with janky wasm, so playing opus can deadlock the javascript engine | | ||||
| | **safari** (14.0.1/iOS)   | same as macos, except it recovers from the deadlocks if you poke it a bit | | ||||
| | **links** (2.21/macports) | can browse, login, upload/mkdir/msg | | ||||
| | **lynx** (2.8.9/macports) | can browse, login, upload/mkdir/msg | | ||||
| | **w3m** (0.5.3/macports)  | can browse, login, upload at 100kB/s, mkdir/msg | | ||||
| | **netsurf** (3.10/arch)   | is basically ie6 with much better css (javascript has almost no effect) |  | ||||
| | **ie4** and **netscape** 4.0  | can browse (text is yellow on white), upload with `?b=u` | | ||||
| | **SerenityOS** (7e98457)  | hits a page fault, works with `?b=u`, file upload not-impl | | ||||
|  | ||||
|  | ||||
| # client examples | ||||
|  | ||||
| * javascript: dump some state into a file (two separate examples) | ||||
|   * `await fetch('https://127.0.0.1:3923/', {method:"PUT", body: JSON.stringify(foo)});` | ||||
|   * `var xhr = new XMLHttpRequest(); xhr.open('POST', 'https://127.0.0.1:3923/msgs?raw'); xhr.send('foo');` | ||||
|  | ||||
| * curl/wget: upload some files (post=file, chunk=stdin) | ||||
|   * `post(){ curl -b cppwd=wark http://127.0.0.1:3923/ -F act=bput -F f=@"$1";}`   | ||||
|     `post movie.mkv` | ||||
|   * `post(){ wget --header='Cookie: cppwd=wark' http://127.0.0.1:3923/?raw --post-file="$1" -O-;}`   | ||||
|     `post movie.mkv` | ||||
|   * `chunk(){ curl -b cppwd=wark http://127.0.0.1:3923/ -T-;}`   | ||||
|     `chunk <movie.mkv` | ||||
|  | ||||
| * FUSE: mount a copyparty server as a local filesystem | ||||
|   * cross-platform python client available in [./bin/](bin/) | ||||
|   * [rclone](https://rclone.org/) as client can give ~5x performance, see [./docs/rclone.md](docs/rclone.md) | ||||
|  | ||||
| * sharex (screenshot utility): see [./contrib/sharex.sxcu](contrib/#sharexsxcu) | ||||
|  | ||||
| copyparty returns a truncated sha512sum of your PUT/POST as base64; you can generate the same checksum locally to verify uplaods: | ||||
|  | ||||
|     b512(){ printf "$((sha512sum||shasum -a512)|sed -E 's/ .*//;s/(..)/\\x\1/g')"|base64|tr '+/' '-_'|head -c44;} | ||||
|     b512 <movie.mkv | ||||
|  | ||||
|  | ||||
| # up2k | ||||
|  | ||||
| quick outline of the up2k protocol, see [uploading](#uploading) for the web-client | ||||
| * the up2k client splits a file into an "optimal" number of chunks | ||||
|   * 1 MiB each, unless that becomes more than 256 chunks | ||||
|   * tries 1.5M, 2M, 3, 4, 6, ... until <= 256 chunks or size >= 32M | ||||
| * client posts the list of hashes, filename, size, last-modified | ||||
| * server creates the `wark`, an identifier for this upload | ||||
|   * `sha512( salt + filesize + chunk_hashes )` | ||||
|   * and a sparse file is created for the chunks to drop into | ||||
| * client uploads each chunk | ||||
|   * header entries for the chunk-hash and wark | ||||
|   * server writes chunks into place based on the hash | ||||
| * client does another handshake with the hashlist; server replies with OK or a list of chunks to reupload | ||||
|  | ||||
|  | ||||
| # performance | ||||
|  | ||||
| defaults are good for most cases, don't mind the `cannot efficiently use multiple CPU cores` message, it's very unlikely to be a problem | ||||
|  | ||||
| below are some tweaks roughly ordered by usefulness: | ||||
|  | ||||
| * `-q` disables logging and can help a bunch, even when combined with `-lo` to redirect logs to file | ||||
| * `--http-only` or `--https-only` (unless you want to support both protocols) will reduce the delay before a new connection is established | ||||
| * `--hist` pointing to a fast location (ssd) will make directory listings and searches faster when `-e2d` or `-e2t` is set | ||||
| * `--no-hash` when indexing a network-disk if you don't care about the actual filehashes and only want the names/tags searchable | ||||
| * `-j` enables multiprocessing (actual multithreading) and can make copyparty perform better in cpu-intensive workloads, for example: | ||||
|   * huge amount of short-lived connections | ||||
|   * really heavy traffic (downloads/uploads) | ||||
|    | ||||
|   ...however it adds an overhead to internal communication so it might be a net loss, see if it works 4 u | ||||
|  | ||||
|  | ||||
| # dependencies | ||||
|  | ||||
| * `jinja2` | ||||
|   * pulls in `markupsafe` as of v2.7; use jinja 2.6 on py3.2 | ||||
| * `jinja2` (is built into the SFX) | ||||
|  | ||||
| optional, enables thumbnails: | ||||
|  | ||||
| ## optional dependencies | ||||
|  | ||||
| enable music tags: | ||||
| * either `mutagen` (fast, pure-python, skips a few tags, makes copyparty GPL? idk) | ||||
| * or `ffprobe` (20x slower, more accurate, possibly dangerous depending on your distro and users) | ||||
|  | ||||
| enable thumbnails of images: | ||||
| * `Pillow` (requires py2.7 or py3.5+) | ||||
|  | ||||
| enable thumbnails of videos: | ||||
| * `ffmpeg` and `ffprobe` somewhere in `$PATH` | ||||
|  | ||||
| enable thumbnails of HEIF pictures: | ||||
| * `pyheif-pillow-opener` (requires Linux or a C compiler) | ||||
|  | ||||
| enable thumbnails of AVIF pictures: | ||||
| * `pillow-avif-plugin` | ||||
|  | ||||
|  | ||||
| ## install recommended deps | ||||
| ``` | ||||
| python -m pip install --user -U jinja2 mutagen Pillow | ||||
| ``` | ||||
|  | ||||
|  | ||||
| ## optional gpl stuff | ||||
|  | ||||
| some bundled tools have copyleft dependencies, see [./bin/#mtag](bin/#mtag) | ||||
|  | ||||
| these are standalone programs and will never be imported / evaluated by copyparty, and must be enabled through `-mtp` configs | ||||
|  | ||||
|  | ||||
| # sfx | ||||
|  | ||||
| currently there are two self-contained binaries: | ||||
| * `copyparty-sfx.sh` for unix (linux and osx) -- smaller, more robust | ||||
| * `copyparty-sfx.py` for windows (unix too) -- crossplatform, beta | ||||
| currently there are two self-contained "binaries": | ||||
| * [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) -- pure python, works everywhere, **recommended** | ||||
| * [copyparty-sfx.sh](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.sh) -- smaller, but only for linux and macos, kinda deprecated | ||||
|  | ||||
| launch either of them and it'll unpack and run copyparty, assuming you have python installed of course | ||||
| launch either of them (**use sfx.py on systemd**) and it'll unpack and run copyparty, assuming you have python installed of course | ||||
|  | ||||
| pls note that `copyparty-sfx.sh` will fail if you rename `copyparty-sfx.py` to `copyparty.py` and keep it in the same folder because `sys.path` is funky | ||||
|  | ||||
| if you don't need all the features you can repack the sfx and save a bunch of space, tho currently the only removable feature is the opus/vorbis javascript decoder which is needed by apple devices to play foss audio files | ||||
|  | ||||
| steps to reduce the sfx size from `720 kB` to `250 kB` roughly: | ||||
| * run one of the sfx'es once to unpack it | ||||
| * `./scripts/make-sfx.sh re no-ogv` creates a new pair of sfx | ||||
| ## sfx repack | ||||
|  | ||||
| no internet connection needed, just download an sfx and the repo zip (also if you're on windows use msys2) | ||||
| if you don't need all the features, you can repack the sfx and save a bunch of space; all you need is an sfx and a copy of this repo (nothing else to download or build, except if you're on windows then you need msys2 or WSL) | ||||
| * `525k` size of original sfx.py as of v0.11.30 | ||||
| * `315k` after `./scripts/make-sfx.sh re no-ogv` | ||||
| * `223k` after `./scripts/make-sfx.sh re no-ogv no-cm` | ||||
|  | ||||
| the features you can opt to drop are | ||||
| * `ogv`.js, the opus/vorbis decoder which is needed by apple devices to play foss audio files | ||||
| * `cm`/easymde, the "fancy" markdown editor | ||||
|  | ||||
| for the `re`pack to work, first run one of the sfx'es once to unpack it | ||||
|  | ||||
| **note:** you can also just download and run [scripts/copyparty-repack.sh](scripts/copyparty-repack.sh) -- this will grab the latest copyparty release from github and do a `no-ogv no-cm` repack; works on linux/macos (and windows with msys2 or WSL) | ||||
|  | ||||
|  | ||||
| # install on android | ||||
|  | ||||
| install [Termux](https://termux.com/) (see [ocv.me/termux](https://ocv.me/termux/)) and then copy-paste this into Termux (long-tap) all at once: | ||||
| ```sh | ||||
| apt update && apt -y full-upgrade && termux-setup-storage && apt -y install curl && cd && curl -L https://github.com/9001/copyparty/raw/master/scripts/copyparty-android.sh > copyparty-android.sh && chmod 755 copyparty-android.sh && ./copyparty-android.sh -h | ||||
| apt update && apt -y full-upgrade && termux-setup-storage && apt -y install python && python -m ensurepip && python -m pip install -U copyparty | ||||
| echo $? | ||||
| ``` | ||||
|  | ||||
| after the initial setup (and restarting bash), you can launch copyparty at any time by running "copyparty" in Termux | ||||
| after the initial setup, you can launch copyparty at any time by running `copyparty` anywhere in Termux | ||||
|  | ||||
|  | ||||
| # dev env setup | ||||
| # building | ||||
|  | ||||
| ## dev env setup | ||||
|  | ||||
| mostly optional; if you need a working env for vscode or similar | ||||
|  | ||||
| ```sh | ||||
| python3 -m venv .venv | ||||
| . .venv/bin/activate | ||||
| pip install jinja2  # mandatory deps | ||||
| pip install Pillow  # thumbnail deps | ||||
| pip install jinja2  # mandatory | ||||
| pip install mutagen  # audio metadata | ||||
| pip install Pillow pyheif-pillow-opener pillow-avif-plugin  # thumbnails | ||||
| pip install black bandit pylint flake8  # vscode tooling | ||||
| ``` | ||||
|  | ||||
|  | ||||
| # how to release | ||||
| ## just the sfx | ||||
|  | ||||
| unless you need to modify something in the web-dependencies, it's faster to grab those from a previous release: | ||||
|  | ||||
| ```sh | ||||
| rm -rf copyparty/web/deps | ||||
| curl -L https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py >x.py | ||||
| python3 x.py -h | ||||
| rm x.py | ||||
| mv /tmp/pe-copyparty/copyparty/web/deps/ copyparty/web/deps/ | ||||
| ``` | ||||
|  | ||||
| then build the sfx using any of the following examples: | ||||
|  | ||||
| ```sh | ||||
| ./scripts/make-sfx.sh  # both python and sh editions | ||||
| ./scripts/make-sfx.sh no-sh gz  # just python with gzip | ||||
| ``` | ||||
|  | ||||
|  | ||||
| ## complete release | ||||
|  | ||||
| also builds the sfx so disregard the sfx section above | ||||
|  | ||||
| in the `scripts` folder: | ||||
|  | ||||
| * run `make -C deps-docker` to build all dependencies | ||||
| * `git tag v1.2.3 && git push origin --tags` | ||||
| * create github release with `make-tgz-release.sh` | ||||
| * upload to pypi with `make-pypi-release.(sh|bat)` | ||||
| * create sfx with `make-sfx.sh` | ||||
| @@ -104,15 +715,32 @@ in the `scripts` folder: | ||||
|  | ||||
| roughly sorted by priority | ||||
|  | ||||
| * sortable browser columns | ||||
| * up2k handle filename too long | ||||
| * up2k fails on empty files? alert then stuck | ||||
| * unexpected filepath on dupe up2k | ||||
| * drop onto folders | ||||
| * look into android thumbnail cache file format | ||||
| * support pillow-simd | ||||
| * hls framework for Someone Else to drop code into :^) | ||||
| * readme.md as epilogue | ||||
|  | ||||
|  | ||||
| ## discarded ideas | ||||
|  | ||||
| * reduce up2k roundtrips | ||||
|   * start from a chunk index and just go | ||||
|   * terminate client on bad data | ||||
|     * not worth the effort, just throw enough conncetions at it | ||||
| * single sha512 across all up2k chunks? | ||||
|   * crypto.subtle cannot into streaming, would have to use hashwasm, expensive | ||||
| * separate sqlite table per tag | ||||
|   * performance fixed by skipping some indexes (`+mt.k`) | ||||
| * audio fingerprinting | ||||
|   * only makes sense if there can be a wasm client and that doesn't exist yet (except for olaf which is agpl hence counts as not existing) | ||||
| * `os.copy_file_range` for up2k cloning | ||||
|   * almost never hit this path anyways | ||||
| * up2k partials ui | ||||
|   * feels like there isn't much point | ||||
| * cache sha512 chunks on client | ||||
| * symlink existing files on upload | ||||
|   * too dangerous | ||||
| * comment field | ||||
| * figure out the deal with pixel3a not being connectable as hotspot | ||||
|   * pixel3a having unpredictable 3sec latency in general :|||| | ||||
|   * nah | ||||
| * look into android thumbnail cache file format | ||||
|   * absolutely not | ||||
| * indexedDB for hashes, cfg enable/clear/sz, 2gb avail, ~9k for 1g, ~4k for 100m, 500k items before autoeviction | ||||
|   * blank hashlist when up-ok to skip handshake | ||||
|     * too many confusing side-effects | ||||
|   | ||||
							
								
								
									
										63
									
								
								bin/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								bin/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| # [`copyparty-fuse.py`](copyparty-fuse.py) | ||||
| * mount a copyparty server as a local filesystem (read-only) | ||||
| * **supports Windows!** -- expect `194 MiB/s` sequential read | ||||
| * **supports Linux** -- expect `117 MiB/s` sequential read | ||||
| * **supports macos** -- expect `85 MiB/s` sequential read | ||||
|  | ||||
| filecache is default-on for windows and macos; | ||||
| * macos readsize is 64kB, so speed ~32 MiB/s without the cache | ||||
| * windows readsize varies by software; explorer=1M, pv=32k | ||||
|  | ||||
| note that copyparty should run with `-ed` to enable dotfiles (hidden otherwise) | ||||
|  | ||||
| also consider using [../docs/rclone.md](../docs/rclone.md) instead for 5x performance | ||||
|  | ||||
|  | ||||
| ## to run this on windows: | ||||
| * install [winfsp](https://github.com/billziss-gh/winfsp/releases/latest) and [python 3](https://www.python.org/downloads/) | ||||
|   * [x] add python 3.x to PATH (it asks during install) | ||||
| * `python -m pip install --user fusepy` | ||||
| * `python ./copyparty-fuse.py n: http://192.168.1.69:3923/` | ||||
|  | ||||
| 10% faster in [msys2](https://www.msys2.org/), 700% faster if debug prints are enabled: | ||||
| * `pacman -S mingw64/mingw-w64-x86_64-python{,-pip}` | ||||
| * `/mingw64/bin/python3 -m pip install --user fusepy` | ||||
| * `/mingw64/bin/python3 ./copyparty-fuse.py [...]` | ||||
|  | ||||
| you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releases/latest), let me know if you [figure out how](https://github.com/dokan-dev/dokany/wiki/FUSE)   | ||||
| (winfsp's sshfs leaks, doesn't look like winfsp itself does, should be fine) | ||||
|  | ||||
|  | ||||
|  | ||||
| # [`copyparty-fuse🅱️.py`](copyparty-fuseb.py) | ||||
| * mount a copyparty server as a local filesystem (read-only) | ||||
| * does the same thing except more correct, `samba` approves | ||||
| * **supports Linux** -- expect `18 MiB/s` (wait what) | ||||
| * **supports Macos** -- probably | ||||
|  | ||||
|  | ||||
|  | ||||
| # [`copyparty-fuse-streaming.py`](copyparty-fuse-streaming.py) | ||||
| * pretend this doesn't exist | ||||
|  | ||||
|  | ||||
|  | ||||
| # [`mtag/`](mtag/) | ||||
| * standalone programs which perform misc. file analysis | ||||
| * copyparty can Popen programs like these during file indexing to collect additional metadata | ||||
|  | ||||
|  | ||||
| # [`dbtool.py`](dbtool.py) | ||||
| upgrade utility which can show db info and help transfer data between databases, for example when a new version of copyparty is incompatible with the old DB and automatically rebuilds the DB from scratch, but you have some really expensive `-mtp` parsers and want to copy over the tags from the old db | ||||
|  | ||||
| for that example (upgrading to v0.11.20), first launch the new version of copyparty like usual, let it make a backup of the old db and rebuild the new db until the point where it starts running mtp (colored messages as it adds the mtp tags), that's when you hit CTRL-C and patch in the old mtp tags from the old db instead | ||||
|  | ||||
| so assuming you have `-mtp` parsers to provide the tags `key` and `.bpm`: | ||||
|  | ||||
| ``` | ||||
| cd /mnt/nas/music/.hist | ||||
| ~/src/copyparty/bin/dbtool.py -ls up2k.db | ||||
| ~/src/copyparty/bin/dbtool.py -src up2k.*.v3 up2k.db -cmp | ||||
| ~/src/copyparty/bin/dbtool.py -src up2k.*.v3 up2k.db -rm-mtp-flag -copy key | ||||
| ~/src/copyparty/bin/dbtool.py -src up2k.*.v3 up2k.db -rm-mtp-flag -copy .bpm -vac | ||||
| ``` | ||||
							
								
								
									
										1100
									
								
								bin/copyparty-fuse-streaming.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1100
									
								
								bin/copyparty-fuse-streaming.py
									
									
									
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										592
									
								
								bin/copyparty-fuseb.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										592
									
								
								bin/copyparty-fuseb.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,592 @@ | ||||
| #!/usr/bin/env python3 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| """copyparty-fuseb: remote copyparty as a local filesystem""" | ||||
| __author__ = "ed <copyparty@ocv.me>" | ||||
| __copyright__ = 2020 | ||||
| __license__ = "MIT" | ||||
| __url__ = "https://github.com/9001/copyparty/" | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import stat | ||||
| import errno | ||||
| import struct | ||||
| import threading | ||||
| import http.client  # py2: httplib | ||||
| import urllib.parse | ||||
| from datetime import datetime | ||||
| from urllib.parse import quote_from_bytes as quote | ||||
|  | ||||
| try: | ||||
|     import fuse | ||||
|     from fuse import Fuse | ||||
|  | ||||
|     fuse.fuse_python_api = (0, 2) | ||||
|     if not hasattr(fuse, "__version__"): | ||||
|         raise Exception("your fuse-python is way old") | ||||
| except: | ||||
|     print( | ||||
|         "\n  could not import fuse; these may help:\n    python3 -m pip install --user fuse-python\n    apt install libfuse\n    modprobe fuse\n" | ||||
|     ) | ||||
|     raise | ||||
|  | ||||
|  | ||||
| """ | ||||
| mount a copyparty server (local or remote) as a filesystem | ||||
|  | ||||
| usage: | ||||
|   python ./copyparty-fuseb.py -f -o allow_other,auto_unmount,nonempty,url=http://192.168.1.69:3923 /mnt/nas | ||||
|  | ||||
| dependencies: | ||||
|   sudo apk add fuse-dev python3-dev | ||||
|   python3 -m pip install --user fuse-python | ||||
|  | ||||
| fork of copyparty-fuse.py based on fuse-python which | ||||
|   appears to be more compliant than fusepy? since this works with samba | ||||
|     (probably just my garbage code tbh) | ||||
| """ | ||||
|  | ||||
|  | ||||
| def threadless_log(msg): | ||||
|     print(msg + "\n", end="") | ||||
|  | ||||
|  | ||||
| def boring_log(msg): | ||||
|     msg = "\033[36m{:012x}\033[0m {}\n".format(threading.current_thread().ident, msg) | ||||
|     print(msg[4:], end="") | ||||
|  | ||||
|  | ||||
| def rice_tid(): | ||||
|     tid = threading.current_thread().ident | ||||
|     c = struct.unpack(b"B" * 5, struct.pack(b">Q", tid)[-5:]) | ||||
|     return "".join("\033[1;37;48;5;{}m{:02x}".format(x, x) for x in c) + "\033[0m" | ||||
|  | ||||
|  | ||||
| def fancy_log(msg): | ||||
|     print("{} {}\n".format(rice_tid(), msg), end="") | ||||
|  | ||||
|  | ||||
| def null_log(msg): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| info = fancy_log | ||||
| log = fancy_log | ||||
| dbg = fancy_log | ||||
| log = null_log | ||||
| dbg = null_log | ||||
|  | ||||
|  | ||||
| def get_tid(): | ||||
|     return threading.current_thread().ident | ||||
|  | ||||
|  | ||||
| def html_dec(txt): | ||||
|     return ( | ||||
|         txt.replace("<", "<") | ||||
|         .replace(">", ">") | ||||
|         .replace(""", '"') | ||||
|         .replace("&", "&") | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class CacheNode(object): | ||||
|     def __init__(self, tag, data): | ||||
|         self.tag = tag | ||||
|         self.data = data | ||||
|         self.ts = time.time() | ||||
|  | ||||
|  | ||||
| class Stat(fuse.Stat): | ||||
|     def __init__(self): | ||||
|         self.st_mode = 0 | ||||
|         self.st_ino = 0 | ||||
|         self.st_dev = 0 | ||||
|         self.st_nlink = 1 | ||||
|         self.st_uid = 1000 | ||||
|         self.st_gid = 1000 | ||||
|         self.st_size = 0 | ||||
|         self.st_atime = 0 | ||||
|         self.st_mtime = 0 | ||||
|         self.st_ctime = 0 | ||||
|  | ||||
|  | ||||
| class Gateway(object): | ||||
|     def __init__(self, base_url): | ||||
|         self.base_url = base_url | ||||
|  | ||||
|         ui = urllib.parse.urlparse(base_url) | ||||
|         self.web_root = ui.path.strip("/") | ||||
|         try: | ||||
|             self.web_host, self.web_port = ui.netloc.split(":") | ||||
|             self.web_port = int(self.web_port) | ||||
|         except: | ||||
|             self.web_host = ui.netloc | ||||
|             if ui.scheme == "http": | ||||
|                 self.web_port = 80 | ||||
|             elif ui.scheme == "https": | ||||
|                 raise Exception("todo") | ||||
|             else: | ||||
|                 raise Exception("bad url?") | ||||
|  | ||||
|         self.conns = {} | ||||
|  | ||||
|     def quotep(self, path): | ||||
|         # TODO: mojibake support | ||||
|         path = path.encode("utf-8", "ignore") | ||||
|         return quote(path, safe="/") | ||||
|  | ||||
|     def getconn(self, tid=None): | ||||
|         tid = tid or get_tid() | ||||
|         try: | ||||
|             return self.conns[tid] | ||||
|         except: | ||||
|             info("new conn [{}] [{}]".format(self.web_host, self.web_port)) | ||||
|  | ||||
|             conn = http.client.HTTPConnection(self.web_host, self.web_port, timeout=260) | ||||
|  | ||||
|             self.conns[tid] = conn | ||||
|             return conn | ||||
|  | ||||
|     def closeconn(self, tid=None): | ||||
|         tid = tid or get_tid() | ||||
|         try: | ||||
|             self.conns[tid].close() | ||||
|             del self.conns[tid] | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def sendreq(self, *args, **kwargs): | ||||
|         tid = get_tid() | ||||
|         try: | ||||
|             c = self.getconn(tid) | ||||
|             c.request(*list(args), **kwargs) | ||||
|             return c.getresponse() | ||||
|         except: | ||||
|             self.closeconn(tid) | ||||
|             c = self.getconn(tid) | ||||
|             c.request(*list(args), **kwargs) | ||||
|             return c.getresponse() | ||||
|  | ||||
|     def listdir(self, path): | ||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots" | ||||
|         r = self.sendreq("GET", web_path) | ||||
|         if r.status != 200: | ||||
|             self.closeconn() | ||||
|             raise Exception( | ||||
|                 "http error {} reading dir {} in {}".format( | ||||
|                     r.status, web_path, rice_tid() | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|         return self.parse_html(r) | ||||
|  | ||||
|     def download_file_range(self, path, ofs1, ofs2): | ||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?raw" | ||||
|         hdr_range = "bytes={}-{}".format(ofs1, ofs2 - 1) | ||||
|         log("downloading {}".format(hdr_range)) | ||||
|  | ||||
|         r = self.sendreq("GET", web_path, headers={"Range": hdr_range}) | ||||
|         if r.status != http.client.PARTIAL_CONTENT: | ||||
|             self.closeconn() | ||||
|             raise Exception( | ||||
|                 "http error {} reading file {} range {} in {}".format( | ||||
|                     r.status, web_path, hdr_range, rice_tid() | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|         return r.read() | ||||
|  | ||||
|     def parse_html(self, datasrc): | ||||
|         ret = [] | ||||
|         remainder = b"" | ||||
|         ptn = re.compile( | ||||
|             r"^<tr><td>(-|DIR)</td><td><a [^>]+>([^<]+)</a></td><td>([^<]+)</td><td>([^<]+)</td></tr>$" | ||||
|         ) | ||||
|  | ||||
|         while True: | ||||
|             buf = remainder + datasrc.read(4096) | ||||
|             # print('[{}]'.format(buf.decode('utf-8'))) | ||||
|             if not buf: | ||||
|                 break | ||||
|  | ||||
|             remainder = b"" | ||||
|             endpos = buf.rfind(b"\n") | ||||
|             if endpos >= 0: | ||||
|                 remainder = buf[endpos + 1 :] | ||||
|                 buf = buf[:endpos] | ||||
|  | ||||
|             lines = buf.decode("utf-8").split("\n") | ||||
|             for line in lines: | ||||
|                 m = ptn.match(line) | ||||
|                 if not m: | ||||
|                     # print(line) | ||||
|                     continue | ||||
|  | ||||
|                 ftype, fname, fsize, fdate = m.groups() | ||||
|                 fname = html_dec(fname) | ||||
|                 ts = datetime.strptime(fdate, "%Y-%m-%d %H:%M:%S").timestamp() | ||||
|                 sz = int(fsize) | ||||
|                 if ftype == "-": | ||||
|                     ret.append([fname, self.stat_file(ts, sz), 0]) | ||||
|                 else: | ||||
|                     ret.append([fname, self.stat_dir(ts, sz), 0]) | ||||
|  | ||||
|         return ret | ||||
|  | ||||
|     def stat_dir(self, ts, sz=4096): | ||||
|         ret = Stat() | ||||
|         ret.st_mode = stat.S_IFDIR | 0o555 | ||||
|         ret.st_nlink = 2 | ||||
|         ret.st_size = sz | ||||
|         ret.st_atime = ts | ||||
|         ret.st_mtime = ts | ||||
|         ret.st_ctime = ts | ||||
|         return ret | ||||
|  | ||||
|     def stat_file(self, ts, sz): | ||||
|         ret = Stat() | ||||
|         ret.st_mode = stat.S_IFREG | 0o444 | ||||
|         ret.st_size = sz | ||||
|         ret.st_atime = ts | ||||
|         ret.st_mtime = ts | ||||
|         ret.st_ctime = ts | ||||
|         return ret | ||||
|  | ||||
|  | ||||
| class CPPF(Fuse): | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         Fuse.__init__(self, *args, **kwargs) | ||||
|  | ||||
|         self.url = None | ||||
|  | ||||
|         self.dircache = [] | ||||
|         self.dircache_mtx = threading.Lock() | ||||
|  | ||||
|         self.filecache = [] | ||||
|         self.filecache_mtx = threading.Lock() | ||||
|  | ||||
|     def init2(self): | ||||
|         # TODO figure out how python-fuse wanted this to go | ||||
|         self.gw = Gateway(self.url)  # .decode('utf-8')) | ||||
|         info("up") | ||||
|  | ||||
|     def clean_dircache(self): | ||||
|         """not threadsafe""" | ||||
|         now = time.time() | ||||
|         cutoff = 0 | ||||
|         for cn in self.dircache: | ||||
|             if now - cn.ts > 1: | ||||
|                 cutoff += 1 | ||||
|             else: | ||||
|                 break | ||||
|  | ||||
|         if cutoff > 0: | ||||
|             self.dircache = self.dircache[cutoff:] | ||||
|  | ||||
|     def get_cached_dir(self, dirpath): | ||||
|         # with self.dircache_mtx: | ||||
|         if True: | ||||
|             self.clean_dircache() | ||||
|             for cn in self.dircache: | ||||
|                 if cn.tag == dirpath: | ||||
|                     return cn | ||||
|  | ||||
|         return None | ||||
|  | ||||
|     """ | ||||
|             ,-------------------------------,  g1>=c1, g2<=c2 | ||||
|             |cache1                   cache2|  buf[g1-c1:(g1-c1)+(g2-g1)] | ||||
|             `-------------------------------' | ||||
|                     ,---------------, | ||||
|                     |get1       get2| | ||||
|                     `---------------' | ||||
|     __________________________________________________________________________ | ||||
|  | ||||
|             ,-------------------------------,  g2<=c2, (g2>=c1) | ||||
|             |cache1                   cache2|  cdr=buf[:g2-c1] | ||||
|             `-------------------------------'  dl car; g1-512K:c1 | ||||
|     ,---------------, | ||||
|     |get1       get2| | ||||
|     `---------------' | ||||
|     __________________________________________________________________________ | ||||
|  | ||||
|             ,-------------------------------,  g1>=c1, (g1<=c2) | ||||
|             |cache1                   cache2|  car=buf[c2-g1:] | ||||
|             `-------------------------------'  dl cdr; c2:c2+1M | ||||
|                                     ,---------------, | ||||
|                                     |get1       get2| | ||||
|                                     `---------------' | ||||
|     """ | ||||
|  | ||||
|     def get_cached_file(self, path, get1, get2, file_sz): | ||||
|         car = None | ||||
|         cdr = None | ||||
|         ncn = -1 | ||||
|         # with self.filecache_mtx: | ||||
|         if True: | ||||
|             dbg("cache request from {} to {}, size {}".format(get1, get2, file_sz)) | ||||
|             for cn in self.filecache: | ||||
|                 ncn += 1 | ||||
|  | ||||
|                 cache_path, cache1 = cn.tag | ||||
|                 if cache_path != path: | ||||
|                     continue | ||||
|  | ||||
|                 cache2 = cache1 + len(cn.data) | ||||
|                 if get2 <= cache1 or get1 >= cache2: | ||||
|                     continue | ||||
|  | ||||
|                 if get1 >= cache1 and get2 <= cache2: | ||||
|                     # keep cache entry alive by moving it to the end | ||||
|                     self.filecache = ( | ||||
|                         self.filecache[:ncn] + self.filecache[ncn + 1 :] + [cn] | ||||
|                     ) | ||||
|                     buf_ofs = get1 - cache1 | ||||
|                     buf_end = buf_ofs + (get2 - get1) | ||||
|                     dbg( | ||||
|                         "found all ({}, {} to {}, len {}) [{}:{}] = {}".format( | ||||
|                             ncn, | ||||
|                             cache1, | ||||
|                             cache2, | ||||
|                             len(cn.data), | ||||
|                             buf_ofs, | ||||
|                             buf_end, | ||||
|                             buf_end - buf_ofs, | ||||
|                         ) | ||||
|                     ) | ||||
|                     return cn.data[buf_ofs:buf_end] | ||||
|  | ||||
|                 if get2 < cache2: | ||||
|                     x = cn.data[: get2 - cache1] | ||||
|                     if not cdr or len(cdr) < len(x): | ||||
|                         dbg( | ||||
|                             "found car ({}, {} to {}, len {}) [:{}-{}] = [:{}] = {}".format( | ||||
|                                 ncn, | ||||
|                                 cache1, | ||||
|                                 cache2, | ||||
|                                 len(cn.data), | ||||
|                                 get2, | ||||
|                                 cache1, | ||||
|                                 get2 - cache1, | ||||
|                                 len(x), | ||||
|                             ) | ||||
|                         ) | ||||
|                         cdr = x | ||||
|  | ||||
|                     continue | ||||
|  | ||||
|                 if get1 > cache1: | ||||
|                     x = cn.data[-(cache2 - get1) :] | ||||
|                     if not car or len(car) < len(x): | ||||
|                         dbg( | ||||
|                             "found cdr ({}, {} to {}, len {}) [-({}-{}):] = [-{}:] = {}".format( | ||||
|                                 ncn, | ||||
|                                 cache1, | ||||
|                                 cache2, | ||||
|                                 len(cn.data), | ||||
|                                 cache2, | ||||
|                                 get1, | ||||
|                                 cache2 - get1, | ||||
|                                 len(x), | ||||
|                             ) | ||||
|                         ) | ||||
|                         car = x | ||||
|  | ||||
|                     continue | ||||
|  | ||||
|                 raise Exception("what") | ||||
|  | ||||
|         if car and cdr: | ||||
|             dbg("<cache> have both") | ||||
|  | ||||
|             ret = car + cdr | ||||
|             if len(ret) == get2 - get1: | ||||
|                 return ret | ||||
|  | ||||
|             raise Exception("{} + {} != {} - {}".format(len(car), len(cdr), get2, get1)) | ||||
|  | ||||
|         elif cdr: | ||||
|             h_end = get1 + (get2 - get1) - len(cdr) | ||||
|             h_ofs = h_end - 512 * 1024 | ||||
|  | ||||
|             if h_ofs < 0: | ||||
|                 h_ofs = 0 | ||||
|  | ||||
|             buf_ofs = (get2 - get1) - len(cdr) | ||||
|  | ||||
|             dbg( | ||||
|                 "<cache> cdr {}, car {}-{}={} [-{}:]".format( | ||||
|                     len(cdr), h_ofs, h_end, h_end - h_ofs, buf_ofs | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|             buf = self.gw.download_file_range(path, h_ofs, h_end) | ||||
|             ret = buf[-buf_ofs:] + cdr | ||||
|  | ||||
|         elif car: | ||||
|             h_ofs = get1 + len(car) | ||||
|             h_end = h_ofs + 1024 * 1024 | ||||
|  | ||||
|             if h_end > file_sz: | ||||
|                 h_end = file_sz | ||||
|  | ||||
|             buf_ofs = (get2 - get1) - len(car) | ||||
|  | ||||
|             dbg( | ||||
|                 "<cache> car {}, cdr {}-{}={} [:{}]".format( | ||||
|                     len(car), h_ofs, h_end, h_end - h_ofs, buf_ofs | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|             buf = self.gw.download_file_range(path, h_ofs, h_end) | ||||
|             ret = car + buf[:buf_ofs] | ||||
|  | ||||
|         else: | ||||
|             h_ofs = get1 - 256 * 1024 | ||||
|             h_end = get2 + 1024 * 1024 | ||||
|  | ||||
|             if h_ofs < 0: | ||||
|                 h_ofs = 0 | ||||
|  | ||||
|             if h_end > file_sz: | ||||
|                 h_end = file_sz | ||||
|  | ||||
|             buf_ofs = get1 - h_ofs | ||||
|             buf_end = buf_ofs + get2 - get1 | ||||
|  | ||||
|             dbg( | ||||
|                 "<cache> {}-{}={} [{}:{}]".format( | ||||
|                     h_ofs, h_end, h_end - h_ofs, buf_ofs, buf_end | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|             buf = self.gw.download_file_range(path, h_ofs, h_end) | ||||
|             ret = buf[buf_ofs:buf_end] | ||||
|  | ||||
|         cn = CacheNode([path, h_ofs], buf) | ||||
|         # with self.filecache_mtx: | ||||
|         if True: | ||||
|             if len(self.filecache) > 6: | ||||
|                 self.filecache = self.filecache[1:] + [cn] | ||||
|             else: | ||||
|                 self.filecache.append(cn) | ||||
|  | ||||
|         return ret | ||||
|  | ||||
|     def _readdir(self, path): | ||||
|         path = path.strip("/") | ||||
|         log("readdir {}".format(path)) | ||||
|  | ||||
|         ret = self.gw.listdir(path) | ||||
|  | ||||
|         # with self.dircache_mtx: | ||||
|         if True: | ||||
|             cn = CacheNode(path, ret) | ||||
|             self.dircache.append(cn) | ||||
|             self.clean_dircache() | ||||
|  | ||||
|         return ret | ||||
|  | ||||
|     def readdir(self, path, offset): | ||||
|         for e in self._readdir(path)[offset:]: | ||||
|             # log("yield [{}]".format(e[0])) | ||||
|             yield fuse.Direntry(e[0]) | ||||
|  | ||||
|     def open(self, path, flags): | ||||
|         if (flags & (os.O_RDONLY | os.O_WRONLY | os.O_RDWR)) != os.O_RDONLY: | ||||
|             return -errno.EACCES | ||||
|  | ||||
|         st = self.getattr(path) | ||||
|         try: | ||||
|             if st.st_nlink > 0: | ||||
|                 return st | ||||
|         except: | ||||
|             return st  # -int(os.errcode) | ||||
|  | ||||
|     def read(self, path, length, offset, fh=None, *args): | ||||
|         if args: | ||||
|             log("unexpected args [" + "] [".join(repr(x) for x in args) + "]") | ||||
|             raise Exception() | ||||
|  | ||||
|         path = path.strip("/") | ||||
|  | ||||
|         ofs2 = offset + length | ||||
|         log("read {} @ {} len {} end {}".format(path, offset, length, ofs2)) | ||||
|  | ||||
|         st = self.getattr(path) | ||||
|         try: | ||||
|             file_sz = st.st_size | ||||
|         except: | ||||
|             return st  # -int(os.errcode) | ||||
|  | ||||
|         if ofs2 > file_sz: | ||||
|             ofs2 = file_sz | ||||
|             log("truncate to len {} end {}".format(ofs2 - offset, ofs2)) | ||||
|  | ||||
|         if file_sz == 0 or offset >= ofs2: | ||||
|             return b"" | ||||
|  | ||||
|         # toggle cache here i suppose | ||||
|         # return self.get_cached_file(path, offset, ofs2, file_sz) | ||||
|         return self.gw.download_file_range(path, offset, ofs2) | ||||
|  | ||||
|     def getattr(self, path): | ||||
|         log("getattr [{}]".format(path)) | ||||
|  | ||||
|         path = path.strip("/") | ||||
|         try: | ||||
|             dirpath, fname = path.rsplit("/", 1) | ||||
|         except: | ||||
|             dirpath = "" | ||||
|             fname = path | ||||
|  | ||||
|         if not path: | ||||
|             ret = self.gw.stat_dir(time.time()) | ||||
|             dbg("=root") | ||||
|             return ret | ||||
|  | ||||
|         cn = self.get_cached_dir(dirpath) | ||||
|         if cn: | ||||
|             log("cache ok") | ||||
|             dents = cn.data | ||||
|         else: | ||||
|             log("cache miss") | ||||
|             dents = self._readdir(dirpath) | ||||
|  | ||||
|         for cache_name, cache_stat, _ in dents: | ||||
|             if cache_name == fname: | ||||
|                 dbg("=file") | ||||
|                 return cache_stat | ||||
|  | ||||
|         log("=404") | ||||
|         return -errno.ENOENT | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||
|  | ||||
|     server = CPPF() | ||||
|     server.parser.add_option(mountopt="url", metavar="BASE_URL", default=None) | ||||
|     server.parse(values=server, errex=1) | ||||
|     if not server.url or not str(server.url).startswith("http"): | ||||
|         print("\nerror:") | ||||
|         print("  need argument: -o url=<...>") | ||||
|         print("  need argument: mount-path") | ||||
|         print("example:") | ||||
|         print( | ||||
|             "  ./copyparty-fuseb.py -f -o allow_other,auto_unmount,nonempty,url=http://192.168.1.69:3923 /mnt/nas" | ||||
|         ) | ||||
|         sys.exit(1) | ||||
|  | ||||
|     server.init2() | ||||
|     threading.Thread(target=server.main, daemon=True).start() | ||||
|     while True: | ||||
|         time.sleep(9001) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										245
									
								
								bin/dbtool.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										245
									
								
								bin/dbtool.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,245 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import shutil | ||||
| import sqlite3 | ||||
| import argparse | ||||
|  | ||||
| DB_VER1 = 3 | ||||
| DB_VER2 = 4 | ||||
|  | ||||
|  | ||||
| def die(msg): | ||||
|     print("\033[31m\n" + msg + "\n\033[0m") | ||||
|     sys.exit(1) | ||||
|  | ||||
|  | ||||
| def read_ver(db): | ||||
|     for tab in ["ki", "kv"]: | ||||
|         try: | ||||
|             c = db.execute(r"select v from {} where k = 'sver'".format(tab)) | ||||
|         except: | ||||
|             continue | ||||
|  | ||||
|         rows = c.fetchall() | ||||
|         if rows: | ||||
|             return int(rows[0][0]) | ||||
|  | ||||
|     return "corrupt" | ||||
|  | ||||
|  | ||||
| def ls(db): | ||||
|     nfiles = next(db.execute("select count(w) from up"))[0] | ||||
|     ntags = next(db.execute("select count(w) from mt"))[0] | ||||
|     print(f"{nfiles} files") | ||||
|     print(f"{ntags} tags\n") | ||||
|  | ||||
|     print("number of occurences for each tag,") | ||||
|     print(" 'x' = file has no tags") | ||||
|     print(" 't:mtp' = the mtp flag (file not mtp processed yet)") | ||||
|     print() | ||||
|     for k, nk in db.execute("select k, count(k) from mt group by k order by k"): | ||||
|         print(f"{nk:9} {k}") | ||||
|  | ||||
|  | ||||
| def compare(n1, d1, n2, d2, verbose): | ||||
|     nt = next(d1.execute("select count(w) from up"))[0] | ||||
|     n = 0 | ||||
|     miss = 0 | ||||
|     for w1, rd, fn in d1.execute("select w, rd, fn from up"): | ||||
|         n += 1 | ||||
|         if n % 25_000 == 0: | ||||
|             m = f"\033[36mchecked {n:,} of {nt:,} files in {n1} against {n2}\033[0m" | ||||
|             print(m) | ||||
|  | ||||
|         if rd.split("/", 1)[0] == ".hist": | ||||
|             continue | ||||
|  | ||||
|         q = "select w from up where rd = ? and fn = ?" | ||||
|         hit = d2.execute(q, (rd, fn)).fetchone() | ||||
|         if not hit: | ||||
|             miss += 1 | ||||
|             if verbose: | ||||
|                 print(f"file in {n1} missing in {n2}: [{w1}] {rd}/{fn}") | ||||
|  | ||||
|     print(f" {miss} files in {n1} missing in {n2}\n") | ||||
|  | ||||
|     nt = next(d1.execute("select count(w) from mt"))[0] | ||||
|     n = 0 | ||||
|     miss = {} | ||||
|     nmiss = 0 | ||||
|     for w1, k, v in d1.execute("select * from mt"): | ||||
|  | ||||
|         n += 1 | ||||
|         if n % 100_000 == 0: | ||||
|             m = f"\033[36mchecked {n:,} of {nt:,} tags in {n1} against {n2}, so far {nmiss} missing tags\033[0m" | ||||
|             print(m) | ||||
|  | ||||
|         q = "select rd, fn from up where substr(w,1,16) = ?" | ||||
|         rd, fn = d1.execute(q, (w1,)).fetchone() | ||||
|         if rd.split("/", 1)[0] == ".hist": | ||||
|             continue | ||||
|  | ||||
|         q = "select substr(w,1,16) from up where rd = ? and fn = ?" | ||||
|         w2 = d2.execute(q, (rd, fn)).fetchone() | ||||
|         if w2: | ||||
|             w2 = w2[0] | ||||
|  | ||||
|         v2 = None | ||||
|         if w2: | ||||
|             v2 = d2.execute( | ||||
|                 "select v from mt where w = ? and +k = ?", (w2, k) | ||||
|             ).fetchone() | ||||
|             if v2: | ||||
|                 v2 = v2[0] | ||||
|  | ||||
|         # if v != v2 and v2 and k in [".bpm", "key"] and n2 == "src": | ||||
|         #    print(f"{w} [{rd}/{fn}] {k} = [{v}] / [{v2}]") | ||||
|  | ||||
|         if v2 is not None: | ||||
|             if k.startswith("."): | ||||
|                 try: | ||||
|                     diff = abs(float(v) - float(v2)) | ||||
|                     if diff > float(v) / 0.9: | ||||
|                         v2 = None | ||||
|                     else: | ||||
|                         v2 = v | ||||
|                 except: | ||||
|                     pass | ||||
|  | ||||
|             if v != v2: | ||||
|                 v2 = None | ||||
|  | ||||
|         if v2 is None: | ||||
|             nmiss += 1 | ||||
|             try: | ||||
|                 miss[k] += 1 | ||||
|             except: | ||||
|                 miss[k] = 1 | ||||
|  | ||||
|             if verbose: | ||||
|                 print(f"missing in {n2}: [{w1}] [{rd}/{fn}] {k} = {v}") | ||||
|  | ||||
|     for k, v in sorted(miss.items()): | ||||
|         if v: | ||||
|             print(f"{n1} has {v:6} more {k:<6} tags than {n2}") | ||||
|  | ||||
|     print(f"in total, {nmiss} missing tags in {n2}\n") | ||||
|  | ||||
|  | ||||
| def copy_mtp(d1, d2, tag, rm): | ||||
|     nt = next(d1.execute("select count(w) from mt where k = ?", (tag,)))[0] | ||||
|     n = 0 | ||||
|     ndone = 0 | ||||
|     for w1, k, v in d1.execute("select * from mt where k = ?", (tag,)): | ||||
|         n += 1 | ||||
|         if n % 25_000 == 0: | ||||
|             m = f"\033[36m{n:,} of {nt:,} tags checked, so far {ndone} copied\033[0m" | ||||
|             print(m) | ||||
|  | ||||
|         q = "select rd, fn from up where substr(w,1,16) = ?" | ||||
|         rd, fn = d1.execute(q, (w1,)).fetchone() | ||||
|         if rd.split("/", 1)[0] == ".hist": | ||||
|             continue | ||||
|  | ||||
|         q = "select substr(w,1,16) from up where rd = ? and fn = ?" | ||||
|         w2 = d2.execute(q, (rd, fn)).fetchone() | ||||
|         if not w2: | ||||
|             continue | ||||
|  | ||||
|         w2 = w2[0] | ||||
|         hit = d2.execute("select v from mt where w = ? and +k = ?", (w2, k)).fetchone() | ||||
|         if hit: | ||||
|             hit = hit[0] | ||||
|  | ||||
|         if hit != v: | ||||
|             ndone += 1 | ||||
|             if hit is not None: | ||||
|                 d2.execute("delete from mt where w = ? and +k = ?", (w2, k)) | ||||
|  | ||||
|             d2.execute("insert into mt values (?,?,?)", (w2, k, v)) | ||||
|             if rm: | ||||
|                 d2.execute("delete from mt where w = ? and +k = 't:mtp'", (w2,)) | ||||
|  | ||||
|     d2.commit() | ||||
|     print(f"copied {ndone} {tag} tags over") | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     os.system("") | ||||
|     print() | ||||
|  | ||||
|     ap = argparse.ArgumentParser() | ||||
|     ap.add_argument("db", help="database to work on") | ||||
|     ap.add_argument("-src", metavar="DB", type=str, help="database to copy from") | ||||
|  | ||||
|     ap2 = ap.add_argument_group("informational / read-only stuff") | ||||
|     ap2.add_argument("-v", action="store_true", help="verbose") | ||||
|     ap2.add_argument("-ls", action="store_true", help="list summary for db") | ||||
|     ap2.add_argument("-cmp", action="store_true", help="compare databases") | ||||
|  | ||||
|     ap2 = ap.add_argument_group("options which modify target db") | ||||
|     ap2.add_argument("-copy", metavar="TAG", type=str, help="mtp tag to copy over") | ||||
|     ap2.add_argument( | ||||
|         "-rm-mtp-flag", | ||||
|         action="store_true", | ||||
|         help="when an mtp tag is copied over, also mark that as done, so copyparty won't run mtp on it", | ||||
|     ) | ||||
|     ap2.add_argument("-vac", action="store_true", help="optimize DB") | ||||
|  | ||||
|     ar = ap.parse_args() | ||||
|  | ||||
|     for v in [ar.db, ar.src]: | ||||
|         if v and not os.path.exists(v): | ||||
|             die("database must exist") | ||||
|  | ||||
|     db = sqlite3.connect(ar.db) | ||||
|     ds = sqlite3.connect(ar.src) if ar.src else None | ||||
|  | ||||
|     # revert journals | ||||
|     for d, p in [[db, ar.db], [ds, ar.src]]: | ||||
|         if not d: | ||||
|             continue | ||||
|  | ||||
|         pj = "{}-journal".format(p) | ||||
|         if not os.path.exists(pj): | ||||
|             continue | ||||
|  | ||||
|         d.execute("create table foo (bar int)") | ||||
|         d.execute("drop table foo") | ||||
|  | ||||
|     if ar.copy: | ||||
|         db.close() | ||||
|         shutil.copy2(ar.db, "{}.bak.dbtool.{:x}".format(ar.db, int(time.time()))) | ||||
|         db = sqlite3.connect(ar.db) | ||||
|  | ||||
|     for d, n in [[ds, "src"], [db, "dst"]]: | ||||
|         if not d: | ||||
|             continue | ||||
|  | ||||
|         ver = read_ver(d) | ||||
|         if ver == "corrupt": | ||||
|             die("{} database appears to be corrupt, sorry") | ||||
|  | ||||
|         if ver < DB_VER1 or ver > DB_VER2: | ||||
|             m = f"{n} db is version {ver}, this tool only supports versions between {DB_VER1} and {DB_VER2}, please upgrade it with copyparty first" | ||||
|             die(m) | ||||
|  | ||||
|     if ar.ls: | ||||
|         ls(db) | ||||
|  | ||||
|     if ar.cmp: | ||||
|         if not ds: | ||||
|             die("need src db to compare against") | ||||
|  | ||||
|         compare("src", ds, "dst", db, ar.v) | ||||
|         compare("dst", db, "src", ds, ar.v) | ||||
|  | ||||
|     if ar.copy: | ||||
|         copy_mtp(ds, db, ar.copy, ar.rm_mtp_flag) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										34
									
								
								bin/mtag/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								bin/mtag/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| standalone programs which take an audio file as argument | ||||
|  | ||||
| some of these rely on libraries which are not MIT-compatible | ||||
|  | ||||
| * [audio-bpm.py](./audio-bpm.py) detects the BPM of music using the BeatRoot Vamp Plugin; imports GPL2 | ||||
| * [audio-key.py](./audio-key.py) detects the melodic key of music using the Mixxx fork of keyfinder; imports GPL3 | ||||
|  | ||||
|  | ||||
| # 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 -mtp key=f,audio-key.py -mtp .bpm=f,audio-bpm.py` | ||||
|  | ||||
| * `f,` makes the detected value replace any existing values | ||||
| * the `.` in `.bpm` indicates numeric value | ||||
| * assumes the python files are in the folder you're launching copyparty from, replace the filename with a relative/absolute path if that's not the case | ||||
| * `mtp` modules will not run if a file has existing tags in the db, so clear out the tags with `-e2tsr` the first time you launch with new `mtp` options | ||||
|  | ||||
|  | ||||
| ## usage with volume-flags | ||||
|  | ||||
| instead of affecting all volumes, you can set the options for just one volume like so: | ||||
| ``` | ||||
| copyparty -v /mnt/nas/music:/music:r:cmtp=key=f,audio-key.py:cmtp=.bpm=f,audio-bpm.py:ce2dsa:ce2ts | ||||
| ``` | ||||
							
								
								
									
										69
									
								
								bin/mtag/audio-bpm.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										69
									
								
								bin/mtag/audio-bpm.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import vamp | ||||
| import tempfile | ||||
| import numpy as np | ||||
| import subprocess as sp | ||||
|  | ||||
| from copyparty.util import fsenc | ||||
|  | ||||
| """ | ||||
| dep: vamp | ||||
| dep: beatroot-vamp | ||||
| dep: ffmpeg | ||||
| """ | ||||
|  | ||||
|  | ||||
| def det(tf): | ||||
|     # fmt: off | ||||
|     sp.check_call([ | ||||
|         "ffmpeg", | ||||
|         "-nostdin", | ||||
|         "-hide_banner", | ||||
|         "-v", "fatal", | ||||
|         "-ss", "13", | ||||
|         "-y", "-i", fsenc(sys.argv[1]), | ||||
|         "-ac", "1", | ||||
|         "-ar", "22050", | ||||
|         "-t", "300", | ||||
|         "-f", "f32le", | ||||
|         tf | ||||
|     ]) | ||||
|     # fmt: on | ||||
|  | ||||
|     with open(tf, "rb") as f: | ||||
|         d = np.fromfile(f, dtype=np.float32) | ||||
|         try: | ||||
|             # 98% accuracy on jcore | ||||
|             c = vamp.collect(d, 22050, "beatroot-vamp:beatroot") | ||||
|             cl = c["list"] | ||||
|         except: | ||||
|             # fallback; 73% accuracy | ||||
|             plug = "vamp-example-plugins:fixedtempo" | ||||
|             c = vamp.collect(d, 22050, plug, parameters={"maxdflen": 40}) | ||||
|             print(c["list"][0]["label"].split(" ")[0]) | ||||
|             return | ||||
|  | ||||
|         # throws if detection failed: | ||||
|         bpm = float(cl[-1]["timestamp"] - cl[1]["timestamp"]) | ||||
|         bpm = round(60 * ((len(cl) - 1) / bpm), 2) | ||||
|         print(f"{bpm:.2f}") | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     with tempfile.NamedTemporaryFile(suffix=".pcm", delete=False) as f: | ||||
|         f.write(b"h") | ||||
|         tf = f.name | ||||
|  | ||||
|     try: | ||||
|         det(tf) | ||||
|     except: | ||||
|         pass  # mute | ||||
|     finally: | ||||
|         os.unlink(tf) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										123
									
								
								bin/mtag/audio-key-slicing.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										123
									
								
								bin/mtag/audio-key-slicing.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,123 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import tempfile | ||||
| import subprocess as sp | ||||
|  | ||||
| import keyfinder | ||||
|  | ||||
| from copyparty.util import fsenc | ||||
|  | ||||
| """ | ||||
| dep: github/mixxxdj/libkeyfinder | ||||
| dep: pypi/keyfinder | ||||
| dep: ffmpeg | ||||
|  | ||||
| note: this is a janky edition of the regular audio-key.py, | ||||
|   slicing the files at 20sec intervals and keeping 5sec from each, | ||||
|   surprisingly accurate but still garbage (446 ok, 69 bad, 13% miss) | ||||
|  | ||||
|   it is fast tho | ||||
| """ | ||||
|  | ||||
|  | ||||
| def get_duration(): | ||||
|     # TODO provide ffprobe tags to mtp as json | ||||
|  | ||||
|     # fmt: off | ||||
|     dur = sp.check_output([ | ||||
|         "ffprobe", | ||||
|         "-hide_banner", | ||||
|         "-v", "fatal", | ||||
|         "-show_streams", | ||||
|         "-show_format", | ||||
|         fsenc(sys.argv[1]) | ||||
|     ]) | ||||
|     # fmt: on | ||||
|  | ||||
|     dur = dur.decode("ascii", "replace").split("\n") | ||||
|     dur = [x.split("=")[1] for x in dur if x.startswith("duration=")] | ||||
|     dur = [float(x) for x in dur if re.match(r"^[0-9\.,]+$", x)] | ||||
|     return list(sorted(dur))[-1] if dur else None | ||||
|  | ||||
|  | ||||
| def get_segs(dur): | ||||
|     # keep first 5s of each 20s, | ||||
|     # keep entire last segment | ||||
|     ofs = 0 | ||||
|     segs = [] | ||||
|     while True: | ||||
|         seg = [ofs, 5] | ||||
|         segs.append(seg) | ||||
|         if dur - ofs < 20: | ||||
|             seg[-1] = int(dur - seg[0]) | ||||
|             break | ||||
|  | ||||
|         ofs += 20 | ||||
|  | ||||
|     return segs | ||||
|  | ||||
|  | ||||
| def slice(tf): | ||||
|     dur = get_duration() | ||||
|     dur = min(dur, 600)  # max 10min | ||||
|     segs = get_segs(dur) | ||||
|  | ||||
|     # fmt: off | ||||
|     cmd = [ | ||||
|         "ffmpeg", | ||||
|         "-nostdin", | ||||
|         "-hide_banner", | ||||
|         "-v", "fatal", | ||||
|         "-y" | ||||
|     ] | ||||
|  | ||||
|     for seg in segs: | ||||
|         cmd.extend([ | ||||
|             "-ss", str(seg[0]), | ||||
|             "-i", fsenc(sys.argv[1]) | ||||
|         ]) | ||||
|      | ||||
|     filt = "" | ||||
|     for n, seg in enumerate(segs): | ||||
|         filt += "[{}:a:0]atrim=duration={}[a{}]; ".format(n, seg[1], n) | ||||
|      | ||||
|     prev = "a0" | ||||
|     for n in range(1, len(segs)): | ||||
|         nxt = "b{}".format(n) | ||||
|         filt += "[{}][a{}]acrossfade=d=0.5[{}]; ".format(prev, n, nxt) | ||||
|         prev = nxt | ||||
|  | ||||
|     cmd.extend([ | ||||
|         "-filter_complex", filt[:-2], | ||||
|         "-map", "[{}]".format(nxt), | ||||
|         "-sample_fmt", "s16", | ||||
|         tf | ||||
|     ]) | ||||
|     # fmt: on | ||||
|  | ||||
|     # print(cmd) | ||||
|     sp.check_call(cmd) | ||||
|  | ||||
|  | ||||
| def det(tf): | ||||
|     slice(tf) | ||||
|     print(keyfinder.key(tf).camelot()) | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     with tempfile.NamedTemporaryFile(suffix=".flac", delete=False) as f: | ||||
|         f.write(b"h") | ||||
|         tf = f.name | ||||
|  | ||||
|     try: | ||||
|         det(tf) | ||||
|     finally: | ||||
|         os.unlink(tf) | ||||
|         pass | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										54
									
								
								bin/mtag/audio-key.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										54
									
								
								bin/mtag/audio-key.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import tempfile | ||||
| import subprocess as sp | ||||
| import keyfinder | ||||
|  | ||||
| from copyparty.util import fsenc | ||||
|  | ||||
| """ | ||||
| dep: github/mixxxdj/libkeyfinder | ||||
| dep: pypi/keyfinder | ||||
| dep: ffmpeg | ||||
| """ | ||||
|  | ||||
|  | ||||
| # tried trimming the first/last 5th, bad idea, | ||||
| # misdetects 9a law field (Sphere Caliber) as 10b, | ||||
| # obvious when mixing 9a ghostly parapara ship | ||||
|  | ||||
|  | ||||
| def det(tf): | ||||
|     # fmt: off | ||||
|     sp.check_call([ | ||||
|         "ffmpeg", | ||||
|         "-nostdin", | ||||
|         "-hide_banner", | ||||
|         "-v", "fatal", | ||||
|         "-y", "-i", fsenc(sys.argv[1]), | ||||
|         "-t", "300", | ||||
|         "-sample_fmt", "s16", | ||||
|         tf | ||||
|     ]) | ||||
|     # fmt: on | ||||
|  | ||||
|     print(keyfinder.key(tf).camelot()) | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     with tempfile.NamedTemporaryFile(suffix=".flac", delete=False) as f: | ||||
|         f.write(b"h") | ||||
|         tf = f.name | ||||
|  | ||||
|     try: | ||||
|         det(tf) | ||||
|     except: | ||||
|         pass  # mute | ||||
|     finally: | ||||
|         os.unlink(tf) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										96
									
								
								bin/mtag/exe.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								bin/mtag/exe.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,96 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import sys | ||||
| import time | ||||
| import json | ||||
| import pefile | ||||
|  | ||||
| """ | ||||
| retrieve exe info, | ||||
| example for multivalue providers | ||||
| """ | ||||
|  | ||||
|  | ||||
| def unk(v): | ||||
|     return "unk({:04x})".format(v) | ||||
|  | ||||
|  | ||||
| class PE2(pefile.PE): | ||||
|     def __init__(self, *a, **ka): | ||||
|         for k in [ | ||||
|             # -- parse_data_directories: | ||||
|             "parse_import_directory", | ||||
|             "parse_export_directory", | ||||
|             # "parse_resources_directory", | ||||
|             "parse_debug_directory", | ||||
|             "parse_relocations_directory", | ||||
|             "parse_directory_tls", | ||||
|             "parse_directory_load_config", | ||||
|             "parse_delay_import_directory", | ||||
|             "parse_directory_bound_imports", | ||||
|             # -- full_load: | ||||
|             "parse_rich_header", | ||||
|         ]: | ||||
|             setattr(self, k, self.noop) | ||||
|  | ||||
|         super(PE2, self).__init__(*a, **ka) | ||||
|  | ||||
|     def noop(*a, **ka): | ||||
|         pass | ||||
|  | ||||
|  | ||||
| try: | ||||
|     pe = PE2(sys.argv[1], fast_load=False) | ||||
| except: | ||||
|     sys.exit(0) | ||||
|  | ||||
| arch = pe.FILE_HEADER.Machine | ||||
| if arch == 0x14C: | ||||
|     arch = "x86" | ||||
| elif arch == 0x8664: | ||||
|     arch = "x64" | ||||
| else: | ||||
|     arch = unk(arch) | ||||
|  | ||||
| try: | ||||
|     buildtime = time.gmtime(pe.FILE_HEADER.TimeDateStamp) | ||||
|     buildtime = time.strftime("%Y-%m-%d_%H:%M:%S", buildtime) | ||||
| except: | ||||
|     buildtime = "invalid" | ||||
|  | ||||
| ui = pe.OPTIONAL_HEADER.Subsystem | ||||
| if ui == 2: | ||||
|     ui = "GUI" | ||||
| elif ui == 3: | ||||
|     ui = "cmdline" | ||||
| else: | ||||
|     ui = unk(ui) | ||||
|  | ||||
| extra = {} | ||||
| if hasattr(pe, "FileInfo"): | ||||
|     for v1 in pe.FileInfo: | ||||
|         for v2 in v1: | ||||
|             if v2.name != "StringFileInfo": | ||||
|                 continue | ||||
|  | ||||
|             for v3 in v2.StringTable: | ||||
|                 for k, v in v3.entries.items(): | ||||
|                     v = v.decode("utf-8", "replace").strip() | ||||
|                     if not v: | ||||
|                         continue | ||||
|  | ||||
|                     if k in [b"FileVersion", b"ProductVersion"]: | ||||
|                         extra["ver"] = v | ||||
|  | ||||
|                     if k in [b"OriginalFilename", b"InternalName"]: | ||||
|                         extra["orig"] = v | ||||
|  | ||||
| r = { | ||||
|     "arch": arch, | ||||
|     "built": buildtime, | ||||
|     "ui": ui, | ||||
|     "cksum": "{:08x}".format(pe.OPTIONAL_HEADER.CheckSum), | ||||
| } | ||||
| r.update(extra) | ||||
|  | ||||
| print(json.dumps(r, indent=4)) | ||||
							
								
								
									
										9
									
								
								bin/mtag/file-ext.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								bin/mtag/file-ext.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| import sys | ||||
|  | ||||
| """ | ||||
| example that just prints the file extension | ||||
| """ | ||||
|  | ||||
| print(sys.argv[1].split(".")[-1]) | ||||
							
								
								
									
										265
									
								
								bin/mtag/install-deps.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										265
									
								
								bin/mtag/install-deps.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,265 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
|  | ||||
| # install dependencies for audio-*.py | ||||
| # | ||||
| # linux: requires {python3,ffmpeg,fftw}-dev py3-{wheel,pip} py3-numpy{,-dev} vamp-sdk-dev patchelf | ||||
| # win64: requires msys2-mingw64 environment | ||||
| # macos: requires macports | ||||
| # | ||||
| # has the following manual dependencies, especially on mac: | ||||
| #   https://www.vamp-plugins.org/pack.html | ||||
| # | ||||
| # installs stuff to the following locations: | ||||
| #   ~/pe/ | ||||
| #   whatever your python uses for --user packages | ||||
| # | ||||
| # does the following terrible things: | ||||
| #   modifies the keyfinder python lib to load the .so in ~/pe | ||||
|  | ||||
|  | ||||
| linux=1 | ||||
|  | ||||
| win= | ||||
| [ ! -z "$MSYSTEM" ] || [ -e /msys2.exe ] && { | ||||
| 	[ "$MSYSTEM" = MINGW64 ] || { | ||||
| 		echo windows detected, msys2-mingw64 required | ||||
| 		exit 1 | ||||
| 	} | ||||
| 	pacman -S --needed mingw-w64-x86_64-{ffmpeg,python,python-pip,vamp-plugin-sdk} | ||||
| 	win=1 | ||||
| 	linux= | ||||
| } | ||||
|  | ||||
| mac= | ||||
| [ $(uname -s) = Darwin ] && { | ||||
| 	#pybin="$(printf '%s\n' /opt/local/bin/python* | (sed -E 's/(.*\/[^/0-9]+)([0-9]?[^/]*)$/\2 \1/' || cat) | (sort -nr || cat) | (sed -E 's/([^ ]*) (.*)/\2\1/' || cat) | grep -E '/(python|pypy)[0-9\.-]*$' | head -n 1)" | ||||
| 	pybin=/opt/local/bin/python3.9 | ||||
| 	[ -e "$pybin" ] || { | ||||
| 		echo mac detected, python3 from macports required | ||||
| 		exit 1 | ||||
| 	} | ||||
| 	pkgs='ffmpeg python39 py39-wheel' | ||||
| 	ninst=$(port installed | awk '/^  /{print$1}' | sort | uniq | grep -E '^('"$(echo "$pkgs" | tr ' ' '|')"')$' | wc -l) | ||||
| 	[ $ninst -eq 3 ] || { | ||||
| 		sudo port install $pkgs | ||||
| 	} | ||||
| 	mac=1 | ||||
| 	linux= | ||||
| } | ||||
|  | ||||
| hash -r | ||||
|  | ||||
| [ $mac ] || { | ||||
| 	command -v python3 && pybin=python3 || pybin=python | ||||
| } | ||||
|  | ||||
| $pybin -m pip install --user numpy | ||||
|  | ||||
|  | ||||
| command -v gnutar && tar() { gnutar "$@"; } | ||||
| command -v gtar && tar() { gtar "$@"; } | ||||
| command -v gsed && sed() { gsed "$@"; } | ||||
|  | ||||
|  | ||||
| need() { | ||||
| 	command -v $1 >/dev/null || { | ||||
| 		echo need $1 | ||||
| 		exit 1 | ||||
| 	} | ||||
| } | ||||
| need cmake | ||||
| need ffmpeg | ||||
| need $pybin | ||||
| #need patchelf | ||||
|  | ||||
|  | ||||
| td="$(mktemp -d)" | ||||
| cln() { | ||||
| 	rm -rf "$td" | ||||
| } | ||||
| trap cln EXIT | ||||
| cd "$td" | ||||
| pwd | ||||
|  | ||||
|  | ||||
| dl_text() { | ||||
| 	command -v curl >/dev/null && exec curl "$@" | ||||
| 	exec wget -O- "$@" | ||||
| } | ||||
| dl_files() { | ||||
| 	local yolo= ex= | ||||
| 	[ $1 = "yolo" ] && yolo=1 && ex=k && shift | ||||
| 	command -v curl >/dev/null && exec curl -${ex}JOL "$@" | ||||
| 	 | ||||
| 	[ $yolo ] && ex=--no-check-certificate | ||||
| 	exec wget --trust-server-names $ex "$@" | ||||
| } | ||||
| export -f dl_files | ||||
|  | ||||
|  | ||||
| github_tarball() { | ||||
| 	dl_text "$1" | | ||||
| 	tee json | | ||||
| 	( | ||||
| 		# prefer jq if available | ||||
| 		jq -r '.tarball_url' || | ||||
|  | ||||
| 		# fallback to awk (sorry) | ||||
| 		awk -F\" '/"tarball_url": "/ {print$4}' | ||||
| 	) | | ||||
| 	tee /dev/stderr | | ||||
| 	tr -d '\r' | tr '\n' '\0' | | ||||
| 	xargs -0 bash -c 'dl_files "$@"' _ | ||||
| } | ||||
|  | ||||
|  | ||||
| gitlab_tarball() { | ||||
| 	dl_text "$1" | | ||||
| 	tee json | | ||||
| 	( | ||||
| 		# prefer jq if available | ||||
| 		jq -r '.[0].assets.sources[]|select(.format|test("tar.gz")).url' || | ||||
|  | ||||
| 		# fallback to abomination | ||||
| 		tr \" '\n' | grep -E '\.tar\.gz$' | head -n 1 | ||||
| 	) | | ||||
| 	tee /dev/stderr | | ||||
| 	tr -d '\r' | tr '\n' '\0' | | ||||
| 	tee links | | ||||
| 	xargs -0 bash -c 'dl_files "$@"' _ | ||||
| } | ||||
|  | ||||
|  | ||||
| install_keyfinder() { | ||||
| 	# windows support: | ||||
| 	#   use msys2 in mingw-w64 mode | ||||
| 	#   pacman -S --needed mingw-w64-x86_64-{ffmpeg,python} | ||||
| 	 | ||||
| 	github_tarball https://api.github.com/repos/mixxxdj/libkeyfinder/releases/latest | ||||
|  | ||||
| 	tar -xf mixxxdj-libkeyfinder-* | ||||
| 	rm -- *.tar.gz | ||||
| 	cd mixxxdj-libkeyfinder* | ||||
| 	 | ||||
| 	h="$HOME" | ||||
| 	so="lib/libkeyfinder.so" | ||||
| 	memes=() | ||||
|  | ||||
| 	[ $win ] && | ||||
| 		so="bin/libkeyfinder.dll" && | ||||
| 		h="$(printf '%s\n' "$USERPROFILE" | tr '\\' '/')" && | ||||
| 		memes+=(-G "MinGW Makefiles" -DBUILD_TESTING=OFF) | ||||
| 	 | ||||
| 	[ $mac ] && | ||||
| 		so="lib/libkeyfinder.dylib" | ||||
|  | ||||
| 	cmake -DCMAKE_INSTALL_PREFIX="$h/pe/keyfinder" "${memes[@]}" -S . -B build | ||||
| 	cmake --build build --parallel $(nproc || echo 4) | ||||
| 	cmake --install build | ||||
|  | ||||
| 	libpath="$h/pe/keyfinder/$so" | ||||
| 	[ $linux ] && [ ! -e "$libpath" ] && | ||||
| 		so=lib64/libkeyfinder.so | ||||
| 	 | ||||
| 	libpath="$h/pe/keyfinder/$so" | ||||
| 	[ -e "$libpath" ] || { | ||||
| 		echo "so not found at $sop" | ||||
| 		exit 1 | ||||
| 	} | ||||
| 	 | ||||
| 	# rm -rf /Users/ed/Library/Python/3.9/lib/python/site-packages/*keyfinder* | ||||
| 	CFLAGS="-I$h/pe/keyfinder/include -I/opt/local/include" \ | ||||
| 	LDFLAGS="-L$h/pe/keyfinder/lib -L$h/pe/keyfinder/lib64 -L/opt/local/lib" \ | ||||
| 	PKG_CONFIG_PATH=/c/msys64/mingw64/lib/pkgconfig \ | ||||
| 	$pybin -m pip install --user keyfinder | ||||
|  | ||||
| 	pypath="$($pybin -c 'import keyfinder; print(keyfinder.__file__)')" | ||||
| 	for pyso in "${pypath%/*}"/*.so; do | ||||
| 		[ -e "$pyso" ] || break | ||||
| 		patchelf --set-rpath "${libpath%/*}" "$pyso" || | ||||
| 			echo "WARNING: patchelf failed (only fatal on musl-based distros)" | ||||
| 	done | ||||
| 	 | ||||
| 	mv "$pypath"{,.bak} | ||||
| 	( | ||||
| 		printf 'import ctypes\nctypes.cdll.LoadLibrary("%s")\n' "$libpath" | ||||
| 		cat "$pypath.bak" | ||||
| 	) >"$pypath" | ||||
|  | ||||
| 	echo | ||||
| 	echo libkeyfinder successfully installed to the following locations: | ||||
| 	echo "  $libpath" | ||||
| 	echo "  $pypath" | ||||
| } | ||||
|  | ||||
|  | ||||
| have_beatroot() { | ||||
| 	$pybin -c 'import vampyhost, sys; plugs = vampyhost.list_plugins(); sys.exit(0 if "beatroot-vamp:beatroot" in plugs else 1)' | ||||
| } | ||||
|  | ||||
|  | ||||
| install_vamp() { | ||||
| 	# windows support: | ||||
| 	#   use msys2 in mingw-w64 mode | ||||
| 	#   pacman -S --needed mingw-w64-x86_64-{ffmpeg,python,python-pip,vamp-plugin-sdk} | ||||
| 	 | ||||
| 	$pybin -m pip install --user vamp | ||||
|  | ||||
| 	have_beatroot || { | ||||
| 		printf '\033[33mcould not find the vamp beatroot plugin, building from source\033[0m\n' | ||||
| 		(dl_files yolo https://code.soundsoftware.ac.uk/attachments/download/885/beatroot-vamp-v1.0.tar.gz) | ||||
| 		sha512sum -c <( | ||||
| 			echo "1f444d1d58ccf565c0adfe99f1a1aa62789e19f5071e46857e2adfbc9d453037bc1c4dcb039b02c16240e9b97f444aaff3afb625c86aa2470233e711f55b6874  -" | ||||
| 		) <beatroot-vamp-v1.0.tar.gz | ||||
| 		tar -xf beatroot-vamp-v1.0.tar.gz  | ||||
| 		cd beatroot-vamp-v1.0 | ||||
| 		make -f Makefile.linux -j4 | ||||
| 		# /home/ed/vamp /home/ed/.vamp /usr/local/lib/vamp | ||||
| 		mkdir ~/vamp | ||||
| 		cp -pv beatroot-vamp.* ~/vamp/ | ||||
| 	} | ||||
| 	 | ||||
| 	have_beatroot && | ||||
| 		printf '\033[32mfound the vamp beatroot plugin, nice\033[0m\n' || | ||||
| 		printf '\033[31mWARNING: could not find the vamp beatroot plugin, please install it for optimal results\033[0m\n' | ||||
| } | ||||
|  | ||||
|  | ||||
| # not in use because it kinda segfaults, also no windows support | ||||
| install_soundtouch() { | ||||
| 	gitlab_tarball https://gitlab.com/api/v4/projects/soundtouch%2Fsoundtouch/releases | ||||
| 	 | ||||
| 	tar -xvf soundtouch-* | ||||
| 	rm -- *.tar.gz | ||||
| 	cd soundtouch-* | ||||
| 	 | ||||
| 	# https://github.com/jrising/pysoundtouch | ||||
| 	./bootstrap | ||||
| 	./configure --enable-integer-samples CXXFLAGS="-fPIC" --prefix="$HOME/pe/soundtouch" | ||||
| 	make -j$(nproc || echo 4) | ||||
| 	make install | ||||
| 	 | ||||
| 	CFLAGS=-I$HOME/pe/soundtouch/include/ \ | ||||
| 	LDFLAGS=-L$HOME/pe/soundtouch/lib \ | ||||
| 	$pybin -m pip install --user git+https://github.com/snowxmas/pysoundtouch.git | ||||
| 	 | ||||
| 	pypath="$($pybin -c 'import importlib; print(importlib.util.find_spec("soundtouch").origin)')" | ||||
| 	libpath="$(echo "$HOME/pe/soundtouch/lib/")" | ||||
| 	patchelf --set-rpath "$libpath" "$pypath" | ||||
|  | ||||
| 	echo | ||||
| 	echo soundtouch successfully installed to the following locations: | ||||
| 	echo "  $libpath" | ||||
| 	echo "  $pypath" | ||||
| } | ||||
|  | ||||
|  | ||||
| [ "$1" = keyfinder ] && { install_keyfinder; exit $?; } | ||||
| [ "$1" = soundtouch ] && { install_soundtouch; exit $?; } | ||||
| [ "$1" = vamp ] && { install_vamp; exit $?; } | ||||
|  | ||||
| echo no args provided, installing keyfinder and vamp | ||||
| install_keyfinder | ||||
| install_vamp | ||||
							
								
								
									
										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}") | ||||
| @@ -118,7 +118,7 @@ printf ']}' >> /dev/shm/$salt.hs | ||||
|  | ||||
| printf '\033[36m' | ||||
|  | ||||
| #curl "http://$target:1234$posturl/handshake.php" -H "Content-Type: text/plain;charset=UTF-8" -H "Cookie: cppwd=$passwd" --data "$(cat "/dev/shm/$salt.hs")" | tee /dev/shm/$salt.res | ||||
| #curl "http://$target:3923$posturl/handshake.php" -H "Content-Type: text/plain;charset=UTF-8" -H "Cookie: cppwd=$passwd" --data "$(cat "/dev/shm/$salt.hs")" | tee /dev/shm/$salt.res | ||||
|  | ||||
| { | ||||
|     { | ||||
| @@ -135,7 +135,7 @@ EOF | ||||
|     cat /dev/shm/$salt.hs | ||||
| } | | ||||
| tee /dev/shm/$salt.hsb | | ||||
| ncat $target 1234 | | ||||
| ncat $target 3923 | | ||||
| tee /dev/shm/$salt.hs1r | ||||
|  | ||||
| wark="$(cat /dev/shm/$salt.hs1r | getwark)" | ||||
| @@ -190,7 +190,7 @@ EOF | ||||
|     nchunk=$((nchunk+1)) | ||||
|  | ||||
| done | | ||||
| ncat $target 1234 | | ||||
| ncat $target 3923 | | ||||
| tee /dev/shm/$salt.pr | ||||
|  | ||||
| t=$(date +%s.%N) | ||||
| @@ -201,7 +201,7 @@ t=$(date +%s.%N) | ||||
|  | ||||
| printf '\033[36m' | ||||
|  | ||||
| ncat $target 1234 < /dev/shm/$salt.hsb | | ||||
| ncat $target 3923 < /dev/shm/$salt.hsb | | ||||
| tee /dev/shm/$salt.hs2r | | ||||
| grep -E '"hash": ?\[ *\]' | ||||
|  | ||||
|   | ||||
							
								
								
									
										37
									
								
								contrib/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								contrib/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| ### [`copyparty.bat`](copyparty.bat) | ||||
| * launches copyparty with no arguments (anon read+write within same folder) | ||||
| * intended for windows machines with no python.exe in PATH | ||||
| * works on windows, linux and macos | ||||
| * assumes `copyparty-sfx.py` was renamed to `copyparty.py` in the same folder as `copyparty.bat` | ||||
|  | ||||
| ### [`index.html`](index.html) | ||||
| * drop-in redirect from an httpd to copyparty | ||||
| * assumes the webserver and copyparty is running on the same server/IP | ||||
| * modify `10.13.1.1` as necessary if you wish to support browsers without javascript | ||||
|  | ||||
| ### [`sharex.sxcu`](sharex.sxcu) | ||||
| * sharex config file to upload screenshots and grab the URL | ||||
| * `RequestURL`: full URL to the target folder | ||||
| * `pw`: password (remove the `pw` line if anon-write) | ||||
|  | ||||
| however if your copyparty is behind a reverse-proxy, you may want to use [`sharex-html.sxcu`](sharex-html.sxcu) instead: | ||||
| * `RequestURL`: full URL to the target folder | ||||
| * `URL`: full URL to the root folder (with trailing slash) followed by `$regex:1|1$` | ||||
| * `pw`: password (remove `Parameters` if anon-write) | ||||
|  | ||||
| ### [`explorer-nothumbs-nofoldertypes.reg`](explorer-nothumbs-nofoldertypes.reg) | ||||
| * disables thumbnails and folder-type detection in windows explorer | ||||
| * makes it way faster (especially for slow/networked locations (such as copyparty-fuse)) | ||||
|  | ||||
| ### [`cfssl.sh`](cfssl.sh) | ||||
| * creates CA and server certificates using cfssl | ||||
| * give a 3rd argument to install it to your copyparty config | ||||
|  | ||||
| # OS integration | ||||
| init-scripts to start copyparty as a service | ||||
| * [`systemd/copyparty.service`](systemd/copyparty.service) | ||||
| * [`openrc/copyparty`](openrc/copyparty) | ||||
|  | ||||
| # Reverse-proxy | ||||
| copyparty has basic support for running behind another webserver | ||||
| * [`nginx/copyparty.conf`](nginx/copyparty.conf) | ||||
							
								
								
									
										72
									
								
								contrib/cfssl.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										72
									
								
								contrib/cfssl.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,72 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
| # ca-name and server-name | ||||
| ca_name="$1" | ||||
| srv_name="$2" | ||||
|  | ||||
| [ -z "$srv_name" ] && { | ||||
| 	echo "need arg 1: ca name" | ||||
| 	echo "need arg 2: server name" | ||||
| 	exit 1 | ||||
| } | ||||
|  | ||||
|  | ||||
| gen_ca() { | ||||
| 	(tee /dev/stderr <<EOF | ||||
| {"CN": "$ca_name ca", | ||||
| "CA": {"expiry":"87600h", "pathlen":0}, | ||||
| "key": {"algo":"rsa", "size":4096}, | ||||
| "names": [{"O":"$ca_name ca"}]} | ||||
| EOF | ||||
| 	)| | ||||
| 	cfssl gencert -initca - | | ||||
| 	cfssljson -bare ca | ||||
| 	 | ||||
| 	mv ca-key.pem ca.key | ||||
| 	rm ca.csr | ||||
| } | ||||
|  | ||||
|  | ||||
| gen_srv() { | ||||
| 	(tee /dev/stderr <<EOF | ||||
| {"key": {"algo":"rsa", "size":4096}, | ||||
| "names": [{"O":"$ca_name - $srv_name"}]} | ||||
| EOF | ||||
| 	)| | ||||
| 	cfssl gencert -ca ca.pem -ca-key ca.key \ | ||||
| 		-profile=www -hostname="$srv_name.$ca_name" - | | ||||
| 	cfssljson -bare "$srv_name" | ||||
|  | ||||
| 	mv "$srv_name-key.pem" "$srv_name.key" | ||||
| 	rm "$srv_name.csr" | ||||
| } | ||||
|  | ||||
|  | ||||
| # create ca if not exist | ||||
| [ -e ca.key ] || | ||||
| 	gen_ca | ||||
|  | ||||
| # always create server cert | ||||
| gen_srv | ||||
|  | ||||
|  | ||||
| # dump cert info | ||||
| show() { | ||||
| 	openssl x509 -text -noout -in $1 | | ||||
| 	awk '!o; {o=0} /[0-9a-f:]{16}/{o=1}' | ||||
| } | ||||
| show ca.pem | ||||
| show "$srv_name.pem" | ||||
|  | ||||
|  | ||||
| # write cert into copyparty config | ||||
| [ -z "$3" ] || { | ||||
| 	mkdir -p ~/.config/copyparty | ||||
| 	cat "$srv_name".{key,pem} ca.pem >~/.config/copyparty/cert.pem  | ||||
| } | ||||
|  | ||||
|  | ||||
| # rm *.key *.pem | ||||
| # cfssl print-defaults config | ||||
| # cfssl print-defaults csr | ||||
							
								
								
									
										33
									
								
								contrib/copyparty.bat
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								contrib/copyparty.bat
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| exec python "$(dirname "$0")"/copyparty.py | ||||
|  | ||||
| @rem on linux, the above will execute and the script will terminate | ||||
| @rem on windows, the rest of this script will run | ||||
|  | ||||
| @echo off | ||||
| cls | ||||
|  | ||||
| set py= | ||||
| for /f %%i in ('where python 2^>nul') do ( | ||||
|     set "py=%%i" | ||||
|     goto c1 | ||||
| ) | ||||
| :c1 | ||||
|  | ||||
| if [%py%] == [] ( | ||||
|     for /f %%i in ('where /r "%localappdata%\programs\python" python 2^>nul') do ( | ||||
|         set "py=%%i" | ||||
|         goto c2 | ||||
|     ) | ||||
| ) | ||||
| :c2 | ||||
|  | ||||
| if [%py%] == [] set "py=c:\python27\python.exe" | ||||
|  | ||||
| if not exist "%py%" ( | ||||
|     echo could not find python | ||||
|     echo( | ||||
|     pause | ||||
|     exit /b | ||||
| ) | ||||
|  | ||||
| start cmd /c %py% "%~dp0\copyparty.py" | ||||
							
								
								
									
										31
									
								
								contrib/explorer-nothumbs-nofoldertypes.reg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								contrib/explorer-nothumbs-nofoldertypes.reg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| Windows Registry Editor Version 5.00 | ||||
|  | ||||
| ; this will do 3 things, all optional: | ||||
| ;  1) disable thumbnails | ||||
| ;  2) delete all existing folder type settings/detections | ||||
| ;  3) disable folder type detection (force default columns) | ||||
| ; | ||||
| ; this makes the file explorer way faster, | ||||
| ; especially on slow/networked locations | ||||
|  | ||||
|  | ||||
| ; ===================================================================== | ||||
| ; 1) disable thumbnails | ||||
|  | ||||
| [HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced] | ||||
| "IconsOnly"=dword:00000001 | ||||
|  | ||||
|  | ||||
| ; ===================================================================== | ||||
| ; 2) delete all existing folder type settings/detections | ||||
|  | ||||
| [-HKEY_CURRENT_USER\Software\Classes\Local Settings\Software\Microsoft\Windows\Shell\Bags] | ||||
|  | ||||
| [-HKEY_CURRENT_USER\Software\Classes\Local Settings\Software\Microsoft\Windows\Shell\BagMRU] | ||||
|  | ||||
|  | ||||
| ; ===================================================================== | ||||
| ; 3) disable folder type detection | ||||
|  | ||||
| [HKEY_CURRENT_USER\Software\Classes\Local Settings\Software\Microsoft\Windows\Shell\Bags\AllFolders\Shell] | ||||
| "FolderType"="NotSpecified" | ||||
							
								
								
									
										43
									
								
								contrib/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								contrib/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|  | ||||
| <head> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>⇆🎉 redirect</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<style> | ||||
|  | ||||
| html, body { | ||||
| 	font-family: sans-serif; | ||||
| } | ||||
| body { | ||||
| 	padding: 1em 2em; | ||||
| 	font-size: 1.5em; | ||||
| } | ||||
| a { | ||||
| 	font-size: 1.2em; | ||||
| 	padding: .1em; | ||||
| } | ||||
|  | ||||
| </style> | ||||
| </head> | ||||
| <body> | ||||
| 	<span id="desc">you probably want</span> <a id="redir" href="//10.13.1.1:3923/">copyparty</a> | ||||
| 	<script> | ||||
|  | ||||
| var a = document.getElementById('redir'), | ||||
| 	proto = window.location.protocol.indexOf('https') === 0 ? 'https' : 'http', | ||||
| 	loc = window.location.hostname || '127.0.0.1', | ||||
| 	port = a.getAttribute('href').split(':').pop().split('/')[0], | ||||
| 	url = proto + '://' + loc + ':' + port + '/'; | ||||
|  | ||||
| a.setAttribute('href', url); | ||||
| document.getElementById('desc').innerHTML = 'redirecting to'; | ||||
|  | ||||
| setTimeout(function() { | ||||
| 	window.location.href = url; | ||||
| }, 500); | ||||
|  | ||||
| </script> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										39
									
								
								contrib/nginx/copyparty.conf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								contrib/nginx/copyparty.conf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| # when running copyparty behind a reverse proxy, | ||||
| # the following arguments are recommended: | ||||
| # | ||||
| #   -nc 512         important, see next paragraph | ||||
| #   --http-only     lower latency on initial connection | ||||
| #   -i 127.0.0.1    only accept connections from nginx | ||||
| # | ||||
| # -nc must match or exceed the webserver's max number of concurrent clients; | ||||
| # nginx default is 512  (worker_processes 1, worker_connections 512) | ||||
| # | ||||
| # you may also consider adding -j0 for CPU-intensive configurations | ||||
| # (not that i can really think of any good examples) | ||||
|  | ||||
| upstream cpp { | ||||
| 	server 127.0.0.1:3923; | ||||
| 	keepalive 120; | ||||
| } | ||||
| server { | ||||
| 	listen 443 ssl; | ||||
| 	listen [::]:443 ssl; | ||||
|  | ||||
| 	server_name fs.example.com; | ||||
| 	 | ||||
| 	location / { | ||||
| 		proxy_pass http://cpp; | ||||
| 		proxy_redirect off; | ||||
| 		# disable buffering (next 4 lines) | ||||
| 		proxy_http_version 1.1; | ||||
| 		client_max_body_size 0; | ||||
| 		proxy_buffering off; | ||||
| 		proxy_request_buffering off; | ||||
|  | ||||
| 		proxy_set_header   Host              $host; | ||||
| 		proxy_set_header   X-Real-IP         $remote_addr; | ||||
| 		proxy_set_header   X-Forwarded-For   $proxy_add_x_forwarded_for; | ||||
| 		proxy_set_header   X-Forwarded-Proto $scheme; | ||||
| 		proxy_set_header   Connection        "Keep-Alive"; | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										18
									
								
								contrib/openrc/copyparty
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								contrib/openrc/copyparty
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| #!/sbin/openrc-run | ||||
|  | ||||
| # this will start `/usr/local/bin/copyparty-sfx.py` | ||||
| # and share '/mnt' with anonymous read+write | ||||
| # | ||||
| # installation: | ||||
| #   cp -pv copyparty /etc/init.d && rc-update add copyparty | ||||
| # | ||||
| # you may want to: | ||||
| #   change '/usr/bin/python' to another interpreter | ||||
| #   change '/mnt::a' 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::a" | ||||
							
								
								
									
										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$" | ||||
| } | ||||
							
								
								
									
										31
									
								
								contrib/systemd/copyparty.service
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								contrib/systemd/copyparty.service
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| # this will start `/usr/local/bin/copyparty-sfx.py` | ||||
| # and share '/mnt' with anonymous read+write | ||||
| # | ||||
| # installation: | ||||
| #   cp -pv copyparty.service /etc/systemd/system && systemctl enable --now copyparty | ||||
| # | ||||
| # you may want to: | ||||
| #   change '/usr/bin/python' to another interpreter | ||||
| #   change '/mnt::a' to another location or permission-set | ||||
| # | ||||
| # with `Type=notify`, copyparty will signal systemd when it is ready to | ||||
| #   accept connections; correctly delaying units depending on copyparty. | ||||
| #   But note that journalctl will get the timestamps wrong due to | ||||
| #   python disabling line-buffering, so messages are out-of-order: | ||||
| #   https://user-images.githubusercontent.com/241032/126040249-cb535cc7-c599-4931-a796-a5d9af691bad.png | ||||
| # | ||||
| # enable line-buffering for realtime logging (slight performance cost): | ||||
| #   modify ExecStart and prefix it with `/bin/stdbuf -oL` like so: | ||||
| #   ExecStart=/bin/stdbuf -oL /usr/bin/python3 [...] | ||||
|  | ||||
| [Unit] | ||||
| Description=copyparty file server | ||||
|  | ||||
| [Service] | ||||
| Type=notify | ||||
| SyslogIdentifier=copyparty | ||||
| ExecStart=/usr/bin/python3 /usr/local/bin/copyparty-sfx.py -q -v /mnt::a | ||||
| ExecStartPre=/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf' | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
| @@ -2,12 +2,16 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import platform | ||||
| import time | ||||
| import sys | ||||
| import os | ||||
|  | ||||
| PY2 = sys.version_info[0] == 2 | ||||
| if PY2: | ||||
|     sys.dont_write_bytecode = True | ||||
|     unicode = unicode | ||||
| else: | ||||
|     unicode = str | ||||
|  | ||||
| WINDOWS = False | ||||
| if platform.system() == "Windows": | ||||
| @@ -16,10 +20,18 @@ if platform.system() == "Windows": | ||||
| VT100 = not WINDOWS or WINDOWS >= [10, 0, 14393] | ||||
| # introduced in anniversary update | ||||
|  | ||||
| ANYWIN = WINDOWS or sys.platform in ["msys"] | ||||
|  | ||||
| MACOS = platform.system() == "Darwin" | ||||
|  | ||||
|  | ||||
| class EnvParams(object): | ||||
|     def __init__(self): | ||||
|         self.t0 = time.time() | ||||
|         self.mod = os.path.dirname(os.path.realpath(__file__)) | ||||
|         if self.mod.endswith("__init__"): | ||||
|             self.mod = os.path.dirname(self.mod) | ||||
|  | ||||
|         if sys.platform == "win32": | ||||
|             self.cfg = os.path.normpath(os.environ["APPDATA"] + "/copyparty") | ||||
|         elif sys.platform == "darwin": | ||||
|   | ||||
| @@ -8,17 +8,30 @@ __copyright__ = 2019 | ||||
| __license__ = "MIT" | ||||
| __url__ = "https://github.com/9001/copyparty/" | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import shutil | ||||
| import filecmp | ||||
| import locale | ||||
| import argparse | ||||
| import threading | ||||
| import traceback | ||||
| from textwrap import dedent | ||||
|  | ||||
| from .__init__ import E, WINDOWS, VT100 | ||||
| from .__init__ import E, WINDOWS, VT100, PY2, unicode | ||||
| from .__version__ import S_VERSION, S_BUILD_DT, CODENAME | ||||
| from .svchub import SvcHub | ||||
| from .util import py_desc | ||||
| from .util import py_desc, align_tab, IMPLICATIONS | ||||
|  | ||||
| HAVE_SSL = True | ||||
| try: | ||||
|     import ssl | ||||
| except: | ||||
|     HAVE_SSL = False | ||||
|  | ||||
| printed = "" | ||||
|  | ||||
|  | ||||
| class RiceFormatter(argparse.HelpFormatter): | ||||
| @@ -44,6 +57,23 @@ class RiceFormatter(argparse.HelpFormatter): | ||||
|         return "".join(indent + line + "\n" for line in text.splitlines()) | ||||
|  | ||||
|  | ||||
| class Dodge11874(RiceFormatter): | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         kwargs["width"] = 9003 | ||||
|         super(Dodge11874, self).__init__(*args, **kwargs) | ||||
|  | ||||
|  | ||||
| def lprint(*a, **ka): | ||||
|     global printed | ||||
|  | ||||
|     printed += " ".join(unicode(x) for x in a) + ka.get("end", "\n") | ||||
|     print(*a, **ka) | ||||
|  | ||||
|  | ||||
| def warn(msg): | ||||
|     lprint("\033[1mwarning:\033[0;33m {}\033[0m\n".format(msg)) | ||||
|  | ||||
|  | ||||
| def ensure_locale(): | ||||
|     for x in [ | ||||
|         "en_US.UTF-8", | ||||
| @@ -52,7 +82,7 @@ def ensure_locale(): | ||||
|     ]: | ||||
|         try: | ||||
|             locale.setlocale(locale.LC_ALL, x) | ||||
|             print("Locale:", x) | ||||
|             lprint("Locale:", x) | ||||
|             break | ||||
|         except: | ||||
|             continue | ||||
| @@ -73,7 +103,7 @@ def ensure_cert(): | ||||
|  | ||||
|     try: | ||||
|         if filecmp.cmp(cert_cfg, cert_insec): | ||||
|             print( | ||||
|             lprint( | ||||
|                 "\033[33m  using default TLS certificate; https will be insecure." | ||||
|                 + "\033[36m\n  certificate location: {}\033[0m\n".format(cert_cfg) | ||||
|             ) | ||||
| @@ -84,59 +114,351 @@ def ensure_cert(): | ||||
|     # printf 'NO\n.\n.\n.\n.\ncopyparty-insecure\n.\n' | faketime '2000-01-01 00:00:00' openssl req -x509 -sha256 -newkey rsa:2048 -keyout insecure.pem -out insecure.pem -days $((($(printf %d 0x7fffffff)-$(date +%s --date=2000-01-01T00:00:00Z))/(60*60*24))) -nodes && ls -al insecure.pem && openssl x509 -in insecure.pem -text -noout | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     if WINDOWS: | ||||
|         os.system("")  # enables colors | ||||
| def configure_ssl_ver(al): | ||||
|     def terse_sslver(txt): | ||||
|         txt = txt.lower() | ||||
|         for c in ["_", "v", "."]: | ||||
|             txt = txt.replace(c, "") | ||||
|  | ||||
|     desc = py_desc().replace("[", "\033[1;30m[") | ||||
|         return txt.replace("tls10", "tls1") | ||||
|  | ||||
|     f = '\033[36mcopyparty v{} "\033[35m{}\033[36m" ({})\n{}\033[0m\n' | ||||
|     print(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc)) | ||||
|     # oh man i love openssl | ||||
|     # check this out | ||||
|     # hold my beer | ||||
|     ptn = re.compile(r"^OP_NO_(TLS|SSL)v") | ||||
|     sslver = terse_sslver(al.ssl_ver).split(",") | ||||
|     flags = [k for k in ssl.__dict__ if ptn.match(k)] | ||||
|     # SSLv2 SSLv3 TLSv1 TLSv1_1 TLSv1_2 TLSv1_3 | ||||
|     if "help" in sslver: | ||||
|         avail = [terse_sslver(x[6:]) for x in flags] | ||||
|         avail = " ".join(sorted(avail) + ["all"]) | ||||
|         lprint("\navailable ssl/tls versions:\n  " + avail) | ||||
|         sys.exit(0) | ||||
|  | ||||
|     ensure_locale() | ||||
|     ensure_cert() | ||||
|     al.ssl_flags_en = 0 | ||||
|     al.ssl_flags_de = 0 | ||||
|     for flag in sorted(flags): | ||||
|         ver = terse_sslver(flag[6:]) | ||||
|         num = getattr(ssl, flag) | ||||
|         if ver in sslver: | ||||
|             al.ssl_flags_en |= num | ||||
|         else: | ||||
|             al.ssl_flags_de |= num | ||||
|  | ||||
|     if sslver == ["all"]: | ||||
|         x = al.ssl_flags_en | ||||
|         al.ssl_flags_en = al.ssl_flags_de | ||||
|         al.ssl_flags_de = x | ||||
|  | ||||
|     for k in ["ssl_flags_en", "ssl_flags_de"]: | ||||
|         num = getattr(al, k) | ||||
|         lprint("{}: {:8x} ({})".format(k, num, num)) | ||||
|  | ||||
|     # think i need that beer now | ||||
|  | ||||
|  | ||||
| def configure_ssl_ciphers(al): | ||||
|     ctx = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH) | ||||
|     if al.ssl_ver: | ||||
|         ctx.options &= ~al.ssl_flags_en | ||||
|         ctx.options |= al.ssl_flags_de | ||||
|  | ||||
|     is_help = al.ciphers == "help" | ||||
|  | ||||
|     if al.ciphers and not is_help: | ||||
|         try: | ||||
|             ctx.set_ciphers(al.ciphers) | ||||
|         except: | ||||
|             lprint("\n\033[1;31mfailed to set ciphers\033[0m\n") | ||||
|  | ||||
|     if not hasattr(ctx, "get_ciphers"): | ||||
|         lprint("cannot read cipher list: openssl or python too old") | ||||
|     else: | ||||
|         ciphers = [x["description"] for x in ctx.get_ciphers()] | ||||
|         lprint("\n  ".join(["\nenabled ciphers:"] + align_tab(ciphers) + [""])) | ||||
|  | ||||
|     if is_help: | ||||
|         sys.exit(0) | ||||
|  | ||||
|  | ||||
| def sighandler(sig=None, frame=None): | ||||
|     msg = [""] * 5 | ||||
|     for th in threading.enumerate(): | ||||
|         msg.append(str(th)) | ||||
|         msg.extend(traceback.format_stack(sys._current_frames()[th.ident])) | ||||
|  | ||||
|     msg.append("\n") | ||||
|     print("\n".join(msg)) | ||||
|  | ||||
|  | ||||
| def run_argparse(argv, formatter): | ||||
|     ap = argparse.ArgumentParser( | ||||
|         formatter_class=RiceFormatter, | ||||
|         formatter_class=formatter, | ||||
|         prog="copyparty", | ||||
|         description="http file sharing hub v{} ({})".format(S_VERSION, S_BUILD_DT), | ||||
|         epilog=dedent( | ||||
|             """ | ||||
|             -a takes username:password, | ||||
|             -v takes src:dst:permset:permset:... where "permset" is | ||||
|                accesslevel followed by username (no separator) | ||||
|             -v takes src:dst:perm1:perm2:permN:cflag1:cflag2:cflagN:... | ||||
|                where "perm" is "accesslevels,username1,username2,..." | ||||
|                and "cflag" is config flags to set on this volume | ||||
|              | ||||
|             list of accesslevels: | ||||
|               "r" (read):   list folder contents, download files | ||||
|               "w" (write):  upload files; need "r" to see the uploads | ||||
|               "m" (move):   move files and folders; need "w" at destination | ||||
|               "d" (delete): permanently delete files and folders | ||||
|  | ||||
|             list of cflags: | ||||
|               "c,nodupe" rejects existing files (instead of symlinking them) | ||||
|               "c,e2d" sets -e2d (all -e2* args can be set using ce2* cflags) | ||||
|               "c,d2t" disables metadata collection, overrides -e2t* | ||||
|               "c,d2d" disables all database stuff, overrides -e2* | ||||
|  | ||||
|             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) | ||||
|  | ||||
|             values for --urlform: | ||||
|               "stash" dumps the data to file and returns length + checksum | ||||
|               "save,get" dumps to file and returns the page like a GET | ||||
|               "print,get" prints the data in the log and returns GET | ||||
|               (leave out the ",get" to return an error instead) | ||||
|  | ||||
|             values for --ls: | ||||
|               "USR" is a user to browse as; * is anonymous, ** is all users | ||||
|               "VOL" is a single volume to scan, default is * (all vols) | ||||
|               "FLAG" is flags; | ||||
|                 "v" in addition to realpaths, print usernames and vpaths | ||||
|                 "ln" only prints symlinks leaving the volume mountpoint | ||||
|                 "p" exits 1 if any such symlinks are found | ||||
|                 "r" 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 | ||||
|             \033[0m | ||||
|             """ | ||||
|         ), | ||||
|     ) | ||||
|     ap.add_argument( | ||||
|         "-c", metavar="PATH", type=str, action="append", help="add config file" | ||||
|     ) | ||||
|     ap.add_argument("-i", metavar="IP", type=str, default="0.0.0.0", help="ip to bind") | ||||
|     ap.add_argument("-p", metavar="PORT", type=int, default=1234, help="port to bind") | ||||
|     ap.add_argument("-nc", metavar="NUM", type=int, default=16, help="max num clients") | ||||
|     ap.add_argument("-j", metavar="CORES", type=int, 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("-nw", action="store_true", help="benchmark: disable writing") | ||||
|     al = ap.parse_args() | ||||
|     # fmt: off | ||||
|     u = unicode | ||||
|     ap2 = ap.add_argument_group('general options') | ||||
|     ap2.add_argument("-c", metavar="PATH", type=u, action="append", help="add config file") | ||||
|     ap2.add_argument("-nc", metavar="NUM", type=int, default=64, help="max num clients") | ||||
|     ap2.add_argument("-j", metavar="CORES", type=int, default=1, help="max num cpu cores") | ||||
|     ap2.add_argument("-a", metavar="ACCT", type=u, action="append", help="add account, USER:PASS; example [ed:wark") | ||||
|     ap2.add_argument("-v", metavar="VOL", type=u, action="append", help="add volume, SRC:DST:FLAG; example [.::r], [/mnt/nas/music:/music:r:aed") | ||||
|     ap2.add_argument("-ed", action="store_true", help="enable ?dots") | ||||
|     ap2.add_argument("-emp", action="store_true", help="enable markdown plugins") | ||||
|     ap2.add_argument("-mcr", metavar="SEC", type=int, default=60, help="md-editor mod-chk rate") | ||||
|     ap2.add_argument("--urlform", metavar="MODE", type=u, default="print,get", help="how to handle url-forms; examples: [stash], [save,get]") | ||||
|  | ||||
|     SvcHub(al).run() | ||||
|     ap2 = ap.add_argument_group('upload options') | ||||
|     ap2.add_argument("--dotpart", action="store_true", help="dotfile incomplete uploads") | ||||
|     ap2.add_argument("--sparse", metavar="MiB", type=int, default=4, help="up2k min.size threshold (mswin-only)") | ||||
|     ap2.add_argument("--unpost", metavar="SEC", type=int, default=3600*12, help="grace period where uploads can be deleted by the uploader, even without delete permissions; 0=disabled") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('network options') | ||||
|     ap2.add_argument("-i", metavar="IP", type=u, default="0.0.0.0", help="ip to bind (comma-sep.)") | ||||
|     ap2.add_argument("-p", metavar="PORT", type=u, default="3923", help="ports to bind (comma/range)") | ||||
|     ap2.add_argument("--rproxy", metavar="DEPTH", type=int, default=1, help="which ip to keep; 0 = tcp, 1 = origin (first x-fwd), 2 = cloudflare, 3 = nginx, -1 = closest proxy") | ||||
|      | ||||
|     ap2 = ap.add_argument_group('SSL/TLS options') | ||||
|     ap2.add_argument("--http-only", action="store_true", help="disable ssl/tls") | ||||
|     ap2.add_argument("--https-only", action="store_true", help="disable plaintext") | ||||
|     ap2.add_argument("--ssl-ver", metavar="LIST", type=u, help="set allowed ssl/tls versions; [help] shows available versions; default is what your python version considers safe") | ||||
|     ap2.add_argument("--ciphers", metavar="LIST", type=u, help="set allowed ssl/tls ciphers; [help] shows available ciphers") | ||||
|     ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info") | ||||
|     ap2.add_argument("--ssl-log", metavar="PATH", type=u, help="log master secrets") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('opt-outs') | ||||
|     ap2.add_argument("-nw", action="store_true", help="disable writes (benchmark)") | ||||
|     ap2.add_argument("--no-del", action="store_true", help="disable delete operations") | ||||
|     ap2.add_argument("--no-mv", action="store_true", help="disable move/rename operations") | ||||
|     ap2.add_argument("-nih", action="store_true", help="no info hostname") | ||||
|     ap2.add_argument("-nid", action="store_true", help="no info disk-usage") | ||||
|     ap2.add_argument("--no-zip", action="store_true", help="disable download as zip/tar") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('safety options') | ||||
|     ap2.add_argument("--ls", metavar="U[,V[,F]]", type=u, help="scan all volumes; arguments USER,VOL,FLAGS; example [**,*,ln,p,r]") | ||||
|     ap2.add_argument("--salt", type=u, default="hunter2", help="up2k file-hash salt") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('logging options') | ||||
|     ap2.add_argument("-q", action="store_true", help="quiet") | ||||
|     ap2.add_argument("-lo", metavar="PATH", type=u, help="logfile, example: cpp-%%Y-%%m%%d-%%H%%M%%S.txt.xz") | ||||
|     ap2.add_argument("--no-voldump", action="store_true", help="do not list volumes and permissions on startup") | ||||
|     ap2.add_argument("--log-conn", action="store_true", help="print tcp-server msgs") | ||||
|     ap2.add_argument("--log-htp", action="store_true", help="print http-server threadpool scaling") | ||||
|     ap2.add_argument("--ihead", metavar="HEADER", type=u, action='append', help="dump incoming header") | ||||
|     ap2.add_argument("--lf-url", metavar="RE", type=u, default=r"^/\.cpr/|\?th=[wj]$", help="dont log URLs matching") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('admin panel options') | ||||
|     ap2.add_argument("--no-rescan", action="store_true", help="disable ?scan (volume reindexing)") | ||||
|     ap2.add_argument("--no-stack", action="store_true", help="disable ?stack (list all stacks)") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('thumbnail options') | ||||
|     ap2.add_argument("--no-thumb", action="store_true", help="disable all thumbnails") | ||||
|     ap2.add_argument("--no-vthumb", action="store_true", help="disable video thumbnails") | ||||
|     ap2.add_argument("--th-size", metavar="WxH", default="320x256", help="thumbnail res") | ||||
|     ap2.add_argument("--th-no-crop", action="store_true", help="dynamic height; show full image") | ||||
|     ap2.add_argument("--th-no-jpg", action="store_true", help="disable jpg output") | ||||
|     ap2.add_argument("--th-no-webp", action="store_true", help="disable webp output") | ||||
|     ap2.add_argument("--th-ff-jpg", action="store_true", help="force jpg for video thumbs") | ||||
|     ap2.add_argument("--th-poke", metavar="SEC", type=int, default=300, help="activity labeling cooldown") | ||||
|     ap2.add_argument("--th-clean", metavar="SEC", type=int, default=43200, help="cleanup interval; 0=disabled") | ||||
|     ap2.add_argument("--th-maxage", metavar="SEC", type=int, default=604800, help="max folder age") | ||||
|     ap2.add_argument("--th-covers", metavar="N,N", type=u, default="folder.png,folder.jpg,cover.png,cover.jpg", help="folder thumbnails to stat for") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('general db options') | ||||
|     ap2.add_argument("-e2d", action="store_true", help="enable up2k database") | ||||
|     ap2.add_argument("-e2ds", action="store_true", help="enable up2k db-scanner, sets -e2d") | ||||
|     ap2.add_argument("-e2dsa", action="store_true", help="scan all folders (for search), sets -e2ds") | ||||
|     ap2.add_argument("--hist", metavar="PATH", type=u, help="where to store volume state") | ||||
|     ap2.add_argument("--no-hash", action="store_true", help="disable hashing during e2ds folder scans") | ||||
|     ap2.add_argument("--re-int", metavar="SEC", type=int, default=30, help="disk rescan check interval") | ||||
|     ap2.add_argument("--re-maxage", metavar="SEC", type=int, default=0, help="disk rescan volume interval (0=off)") | ||||
|     ap2.add_argument("--srch-time", metavar="SEC", type=int, default=30, help="search deadline") | ||||
|      | ||||
|     ap2 = ap.add_argument_group('metadata db options') | ||||
|     ap2.add_argument("-e2t", action="store_true", help="enable metadata indexing") | ||||
|     ap2.add_argument("-e2ts", action="store_true", help="enable metadata scanner, sets -e2t") | ||||
|     ap2.add_argument("-e2tsr", action="store_true", help="rescan all metadata, sets -e2ts") | ||||
|     ap2.add_argument("--no-mutagen", action="store_true", help="use FFprobe for tags instead") | ||||
|     ap2.add_argument("--no-mtag-mt", action="store_true", help="disable tag-read parallelism") | ||||
|     ap2.add_argument("--no-mtag-ff", action="store_true", help="never use FFprobe as tag reader") | ||||
|     ap2.add_argument("-mtm", metavar="M=t,t,t", type=u, action="append", help="add/replace metadata mapping") | ||||
|     ap2.add_argument("-mte", metavar="M,M,M", type=u, help="tags to index/display (comma-sep.)", | ||||
|         default="circle,album,.tn,artist,title,.bpm,key,.dur,.q,.vq,.aq,ac,vc,res,.fps") | ||||
|     ap2.add_argument("-mtp", metavar="M=[f,]bin", type=u, action="append", help="read tag M using bin") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('appearance options') | ||||
|     ap2.add_argument("--css-browser", metavar="L", type=u, help="URL to additional CSS to include") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('debug options') | ||||
|     ap2.add_argument("--no-sendfile", action="store_true", help="disable sendfile") | ||||
|     ap2.add_argument("--no-scandir", action="store_true", help="disable scandir") | ||||
|     ap2.add_argument("--no-fastboot", action="store_true", help="wait for up2k indexing") | ||||
|     ap2.add_argument("--no-htp", action="store_true", help="disable httpserver threadpool, create threads as-needed instead") | ||||
|     ap2.add_argument("--stackmon", metavar="P,S", type=u, help="write stacktrace to Path every S second") | ||||
|     ap2.add_argument("--log-thrs", metavar="SEC", type=float, help="list active threads every SEC") | ||||
|      | ||||
|     return ap.parse_args(args=argv[1:]) | ||||
|     # fmt: on | ||||
|  | ||||
|  | ||||
| def main(argv=None): | ||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||
|     if WINDOWS: | ||||
|         os.system("rem")  # enables colors | ||||
|  | ||||
|     if argv is None: | ||||
|         argv = sys.argv | ||||
|  | ||||
|     desc = py_desc().replace("[", "\033[1;30m[") | ||||
|  | ||||
|     f = '\033[36mcopyparty v{} "\033[35m{}\033[36m" ({})\n{}\033[0m\n' | ||||
|     lprint(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc)) | ||||
|  | ||||
|     ensure_locale() | ||||
|     if HAVE_SSL: | ||||
|         ensure_cert() | ||||
|  | ||||
|     deprecated = [["-e2s", "-e2ds"]] | ||||
|     for dk, nk in deprecated: | ||||
|         try: | ||||
|             idx = argv.index(dk) | ||||
|         except: | ||||
|             continue | ||||
|  | ||||
|         msg = "\033[1;31mWARNING:\033[0;1m\n  {} \033[0;33mwas replaced with\033[0;1m {} \033[0;33mand will be removed\n\033[0m" | ||||
|         lprint(msg.format(dk, nk)) | ||||
|         argv[idx] = nk | ||||
|         time.sleep(2) | ||||
|  | ||||
|     try: | ||||
|         al = run_argparse(argv, RiceFormatter) | ||||
|     except AssertionError: | ||||
|         al = run_argparse(argv, Dodge11874) | ||||
|  | ||||
|     nstrs = [] | ||||
|     anymod = False | ||||
|     for ostr in al.v or []: | ||||
|         mod = False | ||||
|         oa = ostr.split(":") | ||||
|         na = oa[:2] | ||||
|         for opt in oa[2:]: | ||||
|             if re.match("c[^,]", opt): | ||||
|                 mod = True | ||||
|                 na.append("c," + opt[1:]) | ||||
|             elif re.sub("^[rwmd]*", "", opt) and "," not in opt: | ||||
|                 mod = True | ||||
|                 perm = opt[0] | ||||
|                 if perm == "a": | ||||
|                     perm = "rw" | ||||
|                 na.append(perm + "," + opt[1:]) | ||||
|             else: | ||||
|                 na.append(opt) | ||||
|  | ||||
|         nstr = ":".join(na) | ||||
|         nstrs.append(nstr if mod else ostr) | ||||
|         if mod: | ||||
|             msg = "\033[1;31mWARNING:\033[0;1m\n  -v {} \033[0;33mwas replaced with\033[0;1m\n  -v {} \n\033[0m" | ||||
|             lprint(msg.format(ostr, nstr)) | ||||
|             anymod = True | ||||
|  | ||||
|     if anymod: | ||||
|         al.v = nstrs | ||||
|         time.sleep(2) | ||||
|  | ||||
|     # propagate implications | ||||
|     for k1, k2 in IMPLICATIONS: | ||||
|         if getattr(al, k1): | ||||
|             setattr(al, k2, True) | ||||
|  | ||||
|     al.i = al.i.split(",") | ||||
|     try: | ||||
|         if "-" in al.p: | ||||
|             lo, hi = [int(x) for x in al.p.split("-")] | ||||
|             al.p = list(range(lo, hi + 1)) | ||||
|         else: | ||||
|             al.p = [int(x) for x in al.p.split(",")] | ||||
|     except: | ||||
|         raise Exception("invalid value for -p") | ||||
|  | ||||
|     if HAVE_SSL: | ||||
|         if al.ssl_ver: | ||||
|             configure_ssl_ver(al) | ||||
|  | ||||
|         if al.ciphers: | ||||
|             configure_ssl_ciphers(al) | ||||
|     else: | ||||
|         warn("ssl module does not exist; cannot enable https") | ||||
|  | ||||
|     if PY2 and WINDOWS and al.e2d: | ||||
|         warn( | ||||
|             "windows py2 cannot do unicode filenames with -e2d\n" | ||||
|             + "  (if you crash with codec errors then that is why)" | ||||
|         ) | ||||
|  | ||||
|     if sys.version_info < (3, 6): | ||||
|         al.no_scandir = True | ||||
|  | ||||
|     # signal.signal(signal.SIGINT, sighandler) | ||||
|  | ||||
|     SvcHub(al, argv, printed).run() | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| # coding: utf-8 | ||||
|  | ||||
| VERSION = (0, 3, 0) | ||||
| CODENAME = "docuparty" | ||||
| BUILD_DT = (2020, 5, 6) | ||||
| VERSION = (0, 12, 5) | ||||
| CODENAME = "fil\033[33med" | ||||
| BUILD_DT = (2021, 7, 30) | ||||
|  | ||||
| S_VERSION = ".".join(map(str, VERSION)) | ||||
| S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) | ||||
|   | ||||
| @@ -1,23 +1,77 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import stat | ||||
| import base64 | ||||
| import hashlib | ||||
| import threading | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS | ||||
| from .util import undot, Pebkac, fsdec, fsenc | ||||
| from .__init__ import WINDOWS | ||||
| from .util import IMPLICATIONS, uncyg, undot, absreal, Pebkac, fsdec, fsenc, statdir | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| class AXS(object): | ||||
|     def __init__(self, uread=None, uwrite=None, umove=None, udel=None): | ||||
|         self.uread = {} if uread is None else {k: 1 for k in uread} | ||||
|         self.uwrite = {} if uwrite is None else {k: 1 for k in uwrite} | ||||
|         self.umove = {} if umove is None else {k: 1 for k in umove} | ||||
|         self.udel = {} if udel is None else {k: 1 for k in udel} | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return "AXS({})".format( | ||||
|             ", ".join( | ||||
|                 "{}={!r}".format(k, self.__dict__[k]) | ||||
|                 for k in "uread uwrite umove udel".split() | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class VFS(object): | ||||
|     """single level in the virtual fs""" | ||||
|  | ||||
|     def __init__(self, realpath, vpath, uread=[], uwrite=[]): | ||||
|     def __init__(self, log, realpath, vpath, axs, flags): | ||||
|         self.log = log | ||||
|         self.realpath = realpath  # absolute path on host filesystem | ||||
|         self.vpath = vpath  # absolute path in the virtual filesystem | ||||
|         self.uread = uread  # users who can read this | ||||
|         self.uwrite = uwrite  # users who can write this | ||||
|         self.axs = axs  # type: AXS | ||||
|         self.flags = flags  # config options | ||||
|         self.nodes = {}  # child nodes | ||||
|         self.histtab = None  # all realpath->histpath | ||||
|         self.dbv = None  # closest full/non-jump parent | ||||
|  | ||||
|         if realpath: | ||||
|             self.histpath = os.path.join(realpath, ".hist")  # db / thumbcache | ||||
|             self.all_vols = {vpath: self}  # flattened recursive | ||||
|             self.aread = {} | ||||
|             self.awrite = {} | ||||
|             self.amove = {} | ||||
|             self.adel = {} | ||||
|         else: | ||||
|             self.histpath = None | ||||
|             self.all_vols = None | ||||
|             self.aread = None | ||||
|             self.awrite = None | ||||
|             self.amove = None | ||||
|             self.adel = None | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return "VFS({})".format( | ||||
|             ", ".join( | ||||
|                 "{}={!r}".format(k, self.__dict__[k]) | ||||
|                 for k in "realpath vpath axs flags".split() | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|     def get_all_vols(self, outdict): | ||||
|         if self.realpath: | ||||
|             outdict[self.vpath] = self | ||||
|  | ||||
|         for v in self.nodes.values(): | ||||
|             v.get_all_vols(outdict) | ||||
|  | ||||
|     def add(self, src, dst): | ||||
|         """get existing, or add new path to the vfs""" | ||||
| @@ -32,11 +86,13 @@ class VFS(object): | ||||
|                 return self.nodes[name].add(src, dst) | ||||
|  | ||||
|             vn = VFS( | ||||
|                 "{}/{}".format(self.realpath, name), | ||||
|                 self.log, | ||||
|                 os.path.join(self.realpath, name) if self.realpath else None, | ||||
|                 "{}/{}".format(self.vpath, name).lstrip("/"), | ||||
|                 self.uread, | ||||
|                 self.uwrite, | ||||
|                 self.axs, | ||||
|                 self._copy_flags(name), | ||||
|             ) | ||||
|             vn.dbv = self.dbv or self | ||||
|             self.nodes[name] = vn | ||||
|             return vn.add(src, dst) | ||||
|  | ||||
| @@ -46,10 +102,28 @@ class VFS(object): | ||||
|  | ||||
|         # leaf does not exist; create and keep permissions blank | ||||
|         vp = "{}/{}".format(self.vpath, dst).lstrip("/") | ||||
|         vn = VFS(src, vp) | ||||
|         vn = VFS(self.log, src, vp, AXS(), {}) | ||||
|         vn.dbv = self.dbv or self | ||||
|         self.nodes[dst] = vn | ||||
|         return vn | ||||
|  | ||||
|     def _copy_flags(self, name): | ||||
|         flags = {k: v for k, v in self.flags.items()} | ||||
|         hist = flags.get("hist") | ||||
|         if hist and hist != "-": | ||||
|             flags["hist"] = "{}/{}".format(hist.rstrip("/"), name) | ||||
|  | ||||
|         return flags | ||||
|  | ||||
|     def bubble_flags(self): | ||||
|         if self.dbv: | ||||
|             for k, v in self.dbv.flags.items(): | ||||
|                 if k not in ["hist"]: | ||||
|                     self.flags[k] = v | ||||
|  | ||||
|         for v in self.nodes.values(): | ||||
|             v.bubble_flags() | ||||
|  | ||||
|     def _find(self, vpath): | ||||
|         """return [vfs,remainder]""" | ||||
|         vpath = undot(vpath) | ||||
| @@ -68,88 +142,194 @@ class VFS(object): | ||||
|         return [self, vpath] | ||||
|  | ||||
|     def can_access(self, vpath, uname): | ||||
|         """return [readable,writable]""" | ||||
|         # type: (str, str) -> tuple[bool, bool, bool, bool] | ||||
|         """can Read,Write,Move,Delete""" | ||||
|         vn, _ = self._find(vpath) | ||||
|         c = vn.axs | ||||
|         return [ | ||||
|             uname in vn.uread or "*" in vn.uread, | ||||
|             uname in vn.uwrite or "*" in vn.uwrite, | ||||
|             uname in c.uread or "*" in c.uread, | ||||
|             uname in c.uwrite or "*" in c.uwrite, | ||||
|             uname in c.umove or "*" in c.umove, | ||||
|             uname in c.udel or "*" in c.udel, | ||||
|         ] | ||||
|  | ||||
|     def get(self, vpath, uname, will_read, will_write): | ||||
|     def get(self, vpath, uname, will_read, will_write, will_move=False, will_del=False): | ||||
|         # type: (str, str, bool, bool, bool, bool) -> tuple[VFS, str] | ||||
|         """returns [vfsnode,fs_remainder] if user has the requested permissions""" | ||||
|         vn, rem = self._find(vpath) | ||||
|         c = vn.axs | ||||
|  | ||||
|         if will_read and (uname not in vn.uread and "*" not in vn.uread): | ||||
|             raise Pebkac(403, "you don't have read-access for this location") | ||||
|  | ||||
|         if will_write and (uname not in vn.uwrite and "*" not in vn.uwrite): | ||||
|             raise Pebkac(403, "you don't have write-access for this location") | ||||
|         for req, d, msg in [ | ||||
|             [will_read, c.uread, "read"], | ||||
|             [will_write, c.uwrite, "write"], | ||||
|             [will_move, c.umove, "move"], | ||||
|             [will_del, c.udel, "delete"], | ||||
|         ]: | ||||
|             if req and (uname not in d and "*" not in d): | ||||
|                 m = "you don't have {}-access for this location" | ||||
|                 raise Pebkac(403, m.format(msg)) | ||||
|  | ||||
|         return vn, rem | ||||
|  | ||||
|     def canonical(self, rem): | ||||
|     def get_dbv(self, vrem): | ||||
|         dbv = self.dbv | ||||
|         if not dbv: | ||||
|             return self, vrem | ||||
|  | ||||
|         vrem = [self.vpath[len(dbv.vpath) + 1 :], vrem] | ||||
|         vrem = "/".join([x for x in vrem if x]) | ||||
|         return dbv, vrem | ||||
|  | ||||
|     def canonical(self, rem, resolve=True): | ||||
|         """returns the canonical path (fully-resolved absolute fs path)""" | ||||
|         rp = self.realpath | ||||
|         if rem: | ||||
|             rp += "/" + rem | ||||
|  | ||||
|         return fsdec(os.path.realpath(fsenc(rp))) | ||||
|         return absreal(rp) if resolve else rp | ||||
|  | ||||
|     def ls(self, rem, uname): | ||||
|     def ls(self, rem, uname, scandir, permsets, lstat=False): | ||||
|         # type: (str, str, bool, list[list[bool]], bool) -> tuple[str, str, dict[str, VFS]] | ||||
|         """return user-readable [fsdir,real,virt] items at vpath""" | ||||
|         virt_vis = {}  # nodes readable by user | ||||
|         abspath = self.canonical(rem) | ||||
|         items = os.listdir(fsenc(abspath)) | ||||
|         real = [fsdec(x) for x in items] | ||||
|         real = list(statdir(self.log, scandir, lstat, abspath)) | ||||
|         real.sort() | ||||
|         if not rem: | ||||
|             for name, vn2 in sorted(self.nodes.items()): | ||||
|                 if uname in vn2.uread: | ||||
|                     virt_vis[name] = vn2 | ||||
|  | ||||
|             # no vfs nodes in the list of real inodes | ||||
|             real = [x for x in real if x not in self.nodes] | ||||
|             real = [x for x in real if x[0] not in self.nodes] | ||||
|  | ||||
|             for name, vn2 in sorted(self.nodes.items()): | ||||
|                 ok = False | ||||
|                 axs = vn2.axs | ||||
|                 axs = [axs.uread, axs.uwrite, axs.umove, axs.udel] | ||||
|                 for pset in permsets: | ||||
|                     ok = True | ||||
|                     for req, lst in zip(pset, axs): | ||||
|                         if req and uname not in lst and "*" not in lst: | ||||
|                             ok = False | ||||
|                     if ok: | ||||
|                         break | ||||
|  | ||||
|                 if ok: | ||||
|                     virt_vis[name] = vn2 | ||||
|  | ||||
|         return [abspath, real, virt_vis] | ||||
|  | ||||
|     def user_tree(self, uname, readable=False, writable=False): | ||||
|         ret = [] | ||||
|         opt1 = readable and (uname in self.uread or "*" in self.uread) | ||||
|         opt2 = writable and (uname in self.uwrite or "*" in self.uwrite) | ||||
|         if opt1 or opt2: | ||||
|             ret.append(self.vpath) | ||||
|     def walk(self, rel, rem, seen, uname, permsets, dots, scandir, lstat): | ||||
|         """ | ||||
|         recursively yields from ./rem; | ||||
|         rel is a unix-style user-defined vpath (not vfs-related) | ||||
|         """ | ||||
|  | ||||
|         for _, vn in sorted(self.nodes.items()): | ||||
|             ret.extend(vn.user_tree(uname, readable, writable)) | ||||
|         fsroot, vfs_ls, vfs_virt = self.ls(rem, uname, scandir, permsets, lstat=lstat) | ||||
|         dbv, vrem = self.get_dbv(rem) | ||||
|  | ||||
|         return ret | ||||
|         if ( | ||||
|             seen | ||||
|             and (not fsroot.startswith(seen[-1]) or fsroot == seen[-1]) | ||||
|             and fsroot in seen | ||||
|         ): | ||||
|             m = "bailing from symlink loop,\n  prev: {}\n  curr: {}\n  from: {}/{}" | ||||
|             self.log("vfs.walk", m.format(seen[-1], fsroot, self.vpath, rem), 3) | ||||
|             return | ||||
|  | ||||
|         seen = seen[:] + [fsroot] | ||||
|         rfiles = [x for x in vfs_ls if not stat.S_ISDIR(x[1].st_mode)] | ||||
|         rdirs = [x for x in vfs_ls if stat.S_ISDIR(x[1].st_mode)] | ||||
|  | ||||
|         rfiles.sort() | ||||
|         rdirs.sort() | ||||
|  | ||||
|         yield dbv, vrem, rel, fsroot, rfiles, rdirs, vfs_virt | ||||
|  | ||||
|         for rdir, _ in rdirs: | ||||
|             if not dots and rdir.startswith("."): | ||||
|                 continue | ||||
|  | ||||
|             wrel = (rel + "/" + rdir).lstrip("/") | ||||
|             wrem = (rem + "/" + rdir).lstrip("/") | ||||
|             for x in self.walk(wrel, wrem, seen, uname, permsets, dots, scandir, lstat): | ||||
|                 yield x | ||||
|  | ||||
|         for n, vfs in sorted(vfs_virt.items()): | ||||
|             if not dots and n.startswith("."): | ||||
|                 continue | ||||
|  | ||||
|             wrel = (rel + "/" + n).lstrip("/") | ||||
|             for x in vfs.walk(wrel, "", seen, uname, permsets, dots, scandir, lstat): | ||||
|                 yield x | ||||
|  | ||||
|     def zipgen(self, vrem, flt, uname, dots, scandir): | ||||
|         if flt: | ||||
|             flt = {k: True for k in flt} | ||||
|  | ||||
|         f1 = "{0}.hist{0}up2k.".format(os.sep) | ||||
|         f2a = os.sep + "dir.txt" | ||||
|         f2b = "{0}.hist{0}".format(os.sep) | ||||
|  | ||||
|         g = self.walk("", vrem, [], uname, [[True]], dots, scandir, False) | ||||
|         for _, _, vpath, apath, files, rd, vd in g: | ||||
|             if flt: | ||||
|                 files = [x for x in files if x[0] in flt] | ||||
|  | ||||
|                 rm = [x for x in rd if x[0] not in flt] | ||||
|                 [rd.remove(x) for x in rm] | ||||
|  | ||||
|                 rm = [x for x in vd.keys() if x not in flt] | ||||
|                 [vd.pop(x) for x in rm] | ||||
|  | ||||
|                 flt = None | ||||
|  | ||||
|             # print(repr([vpath, apath, [x[0] for x in files]])) | ||||
|             fnames = [n[0] for n in files] | ||||
|             vpaths = [vpath + "/" + n for n in fnames] if vpath else fnames | ||||
|             apaths = [os.path.join(apath, n) for n in fnames] | ||||
|             files = list(zip(vpaths, apaths, files)) | ||||
|  | ||||
|             if not dots: | ||||
|                 # dotfile filtering based on vpath (intended visibility) | ||||
|                 files = [x for x in files if "/." not in "/" + x[0]] | ||||
|  | ||||
|                 rm = [x for x in rd if x[0].startswith(".")] | ||||
|                 for x in rm: | ||||
|                     rd.remove(x) | ||||
|  | ||||
|                 rm = [k for k in vd.keys() if k.startswith(".")] | ||||
|                 for x in rm: | ||||
|                     del vd[x] | ||||
|  | ||||
|             # up2k filetring based on actual abspath | ||||
|             files = [ | ||||
|                 x | ||||
|                 for x in files | ||||
|                 if f1 not in x[1] and (not x[1].endswith(f2a) or f2b not in x[1]) | ||||
|             ] | ||||
|  | ||||
|             for f in [{"vp": v, "ap": a, "st": n[1]} for v, a, n in files]: | ||||
|                 yield f | ||||
|  | ||||
|  | ||||
| class AuthSrv(object): | ||||
|     """verifies users against given paths""" | ||||
|  | ||||
|     def __init__(self, args, log_func): | ||||
|         self.log_func = log_func | ||||
|     def __init__(self, args, log_func, warn_anonwrite=True): | ||||
|         self.args = args | ||||
|  | ||||
|         self.warn_anonwrite = True | ||||
|         self.log_func = log_func | ||||
|         self.warn_anonwrite = warn_anonwrite | ||||
|         self.line_ctr = 0 | ||||
|  | ||||
|         if WINDOWS: | ||||
|             self.re_vol = re.compile(r"^([a-zA-Z]:[\\/][^:]*|[^:]*):([^:]*):(.*)") | ||||
|             self.re_vol = re.compile(r"^([a-zA-Z]:[\\/][^:]*|[^:]*):([^:]*):(.*)$") | ||||
|         else: | ||||
|             self.re_vol = re.compile(r"^([^:]*):([^:]*):(.*)") | ||||
|             self.re_vol = re.compile(r"^([^:]*):([^:]*):(.*)$") | ||||
|  | ||||
|         self.mutex = threading.Lock() | ||||
|         self.reload() | ||||
|  | ||||
|     def log(self, msg): | ||||
|         self.log_func("auth", msg) | ||||
|  | ||||
|     def invert(self, orig): | ||||
|         if PY2: | ||||
|             return {v: k for k, v in orig.iteritems()} | ||||
|         else: | ||||
|             return {v: k for k, v in orig.items()} | ||||
|     def log(self, msg, c=0): | ||||
|         if self.log_func: | ||||
|             self.log_func("auth", msg, c) | ||||
|  | ||||
|     def laggy_iter(self, iterable): | ||||
|         """returns [value,isFinalValue]""" | ||||
| @@ -161,10 +341,13 @@ class AuthSrv(object): | ||||
|  | ||||
|         yield prev, True | ||||
|  | ||||
|     def _parse_config_file(self, fd, user, mread, mwrite, mount): | ||||
|     def _parse_config_file(self, fd, acct, daxs, mflags, mount): | ||||
|         # type: (any, str, dict[str, AXS], any, str) -> None | ||||
|         vol_src = None | ||||
|         vol_dst = None | ||||
|         self.line_ctr = 0 | ||||
|         for ln in [x.decode("utf-8").strip() for x in fd]: | ||||
|             self.line_ctr += 1 | ||||
|             if not ln and vol_src is not None: | ||||
|                 vol_src = None | ||||
|                 vol_dst = None | ||||
| @@ -175,7 +358,7 @@ class AuthSrv(object): | ||||
|             if vol_src is None: | ||||
|                 if ln.startswith("u "): | ||||
|                     u, p = ln[2:].split(":", 1) | ||||
|                     user[u] = p | ||||
|                     acct[u] = p | ||||
|                 else: | ||||
|                     vol_src = ln | ||||
|                 continue | ||||
| @@ -186,18 +369,62 @@ class AuthSrv(object): | ||||
|                     raise Exception('invalid mountpoint "{}"'.format(vol_dst)) | ||||
|  | ||||
|                 # cfg files override arguments and previous files | ||||
|                 vol_src = fsdec(os.path.abspath(fsenc(vol_src))) | ||||
|                 vol_src = bos.path.abspath(vol_src) | ||||
|                 vol_dst = vol_dst.strip("/") | ||||
|                 mount[vol_dst] = vol_src | ||||
|                 mread[vol_dst] = [] | ||||
|                 mwrite[vol_dst] = [] | ||||
|                 daxs[vol_dst] = AXS() | ||||
|                 mflags[vol_dst] = {} | ||||
|                 continue | ||||
|  | ||||
|             lvl, uname = ln.split(" ") | ||||
|             if lvl in "ra": | ||||
|                 mread[vol_dst].append(uname) | ||||
|             if lvl in "wa": | ||||
|                 mwrite[vol_dst].append(uname) | ||||
|             try: | ||||
|                 lvl, uname = ln.split(" ", 1) | ||||
|             except: | ||||
|                 lvl = ln | ||||
|                 uname = "*" | ||||
|  | ||||
|             if lvl == "a": | ||||
|                 m = "WARNING (config-file): permission flag 'a' is deprecated; please use 'rw' instead" | ||||
|                 self.log(m, 1) | ||||
|  | ||||
|             self._read_vol_str(lvl, uname, daxs[vol_dst], mflags[vol_dst]) | ||||
|  | ||||
|     def _read_vol_str(self, lvl, uname, axs, flags): | ||||
|         # type: (str, str, AXS, any) -> None | ||||
|         if lvl == "c": | ||||
|             cval = True | ||||
|             if "=" in uname: | ||||
|                 uname, cval = uname.split("=", 1) | ||||
|  | ||||
|             self._read_volflag(flags, uname, cval, False) | ||||
|             return | ||||
|  | ||||
|         if uname == "": | ||||
|             uname = "*" | ||||
|  | ||||
|         for un in uname.split(","): | ||||
|             if "r" in lvl: | ||||
|                 axs.uread[un] = 1 | ||||
|  | ||||
|             if "w" in lvl: | ||||
|                 axs.uwrite[un] = 1 | ||||
|  | ||||
|             if "m" in lvl: | ||||
|                 axs.umove[un] = 1 | ||||
|  | ||||
|             if "d" in lvl: | ||||
|                 axs.udel[un] = 1 | ||||
|  | ||||
|     def _read_volflag(self, flags, name, value, is_list): | ||||
|         if name not in ["mtp"]: | ||||
|             flags[name] = value | ||||
|             return | ||||
|  | ||||
|         if not is_list: | ||||
|             value = [value] | ||||
|         elif not value: | ||||
|             return | ||||
|  | ||||
|         flags[name] = flags.get(name, []) + value | ||||
|  | ||||
|     def reload(self): | ||||
|         """ | ||||
| @@ -207,51 +434,70 @@ class AuthSrv(object): | ||||
|         before finally building the VFS | ||||
|         """ | ||||
|  | ||||
|         user = {}  # username:password | ||||
|         mread = {}  # mountpoint:[username] | ||||
|         mwrite = {}  # mountpoint:[username] | ||||
|         acct = {}  # username:password | ||||
|         daxs = {}  # type: dict[str, AXS] | ||||
|         mflags = {}  # mountpoint:[flag] | ||||
|         mount = {}  # dst:src (mountpoint:realpath) | ||||
|  | ||||
|         if self.args.a: | ||||
|             # list of username:password | ||||
|             for u, p in [x.split(":", 1) for x in self.args.a]: | ||||
|                 user[u] = p | ||||
|             for x in self.args.a: | ||||
|                 try: | ||||
|                     u, p = x.split(":", 1) | ||||
|                     acct[u] = p | ||||
|                 except: | ||||
|                     m = '\n  invalid value "{}" for argument -a, must be username:password' | ||||
|                     raise Exception(m.format(x)) | ||||
|  | ||||
|         if self.args.v: | ||||
|             # list of src:dst:permset:permset:... | ||||
|             # permset is [rwa]username | ||||
|             for vol_match in [self.re_vol.match(x) for x in self.args.v]: | ||||
|                 try: | ||||
|                     src, dst, perms = vol_match.groups() | ||||
|                 except: | ||||
|                     raise Exception("invalid -v argument") | ||||
|             # permset is <rwmd>[,username][,username] or <c>,<flag>[=args] | ||||
|             for v_str in self.args.v: | ||||
|                 m = self.re_vol.match(v_str) | ||||
|                 if not m: | ||||
|                     raise Exception("invalid -v argument: [{}]".format(v_str)) | ||||
|  | ||||
|                 src = fsdec(os.path.abspath(fsenc(src))) | ||||
|                 src, dst, perms = m.groups() | ||||
|                 if WINDOWS: | ||||
|                     src = uncyg(src) | ||||
|  | ||||
|                 # print("\n".join([src, dst, perms])) | ||||
|                 src = bos.path.abspath(src) | ||||
|                 dst = dst.strip("/") | ||||
|                 mount[dst] = src | ||||
|                 mread[dst] = [] | ||||
|                 mwrite[dst] = [] | ||||
|                 daxs[dst] = AXS() | ||||
|                 mflags[dst] = {} | ||||
|  | ||||
|                 perms = perms.split(":") | ||||
|                 for (lvl, uname) in [[x[0], x[1:]] for x in perms]: | ||||
|                     if uname == "": | ||||
|                         uname = "*" | ||||
|                     if lvl in "ra": | ||||
|                         mread[dst].append(uname) | ||||
|                     if lvl in "wa": | ||||
|                         mwrite[dst].append(uname) | ||||
|                 for x in perms.split(":"): | ||||
|                     lvl, uname = x.split(",", 1) if "," in x else [x, ""] | ||||
|                     self._read_vol_str(lvl, uname, daxs[dst], mflags[dst]) | ||||
|  | ||||
|         if self.args.c: | ||||
|             for cfg_fn in self.args.c: | ||||
|                 with open(cfg_fn, "rb") as f: | ||||
|                     self._parse_config_file(f, user, mread, mwrite, mount) | ||||
|                     try: | ||||
|                         self._parse_config_file(f, acct, daxs, mflags, mount) | ||||
|                     except: | ||||
|                         m = "\n\033[1;31m\nerror in config file {} on line {}:\n\033[0m" | ||||
|                         self.log(m.format(cfg_fn, self.line_ctr), 1) | ||||
|                         raise | ||||
|  | ||||
|         # case-insensitive; normalize | ||||
|         if WINDOWS: | ||||
|             cased = {} | ||||
|             for k, v in mount.items(): | ||||
|                 cased[k] = absreal(v) | ||||
|  | ||||
|             mount = cased | ||||
|  | ||||
|         if not mount: | ||||
|             # -h says our defaults are CWD at root and read/write for everyone | ||||
|             vfs = VFS(os.path.abspath("."), "", ["*"], ["*"]) | ||||
|             axs = AXS(["*"], ["*"], None, None) | ||||
|             vfs = VFS(self.log_func, bos.path.abspath("."), "", axs, {}) | ||||
|         elif "" not in mount: | ||||
|             # there's volumes but no root; make root inaccessible | ||||
|             vfs = VFS(os.path.abspath("."), "", [], []) | ||||
|             vfs = VFS(self.log_func, None, "", AXS(), {}) | ||||
|             vfs.flags["d2d"] = True | ||||
|  | ||||
|         maxdepth = 0 | ||||
|         for dst in sorted(mount.keys(), key=lambda x: (x.count("/"), len(x))): | ||||
| @@ -261,44 +507,321 @@ class AuthSrv(object): | ||||
|  | ||||
|             if dst == "": | ||||
|                 # rootfs was mapped; fully replaces the default CWD vfs | ||||
|                 vfs = VFS(mount[dst], dst, mread[dst], mwrite[dst]) | ||||
|                 vfs = VFS(self.log_func, mount[dst], dst, daxs[dst], mflags[dst]) | ||||
|                 continue | ||||
|  | ||||
|             v = vfs.add(mount[dst], dst) | ||||
|             v.uread = mread[dst] | ||||
|             v.uwrite = mwrite[dst] | ||||
|             v.axs = daxs[dst] | ||||
|             v.flags = mflags[dst] | ||||
|             v.dbv = None | ||||
|  | ||||
|         vfs.all_vols = {} | ||||
|         vfs.get_all_vols(vfs.all_vols) | ||||
|  | ||||
|         for perm in "read write move del".split(): | ||||
|             axs_key = "u" + perm | ||||
|             unames = ["*"] + list(acct.keys()) | ||||
|             umap = {x: [] for x in unames} | ||||
|             for usr in unames: | ||||
|                 for mp, vol in vfs.all_vols.items(): | ||||
|                     if usr in getattr(vol.axs, axs_key): | ||||
|                         umap[usr].append(mp) | ||||
|             setattr(vfs, "a" + perm, umap) | ||||
|  | ||||
|         all_users = {} | ||||
|         missing_users = {} | ||||
|         for d in [mread, mwrite]: | ||||
|             for _, ul in d.items(): | ||||
|                 for usr in ul: | ||||
|                     if usr != "*" and usr not in user: | ||||
|         for axs in daxs.values(): | ||||
|             for d in [axs.uread, axs.uwrite, axs.umove, axs.udel]: | ||||
|                 for usr in d.keys(): | ||||
|                     all_users[usr] = 1 | ||||
|                     if usr != "*" and usr not in acct: | ||||
|                         missing_users[usr] = 1 | ||||
|  | ||||
|         if missing_users: | ||||
|             self.log( | ||||
|                 "\033[31myou must -a the following users: " | ||||
|                 + ", ".join(k for k in sorted(missing_users)) | ||||
|                 + "\033[0m" | ||||
|                 "you must -a the following users: " | ||||
|                 + ", ".join(k for k in sorted(missing_users)), | ||||
|                 c=1, | ||||
|             ) | ||||
|             raise Exception("invalid config") | ||||
|  | ||||
|         promote = [] | ||||
|         demote = [] | ||||
|         for vol in vfs.all_vols.values(): | ||||
|             hid = hashlib.sha512(fsenc(vol.realpath)).digest() | ||||
|             hid = base64.b32encode(hid).decode("ascii").lower() | ||||
|             vflag = vol.flags.get("hist") | ||||
|             if vflag == "-": | ||||
|                 pass | ||||
|             elif vflag: | ||||
|                 vol.histpath = uncyg(vflag) if WINDOWS else vflag | ||||
|             elif self.args.hist: | ||||
|                 for nch in range(len(hid)): | ||||
|                     hpath = os.path.join(self.args.hist, hid[: nch + 1]) | ||||
|                     bos.makedirs(hpath) | ||||
|  | ||||
|                     powner = os.path.join(hpath, "owner.txt") | ||||
|                     try: | ||||
|                         with open(powner, "rb") as f: | ||||
|                             owner = f.read().rstrip() | ||||
|                     except: | ||||
|                         owner = None | ||||
|  | ||||
|                     me = fsenc(vol.realpath).rstrip() | ||||
|                     if owner not in [None, me]: | ||||
|                         continue | ||||
|  | ||||
|                     if owner is None: | ||||
|                         with open(powner, "wb") as f: | ||||
|                             f.write(me) | ||||
|  | ||||
|                     vol.histpath = hpath | ||||
|                     break | ||||
|  | ||||
|             vol.histpath = absreal(vol.histpath) | ||||
|             if vol.dbv: | ||||
|                 if bos.path.exists(os.path.join(vol.histpath, "up2k.db")): | ||||
|                     promote.append(vol) | ||||
|                     vol.dbv = None | ||||
|                 else: | ||||
|                     demote.append(vol) | ||||
|  | ||||
|         # discard jump-vols | ||||
|         for v in demote: | ||||
|             vfs.all_vols.pop(v.vpath) | ||||
|  | ||||
|         if promote: | ||||
|             msg = [ | ||||
|                 "\n  the following jump-volumes were generated to assist the vfs.\n  As they contain a database (probably from v0.11.11 or older),\n  they are promoted to full volumes:" | ||||
|             ] | ||||
|             for vol in promote: | ||||
|                 msg.append( | ||||
|                     "  /{}  ({})  ({})".format(vol.vpath, vol.realpath, vol.histpath) | ||||
|                 ) | ||||
|  | ||||
|             self.log("\n\n".join(msg) + "\n", c=3) | ||||
|  | ||||
|         vfs.histtab = {v.realpath: v.histpath for v in vfs.all_vols.values()} | ||||
|  | ||||
|         all_mte = {} | ||||
|         errors = False | ||||
|         for vol in vfs.all_vols.values(): | ||||
|             if (self.args.e2ds and vol.axs.uwrite) or self.args.e2dsa: | ||||
|                 vol.flags["e2ds"] = True | ||||
|  | ||||
|             if self.args.e2d or "e2ds" in vol.flags: | ||||
|                 vol.flags["e2d"] = True | ||||
|  | ||||
|             if self.args.no_hash: | ||||
|                 if "ehash" not in vol.flags: | ||||
|                     vol.flags["dhash"] = True | ||||
|  | ||||
|             for k in ["e2t", "e2ts", "e2tsr"]: | ||||
|                 if getattr(self.args, k): | ||||
|                     vol.flags[k] = True | ||||
|  | ||||
|             for k1, k2 in IMPLICATIONS: | ||||
|                 if k1 in vol.flags: | ||||
|                     vol.flags[k2] = True | ||||
|  | ||||
|             # default tag-list if unset | ||||
|             if "mte" not in vol.flags: | ||||
|                 vol.flags["mte"] = self.args.mte | ||||
|  | ||||
|             # append parsers from argv to volume-flags | ||||
|             self._read_volflag(vol.flags, "mtp", self.args.mtp, True) | ||||
|  | ||||
|             # d2d drops all database features for a volume | ||||
|             for grp, rm in [["d2d", "e2d"], ["d2t", "e2t"]]: | ||||
|                 if not vol.flags.get(grp, False): | ||||
|                     continue | ||||
|  | ||||
|                 vol.flags["d2t"] = True | ||||
|                 vol.flags = {k: v for k, v in vol.flags.items() if not k.startswith(rm)} | ||||
|  | ||||
|             # mt* needs e2t so drop those too | ||||
|             for grp, rm in [["e2t", "mt"]]: | ||||
|                 if vol.flags.get(grp, False): | ||||
|                     continue | ||||
|  | ||||
|                 vol.flags = {k: v for k, v in vol.flags.items() if not k.startswith(rm)} | ||||
|  | ||||
|             # verify tags mentioned by -mt[mp] are used by -mte | ||||
|             local_mtp = {} | ||||
|             local_only_mtp = {} | ||||
|             tags = vol.flags.get("mtp", []) + vol.flags.get("mtm", []) | ||||
|             tags = [x.split("=")[0] for x in tags] | ||||
|             tags = [y for x in tags for y in x.split(",")] | ||||
|             for a in tags: | ||||
|                 local_mtp[a] = True | ||||
|                 local = True | ||||
|                 for b in self.args.mtp or []: | ||||
|                     b = b.split("=")[0] | ||||
|                     if a == b: | ||||
|                         local = False | ||||
|  | ||||
|                 if local: | ||||
|                     local_only_mtp[a] = True | ||||
|  | ||||
|             local_mte = {} | ||||
|             for a in vol.flags.get("mte", "").split(","): | ||||
|                 local = True | ||||
|                 all_mte[a] = True | ||||
|                 local_mte[a] = True | ||||
|                 for b in self.args.mte.split(","): | ||||
|                     if not a or not b: | ||||
|                         continue | ||||
|  | ||||
|                     if a == b: | ||||
|                         local = False | ||||
|  | ||||
|             for mtp in local_only_mtp.keys(): | ||||
|                 if mtp not in local_mte: | ||||
|                     m = 'volume "/{}" defines metadata tag "{}", but doesnt use it in "-mte" (or with "cmte" in its volume-flags)' | ||||
|                     self.log(m.format(vol.vpath, mtp), 1) | ||||
|                     errors = True | ||||
|  | ||||
|         tags = self.args.mtp or [] | ||||
|         tags = [x.split("=")[0] for x in tags] | ||||
|         tags = [y for x in tags for y in x.split(",")] | ||||
|         for mtp in tags: | ||||
|             if mtp not in all_mte: | ||||
|                 m = 'metadata tag "{}" is defined by "-mtm" or "-mtp", but is not used by "-mte" (or by any "cmte" volume-flag)' | ||||
|                 self.log(m.format(mtp), 1) | ||||
|                 errors = True | ||||
|  | ||||
|         if errors: | ||||
|             sys.exit(1) | ||||
|  | ||||
|         vfs.bubble_flags() | ||||
|  | ||||
|         m = "volumes and permissions:\n" | ||||
|         for v in vfs.all_vols.values(): | ||||
|             if not self.warn_anonwrite: | ||||
|                 break | ||||
|  | ||||
|             m += '\n\033[36m"/{}"  \033[33m{}\033[0m'.format(v.vpath, v.realpath) | ||||
|             for txt, attr in [ | ||||
|                 ["  read", "uread"], | ||||
|                 [" write", "uwrite"], | ||||
|                 ["  move", "umove"], | ||||
|                 ["delete", "udel"], | ||||
|             ]: | ||||
|                 u = list(sorted(getattr(v.axs, attr).keys())) | ||||
|                 u = ", ".join("\033[35meverybody\033[0m" if x == "*" else x for x in u) | ||||
|                 u = u if u else "\033[36m--none--\033[0m" | ||||
|                 m += "\n|  {}:  {}".format(txt, u) | ||||
|             m += "\n" | ||||
|  | ||||
|         if self.warn_anonwrite and not self.args.no_voldump: | ||||
|             self.log(m) | ||||
|  | ||||
|         try: | ||||
|             v, _ = vfs.get("/", "*", False, True) | ||||
|             if self.warn_anonwrite and os.getcwd() == v.realpath: | ||||
|                 self.warn_anonwrite = False | ||||
|                 self.log( | ||||
|                     "\033[31manyone can read/write the current directory: {}\033[0m".format( | ||||
|                         v.realpath | ||||
|                     ) | ||||
|                 ) | ||||
|                 msg = "anyone can read/write the current directory: {}" | ||||
|                 self.log(msg.format(v.realpath), c=1) | ||||
|         except Pebkac: | ||||
|             self.warn_anonwrite = True | ||||
|  | ||||
|         with self.mutex: | ||||
|             self.vfs = vfs | ||||
|             self.user = user | ||||
|             self.iuser = self.invert(user) | ||||
|             self.acct = acct | ||||
|             self.iacct = {v: k for k, v in acct.items()} | ||||
|  | ||||
|         # import pprint | ||||
|         # pprint.pprint({"usr": user, "rd": mread, "wr": mwrite, "mnt": mount}) | ||||
|             self.re_pwd = None | ||||
|             pwds = [re.escape(x) for x in self.iacct.keys()] | ||||
|             if pwds: | ||||
|                 self.re_pwd = re.compile("=(" + "|".join(pwds) + ")([]&; ]|$)") | ||||
|  | ||||
|     def dbg_ls(self): | ||||
|         users = self.args.ls | ||||
|         vols = "*" | ||||
|         flags = [] | ||||
|  | ||||
|         try: | ||||
|             users, vols = users.split(",", 1) | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         try: | ||||
|             vols, flags = vols.split(",", 1) | ||||
|             flags = flags.split(",") | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         if users == "**": | ||||
|             users = list(self.acct.keys()) + ["*"] | ||||
|         else: | ||||
|             users = [users] | ||||
|  | ||||
|         for u in users: | ||||
|             if u not in self.acct and u != "*": | ||||
|                 raise Exception("user not found: " + u) | ||||
|  | ||||
|         if vols == "*": | ||||
|             vols = ["/" + x for x in self.vfs.all_vols.keys()] | ||||
|         else: | ||||
|             vols = [vols] | ||||
|  | ||||
|         for v in vols: | ||||
|             if not v.startswith("/"): | ||||
|                 raise Exception("volumes must start with /") | ||||
|  | ||||
|             if v[1:] not in self.vfs.all_vols: | ||||
|                 raise Exception("volume not found: " + v) | ||||
|  | ||||
|         self.log({"users": users, "vols": vols, "flags": flags}) | ||||
|         m = "/{}: read({}) write({}) move({}) del({})" | ||||
|         for k, v in self.vfs.all_vols.items(): | ||||
|             vc = v.axs | ||||
|             self.log(m.format(k, vc.uread, vc.uwrite, vc.umove, vc.udel)) | ||||
|  | ||||
|         flag_v = "v" in flags | ||||
|         flag_ln = "ln" in flags | ||||
|         flag_p = "p" in flags | ||||
|         flag_r = "r" in flags | ||||
|  | ||||
|         n_bads = 0 | ||||
|         for v in vols: | ||||
|             v = v[1:] | ||||
|             vtop = "/{}/".format(v) if v else "/" | ||||
|             for u in users: | ||||
|                 self.log("checking /{} as {}".format(v, u)) | ||||
|                 try: | ||||
|                     vn, _ = self.vfs.get(v, u, True, False, False, False) | ||||
|                 except: | ||||
|                     continue | ||||
|  | ||||
|                 atop = vn.realpath | ||||
|                 g = vn.walk( | ||||
|                     vn.vpath, "", [], u, [[True]], True, not self.args.no_scandir, False | ||||
|                 ) | ||||
|                 for _, _, vpath, apath, files, _, _ in g: | ||||
|                     fnames = [n[0] for n in files] | ||||
|                     vpaths = [vpath + "/" + n for n in fnames] if vpath else fnames | ||||
|                     vpaths = [vtop + x for x in vpaths] | ||||
|                     apaths = [os.path.join(apath, n) for n in fnames] | ||||
|                     files = [[vpath + "/", apath + os.sep]] + list(zip(vpaths, apaths)) | ||||
|  | ||||
|                     if flag_ln: | ||||
|                         files = [x for x in files if not x[1].startswith(atop + os.sep)] | ||||
|                         n_bads += len(files) | ||||
|  | ||||
|                     if flag_v: | ||||
|                         msg = [ | ||||
|                             '# user "{}", vpath "{}"\n{}'.format(u, vp, ap) | ||||
|                             for vp, ap in files | ||||
|                         ] | ||||
|                     else: | ||||
|                         msg = [x[1] for x in files] | ||||
|  | ||||
|                     if msg: | ||||
|                         self.log("\n" + "\n".join(msg)) | ||||
|  | ||||
|                 if n_bads and flag_p: | ||||
|                     raise Exception("found symlink leaving volume, and strict is set") | ||||
|  | ||||
|         if not flag_r: | ||||
|             sys.exit(0) | ||||
|   | ||||
							
								
								
									
										59
									
								
								copyparty/bos/bos.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								copyparty/bos/bos.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| from ..util import fsenc, fsdec | ||||
| from . import path | ||||
|  | ||||
|  | ||||
| # grep -hRiE '(^|[^a-zA-Z_\.-])os\.' . | gsed -r 's/ /\n/g;s/\(/(\n/g' | grep -hRiE '(^|[^a-zA-Z_\.-])os\.' | sort | uniq -c | ||||
| # printf 'os\.(%s)' "$(grep ^def bos/__init__.py | gsed -r 's/^def //;s/\(.*//' | tr '\n' '|' | gsed -r 's/.$//')" | ||||
|  | ||||
|  | ||||
| def chmod(p, mode): | ||||
|     return os.chmod(fsenc(p), mode) | ||||
|  | ||||
|  | ||||
| def listdir(p="."): | ||||
|     return [fsdec(x) for x in os.listdir(fsenc(p))] | ||||
|  | ||||
|  | ||||
| def lstat(p): | ||||
|     return os.lstat(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def makedirs(name, mode=0o755, exist_ok=True): | ||||
|     bname = fsenc(name) | ||||
|     try: | ||||
|         os.makedirs(bname, mode=mode) | ||||
|     except: | ||||
|         if not exist_ok or not os.path.isdir(bname): | ||||
|             raise | ||||
|  | ||||
|  | ||||
| def mkdir(p, mode=0o755): | ||||
|     return os.mkdir(fsenc(p), mode=mode) | ||||
|  | ||||
|  | ||||
| def rename(src, dst): | ||||
|     return os.rename(fsenc(src), fsenc(dst)) | ||||
|  | ||||
|  | ||||
| def replace(src, dst): | ||||
|     return os.replace(fsenc(src), fsenc(dst)) | ||||
|  | ||||
|  | ||||
| def rmdir(p): | ||||
|     return os.rmdir(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def stat(p): | ||||
|     return os.stat(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def unlink(p): | ||||
|     return os.unlink(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def utime(p, times=None): | ||||
|     return os.utime(fsenc(p), times) | ||||
							
								
								
									
										33
									
								
								copyparty/bos/path.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								copyparty/bos/path.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| from ..util import fsenc, fsdec | ||||
|  | ||||
|  | ||||
| def abspath(p): | ||||
|     return fsdec(os.path.abspath(fsenc(p))) | ||||
|  | ||||
|  | ||||
| def exists(p): | ||||
|     return os.path.exists(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def getmtime(p): | ||||
|     return os.path.getmtime(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def getsize(p): | ||||
|     return os.path.getsize(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def isdir(p): | ||||
|     return os.path.isdir(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def islink(p): | ||||
|     return os.path.islink(fsenc(p)) | ||||
|  | ||||
|  | ||||
| def realpath(p): | ||||
|     return fsdec(os.path.realpath(fsenc(p))) | ||||
| @@ -4,17 +4,11 @@ from __future__ import print_function, unicode_literals | ||||
| import time | ||||
| import threading | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, VT100 | ||||
| from .broker_util import try_exec | ||||
| from .broker_mpw import MpWorker | ||||
| from .util import mp | ||||
|  | ||||
|  | ||||
| if PY2 and not WINDOWS: | ||||
|     from multiprocessing.reduction import ForkingPickler | ||||
|     from StringIO import StringIO as MemesIO  # pylint: disable=import-error | ||||
|  | ||||
|  | ||||
| class BrokerMp(object): | ||||
|     """external api; manages MpWorkers""" | ||||
|  | ||||
| @@ -28,38 +22,33 @@ class BrokerMp(object): | ||||
|         self.retpend_mutex = threading.Lock() | ||||
|         self.mutex = threading.Lock() | ||||
|  | ||||
|         cores = self.args.j | ||||
|         if cores is None: | ||||
|             cores = mp.cpu_count() | ||||
|  | ||||
|         self.log("broker", "booting {} subprocesses".format(cores)) | ||||
|         for n in range(cores): | ||||
|         self.num_workers = self.args.j or mp.cpu_count() | ||||
|         self.log("broker", "booting {} subprocesses".format(self.num_workers)) | ||||
|         for n in range(1, self.num_workers + 1): | ||||
|             q_pend = mp.Queue(1) | ||||
|             q_yield = mp.Queue(64) | ||||
|  | ||||
|             proc = mp.Process(target=MpWorker, args=(q_pend, q_yield, self.args, n)) | ||||
|             proc.q_pend = q_pend | ||||
|             proc.q_yield = q_yield | ||||
|             proc.nid = n | ||||
|             proc.clients = {} | ||||
|             proc.workload = 0 | ||||
|  | ||||
|             thr = threading.Thread(target=self.collector, args=(proc,)) | ||||
|             thr = threading.Thread( | ||||
|                 target=self.collector, args=(proc,), name="mp-sink-{}".format(n) | ||||
|             ) | ||||
|             thr.daemon = True | ||||
|             thr.start() | ||||
|  | ||||
|             self.procs.append(proc) | ||||
|             proc.start() | ||||
|  | ||||
|         if True: | ||||
|             thr = threading.Thread(target=self.debug_load_balancer) | ||||
|             thr.daemon = True | ||||
|             thr.start() | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.log("broker", "shutting down") | ||||
|         for proc in self.procs: | ||||
|             thr = threading.Thread(target=proc.q_pend.put([0, "shutdown", []])) | ||||
|         for n, proc in enumerate(self.procs): | ||||
|             thr = threading.Thread( | ||||
|                 target=proc.q_pend.put([0, "shutdown", []]), | ||||
|                 name="mp-shutdown-{}-{}".format(n, len(self.procs)), | ||||
|             ) | ||||
|             thr.start() | ||||
|  | ||||
|         with self.mutex: | ||||
| @@ -82,20 +71,6 @@ class BrokerMp(object): | ||||
|             if dest == "log": | ||||
|                 self.log(*args) | ||||
|  | ||||
|             elif dest == "workload": | ||||
|                 with self.mutex: | ||||
|                     proc.workload = args[0] | ||||
|  | ||||
|             elif dest == "httpdrop": | ||||
|                 addr = args[0] | ||||
|  | ||||
|                 with self.mutex: | ||||
|                     del proc.clients[addr] | ||||
|                     if not proc.clients: | ||||
|                         proc.workload = 0 | ||||
|  | ||||
|                 self.hub.tcpsrv.num_clients.add(-1) | ||||
|  | ||||
|             elif dest == "retq": | ||||
|                 # response from previous ipc call | ||||
|                 with self.retpend_mutex: | ||||
| @@ -121,38 +96,12 @@ class BrokerMp(object): | ||||
|         returns a Queue object which eventually contains the response if want_retval | ||||
|         (not-impl here since nothing uses it yet) | ||||
|         """ | ||||
|         if dest == "httpconn": | ||||
|             sck, addr = args | ||||
|             sck2 = sck | ||||
|             if PY2: | ||||
|                 buf = MemesIO() | ||||
|                 ForkingPickler(buf).dump(sck) | ||||
|                 sck2 = buf.getvalue() | ||||
|         if dest == "listen": | ||||
|             for p in self.procs: | ||||
|                 p.q_pend.put([0, dest, [args[0], len(self.procs)]]) | ||||
|  | ||||
|             proc = sorted(self.procs, key=lambda x: x.workload)[0] | ||||
|             proc.q_pend.put([0, dest, [sck2, addr]]) | ||||
|  | ||||
|             with self.mutex: | ||||
|                 proc.clients[addr] = 50 | ||||
|                 proc.workload += 50 | ||||
|         elif dest == "cb_httpsrv_up": | ||||
|             self.hub.cb_httpsrv_up() | ||||
|  | ||||
|         else: | ||||
|             raise Exception("what is " + str(dest)) | ||||
|  | ||||
|     def debug_load_balancer(self): | ||||
|         fmt = "\033[1m{}\033[0;36m{:4}\033[0m " | ||||
|         if not VT100: | ||||
|             fmt = "({}{:4})" | ||||
|  | ||||
|         last = "" | ||||
|         while self.procs: | ||||
|             msg = "" | ||||
|             for proc in self.procs: | ||||
|                 msg += fmt.format(len(proc.clients), proc.workload) | ||||
|  | ||||
|             if msg != last: | ||||
|                 last = msg | ||||
|                 with self.hub.log_mutex: | ||||
|                     print(msg) | ||||
|  | ||||
|             time.sleep(0.1) | ||||
|   | ||||
| @@ -1,19 +1,15 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
| from copyparty.authsrv import AuthSrv | ||||
|  | ||||
| import sys | ||||
| import time | ||||
| import signal | ||||
| import threading | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS | ||||
| from .broker_util import ExceptionalQueue | ||||
| from .httpsrv import HttpSrv | ||||
| from .util import FAKE_MP | ||||
|  | ||||
| if PY2 and not WINDOWS: | ||||
|     import pickle  # nosec | ||||
|  | ||||
|  | ||||
| class MpWorker(object): | ||||
|     """one single mp instance""" | ||||
| @@ -24,39 +20,43 @@ class MpWorker(object): | ||||
|         self.args = args | ||||
|         self.n = n | ||||
|  | ||||
|         self.log = self._log_disabled if args.q and not args.lo else self._log_enabled | ||||
|  | ||||
|         self.retpend = {} | ||||
|         self.retpend_mutex = threading.Lock() | ||||
|         self.mutex = threading.Lock() | ||||
|         self.workload_thr_active = False | ||||
|  | ||||
|         # we inherited signal_handler from parent, | ||||
|         # replace it with something harmless | ||||
|         if not FAKE_MP: | ||||
|             signal.signal(signal.SIGINT, self.signal_handler) | ||||
|             for sig in [signal.SIGINT, signal.SIGTERM]: | ||||
|                 signal.signal(sig, self.signal_handler) | ||||
|  | ||||
|         # starting to look like a good idea | ||||
|         self.asrv = AuthSrv(args, None, False) | ||||
|  | ||||
|         # instantiate all services here (TODO: inheritance?) | ||||
|         self.httpsrv = HttpSrv(self) | ||||
|         self.httpsrv.disconnect_func = self.httpdrop | ||||
|         self.httpsrv = HttpSrv(self, n) | ||||
|  | ||||
|         # on winxp and some other platforms, | ||||
|         # use thr.join() to block all signals | ||||
|         thr = threading.Thread(target=self.main) | ||||
|         thr = threading.Thread(target=self.main, name="mpw-main") | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|         thr.join() | ||||
|  | ||||
|     def signal_handler(self, signal, frame): | ||||
|     def signal_handler(self, sig, frame): | ||||
|         # print('k') | ||||
|         pass | ||||
|  | ||||
|     def log(self, src, msg): | ||||
|         self.q_yield.put([0, "log", [src, msg]]) | ||||
|     def _log_enabled(self, src, msg, c=0): | ||||
|         self.q_yield.put([0, "log", [src, msg, c]]) | ||||
|  | ||||
|     def logw(self, msg): | ||||
|         self.log("mp{}".format(self.n), msg) | ||||
|     def _log_disabled(self, src, msg, c=0): | ||||
|         pass | ||||
|  | ||||
|     def httpdrop(self, addr): | ||||
|         self.q_yield.put([0, "httpdrop", [addr]]) | ||||
|     def logw(self, msg, c=0): | ||||
|         self.log("mp{}".format(self.n), msg, c) | ||||
|  | ||||
|     def main(self): | ||||
|         while True: | ||||
| @@ -64,24 +64,13 @@ class MpWorker(object): | ||||
|  | ||||
|             # self.logw("work: [{}]".format(d[0])) | ||||
|             if dest == "shutdown": | ||||
|                 self.httpsrv.shutdown() | ||||
|                 self.logw("ok bye") | ||||
|                 sys.exit(0) | ||||
|                 return | ||||
|  | ||||
|             elif dest == "httpconn": | ||||
|                 sck, addr = args | ||||
|                 if PY2: | ||||
|                     sck = pickle.loads(sck)  # nosec | ||||
|  | ||||
|                 self.log("%s %s" % addr, "-" * 4 + "C-qpop") | ||||
|                 self.httpsrv.accept(sck, addr) | ||||
|  | ||||
|                 with self.mutex: | ||||
|                     if not self.workload_thr_active: | ||||
|                         self.workload_thr_alive = True | ||||
|                         thr = threading.Thread(target=self.thr_workload) | ||||
|                         thr.daemon = True | ||||
|                         thr.start() | ||||
|             elif dest == "listen": | ||||
|                 self.httpsrv.listen(args[0], args[1]) | ||||
|  | ||||
|             elif dest == "retq": | ||||
|                 # response from previous ipc call | ||||
| @@ -105,16 +94,3 @@ class MpWorker(object): | ||||
|  | ||||
|         self.q_yield.put([retq_id, dest, args]) | ||||
|         return retq | ||||
|  | ||||
|     def thr_workload(self): | ||||
|         """announce workloads to MpSrv (the mp controller / loadbalancer)""" | ||||
|         # avoid locking in extract_filedata by tracking difference here | ||||
|         while True: | ||||
|             time.sleep(0.2) | ||||
|             with self.mutex: | ||||
|                 if self.httpsrv.num_clients() == 0: | ||||
|                     # no clients rn, termiante thread | ||||
|                     self.workload_thr_alive = False | ||||
|                     return | ||||
|  | ||||
|             self.q_yield.put([0, "workload", [self.httpsrv.workload]]) | ||||
|   | ||||
| @@ -14,22 +14,22 @@ class BrokerThr(object): | ||||
|         self.hub = hub | ||||
|         self.log = hub.log | ||||
|         self.args = hub.args | ||||
|         self.asrv = hub.asrv | ||||
|  | ||||
|         self.mutex = threading.Lock() | ||||
|         self.num_workers = 1 | ||||
|  | ||||
|         # instantiate all services here (TODO: inheritance?) | ||||
|         self.httpsrv = HttpSrv(self) | ||||
|         self.httpsrv.disconnect_func = self.httpdrop | ||||
|         self.httpsrv = HttpSrv(self, None) | ||||
|  | ||||
|     def shutdown(self): | ||||
|         # self.log("broker", "shutting down") | ||||
|         self.httpsrv.shutdown() | ||||
|         pass | ||||
|  | ||||
|     def put(self, want_retval, dest, *args): | ||||
|         if dest == "httpconn": | ||||
|             sck, addr = args | ||||
|             self.log("%s %s" % addr, "-" * 4 + "C-qpop") | ||||
|             self.httpsrv.accept(sck, addr) | ||||
|         if dest == "listen": | ||||
|             self.httpsrv.listen(args[0], 1) | ||||
|  | ||||
|         else: | ||||
|             # new ipc invoking managed service in hub | ||||
| @@ -46,6 +46,3 @@ class BrokerThr(object): | ||||
|             retq = ExceptionalQueue(1) | ||||
|             retq.put(rv) | ||||
|             return retq | ||||
|  | ||||
|     def httpdrop(self, addr): | ||||
|         self.hub.tcpsrv.num_clients.add(-1) | ||||
|   | ||||
| @@ -4,7 +4,6 @@ from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import traceback | ||||
|  | ||||
| from .__init__ import PY2 | ||||
| from .util import Pebkac, Queue | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										1567
									
								
								copyparty/httpcli.py
									
									
									
									
									
								
							
							
						
						
									
										1567
									
								
								copyparty/httpcli.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,29 +1,24 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import ssl | ||||
| import time | ||||
| import socket | ||||
|  | ||||
| HAVE_SSL = True | ||||
| try: | ||||
|     import jinja2 | ||||
| except ImportError: | ||||
|     print( | ||||
|         """\033[1;31m | ||||
|   you do not have jinja2 installed,\033[33m | ||||
|   choose one of these:\033[0m | ||||
|    * apt install python-jinja2 | ||||
|    * python3 -m pip install --user jinja2 | ||||
|    * (try another python version, if you have one) | ||||
|    * (try copyparty.sfx instead) | ||||
| """ | ||||
|     ) | ||||
|     sys.exit(1) | ||||
|     import ssl | ||||
| except: | ||||
|     HAVE_SSL = False | ||||
|  | ||||
| from .__init__ import E | ||||
| from .util import Unrecv | ||||
| from .httpcli import HttpCli | ||||
| from .u2idx import U2idx | ||||
| from .th_cli import ThumbCli | ||||
| from .th_srv import HAVE_PIL | ||||
| from .ico import Ico | ||||
|  | ||||
|  | ||||
| class HttpConn(object): | ||||
| @@ -38,30 +33,57 @@ class HttpConn(object): | ||||
|         self.hsrv = hsrv | ||||
|  | ||||
|         self.args = hsrv.args | ||||
|         self.auth = hsrv.auth | ||||
|         self.asrv = hsrv.asrv | ||||
|         self.cert_path = hsrv.cert_path | ||||
|  | ||||
|         self.workload = 0 | ||||
|         self.log_func = hsrv.log | ||||
|         self.log_src = "{} \033[36m{}".format(addr[0], addr[1]).ljust(26) | ||||
|         enth = HAVE_PIL and not self.args.no_thumb | ||||
|         self.thumbcli = ThumbCli(hsrv.broker) if enth else None | ||||
|         self.ico = Ico(self.args) | ||||
|  | ||||
|         env = jinja2.Environment() | ||||
|         env.loader = jinja2.FileSystemLoader(os.path.join(E.mod, "web")) | ||||
|         self.tpl_mounts = env.get_template("splash.html") | ||||
|         self.tpl_browser = env.get_template("browser.html") | ||||
|         self.tpl_msg = env.get_template("msg.html") | ||||
|         self.tpl_md = env.get_template("md.html") | ||||
|         self.tpl_mde = env.get_template("mde.html") | ||||
|         self.t0 = time.time() | ||||
|         self.stopping = False | ||||
|         self.nreq = 0 | ||||
|         self.nbyte = 0 | ||||
|         self.u2idx = None | ||||
|         self.log_func = hsrv.log | ||||
|         self.lf_url = re.compile(self.args.lf_url) if self.args.lf_url else None | ||||
|         self.set_rproxy() | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.stopping = True | ||||
|         try: | ||||
|             self.s.shutdown(socket.SHUT_RDWR) | ||||
|             self.s.close() | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def set_rproxy(self, ip=None): | ||||
|         if ip is None: | ||||
|             color = 36 | ||||
|             ip = self.addr[0] | ||||
|             self.rproxy = None | ||||
|         else: | ||||
|             color = 34 | ||||
|             self.rproxy = ip | ||||
|  | ||||
|         self.ip = ip | ||||
|         self.log_src = "{} \033[{}m{}".format(ip, color, self.addr[1]).ljust(26) | ||||
|         return self.log_src | ||||
|  | ||||
|     def respath(self, res_name): | ||||
|         return os.path.join(E.mod, "web", res_name) | ||||
|  | ||||
|     def log(self, msg): | ||||
|         self.log_func(self.log_src, msg) | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func(self.log_src, msg, c) | ||||
|  | ||||
|     def run(self): | ||||
|     def get_u2idx(self): | ||||
|         if not self.u2idx: | ||||
|             self.u2idx = U2idx(self) | ||||
|  | ||||
|         return self.u2idx | ||||
|  | ||||
|     def _detect_https(self): | ||||
|         method = None | ||||
|         self.sr = None | ||||
|         if self.cert_path: | ||||
|             try: | ||||
|                 method = self.s.recv(4, socket.MSG_PEEK) | ||||
| @@ -82,20 +104,64 @@ class HttpConn(object): | ||||
|                 err = "need at least 4 bytes in the first packet; got {}".format( | ||||
|                     len(method) | ||||
|                 ) | ||||
|                 self.log(err) | ||||
|                 if method: | ||||
|                     self.log(err) | ||||
|  | ||||
|                 self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8")) | ||||
|                 return | ||||
|  | ||||
|         if method not in [None, b"GET ", b"HEAD", b"POST"]: | ||||
|         return method not in [None, b"GET ", b"HEAD", b"POST", b"PUT ", b"OPTI"] | ||||
|  | ||||
|     def run(self): | ||||
|         self.sr = None | ||||
|         if self.args.https_only: | ||||
|             is_https = True | ||||
|         elif self.args.http_only or not HAVE_SSL: | ||||
|             is_https = False | ||||
|         else: | ||||
|             is_https = self._detect_https() | ||||
|  | ||||
|         if is_https: | ||||
|             if self.sr: | ||||
|                 self.log("\033[1;31mTODO: cannot do https in jython\033[0m") | ||||
|                 self.log("TODO: cannot do https in jython", c="1;31") | ||||
|                 return | ||||
|  | ||||
|             self.log_src = self.log_src.replace("[36m", "[35m") | ||||
|             try: | ||||
|                 self.s = ssl.wrap_socket( | ||||
|                     self.s, server_side=True, certfile=self.cert_path | ||||
|                 ) | ||||
|                 ctx = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH) | ||||
|                 ctx.load_cert_chain(self.cert_path) | ||||
|                 if self.args.ssl_ver: | ||||
|                     ctx.options &= ~self.args.ssl_flags_en | ||||
|                     ctx.options |= self.args.ssl_flags_de | ||||
|                     # print(repr(ctx.options)) | ||||
|  | ||||
|                 if self.args.ssl_log: | ||||
|                     try: | ||||
|                         ctx.keylog_filename = self.args.ssl_log | ||||
|                     except: | ||||
|                         self.log("keylog failed; openssl or python too old") | ||||
|  | ||||
|                 if self.args.ciphers: | ||||
|                     ctx.set_ciphers(self.args.ciphers) | ||||
|  | ||||
|                 self.s = ctx.wrap_socket(self.s, server_side=True) | ||||
|                 msg = [ | ||||
|                     "\033[1;3{:d}m{}".format(c, s) | ||||
|                     for c, s in zip([0, 5, 0], self.s.cipher()) | ||||
|                 ] | ||||
|                 self.log(" ".join(msg) + "\033[0m") | ||||
|  | ||||
|                 if self.args.ssl_dbg and hasattr(self.s, "shared_ciphers"): | ||||
|                     overlap = [y[::-1] for y in self.s.shared_ciphers()] | ||||
|                     lines = [str(x) for x in (["TLS cipher overlap:"] + overlap)] | ||||
|                     self.log("\n".join(lines)) | ||||
|                     for k, v in [ | ||||
|                         ["compression", self.s.compression()], | ||||
|                         ["ALPN proto", self.s.selected_alpn_protocol()], | ||||
|                         ["NPN proto", self.s.selected_npn_protocol()], | ||||
|                     ]: | ||||
|                         self.log("TLS {}: {}".format(k, v or "nah")) | ||||
|  | ||||
|             except Exception as ex: | ||||
|                 em = str(ex) | ||||
|  | ||||
| @@ -104,18 +170,19 @@ class HttpConn(object): | ||||
|                     self.log("client rejected our certificate (nice)") | ||||
|  | ||||
|                 elif "ALERT_CERTIFICATE_UNKNOWN" in em: | ||||
|                     # chrome-android keeps doing this | ||||
|                     # android-chrome keeps doing this | ||||
|                     pass | ||||
|  | ||||
|                 else: | ||||
|                     self.log("\033[35mhandshake\033[0m " + em) | ||||
|                     self.log("handshake\033[0m " + em, c=5) | ||||
|  | ||||
|                 return | ||||
|  | ||||
|         if not self.sr: | ||||
|             self.sr = Unrecv(self.s) | ||||
|  | ||||
|         while True: | ||||
|         while not self.stopping: | ||||
|             self.nreq += 1 | ||||
|             cli = HttpCli(self) | ||||
|             if not cli.run(): | ||||
|                 return | ||||
|   | ||||
| @@ -2,13 +2,39 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import math | ||||
| import base64 | ||||
| import socket | ||||
| import threading | ||||
|  | ||||
| from .__init__ import E | ||||
| try: | ||||
|     import jinja2 | ||||
| except ImportError: | ||||
|     print( | ||||
|         """\033[1;31m | ||||
|   you do not have jinja2 installed,\033[33m | ||||
|   choose one of these:\033[0m | ||||
|    * apt install python-jinja2 | ||||
|    * {} -m pip install --user jinja2 | ||||
|    * (try another python version, if you have one) | ||||
|    * (try copyparty.sfx instead) | ||||
| """.format( | ||||
|             os.path.basename(sys.executable) | ||||
|         ) | ||||
|     ) | ||||
|     sys.exit(1) | ||||
|  | ||||
| from .__init__ import E, PY2, MACOS | ||||
| from .util import spack, min_ex, start_stackmon, start_log_thrs | ||||
| from .bos import bos | ||||
| from .httpconn import HttpConn | ||||
| from .authsrv import AuthSrv | ||||
|  | ||||
| if PY2: | ||||
|     import Queue as queue | ||||
| else: | ||||
|     import queue | ||||
|  | ||||
|  | ||||
| class HttpSrv(object): | ||||
| @@ -17,38 +43,207 @@ class HttpSrv(object): | ||||
|     relying on MpSrv for performance (HttpSrv is just plain threads) | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, broker): | ||||
|     def __init__(self, broker, nid): | ||||
|         self.broker = broker | ||||
|         self.nid = nid | ||||
|         self.args = broker.args | ||||
|         self.log = broker.log | ||||
|         self.asrv = broker.asrv | ||||
|  | ||||
|         self.disconnect_func = None | ||||
|         self.name = "httpsrv" + ("-n{}-i{:x}".format(nid, os.getpid()) if nid else "") | ||||
|         self.mutex = threading.Lock() | ||||
|         self.stopping = False | ||||
|  | ||||
|         self.clients = {} | ||||
|         self.workload = 0 | ||||
|         self.workload_thr_alive = False | ||||
|         self.auth = AuthSrv(self.args, self.log) | ||||
|         self.tp_nthr = 0  # actual | ||||
|         self.tp_ncli = 0  # fading | ||||
|         self.tp_time = None  # latest worker collect | ||||
|         self.tp_q = None if self.args.no_htp else queue.LifoQueue() | ||||
|  | ||||
|         self.srvs = [] | ||||
|         self.ncli = 0  # exact | ||||
|         self.clients = {}  # laggy | ||||
|         self.nclimax = 0 | ||||
|         self.cb_ts = 0 | ||||
|         self.cb_v = 0 | ||||
|  | ||||
|         env = jinja2.Environment() | ||||
|         env.loader = jinja2.FileSystemLoader(os.path.join(E.mod, "web")) | ||||
|         self.j2 = { | ||||
|             x: env.get_template(x + ".html") | ||||
|             for x in ["splash", "browser", "browser2", "msg", "md", "mde"] | ||||
|         } | ||||
|  | ||||
|         cert_path = os.path.join(E.cfg, "cert.pem") | ||||
|         if os.path.exists(cert_path): | ||||
|         if bos.path.exists(cert_path): | ||||
|             self.cert_path = cert_path | ||||
|         else: | ||||
|             self.cert_path = None | ||||
|  | ||||
|         if self.tp_q: | ||||
|             self.start_threads(4) | ||||
|  | ||||
|             name = "httpsrv-scaler" + ("-{}".format(nid) if nid else "") | ||||
|             t = threading.Thread(target=self.thr_scaler, name=name) | ||||
|             t.daemon = True | ||||
|             t.start() | ||||
|  | ||||
|         if nid: | ||||
|             if self.args.stackmon: | ||||
|                 start_stackmon(self.args.stackmon, nid) | ||||
|  | ||||
|             if self.args.log_thrs: | ||||
|                 start_log_thrs(self.log, self.args.log_thrs, nid) | ||||
|  | ||||
|     def start_threads(self, n): | ||||
|         self.tp_nthr += n | ||||
|         if self.args.log_htp: | ||||
|             self.log(self.name, "workers += {} = {}".format(n, self.tp_nthr), 6) | ||||
|  | ||||
|         for _ in range(n): | ||||
|             thr = threading.Thread( | ||||
|                 target=self.thr_poolw, | ||||
|                 name=self.name + "-poolw", | ||||
|             ) | ||||
|             thr.daemon = True | ||||
|             thr.start() | ||||
|  | ||||
|     def stop_threads(self, n): | ||||
|         self.tp_nthr -= n | ||||
|         if self.args.log_htp: | ||||
|             self.log(self.name, "workers -= {} = {}".format(n, self.tp_nthr), 6) | ||||
|  | ||||
|         for _ in range(n): | ||||
|             self.tp_q.put(None) | ||||
|  | ||||
|     def thr_scaler(self): | ||||
|         while True: | ||||
|             time.sleep(2 if self.tp_ncli else 30) | ||||
|             with self.mutex: | ||||
|                 self.tp_ncli = max(self.ncli, self.tp_ncli - 2) | ||||
|                 if self.tp_nthr > self.tp_ncli + 8: | ||||
|                     self.stop_threads(4) | ||||
|  | ||||
|     def listen(self, sck, nlisteners): | ||||
|         ip, port = sck.getsockname() | ||||
|         self.srvs.append(sck) | ||||
|         self.nclimax = math.ceil(self.args.nc * 1.0 / nlisteners) | ||||
|         t = threading.Thread( | ||||
|             target=self.thr_listen, | ||||
|             args=(sck,), | ||||
|             name="httpsrv-n{}-listen-{}-{}".format(self.nid or "0", ip, port), | ||||
|         ) | ||||
|         t.daemon = True | ||||
|         t.start() | ||||
|  | ||||
|     def thr_listen(self, srv_sck): | ||||
|         """listens on a shared tcp server""" | ||||
|         ip, port = srv_sck.getsockname() | ||||
|         fno = srv_sck.fileno() | ||||
|         msg = "subscribed @ {}:{}  f{}".format(ip, port, fno) | ||||
|         self.log(self.name, msg) | ||||
|         self.broker.put(False, "cb_httpsrv_up") | ||||
|         while not self.stopping: | ||||
|             if self.args.log_conn: | ||||
|                 self.log(self.name, "|%sC-ncli" % ("-" * 1,), c="1;30") | ||||
|  | ||||
|             if self.ncli >= self.nclimax: | ||||
|                 self.log(self.name, "at connection limit; waiting", 3) | ||||
|                 while self.ncli >= self.nclimax: | ||||
|                     time.sleep(0.1) | ||||
|  | ||||
|             if self.args.log_conn: | ||||
|                 self.log(self.name, "|%sC-acc1" % ("-" * 2,), c="1;30") | ||||
|  | ||||
|             try: | ||||
|                 sck, addr = srv_sck.accept() | ||||
|             except (OSError, socket.error) as ex: | ||||
|                 self.log(self.name, "accept({}): {}".format(fno, ex), c=6) | ||||
|                 time.sleep(0.02) | ||||
|                 continue | ||||
|  | ||||
|             if self.args.log_conn: | ||||
|                 m = "|{}C-acc2 \033[0;36m{} \033[3{}m{}".format( | ||||
|                     "-" * 3, ip, port % 8, port | ||||
|                 ) | ||||
|                 self.log("%s %s" % addr, m, c="1;30") | ||||
|  | ||||
|             self.accept(sck, addr) | ||||
|  | ||||
|     def accept(self, sck, addr): | ||||
|         """takes an incoming tcp connection and creates a thread to handle it""" | ||||
|         self.log("%s %s" % addr, "-" * 5 + "C-cthr") | ||||
|         thr = threading.Thread(target=self.thr_client, args=(sck, addr)) | ||||
|         now = time.time() | ||||
|  | ||||
|         if now - (self.tp_time or now) > 300: | ||||
|             self.tp_q = None | ||||
|  | ||||
|         if self.tp_q: | ||||
|             self.tp_q.put((sck, addr)) | ||||
|             with self.mutex: | ||||
|                 self.ncli += 1 | ||||
|                 self.tp_time = self.tp_time or now | ||||
|                 self.tp_ncli = max(self.tp_ncli, self.ncli + 1) | ||||
|                 if self.tp_nthr < self.ncli + 4: | ||||
|                     self.start_threads(8) | ||||
|             return | ||||
|  | ||||
|         if not self.args.no_htp: | ||||
|             m = "looks like the httpserver threadpool died; please make an issue on github and tell me the story of how you pulled that off, thanks and dog bless\n" | ||||
|             self.log(self.name, m, 1) | ||||
|  | ||||
|         with self.mutex: | ||||
|             self.ncli += 1 | ||||
|  | ||||
|         thr = threading.Thread( | ||||
|             target=self.thr_client, | ||||
|             args=(sck, addr), | ||||
|             name="httpconn-{}-{}".format(addr[0].split(".", 2)[-1][-6:], addr[1]), | ||||
|         ) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|     def num_clients(self): | ||||
|         with self.mutex: | ||||
|             return len(self.clients) | ||||
|     def thr_poolw(self): | ||||
|         while True: | ||||
|             task = self.tp_q.get() | ||||
|             if not task: | ||||
|                 break | ||||
|  | ||||
|             with self.mutex: | ||||
|                 self.tp_time = None | ||||
|  | ||||
|             try: | ||||
|                 sck, addr = task | ||||
|                 me = threading.current_thread() | ||||
|                 me.name = "httpconn-{}-{}".format( | ||||
|                     addr[0].split(".", 2)[-1][-6:], addr[1] | ||||
|                 ) | ||||
|                 self.thr_client(sck, addr) | ||||
|                 me.name = self.name + "-poolw" | ||||
|             except: | ||||
|                 self.log(self.name, "thr_client: " + min_ex(), 3) | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.log("ok bye") | ||||
|         self.stopping = True | ||||
|         for srv in self.srvs: | ||||
|             try: | ||||
|                 srv.close() | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         clients = list(self.clients.keys()) | ||||
|         for cli in clients: | ||||
|             try: | ||||
|                 cli.shutdown() | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         if self.tp_q: | ||||
|             self.stop_threads(self.tp_nthr) | ||||
|             for _ in range(10): | ||||
|                 time.sleep(0.05) | ||||
|                 if self.tp_q.empty(): | ||||
|                     break | ||||
|  | ||||
|         self.log(self.name, "ok bye") | ||||
|  | ||||
|     def thr_client(self, sck, addr): | ||||
|         """thread managing one tcp client""" | ||||
| @@ -57,62 +252,69 @@ class HttpSrv(object): | ||||
|         cli = HttpConn(sck, addr, self) | ||||
|         with self.mutex: | ||||
|             self.clients[cli] = 0 | ||||
|             self.workload += 50 | ||||
|  | ||||
|             if not self.workload_thr_alive: | ||||
|                 self.workload_thr_alive = True | ||||
|                 thr = threading.Thread(target=self.thr_workload) | ||||
|                 thr.daemon = True | ||||
|                 thr.start() | ||||
|  | ||||
|         fno = sck.fileno() | ||||
|         try: | ||||
|             self.log("%s %s" % addr, "-" * 6 + "C-crun") | ||||
|             if self.args.log_conn: | ||||
|                 self.log("%s %s" % addr, "|%sC-crun" % ("-" * 4,), c="1;30") | ||||
|  | ||||
|             cli.run() | ||||
|  | ||||
|         except (OSError, socket.error) as ex: | ||||
|             if ex.errno not in [10038, 10054, 107, 57, 49, 9]: | ||||
|                 self.log( | ||||
|                     "%s %s" % addr, | ||||
|                     "run({}): {}".format(fno, ex), | ||||
|                     c=6, | ||||
|                 ) | ||||
|  | ||||
|         finally: | ||||
|             self.log("%s %s" % addr, "-" * 7 + "C-done") | ||||
|             sck = cli.s | ||||
|             if self.args.log_conn: | ||||
|                 self.log("%s %s" % addr, "|%sC-cdone" % ("-" * 5,), c="1;30") | ||||
|  | ||||
|             try: | ||||
|                 fno = sck.fileno() | ||||
|                 sck.shutdown(socket.SHUT_RDWR) | ||||
|                 sck.close() | ||||
|             except (OSError, socket.error) as ex: | ||||
|                 self.log( | ||||
|                     "%s %s" % addr, "shut_rdwr err:\n  {}\n  {}".format(repr(sck), ex), | ||||
|                 ) | ||||
|                 if ex.errno not in [10038, 107, 57, 9]: | ||||
|                 if not MACOS: | ||||
|                     self.log( | ||||
|                         "%s %s" % addr, | ||||
|                         "shut({}): {}".format(fno, ex), | ||||
|                         c="1;30", | ||||
|                     ) | ||||
|                 if ex.errno not in [10038, 10054, 107, 57, 49, 9]: | ||||
|                     # 10038 No longer considered a socket | ||||
|                     # 10054 Foribly closed by remote | ||||
|                     #   107 Transport endpoint not connected | ||||
|                     #    57 Socket is not connected | ||||
|                     #    49 Can't assign requested address (wifi down) | ||||
|                     #     9 Bad file descriptor | ||||
|                     raise | ||||
|             finally: | ||||
|                 with self.mutex: | ||||
|                     del self.clients[cli] | ||||
|                     self.ncli -= 1 | ||||
|  | ||||
|                 if self.disconnect_func: | ||||
|                     self.disconnect_func(addr)  # pylint: disable=not-callable | ||||
|     def cachebuster(self): | ||||
|         if time.time() - self.cb_ts < 1: | ||||
|             return self.cb_v | ||||
|  | ||||
|     def thr_workload(self): | ||||
|         """indicates the python interpreter workload caused by this HttpSrv""" | ||||
|         # avoid locking in extract_filedata by tracking difference here | ||||
|         while True: | ||||
|             time.sleep(0.2) | ||||
|             with self.mutex: | ||||
|                 if not self.clients: | ||||
|                     # no clients rn, termiante thread | ||||
|                     self.workload_thr_alive = False | ||||
|                     self.workload = 0 | ||||
|                     return | ||||
|         with self.mutex: | ||||
|             if time.time() - self.cb_ts < 1: | ||||
|                 return self.cb_v | ||||
|  | ||||
|             total = 0 | ||||
|             with self.mutex: | ||||
|                 for cli in self.clients.keys(): | ||||
|                     now = cli.workload | ||||
|                     delta = now - self.clients[cli] | ||||
|                     if delta < 0: | ||||
|                         # was reset in HttpCli to prevent overflow | ||||
|                         delta = now | ||||
|             v = E.t0 | ||||
|             try: | ||||
|                 with os.scandir(os.path.join(E.mod, "web")) as dh: | ||||
|                     for fh in dh: | ||||
|                         inf = fh.stat() | ||||
|                         v = max(v, inf.st_mtime) | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|                     total += delta | ||||
|                     self.clients[cli] = now | ||||
|  | ||||
|             self.workload = total | ||||
|             v = base64.urlsafe_b64encode(spack(b">xxL", int(v))) | ||||
|             self.cb_v = v.decode("ascii")[-4:] | ||||
|             self.cb_ts = time.time() | ||||
|             return self.cb_v | ||||
|   | ||||
							
								
								
									
										42
									
								
								copyparty/ico.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								copyparty/ico.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import hashlib | ||||
| import colorsys | ||||
|  | ||||
| from .__init__ import PY2 | ||||
|  | ||||
|  | ||||
| class Ico(object): | ||||
|     def __init__(self, args): | ||||
|         self.args = args | ||||
|  | ||||
|     def get(self, ext, as_thumb): | ||||
|         """placeholder to make thumbnails not break""" | ||||
|  | ||||
|         h = hashlib.md5(ext.encode("utf-8")).digest()[:2] | ||||
|         if PY2: | ||||
|             h = [ord(x) for x in h] | ||||
|  | ||||
|         c1 = colorsys.hsv_to_rgb(h[0] / 256.0, 1, 0.3) | ||||
|         c2 = colorsys.hsv_to_rgb(h[0] / 256.0, 1, 1) | ||||
|         c = list(c1) + list(c2) | ||||
|         c = [int(x * 255) for x in c] | ||||
|         c = "".join(["{:02x}".format(x) for x in c]) | ||||
|  | ||||
|         h = 30 | ||||
|         if not self.args.th_no_crop and as_thumb: | ||||
|             w, h = self.args.th_size.split("x") | ||||
|             h = int(100 / (float(w) / float(h))) | ||||
|  | ||||
|         svg = """\ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <svg version="1.1" viewBox="0 0 100 {}" xmlns="http://www.w3.org/2000/svg"><g> | ||||
| <rect width="100%" height="100%" fill="#{}" /> | ||||
| <text x="50%" y="50%" dominant-baseline="middle" text-anchor="middle" xml:space="preserve" | ||||
|   fill="#{}" font-family="monospace" font-size="14px" style="letter-spacing:.5px">{}</text> | ||||
| </g></svg> | ||||
| """ | ||||
|         svg = svg.format(h, c[:6], c[6:], ext).encode("utf-8") | ||||
|  | ||||
|         return ["image/svg+xml", svg] | ||||
							
								
								
									
										489
									
								
								copyparty/mtag.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										489
									
								
								copyparty/mtag.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,489 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import json | ||||
| import shutil | ||||
| import subprocess as sp | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, unicode | ||||
| from .util import fsenc, fsdec, uncyg, REKOBO_LKEY | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| def have_ff(cmd): | ||||
|     if PY2: | ||||
|         print("# checking {}".format(cmd)) | ||||
|         cmd = (cmd + " -version").encode("ascii").split(b" ") | ||||
|         try: | ||||
|             sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE).communicate() | ||||
|             return True | ||||
|         except: | ||||
|             return False | ||||
|     else: | ||||
|         return bool(shutil.which(cmd)) | ||||
|  | ||||
|  | ||||
| HAVE_FFMPEG = have_ff("ffmpeg") | ||||
| HAVE_FFPROBE = have_ff("ffprobe") | ||||
|  | ||||
|  | ||||
| class MParser(object): | ||||
|     def __init__(self, cmdline): | ||||
|         self.tag, args = cmdline.split("=", 1) | ||||
|         self.tags = self.tag.split(",") | ||||
|  | ||||
|         self.timeout = 30 | ||||
|         self.force = False | ||||
|         self.audio = "y" | ||||
|         self.ext = [] | ||||
|  | ||||
|         while True: | ||||
|             try: | ||||
|                 bp = os.path.expanduser(args) | ||||
|                 if WINDOWS: | ||||
|                     bp = uncyg(bp) | ||||
|  | ||||
|                 if bos.path.exists(bp): | ||||
|                     self.bin = bp | ||||
|                     return | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|             arg, args = args.split(",", 1) | ||||
|             arg = arg.lower() | ||||
|  | ||||
|             if arg.startswith("a"): | ||||
|                 self.audio = arg[1:]  # [r]equire [n]ot [d]ontcare | ||||
|                 continue | ||||
|  | ||||
|             if arg == "f": | ||||
|                 self.force = True | ||||
|                 continue | ||||
|  | ||||
|             if arg.startswith("t"): | ||||
|                 self.timeout = int(arg[1:]) | ||||
|                 continue | ||||
|  | ||||
|             if arg.startswith("e"): | ||||
|                 self.ext.append(arg[1:]) | ||||
|                 continue | ||||
|  | ||||
|             raise Exception() | ||||
|  | ||||
|  | ||||
| def ffprobe(abspath): | ||||
|     cmd = [ | ||||
|         b"ffprobe", | ||||
|         b"-hide_banner", | ||||
|         b"-show_streams", | ||||
|         b"-show_format", | ||||
|         b"--", | ||||
|         fsenc(abspath), | ||||
|     ] | ||||
|     p = sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE) | ||||
|     r = p.communicate() | ||||
|     txt = r[0].decode("utf-8", "replace") | ||||
|     return parse_ffprobe(txt) | ||||
|  | ||||
|  | ||||
| def parse_ffprobe(txt): | ||||
|     """ffprobe -show_format -show_streams""" | ||||
|     streams = [] | ||||
|     fmt = {} | ||||
|     g = None | ||||
|     for ln in [x.rstrip("\r") for x in txt.split("\n")]: | ||||
|         try: | ||||
|             k, v = ln.split("=", 1) | ||||
|             g[k] = v | ||||
|             continue | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         if ln == "[STREAM]": | ||||
|             g = {} | ||||
|             streams.append(g) | ||||
|  | ||||
|         if ln == "[FORMAT]": | ||||
|             g = {"codec_type": "format"}  # heh | ||||
|             fmt = g | ||||
|  | ||||
|     streams = [fmt] + streams | ||||
|     ret = {}  # processed | ||||
|     md = {}  # raw tags | ||||
|  | ||||
|     is_audio = fmt.get("format_name") in ["mp3", "ogg", "flac", "wav"] | ||||
|     if fmt.get("filename", "").split(".")[-1].lower() in ["m4a", "aac"]: | ||||
|         is_audio = True | ||||
|  | ||||
|     # if audio file, ensure audio stream appears first | ||||
|     if ( | ||||
|         is_audio | ||||
|         and len(streams) > 2 | ||||
|         and streams[1].get("codec_type") != "audio" | ||||
|         and streams[2].get("codec_type") == "audio" | ||||
|     ): | ||||
|         streams = [fmt, streams[2], streams[1]] + streams[3:] | ||||
|  | ||||
|     have = {} | ||||
|     for strm in streams: | ||||
|         typ = strm.get("codec_type") | ||||
|         if typ in have: | ||||
|             continue | ||||
|  | ||||
|         have[typ] = True | ||||
|         kvm = [] | ||||
|  | ||||
|         if typ == "audio": | ||||
|             kvm = [ | ||||
|                 ["codec_name", "ac"], | ||||
|                 ["channel_layout", "chs"], | ||||
|                 ["sample_rate", ".hz"], | ||||
|                 ["bit_rate", ".aq"], | ||||
|                 ["duration", ".dur"], | ||||
|             ] | ||||
|  | ||||
|         if typ == "video": | ||||
|             if strm.get("DISPOSITION:attached_pic") == "1" or is_audio: | ||||
|                 continue | ||||
|  | ||||
|             kvm = [ | ||||
|                 ["codec_name", "vc"], | ||||
|                 ["pix_fmt", "pixfmt"], | ||||
|                 ["r_frame_rate", ".fps"], | ||||
|                 ["bit_rate", ".vq"], | ||||
|                 ["width", ".resw"], | ||||
|                 ["height", ".resh"], | ||||
|                 ["duration", ".dur"], | ||||
|             ] | ||||
|  | ||||
|         if typ == "format": | ||||
|             kvm = [["duration", ".dur"], ["bit_rate", ".q"]] | ||||
|  | ||||
|         for sk, rk in kvm: | ||||
|             v = strm.get(sk) | ||||
|             if v is None: | ||||
|                 continue | ||||
|  | ||||
|             if rk.startswith("."): | ||||
|                 try: | ||||
|                     v = float(v) | ||||
|                     v2 = ret.get(rk) | ||||
|                     if v2 is None or v > v2: | ||||
|                         ret[rk] = v | ||||
|                 except: | ||||
|                     # sqlite doesnt care but the code below does | ||||
|                     if v not in ["N/A"]: | ||||
|                         ret[rk] = v | ||||
|             else: | ||||
|                 ret[rk] = v | ||||
|  | ||||
|     if ret.get("vc") == "ansi":  # shellscript | ||||
|         return {}, {} | ||||
|  | ||||
|     for strm in streams: | ||||
|         for k, v in strm.items(): | ||||
|             if not k.startswith("TAG:"): | ||||
|                 continue | ||||
|  | ||||
|             k = k[4:].strip() | ||||
|             v = v.strip() | ||||
|             if k and v and k not in md: | ||||
|                 md[k] = [v] | ||||
|  | ||||
|     for k in [".q", ".vq", ".aq"]: | ||||
|         if k in ret: | ||||
|             ret[k] /= 1000  # bit_rate=320000 | ||||
|  | ||||
|     for k in [".q", ".vq", ".aq", ".resw", ".resh"]: | ||||
|         if k in ret: | ||||
|             ret[k] = int(ret[k]) | ||||
|  | ||||
|     if ".fps" in ret: | ||||
|         fps = ret[".fps"] | ||||
|         if "/" in fps: | ||||
|             fa, fb = fps.split("/") | ||||
|             fps = int(fa) * 1.0 / int(fb) | ||||
|  | ||||
|         if fps < 1000 and fmt.get("format_name") not in ["image2", "png_pipe"]: | ||||
|             ret[".fps"] = round(fps, 3) | ||||
|         else: | ||||
|             del ret[".fps"] | ||||
|  | ||||
|     if ".dur" in ret: | ||||
|         if ret[".dur"] < 0.1: | ||||
|             del ret[".dur"] | ||||
|             if ".q" in ret: | ||||
|                 del ret[".q"] | ||||
|  | ||||
|     if ".resw" in ret and ".resh" in ret: | ||||
|         ret["res"] = "{}x{}".format(ret[".resw"], ret[".resh"]) | ||||
|  | ||||
|     ret = {k: [0, v] for k, v in ret.items()} | ||||
|  | ||||
|     return ret, md | ||||
|  | ||||
|  | ||||
| class MTag(object): | ||||
|     def __init__(self, log_func, args): | ||||
|         self.log_func = log_func | ||||
|         self.args = args | ||||
|         self.usable = True | ||||
|         self.prefer_mt = not args.no_mtag_ff | ||||
|         self.backend = "ffprobe" if args.no_mutagen else "mutagen" | ||||
|         self.can_ffprobe = ( | ||||
|             HAVE_FFPROBE | ||||
|             and not args.no_mtag_ff | ||||
|             and (not WINDOWS or sys.version_info >= (3, 8)) | ||||
|         ) | ||||
|         mappings = args.mtm | ||||
|         or_ffprobe = " or FFprobe" | ||||
|  | ||||
|         if self.backend == "mutagen": | ||||
|             self.get = self.get_mutagen | ||||
|             try: | ||||
|                 import mutagen | ||||
|             except: | ||||
|                 self.log("could not load Mutagen, trying FFprobe instead", c=3) | ||||
|                 self.backend = "ffprobe" | ||||
|  | ||||
|         if self.backend == "ffprobe": | ||||
|             self.usable = self.can_ffprobe | ||||
|             self.get = self.get_ffprobe | ||||
|             self.prefer_mt = True | ||||
|  | ||||
|             if not HAVE_FFPROBE: | ||||
|                 pass | ||||
|  | ||||
|             elif args.no_mtag_ff: | ||||
|                 msg = "found FFprobe but it was disabled by --no-mtag-ff" | ||||
|                 self.log(msg, c=3) | ||||
|  | ||||
|             elif WINDOWS and sys.version_info < (3, 8): | ||||
|                 or_ffprobe = " or python >= 3.8" | ||||
|                 msg = "found FFprobe but your python is too old; need 3.8 or newer" | ||||
|                 self.log(msg, c=1) | ||||
|  | ||||
|         if not self.usable: | ||||
|             msg = "need Mutagen{} to read media tags so please run this:\n{}{} -m pip install --user mutagen\n" | ||||
|             pybin = os.path.basename(sys.executable) | ||||
|             self.log(msg.format(or_ffprobe, " " * 37, pybin), c=1) | ||||
|             return | ||||
|  | ||||
|         # https://picard-docs.musicbrainz.org/downloads/MusicBrainz_Picard_Tag_Map.html | ||||
|         tagmap = { | ||||
|             "album": ["album", "talb", "\u00a9alb", "original-album", "toal"], | ||||
|             "artist": [ | ||||
|                 "artist", | ||||
|                 "tpe1", | ||||
|                 "\u00a9art", | ||||
|                 "composer", | ||||
|                 "performer", | ||||
|                 "arranger", | ||||
|                 "\u00a9wrt", | ||||
|                 "tcom", | ||||
|                 "tpe3", | ||||
|                 "original-artist", | ||||
|                 "tope", | ||||
|             ], | ||||
|             "title": ["title", "tit2", "\u00a9nam"], | ||||
|             "circle": [ | ||||
|                 "album-artist", | ||||
|                 "tpe2", | ||||
|                 "aart", | ||||
|                 "conductor", | ||||
|                 "organization", | ||||
|                 "band", | ||||
|             ], | ||||
|             ".tn": ["tracknumber", "trck", "trkn", "track"], | ||||
|             "genre": ["genre", "tcon", "\u00a9gen"], | ||||
|             "date": [ | ||||
|                 "original-release-date", | ||||
|                 "release-date", | ||||
|                 "date", | ||||
|                 "tdrc", | ||||
|                 "\u00a9day", | ||||
|                 "original-date", | ||||
|                 "original-year", | ||||
|                 "tyer", | ||||
|                 "tdor", | ||||
|                 "tory", | ||||
|                 "year", | ||||
|                 "creation-time", | ||||
|             ], | ||||
|             ".bpm": ["bpm", "tbpm", "tmpo", "tbp"], | ||||
|             "key": ["initial-key", "tkey", "key"], | ||||
|             "comment": ["comment", "comm", "\u00a9cmt", "comments", "description"], | ||||
|         } | ||||
|  | ||||
|         if mappings: | ||||
|             for k, v in [x.split("=") for x in mappings]: | ||||
|                 tagmap[k] = v.split(",") | ||||
|  | ||||
|         self.tagmap = {} | ||||
|         for k, vs in tagmap.items(): | ||||
|             vs2 = [] | ||||
|             for v in vs: | ||||
|                 if "-" not in v: | ||||
|                     vs2.append(v) | ||||
|                     continue | ||||
|  | ||||
|                 vs2.append(v.replace("-", " ")) | ||||
|                 vs2.append(v.replace("-", "_")) | ||||
|                 vs2.append(v.replace("-", "")) | ||||
|  | ||||
|             self.tagmap[k] = vs2 | ||||
|  | ||||
|         self.rmap = { | ||||
|             v: [n, k] for k, vs in self.tagmap.items() for n, v in enumerate(vs) | ||||
|         } | ||||
|         # self.get = self.compare | ||||
|  | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func("mtag", msg, c) | ||||
|  | ||||
|     def normalize_tags(self, ret, md): | ||||
|         for k, v in dict(md).items(): | ||||
|             if not v: | ||||
|                 continue | ||||
|  | ||||
|             k = k.lower().split("::")[0].strip() | ||||
|             mk = self.rmap.get(k) | ||||
|             if not mk: | ||||
|                 continue | ||||
|  | ||||
|             pref, mk = mk | ||||
|             if mk not in ret or ret[mk][0] > pref: | ||||
|                 ret[mk] = [pref, v[0]] | ||||
|  | ||||
|         # take first value | ||||
|         ret = {k: unicode(v[1]).strip() for k, v in ret.items()} | ||||
|  | ||||
|         # track 3/7 => track 3 | ||||
|         for k, v in ret.items(): | ||||
|             if k[0] == ".": | ||||
|                 v = v.split("/")[0].strip().lstrip("0") | ||||
|                 ret[k] = v or 0 | ||||
|  | ||||
|         # normalize key notation to rkeobo | ||||
|         okey = ret.get("key") | ||||
|         if okey: | ||||
|             key = okey.replace(" ", "").replace("maj", "").replace("min", "m") | ||||
|             ret["key"] = REKOBO_LKEY.get(key.lower(), okey) | ||||
|  | ||||
|         return ret | ||||
|  | ||||
|     def compare(self, abspath): | ||||
|         if abspath.endswith(".au"): | ||||
|             return {} | ||||
|  | ||||
|         print("\n" + abspath) | ||||
|         r1 = self.get_mutagen(abspath) | ||||
|         r2 = self.get_ffprobe(abspath) | ||||
|  | ||||
|         keys = {} | ||||
|         for d in [r1, r2]: | ||||
|             for k in d.keys(): | ||||
|                 keys[k] = True | ||||
|  | ||||
|         diffs = [] | ||||
|         l1 = [] | ||||
|         l2 = [] | ||||
|         for k in sorted(keys.keys()): | ||||
|             if k in [".q", ".dur"]: | ||||
|                 continue  # lenient | ||||
|  | ||||
|             v1 = r1.get(k) | ||||
|             v2 = r2.get(k) | ||||
|             if v1 == v2: | ||||
|                 print("  ", k, v1) | ||||
|             elif v1 != "0000":  # FFprobe date=0 | ||||
|                 diffs.append(k) | ||||
|                 print(" 1", k, v1) | ||||
|                 print(" 2", k, v2) | ||||
|                 if v1: | ||||
|                     l1.append(k) | ||||
|                 if v2: | ||||
|                     l2.append(k) | ||||
|  | ||||
|         if diffs: | ||||
|             raise Exception() | ||||
|  | ||||
|         return r1 | ||||
|  | ||||
|     def get_mutagen(self, abspath): | ||||
|         import mutagen | ||||
|  | ||||
|         try: | ||||
|             md = mutagen.File(fsenc(abspath), easy=True) | ||||
|             x = md.info.length | ||||
|         except Exception as ex: | ||||
|             return self.get_ffprobe(abspath) if self.can_ffprobe else {} | ||||
|  | ||||
|         sz = bos.path.getsize(abspath) | ||||
|         ret = {".q": [0, int((sz / md.info.length) / 128)]} | ||||
|  | ||||
|         for attr, k, norm in [ | ||||
|             ["codec", "ac", unicode], | ||||
|             ["channels", "chs", int], | ||||
|             ["sample_rate", ".hz", int], | ||||
|             ["bitrate", ".aq", int], | ||||
|             ["length", ".dur", int], | ||||
|         ]: | ||||
|             try: | ||||
|                 v = getattr(md.info, attr) | ||||
|             except: | ||||
|                 continue | ||||
|  | ||||
|             if not v: | ||||
|                 continue | ||||
|  | ||||
|             if k == ".aq": | ||||
|                 v /= 1000 | ||||
|  | ||||
|             if k == "ac" and v.startswith("mp4a.40."): | ||||
|                 v = "aac" | ||||
|  | ||||
|             ret[k] = [0, norm(v)] | ||||
|  | ||||
|         return self.normalize_tags(ret, md) | ||||
|  | ||||
|     def get_ffprobe(self, abspath): | ||||
|         ret, md = ffprobe(abspath) | ||||
|         return self.normalize_tags(ret, md) | ||||
|  | ||||
|     def get_bin(self, parsers, abspath): | ||||
|         pypath = os.path.abspath(os.path.dirname(os.path.dirname(__file__))) | ||||
|         pypath = [str(pypath)] + [str(x) for x in sys.path if x] | ||||
|         pypath = str(os.pathsep.join(pypath)) | ||||
|         env = os.environ.copy() | ||||
|         env["PYTHONPATH"] = pypath | ||||
|  | ||||
|         ret = {} | ||||
|         for tagname, mp in parsers.items(): | ||||
|             try: | ||||
|                 cmd = [sys.executable, mp.bin, abspath] | ||||
|                 args = {"env": env, "timeout": mp.timeout} | ||||
|  | ||||
|                 if WINDOWS: | ||||
|                     args["creationflags"] = 0x4000 | ||||
|                 else: | ||||
|                     cmd = ["nice"] + cmd | ||||
|  | ||||
|                 cmd = [fsenc(x) for x in cmd] | ||||
|                 v = sp.check_output(cmd, **args).strip() | ||||
|                 if not v: | ||||
|                     continue | ||||
|  | ||||
|                 if "," not in tagname: | ||||
|                     ret[tagname] = v.decode("utf-8") | ||||
|                 else: | ||||
|                     v = json.loads(v) | ||||
|                     for tag in tagname.split(","): | ||||
|                         if tag and tag in v: | ||||
|                             ret[tag] = v[tag] | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         return ret | ||||
							
								
								
									
										100
									
								
								copyparty/star.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								copyparty/star.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import tarfile | ||||
| import threading | ||||
|  | ||||
| from .sutil import errdesc | ||||
| from .util import Queue, fsenc | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| class QFile(object): | ||||
|     """file-like object which buffers writes into a queue""" | ||||
|  | ||||
|     def __init__(self): | ||||
|         self.q = Queue(64) | ||||
|         self.bq = [] | ||||
|         self.nq = 0 | ||||
|  | ||||
|     def write(self, buf): | ||||
|         if buf is None or self.nq >= 240 * 1024: | ||||
|             self.q.put(b"".join(self.bq)) | ||||
|             self.bq = [] | ||||
|             self.nq = 0 | ||||
|  | ||||
|         if buf is None: | ||||
|             self.q.put(None) | ||||
|         else: | ||||
|             self.bq.append(buf) | ||||
|             self.nq += len(buf) | ||||
|  | ||||
|  | ||||
| class StreamTar(object): | ||||
|     """construct in-memory tar file from the given path""" | ||||
|  | ||||
|     def __init__(self, log, fgen, **kwargs): | ||||
|         self.ci = 0 | ||||
|         self.co = 0 | ||||
|         self.qfile = QFile() | ||||
|         self.log = log | ||||
|         self.fgen = fgen | ||||
|         self.errf = None | ||||
|  | ||||
|         # python 3.8 changed to PAX_FORMAT as default, | ||||
|         # waste of space and don't care about the new features | ||||
|         fmt = tarfile.GNU_FORMAT | ||||
|         self.tar = tarfile.open(fileobj=self.qfile, mode="w|", format=fmt) | ||||
|  | ||||
|         w = threading.Thread(target=self._gen, name="star-gen") | ||||
|         w.daemon = True | ||||
|         w.start() | ||||
|  | ||||
|     def gen(self): | ||||
|         while True: | ||||
|             buf = self.qfile.q.get() | ||||
|             if not buf: | ||||
|                 break | ||||
|  | ||||
|             self.co += len(buf) | ||||
|             yield buf | ||||
|  | ||||
|         yield None | ||||
|         if self.errf: | ||||
|             bos.unlink(self.errf["ap"]) | ||||
|  | ||||
|     def ser(self, f): | ||||
|         name = f["vp"] | ||||
|         src = f["ap"] | ||||
|         fsi = f["st"] | ||||
|  | ||||
|         inf = tarfile.TarInfo(name=name) | ||||
|         inf.mode = fsi.st_mode | ||||
|         inf.size = fsi.st_size | ||||
|         inf.mtime = fsi.st_mtime | ||||
|         inf.uid = 0 | ||||
|         inf.gid = 0 | ||||
|  | ||||
|         self.ci += inf.size | ||||
|         with open(fsenc(src), "rb", 512 * 1024) as f: | ||||
|             self.tar.addfile(inf, f) | ||||
|  | ||||
|     def _gen(self): | ||||
|         errors = [] | ||||
|         for f in self.fgen: | ||||
|             if "err" in f: | ||||
|                 errors.append([f["vp"], f["err"]]) | ||||
|                 continue | ||||
|  | ||||
|             try: | ||||
|                 self.ser(f) | ||||
|             except Exception as ex: | ||||
|                 errors.append([f["vp"], repr(ex)]) | ||||
|  | ||||
|         if errors: | ||||
|             self.errf, txt = errdesc(errors) | ||||
|             self.log("\n".join(([repr(self.errf)] + txt[1:]))) | ||||
|             self.ser(self.errf) | ||||
|  | ||||
|         self.tar.close() | ||||
|         self.qfile.write(None) | ||||
							
								
								
									
										29
									
								
								copyparty/sutil.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								copyparty/sutil.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import time | ||||
| import tempfile | ||||
| from datetime import datetime | ||||
|  | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| def errdesc(errors): | ||||
|     report = ["copyparty failed to add the following files to the archive:", ""] | ||||
|  | ||||
|     for fn, err in errors: | ||||
|         report.extend([" file: {}".format(fn), "error: {}".format(err), ""]) | ||||
|  | ||||
|     with tempfile.NamedTemporaryFile(prefix="copyparty-", delete=False) as tf: | ||||
|         tf_path = tf.name | ||||
|         tf.write("\r\n".join(report).encode("utf-8", "replace")) | ||||
|  | ||||
|     dt = datetime.utcfromtimestamp(time.time()) | ||||
|     dt = dt.strftime("%Y-%m%d-%H%M%S") | ||||
|  | ||||
|     bos.chmod(tf_path, 0o444) | ||||
|     return { | ||||
|         "vp": "archive-errors-{}.txt".format(dt), | ||||
|         "ap": tf_path, | ||||
|         "st": bos.stat(tf_path), | ||||
|     }, report | ||||
| @@ -2,16 +2,23 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import shlex | ||||
| import string | ||||
| import signal | ||||
| import socket | ||||
| import threading | ||||
| from datetime import datetime, timedelta | ||||
| import calendar | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, VT100 | ||||
| from .__init__ import E, PY2, WINDOWS, ANYWIN, MACOS, VT100, unicode | ||||
| from .util import mp, start_log_thrs, start_stackmon, min_ex | ||||
| from .authsrv import AuthSrv | ||||
| from .tcpsrv import TcpSrv | ||||
| from .up2k import Up2k | ||||
| from .util import mp | ||||
| from .th_srv import ThumbSrv, HAVE_PIL, HAVE_WEBP | ||||
|  | ||||
|  | ||||
| class SvcHub(object): | ||||
| @@ -25,19 +32,52 @@ class SvcHub(object): | ||||
|     put() can return a queue (if want_reply=True) which has a blocking get() with the response. | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, args): | ||||
|     def __init__(self, args, argv, printed): | ||||
|         self.args = args | ||||
|         self.argv = argv | ||||
|         self.logf = None | ||||
|         self.stop_req = False | ||||
|         self.stopping = False | ||||
|         self.stop_cond = threading.Condition() | ||||
|         self.httpsrv_up = 0 | ||||
|  | ||||
|         self.ansi_re = re.compile("\033\\[[^m]*m") | ||||
|         self.log_mutex = threading.Lock() | ||||
|         self.next_day = 0 | ||||
|  | ||||
|         self.log = self._log_disabled if args.q else self._log_enabled | ||||
|         if args.lo: | ||||
|             self._setup_logfile(printed) | ||||
|  | ||||
|         if args.stackmon: | ||||
|             start_stackmon(args.stackmon, 0) | ||||
|  | ||||
|         if args.log_thrs: | ||||
|             start_log_thrs(self.log, args.log_thrs, 0) | ||||
|  | ||||
|         # initiate all services to manage | ||||
|         self.asrv = AuthSrv(self.args, self.log) | ||||
|         if args.ls: | ||||
|             self.asrv.dbg_ls() | ||||
|  | ||||
|         self.tcpsrv = TcpSrv(self) | ||||
|         self.up2k = Up2k(self) | ||||
|  | ||||
|         self.thumbsrv = None | ||||
|         if not args.no_thumb: | ||||
|             if HAVE_PIL: | ||||
|                 if not HAVE_WEBP: | ||||
|                     args.th_no_webp = True | ||||
|                     msg = "setting --th-no-webp because either libwebp is not available or your Pillow is too old" | ||||
|                     self.log("thumb", msg, c=3) | ||||
|  | ||||
|                 self.thumbsrv = ThumbSrv(self) | ||||
|             else: | ||||
|                 msg = "need Pillow to create thumbnails; for example:\n{}{} -m pip install --user Pillow\n" | ||||
|                 self.log( | ||||
|                     "thumb", msg.format(" " * 37, os.path.basename(sys.executable)), c=3 | ||||
|                 ) | ||||
|  | ||||
|         # decide which worker impl to use | ||||
|         if self.check_mp_enable(): | ||||
|             from .broker_mp import BrokerMp as Broker | ||||
| @@ -47,73 +87,234 @@ class SvcHub(object): | ||||
|  | ||||
|         self.broker = Broker(self) | ||||
|  | ||||
|     def run(self): | ||||
|         thr = threading.Thread(target=self.tcpsrv.run) | ||||
|     def thr_httpsrv_up(self): | ||||
|         time.sleep(5) | ||||
|         failed = self.broker.num_workers - self.httpsrv_up | ||||
|         if not failed: | ||||
|             return | ||||
|  | ||||
|         m = "{}/{} workers failed to start" | ||||
|         m = m.format(failed, self.broker.num_workers) | ||||
|         self.log("root", m, 1) | ||||
|         os._exit(1) | ||||
|  | ||||
|     def cb_httpsrv_up(self): | ||||
|         self.httpsrv_up += 1 | ||||
|         if self.httpsrv_up != self.broker.num_workers: | ||||
|             return | ||||
|  | ||||
|         self.log("root", "workers OK\n") | ||||
|         self.up2k.init_vols() | ||||
|  | ||||
|         thr = threading.Thread(target=self.sd_notify, name="sd-notify") | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|         # winxp/py2.7 support: thr.join() kills signals | ||||
|         try: | ||||
|             while True: | ||||
|                 time.sleep(9001) | ||||
|     def _logname(self): | ||||
|         dt = datetime.utcfromtimestamp(time.time()) | ||||
|         fn = self.args.lo | ||||
|         for fs in "YmdHMS": | ||||
|             fs = "%" + fs | ||||
|             if fs in fn: | ||||
|                 fn = fn.replace(fs, dt.strftime(fs)) | ||||
|  | ||||
|         except KeyboardInterrupt: | ||||
|         return fn | ||||
|  | ||||
|     def _setup_logfile(self, printed): | ||||
|         base_fn = fn = sel_fn = self._logname() | ||||
|         if fn != self.args.lo: | ||||
|             ctr = 0 | ||||
|             # yup this is a race; if started sufficiently concurrently, two | ||||
|             # copyparties can grab the same logfile (considered and ignored) | ||||
|             while os.path.exists(sel_fn): | ||||
|                 ctr += 1 | ||||
|                 sel_fn = "{}.{}".format(fn, ctr) | ||||
|  | ||||
|         fn = sel_fn | ||||
|  | ||||
|         try: | ||||
|             import lzma | ||||
|  | ||||
|             lh = lzma.open(fn, "wt", encoding="utf-8", errors="replace", preset=0) | ||||
|  | ||||
|         except: | ||||
|             import codecs | ||||
|  | ||||
|             lh = codecs.open(fn, "w", encoding="utf-8", errors="replace") | ||||
|  | ||||
|         lh.base_fn = base_fn | ||||
|  | ||||
|         argv = [sys.executable] + self.argv | ||||
|         if hasattr(shlex, "quote"): | ||||
|             argv = [shlex.quote(x) for x in argv] | ||||
|         else: | ||||
|             argv = ['"{}"'.format(x) for x in argv] | ||||
|  | ||||
|         msg = "[+] opened logfile [{}]\n".format(fn) | ||||
|         printed += msg | ||||
|         lh.write("t0: {:.3f}\nargv: {}\n\n{}".format(E.t0, " ".join(argv), printed)) | ||||
|         self.logf = lh | ||||
|         print(msg, end="") | ||||
|  | ||||
|     def run(self): | ||||
|         self.tcpsrv.run() | ||||
|  | ||||
|         thr = threading.Thread(target=self.thr_httpsrv_up) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|         for sig in [signal.SIGINT, signal.SIGTERM]: | ||||
|             signal.signal(sig, self.signal_handler) | ||||
|  | ||||
|         # macos hangs after shutdown on sigterm with while-sleep, | ||||
|         # windows cannot ^c stop_cond (and win10 does the macos thing but winxp is fine??) | ||||
|         # linux is fine with both, | ||||
|         # never lucky | ||||
|         if ANYWIN: | ||||
|             # msys-python probably fine but >msys-python | ||||
|             thr = threading.Thread(target=self.stop_thr, name="svchub-sig") | ||||
|             thr.daemon = True | ||||
|             thr.start() | ||||
|  | ||||
|             try: | ||||
|                 while not self.stop_req: | ||||
|                     time.sleep(1) | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|             self.shutdown() | ||||
|             thr.join() | ||||
|         else: | ||||
|             self.stop_thr() | ||||
|  | ||||
|     def stop_thr(self): | ||||
|         while not self.stop_req: | ||||
|             with self.stop_cond: | ||||
|                 self.stop_cond.wait(9001) | ||||
|  | ||||
|         self.shutdown() | ||||
|  | ||||
|     def signal_handler(self, sig, frame): | ||||
|         if self.stopping: | ||||
|             return | ||||
|  | ||||
|         self.stop_req = True | ||||
|         with self.stop_cond: | ||||
|             self.stop_cond.notify_all() | ||||
|  | ||||
|     def shutdown(self): | ||||
|         if self.stopping: | ||||
|             return | ||||
|  | ||||
|         self.stopping = True | ||||
|         self.stop_req = True | ||||
|         with self.stop_cond: | ||||
|             self.stop_cond.notify_all() | ||||
|  | ||||
|         ret = 1 | ||||
|         try: | ||||
|             with self.log_mutex: | ||||
|                 print("OPYTHAT") | ||||
|  | ||||
|             self.tcpsrv.shutdown() | ||||
|             self.broker.shutdown() | ||||
|             print("nailed it") | ||||
|             self.up2k.shutdown() | ||||
|             if self.thumbsrv: | ||||
|                 self.thumbsrv.shutdown() | ||||
|  | ||||
|     def _log_disabled(self, src, msg): | ||||
|         pass | ||||
|                 for n in range(200):  # 10s | ||||
|                     time.sleep(0.05) | ||||
|                     if self.thumbsrv.stopped(): | ||||
|                         break | ||||
|  | ||||
|     def _log_enabled(self, src, msg): | ||||
|                     if n == 3: | ||||
|                         print("waiting for thumbsrv (10sec)...") | ||||
|  | ||||
|             print("nailed it", end="") | ||||
|             ret = 0 | ||||
|         finally: | ||||
|             print("\033[0m") | ||||
|             if self.logf: | ||||
|                 self.logf.close() | ||||
|  | ||||
|             sys.exit(ret) | ||||
|  | ||||
|     def _log_disabled(self, src, msg, c=0): | ||||
|         if not self.logf: | ||||
|             return | ||||
|  | ||||
|         with self.log_mutex: | ||||
|             ts = datetime.utcfromtimestamp(time.time()) | ||||
|             ts = ts.strftime("%Y-%m%d-%H%M%S.%f")[:-3] | ||||
|             self.logf.write("@{} [{}] {}\n".format(ts, src, msg)) | ||||
|  | ||||
|             now = time.time() | ||||
|             if now >= self.next_day: | ||||
|                 self._set_next_day() | ||||
|  | ||||
|     def _set_next_day(self): | ||||
|         if self.next_day and self.logf and self.logf.base_fn != self._logname(): | ||||
|             self.logf.close() | ||||
|             self._setup_logfile("") | ||||
|  | ||||
|         dt = datetime.utcfromtimestamp(time.time()) | ||||
|  | ||||
|         # unix timestamp of next 00:00:00 (leap-seconds safe) | ||||
|         day_now = dt.day | ||||
|         while dt.day == day_now: | ||||
|             dt += timedelta(hours=12) | ||||
|  | ||||
|         dt = dt.replace(hour=0, minute=0, second=0) | ||||
|         self.next_day = calendar.timegm(dt.utctimetuple()) | ||||
|  | ||||
|     def _log_enabled(self, src, msg, c=0): | ||||
|         """handles logging from all components""" | ||||
|         with self.log_mutex: | ||||
|             now = time.time() | ||||
|             if now >= self.next_day: | ||||
|                 dt = datetime.utcfromtimestamp(now) | ||||
|                 print("\033[36m{}\033[0m".format(dt.strftime("%Y-%m-%d"))) | ||||
|                 print("\033[36m{}\033[0m\n".format(dt.strftime("%Y-%m-%d")), end="") | ||||
|                 self._set_next_day() | ||||
|  | ||||
|                 # unix timestamp of next 00:00:00 (leap-seconds safe) | ||||
|                 day_now = dt.day | ||||
|                 while dt.day == day_now: | ||||
|                     dt += timedelta(hours=12) | ||||
|  | ||||
|                 dt = dt.replace(hour=0, minute=0, second=0) | ||||
|                 self.next_day = calendar.timegm(dt.utctimetuple()) | ||||
|  | ||||
|             fmt = "\033[36m{} \033[33m{:21} \033[0m{}" | ||||
|             fmt = "\033[36m{} \033[33m{:21} \033[0m{}\n" | ||||
|             if not VT100: | ||||
|                 fmt = "{} {:21} {}" | ||||
|                 fmt = "{} {:21} {}\n" | ||||
|                 if "\033" in msg: | ||||
|                     msg = self.ansi_re.sub("", msg) | ||||
|                 if "\033" in src: | ||||
|                     src = self.ansi_re.sub("", src) | ||||
|             elif c: | ||||
|                 if isinstance(c, int): | ||||
|                     msg = "\033[3{}m{}".format(c, msg) | ||||
|                 elif "\033" not in c: | ||||
|                     msg = "\033[{}m{}\033[0m".format(c, msg) | ||||
|                 else: | ||||
|                     msg = "{}{}\033[0m".format(c, msg) | ||||
|  | ||||
|             ts = datetime.utcfromtimestamp(now).strftime("%H:%M:%S.%f")[:-3] | ||||
|             msg = fmt.format(ts, src, msg) | ||||
|             try: | ||||
|                 print(msg) | ||||
|                 print(msg, end="") | ||||
|             except UnicodeEncodeError: | ||||
|                 try: | ||||
|                     print(msg.encode("utf-8", "replace").decode()) | ||||
|                     print(msg.encode("utf-8", "replace").decode(), end="") | ||||
|                 except: | ||||
|                     print(msg.encode("ascii", "replace").decode()) | ||||
|                     print(msg.encode("ascii", "replace").decode(), end="") | ||||
|  | ||||
|             if self.logf: | ||||
|                 self.logf.write(msg) | ||||
|  | ||||
|     def check_mp_support(self): | ||||
|         vmin = sys.version_info[1] | ||||
|         if WINDOWS: | ||||
|             msg = "need python 3.3 or newer for multiprocessing;" | ||||
|             if PY2: | ||||
|                 # py2 pickler doesn't support winsock | ||||
|                 return msg | ||||
|             elif vmin < 3: | ||||
|             if PY2 or vmin < 3: | ||||
|                 return msg | ||||
|         elif MACOS: | ||||
|             return "multiprocessing is wonky on mac osx;" | ||||
|         else: | ||||
|             msg = "need python 2.7 or 3.3+ for multiprocessing;" | ||||
|             if not PY2 and vmin < 3: | ||||
|             msg = "need python 3.3+ for multiprocessing;" | ||||
|             if PY2 or vmin < 3: | ||||
|                 return msg | ||||
|  | ||||
|         try: | ||||
| @@ -127,13 +328,13 @@ class SvcHub(object): | ||||
|         return None | ||||
|  | ||||
|     def check_mp_enable(self): | ||||
|         if self.args.j == 0: | ||||
|             self.log("root", "multiprocessing disabled by argument -j 0;") | ||||
|         if self.args.j == 1: | ||||
|             self.log("root", "multiprocessing disabled by argument -j 1;") | ||||
|             return False | ||||
|  | ||||
|         if mp.cpu_count() <= 1: | ||||
|             return False | ||||
|          | ||||
|  | ||||
|         try: | ||||
|             # support vscode debugger (bonus: same behavior as on windows) | ||||
|             mp.set_start_method("spawn", True) | ||||
| @@ -145,5 +346,24 @@ class SvcHub(object): | ||||
|         if not err: | ||||
|             return True | ||||
|         else: | ||||
|             self.log("root", err) | ||||
|             self.log("svchub", err) | ||||
|             return False | ||||
|  | ||||
|     def sd_notify(self): | ||||
|         try: | ||||
|             addr = os.getenv("NOTIFY_SOCKET") | ||||
|             if not addr: | ||||
|                 return | ||||
|  | ||||
|             addr = unicode(addr) | ||||
|             if addr.startswith("@"): | ||||
|                 addr = "\0" + addr[1:] | ||||
|  | ||||
|             m = "".join(x for x in addr if x in string.printable) | ||||
|             self.log("sd_notify", m) | ||||
|  | ||||
|             sck = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM) | ||||
|             sck.connect(addr) | ||||
|             sck.sendall(b"READY=1") | ||||
|         except: | ||||
|             self.log("sd_notify", min_ex()) | ||||
|   | ||||
							
								
								
									
										275
									
								
								copyparty/szip.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										275
									
								
								copyparty/szip.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,275 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import time | ||||
| import zlib | ||||
| from datetime import datetime | ||||
|  | ||||
| from .sutil import errdesc | ||||
| from .util import yieldfile, sanitize_fn, spack, sunpack | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| def dostime2unix(buf): | ||||
|     t, d = sunpack(b"<HH", buf) | ||||
|  | ||||
|     ts = (t & 0x1F) * 2 | ||||
|     tm = (t >> 5) & 0x3F | ||||
|     th = t >> 11 | ||||
|  | ||||
|     dd = d & 0x1F | ||||
|     dm = (d >> 5) & 0xF | ||||
|     dy = (d >> 9) + 1980 | ||||
|  | ||||
|     tt = (dy, dm, dd, th, tm, ts) | ||||
|     tf = "{:04d}-{:02d}-{:02d} {:02d}:{:02d}:{:02d}" | ||||
|     iso = tf.format(*tt) | ||||
|  | ||||
|     dt = datetime.strptime(iso, "%Y-%m-%d %H:%M:%S") | ||||
|     return int(dt.timestamp()) | ||||
|  | ||||
|  | ||||
| def unixtime2dos(ts): | ||||
|     tt = time.gmtime(ts) | ||||
|     dy, dm, dd, th, tm, ts = list(tt)[:6] | ||||
|  | ||||
|     bd = ((dy - 1980) << 9) + (dm << 5) + dd | ||||
|     bt = (th << 11) + (tm << 5) + ts // 2 | ||||
|     return spack(b"<HH", bt, bd) | ||||
|  | ||||
|  | ||||
| def gen_fdesc(sz, crc32, z64): | ||||
|     ret = b"\x50\x4b\x07\x08" | ||||
|     fmt = b"<LQQ" if z64 else b"<LLL" | ||||
|     ret += spack(fmt, crc32, sz, sz) | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def gen_hdr(h_pos, fn, sz, lastmod, utf8, crc32, pre_crc): | ||||
|     """ | ||||
|     does regular file headers | ||||
|     and the central directory meme if h_pos is set | ||||
|     (h_pos = absolute position of the regular header) | ||||
|     """ | ||||
|  | ||||
|     # appnote 4.5 / zip 3.0 (2008) / unzip 6.0 (2009) says to add z64 | ||||
|     # extinfo for values which exceed H, but that becomes an off-by-one | ||||
|     # (can't tell if it was clamped or exactly maxval), make it obvious | ||||
|     z64 = sz >= 0xFFFFFFFF | ||||
|     z64v = [sz, sz] if z64 else [] | ||||
|     if h_pos and h_pos >= 0xFFFFFFFF: | ||||
|         # central, also consider ptr to original header | ||||
|         z64v.append(h_pos) | ||||
|  | ||||
|     # confusingly this doesn't bump if h_pos | ||||
|     req_ver = b"\x2d\x00" if z64 else b"\x0a\x00" | ||||
|  | ||||
|     if crc32: | ||||
|         crc32 = spack(b"<L", crc32) | ||||
|     else: | ||||
|         crc32 = b"\x00" * 4 | ||||
|  | ||||
|     if h_pos is None: | ||||
|         # 4b magic, 2b min-ver | ||||
|         ret = b"\x50\x4b\x03\x04" + req_ver | ||||
|     else: | ||||
|         # 4b magic, 2b spec-ver, 2b min-ver | ||||
|         ret = b"\x50\x4b\x01\x02\x1e\x03" + req_ver | ||||
|  | ||||
|     ret += b"\x00" if pre_crc else b"\x08"  # streaming | ||||
|     ret += b"\x08" if utf8 else b"\x00"  # appnote 6.3.2 (2007) | ||||
|  | ||||
|     # 2b compression, 4b time, 4b crc | ||||
|     ret += b"\x00\x00" + unixtime2dos(lastmod) + crc32 | ||||
|  | ||||
|     # spec says to put zeros when !crc if bit3 (streaming) | ||||
|     # however infozip does actual sz and it even works on winxp | ||||
|     # (same reasning for z64 extradata later) | ||||
|     vsz = 0xFFFFFFFF if z64 else sz | ||||
|     ret += spack(b"<LL", vsz, vsz) | ||||
|  | ||||
|     # windows support (the "?" replace below too) | ||||
|     fn = sanitize_fn(fn, "/", []) | ||||
|     bfn = fn.encode("utf-8" if utf8 else "cp437", "replace").replace(b"?", b"_") | ||||
|  | ||||
|     z64_len = len(z64v) * 8 + 4 if z64v else 0 | ||||
|     ret += spack(b"<HH", len(bfn), z64_len) | ||||
|  | ||||
|     if h_pos is not None: | ||||
|         # 2b comment, 2b diskno | ||||
|         ret += b"\x00" * 4 | ||||
|  | ||||
|         # 2b internal.attr, 4b external.attr | ||||
|         # infozip-macos: 0100 0000 a481 file:644 | ||||
|         # infozip-macos: 0100 0100 0080 file:000 | ||||
|         ret += b"\x01\x00\x00\x00\xa4\x81" | ||||
|  | ||||
|         # 4b local-header-ofs | ||||
|         ret += spack(b"<L", min(h_pos, 0xFFFFFFFF)) | ||||
|  | ||||
|     ret += bfn | ||||
|  | ||||
|     if z64v: | ||||
|         ret += spack(b"<HH" + b"Q" * len(z64v), 1, len(z64v) * 8, *z64v) | ||||
|  | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def gen_ecdr(items, cdir_pos, cdir_end): | ||||
|     """ | ||||
|     summary of all file headers, | ||||
|     usually the zipfile footer unless something clamps | ||||
|     """ | ||||
|  | ||||
|     ret = b"\x50\x4b\x05\x06" | ||||
|  | ||||
|     # 2b ndisk, 2b disk0 | ||||
|     ret += b"\x00" * 4 | ||||
|  | ||||
|     cdir_sz = cdir_end - cdir_pos | ||||
|  | ||||
|     nitems = min(0xFFFF, len(items)) | ||||
|     csz = min(0xFFFFFFFF, cdir_sz) | ||||
|     cpos = min(0xFFFFFFFF, cdir_pos) | ||||
|  | ||||
|     need_64 = nitems == 0xFFFF or 0xFFFFFFFF in [csz, cpos] | ||||
|  | ||||
|     # 2b tnfiles, 2b dnfiles, 4b dir sz, 4b dir pos | ||||
|     ret += spack(b"<HHLL", nitems, nitems, csz, cpos) | ||||
|  | ||||
|     # 2b comment length | ||||
|     ret += b"\x00\x00" | ||||
|  | ||||
|     return [ret, need_64] | ||||
|  | ||||
|  | ||||
| def gen_ecdr64(items, cdir_pos, cdir_end): | ||||
|     """ | ||||
|     z64 end of central directory | ||||
|     added when numfiles or a headerptr clamps | ||||
|     """ | ||||
|  | ||||
|     ret = b"\x50\x4b\x06\x06" | ||||
|  | ||||
|     # 8b own length from hereon | ||||
|     ret += b"\x2c" + b"\x00" * 7 | ||||
|  | ||||
|     # 2b spec-ver, 2b min-ver | ||||
|     ret += b"\x1e\x03\x2d\x00" | ||||
|  | ||||
|     # 4b ndisk, 4b disk0 | ||||
|     ret += b"\x00" * 8 | ||||
|  | ||||
|     # 8b tnfiles, 8b dnfiles, 8b dir sz, 8b dir pos | ||||
|     cdir_sz = cdir_end - cdir_pos | ||||
|     ret += spack(b"<QQQQ", len(items), len(items), cdir_sz, cdir_pos) | ||||
|  | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def gen_ecdr64_loc(ecdr64_pos): | ||||
|     """ | ||||
|     z64 end of central directory locator | ||||
|     points to ecdr64 | ||||
|     why | ||||
|     """ | ||||
|  | ||||
|     ret = b"\x50\x4b\x06\x07" | ||||
|  | ||||
|     # 4b cdisk, 8b start of ecdr64, 4b ndisks | ||||
|     ret += spack(b"<LQL", 0, ecdr64_pos, 1) | ||||
|  | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| class StreamZip(object): | ||||
|     def __init__(self, log, fgen, utf8=False, pre_crc=False): | ||||
|         self.log = log | ||||
|         self.fgen = fgen | ||||
|         self.utf8 = utf8 | ||||
|         self.pre_crc = pre_crc | ||||
|  | ||||
|         self.pos = 0 | ||||
|         self.items = [] | ||||
|  | ||||
|     def _ct(self, buf): | ||||
|         self.pos += len(buf) | ||||
|         return buf | ||||
|  | ||||
|     def ser(self, f): | ||||
|         name = f["vp"] | ||||
|         src = f["ap"] | ||||
|         st = f["st"] | ||||
|  | ||||
|         sz = st.st_size | ||||
|         ts = st.st_mtime + 1 | ||||
|  | ||||
|         crc = None | ||||
|         if self.pre_crc: | ||||
|             crc = 0 | ||||
|             for buf in yieldfile(src): | ||||
|                 crc = zlib.crc32(buf, crc) | ||||
|  | ||||
|             crc &= 0xFFFFFFFF | ||||
|  | ||||
|         h_pos = self.pos | ||||
|         buf = gen_hdr(None, name, sz, ts, self.utf8, crc, self.pre_crc) | ||||
|         yield self._ct(buf) | ||||
|  | ||||
|         crc = crc or 0 | ||||
|         for buf in yieldfile(src): | ||||
|             if not self.pre_crc: | ||||
|                 crc = zlib.crc32(buf, crc) | ||||
|  | ||||
|             yield self._ct(buf) | ||||
|  | ||||
|         crc &= 0xFFFFFFFF | ||||
|  | ||||
|         self.items.append([name, sz, ts, crc, h_pos]) | ||||
|  | ||||
|         z64 = sz >= 4 * 1024 * 1024 * 1024 | ||||
|  | ||||
|         if z64 or not self.pre_crc: | ||||
|             buf = gen_fdesc(sz, crc, z64) | ||||
|             yield self._ct(buf) | ||||
|  | ||||
|     def gen(self): | ||||
|         errors = [] | ||||
|         for f in self.fgen: | ||||
|             if "err" in f: | ||||
|                 errors.append([f["vp"], f["err"]]) | ||||
|                 continue | ||||
|  | ||||
|             try: | ||||
|                 for x in self.ser(f): | ||||
|                     yield x | ||||
|             except Exception as ex: | ||||
|                 errors.append([f["vp"], repr(ex)]) | ||||
|  | ||||
|         if errors: | ||||
|             errf, txt = errdesc(errors) | ||||
|             self.log("\n".join(([repr(errf)] + txt[1:]))) | ||||
|             for x in self.ser(errf): | ||||
|                 yield x | ||||
|  | ||||
|         cdir_pos = self.pos | ||||
|         for name, sz, ts, crc, h_pos in self.items: | ||||
|             buf = gen_hdr(h_pos, name, sz, ts, self.utf8, crc, self.pre_crc) | ||||
|             yield self._ct(buf) | ||||
|         cdir_end = self.pos | ||||
|  | ||||
|         _, need_64 = gen_ecdr(self.items, cdir_pos, cdir_end) | ||||
|         if need_64: | ||||
|             ecdir64_pos = self.pos | ||||
|             buf = gen_ecdr64(self.items, cdir_pos, cdir_end) | ||||
|             yield self._ct(buf) | ||||
|  | ||||
|             buf = gen_ecdr64_loc(ecdir64_pos) | ||||
|             yield self._ct(buf) | ||||
|  | ||||
|         ecdr, _ = gen_ecdr(self.items, cdir_pos, cdir_end) | ||||
|         yield self._ct(ecdr) | ||||
|  | ||||
|         if errors: | ||||
|             bos.unlink(errf["ap"]) | ||||
| @@ -2,10 +2,10 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import time | ||||
| import socket | ||||
|  | ||||
| from .util import chkcmd, Counter | ||||
| from .__init__ import MACOS, ANYWIN | ||||
| from .util import chkcmd | ||||
|  | ||||
|  | ||||
| class TcpSrv(object): | ||||
| @@ -19,76 +19,165 @@ class TcpSrv(object): | ||||
|         self.args = hub.args | ||||
|         self.log = hub.log | ||||
|  | ||||
|         self.num_clients = Counter() | ||||
|         self.stopping = False | ||||
|  | ||||
|         ip = "127.0.0.1" | ||||
|         eps = {ip: "local only"} | ||||
|         if self.args.i != ip: | ||||
|             eps = self.detect_interfaces(self.args.i) or eps | ||||
|         nonlocals = [x for x in self.args.i if x != ip] | ||||
|         if nonlocals: | ||||
|             eps = self.detect_interfaces(self.args.i) | ||||
|             if not eps: | ||||
|                 for x in nonlocals: | ||||
|                     eps[x] = "external" | ||||
|  | ||||
|         msgs = [] | ||||
|         m = "available @ http://{}:{}/  (\033[33m{}\033[0m)" | ||||
|         for ip, desc in sorted(eps.items(), key=lambda x: x[1]): | ||||
|             self.log( | ||||
|                 "tcpsrv", | ||||
|                 "available @ http://{}:{}/  (\033[33m{}\033[0m)".format( | ||||
|                     ip, self.args.p, desc | ||||
|                 ), | ||||
|             ) | ||||
|             for port in sorted(self.args.p): | ||||
|                 msgs.append(m.format(ip, port, desc)) | ||||
|  | ||||
|         self.srv = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | ||||
|         self.srv.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | ||||
|         if msgs: | ||||
|             msgs[-1] += "\n" | ||||
|             for m in msgs: | ||||
|                 self.log("tcpsrv", m) | ||||
|  | ||||
|         self.srv = [] | ||||
|         for ip in self.args.i: | ||||
|             for port in self.args.p: | ||||
|                 self.srv.append(self._listen(ip, port)) | ||||
|  | ||||
|     def _listen(self, ip, port): | ||||
|         srv = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | ||||
|         srv.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | ||||
|         srv.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1) | ||||
|         try: | ||||
|             self.srv.bind((self.args.i, self.args.p)) | ||||
|             srv.bind((ip, port)) | ||||
|             return srv | ||||
|         except (OSError, socket.error) as ex: | ||||
|             if ex.errno == 98: | ||||
|                 raise Exception( | ||||
|                     "\033[1;31mport {} is busy on interface {}\033[0m".format( | ||||
|                         self.args.p, self.args.i | ||||
|                     ) | ||||
|                 ) | ||||
|  | ||||
|             if ex.errno == 99: | ||||
|                 raise Exception( | ||||
|                     "\033[1;31minterface {} does not exist\033[0m".format(self.args.i) | ||||
|                 ) | ||||
|             if ex.errno in [98, 48]: | ||||
|                 e = "\033[1;31mport {} is busy on interface {}\033[0m".format(port, ip) | ||||
|             elif ex.errno in [99, 49]: | ||||
|                 e = "\033[1;31minterface {} does not exist\033[0m".format(ip) | ||||
|             else: | ||||
|                 raise | ||||
|             raise Exception(e) | ||||
|  | ||||
|     def run(self): | ||||
|         self.srv.listen(self.args.nc) | ||||
|         for srv in self.srv: | ||||
|             srv.listen(self.args.nc) | ||||
|             ip, port = srv.getsockname() | ||||
|             fno = srv.fileno() | ||||
|             msg = "listening @ {}:{}  f{}".format(ip, port, fno) | ||||
|             self.log("tcpsrv", msg) | ||||
|             if self.args.q: | ||||
|                 print(msg) | ||||
|  | ||||
|         self.log("tcpsrv", "listening @ {0}:{1}".format(self.args.i, self.args.p)) | ||||
|  | ||||
|         while True: | ||||
|             self.log("tcpsrv", "-" * 1 + "C-ncli") | ||||
|             if self.num_clients.v >= self.args.nc: | ||||
|                 time.sleep(0.1) | ||||
|                 continue | ||||
|  | ||||
|             self.log("tcpsrv", "-" * 2 + "C-acc1") | ||||
|             sck, addr = self.srv.accept() | ||||
|             self.log("%s %s" % addr, "-" * 3 + "C-acc2") | ||||
|             self.num_clients.add() | ||||
|             self.hub.broker.put(False, "httpconn", sck, addr) | ||||
|             self.hub.broker.put(False, "listen", srv) | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.stopping = True | ||||
|         try: | ||||
|             for srv in self.srv: | ||||
|                 srv.close() | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         self.log("tcpsrv", "ok bye") | ||||
|  | ||||
|     def detect_interfaces(self, listen_ip): | ||||
|     def ips_linux(self): | ||||
|         eps = {} | ||||
|  | ||||
|         # get all ips and their interfaces | ||||
|         try: | ||||
|             ip_addr, _ = chkcmd("ip", "addr") | ||||
|             txt, _ = chkcmd(["ip", "addr"]) | ||||
|         except: | ||||
|             ip_addr = None | ||||
|             return eps | ||||
|  | ||||
|         if ip_addr: | ||||
|             r = re.compile(r"^\s+inet ([^ ]+)/.* (.*)") | ||||
|             for ln in ip_addr.split("\n"): | ||||
|                 try: | ||||
|                     ip, dev = r.match(ln.rstrip()).groups() | ||||
|                     if listen_ip in ["0.0.0.0", ip]: | ||||
|                         eps[ip] = dev | ||||
|                 except: | ||||
|                     pass | ||||
|         r = re.compile(r"^\s+inet ([^ ]+)/.* (.*)") | ||||
|         for ln in txt.split("\n"): | ||||
|             try: | ||||
|                 ip, dev = r.match(ln.rstrip()).groups() | ||||
|                 eps[ip] = dev | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def ips_macos(self): | ||||
|         eps = {} | ||||
|         try: | ||||
|             txt, _ = chkcmd(["ifconfig"]) | ||||
|         except: | ||||
|             return eps | ||||
|  | ||||
|         rdev = re.compile(r"^([^ ]+):") | ||||
|         rip = re.compile(r"^\tinet ([0-9\.]+) ") | ||||
|         dev = None | ||||
|         for ln in txt.split("\n"): | ||||
|             m = rdev.match(ln) | ||||
|             if m: | ||||
|                 dev = m.group(1) | ||||
|  | ||||
|             m = rip.match(ln) | ||||
|             if m: | ||||
|                 eps[m.group(1)] = dev | ||||
|                 dev = None | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def ips_windows_ipconfig(self): | ||||
|         eps = {} | ||||
|         try: | ||||
|             txt, _ = chkcmd(["ipconfig"]) | ||||
|         except: | ||||
|             return eps | ||||
|  | ||||
|         rdev = re.compile(r"(^[^ ].*):$") | ||||
|         rip = re.compile(r"^ +IPv?4? [^:]+: *([0-9\.]{7,15})$") | ||||
|         dev = None | ||||
|         for ln in txt.replace("\r", "").split("\n"): | ||||
|             m = rdev.match(ln) | ||||
|             if m: | ||||
|                 dev = m.group(1).split(" adapter ", 1)[-1] | ||||
|  | ||||
|             m = rip.match(ln) | ||||
|             if m and dev: | ||||
|                 eps[m.group(1)] = dev | ||||
|                 dev = None | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def ips_windows_netsh(self): | ||||
|         eps = {} | ||||
|         try: | ||||
|             txt, _ = chkcmd("netsh interface ip show address".split()) | ||||
|         except: | ||||
|             return eps | ||||
|  | ||||
|         rdev = re.compile(r'.* "([^"]+)"$') | ||||
|         rip = re.compile(r".* IP\b.*: +([0-9\.]{7,15})$") | ||||
|         dev = None | ||||
|         for ln in txt.replace("\r", "").split("\n"): | ||||
|             m = rdev.match(ln) | ||||
|             if m: | ||||
|                 dev = m.group(1) | ||||
|  | ||||
|             m = rip.match(ln) | ||||
|             if m and dev: | ||||
|                 eps[m.group(1)] = dev | ||||
|                 dev = None | ||||
|  | ||||
|         return eps | ||||
|  | ||||
|     def detect_interfaces(self, listen_ips): | ||||
|         if MACOS: | ||||
|             eps = self.ips_macos() | ||||
|         elif ANYWIN: | ||||
|             eps = self.ips_windows_ipconfig()  # sees more interfaces | ||||
|             eps.update(self.ips_windows_netsh())  # has better names | ||||
|         else: | ||||
|             eps = self.ips_linux() | ||||
|  | ||||
|         if "0.0.0.0" not in listen_ips: | ||||
|             eps = {k: v for k, v in eps if k in listen_ips} | ||||
|  | ||||
|         default_route = None | ||||
|         s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) | ||||
| @@ -113,11 +202,12 @@ class TcpSrv(object): | ||||
|  | ||||
|         s.close() | ||||
|  | ||||
|         if default_route and listen_ip in ["0.0.0.0", default_route]: | ||||
|             desc = "\033[32mexternal" | ||||
|             try: | ||||
|                 eps[default_route] += ", " + desc | ||||
|             except: | ||||
|                 eps[default_route] = desc | ||||
|         for lip in listen_ips: | ||||
|             if default_route and lip in ["0.0.0.0", default_route]: | ||||
|                 desc = "\033[32mexternal" | ||||
|                 try: | ||||
|                     eps[default_route] += ", " + desc | ||||
|                 except: | ||||
|                     eps[default_route] = desc | ||||
|  | ||||
|         return eps | ||||
|   | ||||
							
								
								
									
										56
									
								
								copyparty/th_cli.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								copyparty/th_cli.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
|  | ||||
| from .util import Cooldown | ||||
| from .th_srv import thumb_path, THUMBABLE, FMT_FF | ||||
| from .bos import bos | ||||
|  | ||||
|  | ||||
| class ThumbCli(object): | ||||
|     def __init__(self, broker): | ||||
|         self.broker = broker | ||||
|         self.args = broker.args | ||||
|         self.asrv = broker.asrv | ||||
|  | ||||
|         # cache on both sides for less broker spam | ||||
|         self.cooldown = Cooldown(self.args.th_poke) | ||||
|  | ||||
|     def get(self, ptop, rem, mtime, fmt): | ||||
|         ext = rem.rsplit(".")[-1].lower() | ||||
|         if ext not in THUMBABLE: | ||||
|             return None | ||||
|  | ||||
|         is_vid = ext in FMT_FF | ||||
|         if is_vid and self.args.no_vthumb: | ||||
|             return None | ||||
|  | ||||
|         if fmt == "j" and self.args.th_no_jpg: | ||||
|             fmt = "w" | ||||
|  | ||||
|         if fmt == "w": | ||||
|             if self.args.th_no_webp or (is_vid and self.args.th_ff_jpg): | ||||
|                 fmt = "j" | ||||
|  | ||||
|         histpath = self.asrv.vfs.histtab[ptop] | ||||
|         tpath = thumb_path(histpath, rem, mtime, fmt) | ||||
|         ret = None | ||||
|         try: | ||||
|             st = bos.stat(tpath) | ||||
|             if st.st_size: | ||||
|                 ret = tpath | ||||
|             else: | ||||
|                 return None | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         if ret: | ||||
|             tdir = os.path.dirname(tpath) | ||||
|             if self.cooldown.poke(tdir): | ||||
|                 self.broker.put(False, "thumbsrv.poke", tdir) | ||||
|  | ||||
|             return ret | ||||
|  | ||||
|         x = self.broker.put(True, "thumbsrv.get", ptop, rem, mtime, fmt) | ||||
|         return x.get() | ||||
							
								
								
									
										399
									
								
								copyparty/th_srv.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										399
									
								
								copyparty/th_srv.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,399 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import time | ||||
| import shutil | ||||
| import base64 | ||||
| import hashlib | ||||
| import threading | ||||
| import subprocess as sp | ||||
|  | ||||
| from .__init__ import PY2, unicode | ||||
| from .util import fsenc, vsplit, runcmd, Queue, Cooldown, BytesIO, min_ex | ||||
| from .bos import bos | ||||
| from .mtag import HAVE_FFMPEG, HAVE_FFPROBE, ffprobe | ||||
|  | ||||
|  | ||||
| HAVE_PIL = False | ||||
| HAVE_HEIF = False | ||||
| HAVE_AVIF = False | ||||
| HAVE_WEBP = False | ||||
|  | ||||
| try: | ||||
|     from PIL import Image, ImageOps | ||||
|  | ||||
|     HAVE_PIL = True | ||||
|     try: | ||||
|         Image.new("RGB", (2, 2)).save(BytesIO(), format="webp") | ||||
|         HAVE_WEBP = True | ||||
|     except: | ||||
|         pass | ||||
|  | ||||
|     try: | ||||
|         from pyheif_pillow_opener import register_heif_opener | ||||
|  | ||||
|         register_heif_opener() | ||||
|         HAVE_HEIF = True | ||||
|     except: | ||||
|         pass | ||||
|  | ||||
|     try: | ||||
|         import pillow_avif | ||||
|  | ||||
|         HAVE_AVIF = True | ||||
|     except: | ||||
|         pass | ||||
| except: | ||||
|     pass | ||||
|  | ||||
| # https://pillow.readthedocs.io/en/stable/handbook/image-file-formats.html | ||||
| # ffmpeg -formats | ||||
| FMT_PIL = "bmp dib gif icns ico jpg jpeg jp2 jpx pcx png pbm pgm ppm pnm sgi tga tif tiff webp xbm dds xpm" | ||||
| FMT_FF = "av1 asf avi flv m4v mkv mjpeg mjpg mpg mpeg mpg2 mpeg2 h264 avc mts h265 hevc mov 3gp mp4 ts mpegts nut ogv ogm rm vob webm wmv" | ||||
|  | ||||
| if HAVE_HEIF: | ||||
|     FMT_PIL += " heif heifs heic heics" | ||||
|  | ||||
| if HAVE_AVIF: | ||||
|     FMT_PIL += " avif avifs" | ||||
|  | ||||
| FMT_PIL, FMT_FF = [{x: True for x in y.split(" ") if x} for y in [FMT_PIL, FMT_FF]] | ||||
|  | ||||
|  | ||||
| THUMBABLE = {} | ||||
|  | ||||
| if HAVE_PIL: | ||||
|     THUMBABLE.update(FMT_PIL) | ||||
|  | ||||
| if HAVE_FFMPEG and HAVE_FFPROBE: | ||||
|     THUMBABLE.update(FMT_FF) | ||||
|  | ||||
|  | ||||
| def thumb_path(histpath, rem, mtime, fmt): | ||||
|     # base16 = 16 = 256 | ||||
|     # b64-lc = 38 = 1444 | ||||
|     # base64 = 64 = 4096 | ||||
|     rd, fn = vsplit(rem) | ||||
|     if rd: | ||||
|         h = hashlib.sha512(fsenc(rd)).digest() | ||||
|         b64 = base64.urlsafe_b64encode(h).decode("ascii")[:24] | ||||
|         rd = "{}/{}/".format(b64[:2], b64[2:4]).lower() + b64 | ||||
|     else: | ||||
|         rd = "top" | ||||
|  | ||||
|     # could keep original filenames but this is safer re pathlen | ||||
|     h = hashlib.sha512(fsenc(fn)).digest() | ||||
|     fn = base64.urlsafe_b64encode(h).decode("ascii")[:24] | ||||
|  | ||||
|     return "{}/th/{}/{}.{:x}.{}".format( | ||||
|         histpath, rd, fn, int(mtime), "webp" if fmt == "w" else "jpg" | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class ThumbSrv(object): | ||||
|     def __init__(self, hub): | ||||
|         self.hub = hub | ||||
|         self.asrv = hub.asrv | ||||
|         self.args = hub.args | ||||
|         self.log_func = hub.log | ||||
|  | ||||
|         res = hub.args.th_size.split("x") | ||||
|         self.res = tuple([int(x) for x in res]) | ||||
|         self.poke_cd = Cooldown(self.args.th_poke) | ||||
|  | ||||
|         self.mutex = threading.Lock() | ||||
|         self.busy = {} | ||||
|         self.stopping = False | ||||
|         self.nthr = os.cpu_count() if hasattr(os, "cpu_count") else 4 | ||||
|         self.q = Queue(self.nthr * 4) | ||||
|         for n in range(self.nthr): | ||||
|             t = threading.Thread( | ||||
|                 target=self.worker, name="thumb-{}-{}".format(n, self.nthr) | ||||
|             ) | ||||
|             t.daemon = True | ||||
|             t.start() | ||||
|  | ||||
|         if not self.args.no_vthumb and (not HAVE_FFMPEG or not HAVE_FFPROBE): | ||||
|             missing = [] | ||||
|             if not HAVE_FFMPEG: | ||||
|                 missing.append("FFmpeg") | ||||
|  | ||||
|             if not HAVE_FFPROBE: | ||||
|                 missing.append("FFprobe") | ||||
|  | ||||
|             msg = "cannot create video thumbnails because some of the required programs are not available: " | ||||
|             msg += ", ".join(missing) | ||||
|             self.log(msg, c=3) | ||||
|  | ||||
|         if self.args.th_clean: | ||||
|             t = threading.Thread(target=self.cleaner, name="thumb-cleaner") | ||||
|             t.daemon = True | ||||
|             t.start() | ||||
|  | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func("thumb", msg, c) | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.stopping = True | ||||
|         for _ in range(self.nthr): | ||||
|             self.q.put(None) | ||||
|  | ||||
|     def stopped(self): | ||||
|         with self.mutex: | ||||
|             return not self.nthr | ||||
|  | ||||
|     def get(self, ptop, rem, mtime, fmt): | ||||
|         histpath = self.asrv.vfs.histtab[ptop] | ||||
|         tpath = thumb_path(histpath, rem, mtime, fmt) | ||||
|         abspath = os.path.join(ptop, rem) | ||||
|         cond = threading.Condition(self.mutex) | ||||
|         do_conv = False | ||||
|         with self.mutex: | ||||
|             try: | ||||
|                 self.busy[tpath].append(cond) | ||||
|                 self.log("wait {}".format(tpath)) | ||||
|             except: | ||||
|                 thdir = os.path.dirname(tpath) | ||||
|                 bos.makedirs(thdir) | ||||
|  | ||||
|                 inf_path = os.path.join(thdir, "dir.txt") | ||||
|                 if not bos.path.exists(inf_path): | ||||
|                     with open(inf_path, "wb") as f: | ||||
|                         f.write(fsenc(os.path.dirname(abspath))) | ||||
|  | ||||
|                 self.busy[tpath] = [cond] | ||||
|                 do_conv = True | ||||
|  | ||||
|         if do_conv: | ||||
|             self.q.put([abspath, tpath]) | ||||
|             self.log("conv {} \033[0m{}".format(tpath, abspath), c=6) | ||||
|  | ||||
|         while not self.stopping: | ||||
|             with self.mutex: | ||||
|                 if tpath not in self.busy: | ||||
|                     break | ||||
|  | ||||
|             with cond: | ||||
|                 cond.wait(3) | ||||
|  | ||||
|         try: | ||||
|             st = bos.stat(tpath) | ||||
|             if st.st_size: | ||||
|                 return tpath | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         return None | ||||
|  | ||||
|     def worker(self): | ||||
|         while not self.stopping: | ||||
|             task = self.q.get() | ||||
|             if not task: | ||||
|                 break | ||||
|  | ||||
|             abspath, tpath = task | ||||
|             ext = abspath.split(".")[-1].lower() | ||||
|             fun = None | ||||
|             if not bos.path.exists(tpath): | ||||
|                 if ext in FMT_PIL: | ||||
|                     fun = self.conv_pil | ||||
|                 elif ext in FMT_FF: | ||||
|                     fun = self.conv_ffmpeg | ||||
|  | ||||
|             if fun: | ||||
|                 try: | ||||
|                     fun(abspath, tpath) | ||||
|                 except: | ||||
|                     msg = "{} failed on {}\n{}" | ||||
|                     self.log(msg.format(fun.__name__, abspath, min_ex()), 3) | ||||
|                     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 conv_pil(self, abspath, tpath): | ||||
|         with Image.open(fsenc(abspath)) as im: | ||||
|             crop = not self.args.th_no_crop | ||||
|             res2 = self.res | ||||
|             if crop: | ||||
|                 res2 = (res2[0] * 2, res2[1] * 2) | ||||
|  | ||||
|             try: | ||||
|                 im.thumbnail(res2, resample=Image.LANCZOS) | ||||
|                 if crop: | ||||
|                     iw, ih = im.size | ||||
|                     dw, dh = self.res | ||||
|                     res = (min(iw, dw), min(ih, dh)) | ||||
|                     im = ImageOps.fit(im, res, method=Image.LANCZOS) | ||||
|             except: | ||||
|                 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: | ||||
|                 pass  # default q = 75 | ||||
|  | ||||
|             if im.mode not in fmts: | ||||
|                 # print("conv {}".format(im.mode)) | ||||
|                 im = im.convert("RGB") | ||||
|  | ||||
|             im.save(tpath, quality=40, method=6) | ||||
|  | ||||
|     def conv_ffmpeg(self, abspath, tpath): | ||||
|         ret, _ = ffprobe(abspath) | ||||
|  | ||||
|         ext = abspath.rsplit(".")[-1] | ||||
|         if ext in ["h264", "h265"]: | ||||
|             seek = [] | ||||
|         else: | ||||
|             dur = ret[".dur"][1] if ".dur" in ret else 4 | ||||
|             seek = "{:.0f}".format(dur / 3) | ||||
|             seek = [b"-ss", seek.encode("utf-8")] | ||||
|  | ||||
|         scale = "scale={0}:{1}:force_original_aspect_ratio=" | ||||
|         if self.args.th_no_crop: | ||||
|             scale += "decrease,setsar=1:1" | ||||
|         else: | ||||
|             scale += "increase,crop={0}:{1},setsar=1:1" | ||||
|  | ||||
|         scale = scale.format(*list(self.res)).encode("utf-8") | ||||
|         # fmt: off | ||||
|         cmd = [ | ||||
|             b"ffmpeg", | ||||
|             b"-nostdin", | ||||
|             b"-v", b"error", | ||||
|             b"-hide_banner" | ||||
|         ] | ||||
|         cmd += seek | ||||
|         cmd += [ | ||||
|             b"-i", fsenc(abspath), | ||||
|             b"-vf", scale, | ||||
|             b"-vframes", b"1", | ||||
|         ] | ||||
|         # 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)] | ||||
|  | ||||
|         ret, sout, serr = runcmd(cmd) | ||||
|         if ret != 0: | ||||
|             msg = ["ff: {}".format(x) for x in serr.split("\n")] | ||||
|             self.log("FFmpeg failed:\n" + "\n".join(msg), c="1;30") | ||||
|             raise sp.CalledProcessError(ret, (cmd[0], b"...", cmd[-1])) | ||||
|  | ||||
|     def poke(self, tdir): | ||||
|         if not self.poke_cd.poke(tdir): | ||||
|             return | ||||
|  | ||||
|         ts = int(time.time()) | ||||
|         try: | ||||
|             p1 = os.path.dirname(tdir) | ||||
|             p2 = os.path.dirname(p1) | ||||
|             for dp in [tdir, p1, p2]: | ||||
|                 bos.utime(dp, (ts, ts)) | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     def cleaner(self): | ||||
|         interval = self.args.th_clean | ||||
|         while True: | ||||
|             time.sleep(interval) | ||||
|             ndirs = 0 | ||||
|             for vol, histpath in self.asrv.vfs.histtab.items(): | ||||
|                 if histpath.startswith(vol): | ||||
|                     self.log("\033[Jcln {}/\033[A".format(histpath)) | ||||
|                 else: | ||||
|                     self.log("\033[Jcln {} ({})/\033[A".format(histpath, vol)) | ||||
|  | ||||
|                 ndirs += self.clean(histpath) | ||||
|  | ||||
|             self.log("\033[Jcln ok; rm {} dirs".format(ndirs)) | ||||
|  | ||||
|     def clean(self, histpath): | ||||
|         thumbpath = os.path.join(histpath, "th") | ||||
|         # self.log("cln {}".format(thumbpath)) | ||||
|         maxage = self.args.th_maxage | ||||
|         now = time.time() | ||||
|         prev_b64 = None | ||||
|         prev_fp = None | ||||
|         try: | ||||
|             ents = bos.listdir(thumbpath) | ||||
|         except: | ||||
|             return 0 | ||||
|  | ||||
|         ndirs = 0 | ||||
|         for f in sorted(ents): | ||||
|             fp = os.path.join(thumbpath, f) | ||||
|             cmp = fp.lower().replace("\\", "/") | ||||
|  | ||||
|             # "top" or b64 prefix/full (a folder) | ||||
|             if len(f) <= 3 or len(f) == 24: | ||||
|                 age = now - bos.path.getmtime(fp) | ||||
|                 if age > maxage: | ||||
|                     with self.mutex: | ||||
|                         safe = True | ||||
|                         for k in self.busy.keys(): | ||||
|                             if k.lower().replace("\\", "/").startswith(cmp): | ||||
|                                 safe = False | ||||
|                                 break | ||||
|  | ||||
|                         if safe: | ||||
|                             ndirs += 1 | ||||
|                             self.log("rm -rf [{}]".format(fp)) | ||||
|                             shutil.rmtree(fp, ignore_errors=True) | ||||
|                 else: | ||||
|                     ndirs += self.clean(fp) | ||||
|                 continue | ||||
|  | ||||
|             # thumb file | ||||
|             try: | ||||
|                 b64, ts, ext = f.split(".") | ||||
|                 if len(b64) != 24 or len(ts) != 8 or ext not in ["jpg", "webp"]: | ||||
|                     raise Exception() | ||||
|  | ||||
|                 ts = int(ts, 16) | ||||
|             except: | ||||
|                 if f != "dir.txt": | ||||
|                     self.log("foreign file in thumbs dir: [{}]".format(fp), 1) | ||||
|  | ||||
|                 continue | ||||
|  | ||||
|             if b64 == prev_b64: | ||||
|                 self.log("rm replaced [{}]".format(fp)) | ||||
|                 bos.unlink(prev_fp) | ||||
|  | ||||
|             prev_b64 = b64 | ||||
|             prev_fp = fp | ||||
|  | ||||
|         return ndirs | ||||
							
								
								
									
										290
									
								
								copyparty/u2idx.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										290
									
								
								copyparty/u2idx.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,290 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import time | ||||
| import threading | ||||
| from datetime import datetime | ||||
|  | ||||
| from .__init__ import unicode | ||||
| from .util import s3dec, Pebkac, min_ex | ||||
| from .bos import bos | ||||
| from .up2k import up2k_wark_from_hashlist | ||||
|  | ||||
|  | ||||
| try: | ||||
|     HAVE_SQLITE3 = True | ||||
|     import sqlite3 | ||||
| except: | ||||
|     HAVE_SQLITE3 = False | ||||
|  | ||||
|  | ||||
| class U2idx(object): | ||||
|     def __init__(self, conn): | ||||
|         self.log_func = conn.log_func | ||||
|         self.asrv = conn.asrv | ||||
|         self.args = conn.args | ||||
|         self.timeout = self.args.srch_time | ||||
|  | ||||
|         if not HAVE_SQLITE3: | ||||
|             self.log("your python does not have sqlite3; searching will be disabled") | ||||
|             return | ||||
|  | ||||
|         self.cur = {} | ||||
|         self.mem_cur = sqlite3.connect(":memory:") | ||||
|         self.mem_cur.execute(r"create table a (b text)") | ||||
|  | ||||
|         self.p_end = None | ||||
|         self.p_dur = 0 | ||||
|  | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func("u2idx", msg, c) | ||||
|  | ||||
|     def fsearch(self, vols, body): | ||||
|         """search by up2k hashlist""" | ||||
|         if not HAVE_SQLITE3: | ||||
|             return [] | ||||
|  | ||||
|         fsize = body["size"] | ||||
|         fhash = body["hash"] | ||||
|         wark = up2k_wark_from_hashlist(self.args.salt, fsize, fhash) | ||||
|  | ||||
|         uq = "where substr(w,1,16) = ? and w = ?" | ||||
|         uv = [wark[:16], wark] | ||||
|  | ||||
|         try: | ||||
|             return self.run_query(vols, uq, uv)[0] | ||||
|         except: | ||||
|             raise Pebkac(500, min_ex()) | ||||
|  | ||||
|     def get_cur(self, ptop): | ||||
|         if not HAVE_SQLITE3: | ||||
|             return None | ||||
|  | ||||
|         cur = self.cur.get(ptop) | ||||
|         if cur: | ||||
|             return cur | ||||
|  | ||||
|         histpath = self.asrv.vfs.histtab[ptop] | ||||
|         db_path = os.path.join(histpath, "up2k.db") | ||||
|         if not bos.path.exists(db_path): | ||||
|             return None | ||||
|  | ||||
|         cur = sqlite3.connect(db_path, 2).cursor() | ||||
|         self.cur[ptop] = cur | ||||
|         return cur | ||||
|  | ||||
|     def search(self, vols, uq): | ||||
|         """search by query params""" | ||||
|         if not HAVE_SQLITE3: | ||||
|             return [] | ||||
|  | ||||
|         q = "" | ||||
|         va = [] | ||||
|         joins = "" | ||||
|         is_key = True | ||||
|         is_size = False | ||||
|         is_date = False | ||||
|         kw_key = ["(", ")", "and ", "or ", "not "] | ||||
|         kw_val = ["==", "=", "!=", ">", ">=", "<", "<=", "like "] | ||||
|         ptn_mt = re.compile(r"^\.?[a-z]+$") | ||||
|         mt_ctr = 0 | ||||
|         mt_keycmp = "substr(up.w,1,16)" | ||||
|         mt_keycmp2 = None | ||||
|         ptn_lc = re.compile(r" (mt[0-9]+\.v) ([=<!>]+) \? $") | ||||
|         ptn_lcv = re.compile(r"[a-zA-Z]") | ||||
|  | ||||
|         while True: | ||||
|             uq = uq.strip() | ||||
|             if not uq: | ||||
|                 break | ||||
|  | ||||
|             ok = False | ||||
|             for kw in kw_key + kw_val: | ||||
|                 if uq.startswith(kw): | ||||
|                     is_key = kw in kw_key | ||||
|                     uq = uq[len(kw) :] | ||||
|                     ok = True | ||||
|                     q += kw | ||||
|                     break | ||||
|  | ||||
|             if ok: | ||||
|                 continue | ||||
|  | ||||
|             v, uq = (uq + " ").split(" ", 1) | ||||
|             if is_key: | ||||
|                 is_key = False | ||||
|  | ||||
|                 if v == "size": | ||||
|                     v = "up.sz" | ||||
|                     is_size = True | ||||
|  | ||||
|                 elif v == "date": | ||||
|                     v = "up.mt" | ||||
|                     is_date = True | ||||
|  | ||||
|                 elif v == "path": | ||||
|                     v = "up.rd" | ||||
|  | ||||
|                 elif v == "name": | ||||
|                     v = "up.fn" | ||||
|  | ||||
|                 elif v == "tags" or ptn_mt.match(v): | ||||
|                     mt_ctr += 1 | ||||
|                     mt_keycmp2 = "mt{}.w".format(mt_ctr) | ||||
|                     joins += "inner join mt mt{} on {} = {} ".format( | ||||
|                         mt_ctr, mt_keycmp, mt_keycmp2 | ||||
|                     ) | ||||
|                     mt_keycmp = mt_keycmp2 | ||||
|                     if v == "tags": | ||||
|                         v = "mt{0}.v".format(mt_ctr) | ||||
|                     else: | ||||
|                         v = "+mt{0}.k = '{1}' and mt{0}.v".format(mt_ctr, v) | ||||
|  | ||||
|                 else: | ||||
|                     raise Pebkac(400, "invalid key [" + v + "]") | ||||
|  | ||||
|                 q += v + " " | ||||
|                 continue | ||||
|  | ||||
|             head = "" | ||||
|             tail = "" | ||||
|  | ||||
|             if is_date: | ||||
|                 is_date = False | ||||
|                 v = v.upper().rstrip("Z").replace(",", " ").replace("T", " ") | ||||
|                 while "  " in v: | ||||
|                     v = v.replace("  ", " ") | ||||
|  | ||||
|                 for fmt in [ | ||||
|                     "%Y-%m-%d %H:%M:%S", | ||||
|                     "%Y-%m-%d %H:%M", | ||||
|                     "%Y-%m-%d %H", | ||||
|                     "%Y-%m-%d", | ||||
|                 ]: | ||||
|                     try: | ||||
|                         v = datetime.strptime(v, fmt).timestamp() | ||||
|                         break | ||||
|                     except: | ||||
|                         pass | ||||
|  | ||||
|             elif is_size: | ||||
|                 is_size = False | ||||
|                 v = int(float(v) * 1024 * 1024) | ||||
|  | ||||
|             else: | ||||
|                 if v.startswith("*"): | ||||
|                     head = "'%'||" | ||||
|                     v = v[1:] | ||||
|  | ||||
|                 if v.endswith("*"): | ||||
|                     tail = "||'%'" | ||||
|                     v = v[:-1] | ||||
|  | ||||
|             q += " {}?{} ".format(head, tail) | ||||
|             va.append(v) | ||||
|             is_key = True | ||||
|  | ||||
|             # lowercase tag searches | ||||
|             m = ptn_lc.search(q) | ||||
|             if not m or not ptn_lcv.search(unicode(v)): | ||||
|                 continue | ||||
|  | ||||
|             va.pop() | ||||
|             va.append(v.lower()) | ||||
|             q = q[: m.start()] | ||||
|  | ||||
|             field, oper = m.groups() | ||||
|             if oper in ["=", "=="]: | ||||
|                 q += " {} like ? ".format(field) | ||||
|             else: | ||||
|                 q += " lower({}) {} ? ".format(field, oper) | ||||
|  | ||||
|         try: | ||||
|             return self.run_query(vols, joins + "where " + q, va) | ||||
|         except Exception as ex: | ||||
|             raise Pebkac(500, repr(ex)) | ||||
|  | ||||
|     def run_query(self, vols, uq, uv): | ||||
|         done_flag = [] | ||||
|         self.active_id = "{:.6f}_{}".format( | ||||
|             time.time(), threading.current_thread().ident | ||||
|         ) | ||||
|         thr = threading.Thread( | ||||
|             target=self.terminator, | ||||
|             args=( | ||||
|                 self.active_id, | ||||
|                 done_flag, | ||||
|             ), | ||||
|             name="u2idx-terminator", | ||||
|         ) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|         if not uq or not uv: | ||||
|             q = "select * from up" | ||||
|             v = () | ||||
|         else: | ||||
|             q = "select up.* from up " + uq | ||||
|             v = tuple(uv) | ||||
|  | ||||
|         self.log("qs: {!r} {!r}".format(q, v)) | ||||
|  | ||||
|         ret = [] | ||||
|         lim = 1000 | ||||
|         taglist = {} | ||||
|         for (vtop, ptop, flags) in vols: | ||||
|             cur = self.get_cur(ptop) | ||||
|             if not cur: | ||||
|                 continue | ||||
|  | ||||
|             self.active_cur = cur | ||||
|  | ||||
|             sret = [] | ||||
|             c = cur.execute(q, v) | ||||
|             for hit in c: | ||||
|                 w, ts, sz, rd, fn, ip, at = hit | ||||
|                 lim -= 1 | ||||
|                 if lim <= 0: | ||||
|                     break | ||||
|  | ||||
|                 if rd.startswith("//") or fn.startswith("//"): | ||||
|                     rd, fn = s3dec(rd, fn) | ||||
|  | ||||
|                 rp = "/".join([x for x in [vtop, rd, fn] if x]) | ||||
|                 sret.append({"ts": int(ts), "sz": sz, "rp": rp, "w": w[:16]}) | ||||
|  | ||||
|             for hit in sret: | ||||
|                 w = hit["w"] | ||||
|                 del hit["w"] | ||||
|                 tags = {} | ||||
|                 q2 = "select k, v from mt where w = ? and k != 'x'" | ||||
|                 for k, v2 in cur.execute(q2, (w,)): | ||||
|                     taglist[k] = True | ||||
|                     tags[k] = v2 | ||||
|  | ||||
|                 hit["tags"] = tags | ||||
|  | ||||
|             ret.extend(sret) | ||||
|             # print("[{}] {}".format(ptop, sret)) | ||||
|  | ||||
|         done_flag.append(True) | ||||
|         self.active_id = None | ||||
|  | ||||
|         # undupe hits from multiple metadata keys | ||||
|         if len(ret) > 1: | ||||
|             ret = [ret[0]] + [ | ||||
|                 y for x, y in zip(ret[:-1], ret[1:]) if x["rp"] != y["rp"] | ||||
|             ] | ||||
|  | ||||
|         return ret, list(taglist.keys()) | ||||
|  | ||||
|     def terminator(self, identifier, done_flag): | ||||
|         for _ in range(self.timeout): | ||||
|             time.sleep(1) | ||||
|             if done_flag: | ||||
|                 return | ||||
|  | ||||
|         if identifier == self.active_id: | ||||
|             self.active_cur.connection.interrupt() | ||||
							
								
								
									
										1841
									
								
								copyparty/up2k.py
									
									
									
									
									
								
							
							
						
						
									
										1841
									
								
								copyparty/up2k.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -2,16 +2,24 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import stat | ||||
| import time | ||||
| import base64 | ||||
| import select | ||||
| import struct | ||||
| import hashlib | ||||
| import platform | ||||
| import traceback | ||||
| import threading | ||||
| import mimetypes | ||||
| import contextlib | ||||
| import subprocess as sp  # nosec | ||||
| from datetime import datetime | ||||
| from collections import Counter | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS | ||||
| from .__init__ import PY2, WINDOWS, ANYWIN | ||||
| from .stolen import surrogateescape | ||||
|  | ||||
| FAKE_MP = False | ||||
| @@ -29,10 +37,26 @@ if not PY2: | ||||
|     from urllib.parse import unquote_to_bytes as unquote | ||||
|     from urllib.parse import quote_from_bytes as quote | ||||
|     from queue import Queue | ||||
|     from io import BytesIO | ||||
| else: | ||||
|     from urllib import unquote  # pylint: disable=no-name-in-module | ||||
|     from urllib import quote  # pylint: disable=no-name-in-module | ||||
|     from Queue import Queue  # pylint: disable=import-error,no-name-in-module | ||||
|     from StringIO import StringIO as BytesIO | ||||
|  | ||||
|  | ||||
| try: | ||||
|     struct.unpack(b">i", b"idgi") | ||||
|     spack = struct.pack | ||||
|     sunpack = struct.unpack | ||||
| except: | ||||
|  | ||||
|     def spack(f, *a, **ka): | ||||
|         return struct.pack(f.decode("ascii"), *a, **ka) | ||||
|  | ||||
|     def sunpack(f, *a, **ka): | ||||
|         return struct.unpack(f.decode("ascii"), *a, **ka) | ||||
|  | ||||
|  | ||||
| surrogateescape.register_surrogateescape() | ||||
| FS_ENCODING = sys.getfilesystemencoding() | ||||
| @@ -40,33 +64,105 @@ if WINDOWS and PY2: | ||||
|     FS_ENCODING = "utf-8" | ||||
|  | ||||
|  | ||||
| HTTP_TS_FMT = "%a, %d %b %Y %H:%M:%S GMT" | ||||
|  | ||||
|  | ||||
| HTTPCODE = { | ||||
|     200: "OK", | ||||
|     204: "No Content", | ||||
|     206: "Partial Content", | ||||
|     302: "Found", | ||||
|     304: "Not Modified", | ||||
|     400: "Bad Request", | ||||
|     403: "Forbidden", | ||||
|     404: "Not Found", | ||||
|     405: "Method Not Allowed", | ||||
|     413: "Payload Too Large", | ||||
|     416: "Requested Range Not Satisfiable", | ||||
|     422: "Unprocessable Entity", | ||||
|     429: "Too Many Requests", | ||||
|     500: "Internal Server Error", | ||||
|     501: "Not Implemented", | ||||
| } | ||||
|  | ||||
|  | ||||
| class Counter(object): | ||||
|     def __init__(self, v=0): | ||||
|         self.v = v | ||||
| IMPLICATIONS = [ | ||||
|     ["e2dsa", "e2ds"], | ||||
|     ["e2ds", "e2d"], | ||||
|     ["e2tsr", "e2ts"], | ||||
|     ["e2ts", "e2t"], | ||||
|     ["e2t", "e2d"], | ||||
| ] | ||||
|  | ||||
|  | ||||
| MIMES = { | ||||
|     "md": "text/plain; charset=UTF-8", | ||||
|     "opus": "audio/ogg; codecs=opus", | ||||
|     "webp": "image/webp", | ||||
| } | ||||
|  | ||||
|  | ||||
| REKOBO_KEY = { | ||||
|     v: ln.split(" ", 1)[0] | ||||
|     for ln in """ | ||||
| 1B 6d B | ||||
| 2B 7d Gb F# | ||||
| 3B 8d Db C# | ||||
| 4B 9d Ab G# | ||||
| 5B 10d Eb D# | ||||
| 6B 11d Bb A# | ||||
| 7B 12d F | ||||
| 8B 1d C | ||||
| 9B 2d G | ||||
| 10B 3d D | ||||
| 11B 4d A | ||||
| 12B 5d E | ||||
| 1A 6m Abm G#m | ||||
| 2A 7m Ebm D#m | ||||
| 3A 8m Bbm A#m | ||||
| 4A 9m Fm | ||||
| 5A 10m Cm | ||||
| 6A 11m Gm | ||||
| 7A 12m Dm | ||||
| 8A 1m Am | ||||
| 9A 2m Em | ||||
| 10A 3m Bm | ||||
| 11A 4m Gbm F#m | ||||
| 12A 5m Dbm C#m | ||||
| """.strip().split( | ||||
|         "\n" | ||||
|     ) | ||||
|     for v in ln.strip().split(" ")[1:] | ||||
|     if v | ||||
| } | ||||
|  | ||||
| REKOBO_LKEY = {k.lower(): v for k, v in REKOBO_KEY.items()} | ||||
|  | ||||
|  | ||||
| class Cooldown(object): | ||||
|     def __init__(self, maxage): | ||||
|         self.maxage = maxage | ||||
|         self.mutex = threading.Lock() | ||||
|         self.hist = {} | ||||
|         self.oldest = 0 | ||||
|  | ||||
|     def add(self, delta=1): | ||||
|     def poke(self, key): | ||||
|         with self.mutex: | ||||
|             self.v += delta | ||||
|             now = time.time() | ||||
|  | ||||
|     def set(self, absval): | ||||
|         with self.mutex: | ||||
|             self.v = absval | ||||
|             ret = False | ||||
|             v = self.hist.get(key, 0) | ||||
|             if now - v > self.maxage: | ||||
|                 self.hist[key] = now | ||||
|                 ret = True | ||||
|  | ||||
|             if self.oldest - now > self.maxage * 2: | ||||
|                 self.hist = { | ||||
|                     k: v for k, v in self.hist.items() if now - v < self.maxage | ||||
|                 } | ||||
|                 self.oldest = sorted(self.hist.values())[0] | ||||
|  | ||||
|             return ret | ||||
|  | ||||
|  | ||||
| class Unrecv(object): | ||||
| @@ -93,6 +189,245 @@ class Unrecv(object): | ||||
|         self.buf = buf + self.buf | ||||
|  | ||||
|  | ||||
| class ProgressPrinter(threading.Thread): | ||||
|     """ | ||||
|     periodically print progress info without linefeeds | ||||
|     """ | ||||
|  | ||||
|     def __init__(self): | ||||
|         threading.Thread.__init__(self, name="pp") | ||||
|         self.daemon = True | ||||
|         self.msg = None | ||||
|         self.end = False | ||||
|         self.start() | ||||
|  | ||||
|     def run(self): | ||||
|         msg = None | ||||
|         while not self.end: | ||||
|             time.sleep(0.1) | ||||
|             if msg == self.msg or self.end: | ||||
|                 continue | ||||
|  | ||||
|             msg = self.msg | ||||
|             uprint(" {}\033[K\r".format(msg)) | ||||
|             if PY2: | ||||
|                 sys.stdout.flush() | ||||
|  | ||||
|         print("\033[K", end="") | ||||
|         sys.stdout.flush()  # necessary on win10 even w/ stderr btw | ||||
|  | ||||
|  | ||||
| def uprint(msg): | ||||
|     try: | ||||
|         print(msg, end="") | ||||
|     except UnicodeEncodeError: | ||||
|         try: | ||||
|             print(msg.encode("utf-8", "replace").decode(), end="") | ||||
|         except: | ||||
|             print(msg.encode("ascii", "replace").decode(), end="") | ||||
|  | ||||
|  | ||||
| def nuprint(msg): | ||||
|     uprint("{}\n".format(msg)) | ||||
|  | ||||
|  | ||||
| def rice_tid(): | ||||
|     tid = threading.current_thread().ident | ||||
|     c = sunpack(b"B" * 5, spack(b">Q", tid)[-5:]) | ||||
|     return "".join("\033[1;37;48;5;{}m{:02x}".format(x, x) for x in c) + "\033[0m" | ||||
|  | ||||
|  | ||||
| def trace(*args, **kwargs): | ||||
|     t = time.time() | ||||
|     stack = "".join( | ||||
|         "\033[36m{}\033[33m{}".format(x[0].split(os.sep)[-1][:-3], x[1]) | ||||
|         for x in traceback.extract_stack()[3:-1] | ||||
|     ) | ||||
|     parts = ["{:.6f}".format(t), rice_tid(), stack] | ||||
|  | ||||
|     if args: | ||||
|         parts.append(repr(args)) | ||||
|  | ||||
|     if kwargs: | ||||
|         parts.append(repr(kwargs)) | ||||
|  | ||||
|     msg = "\033[0m ".join(parts) | ||||
|     # _tracebuf.append(msg) | ||||
|     nuprint(msg) | ||||
|  | ||||
|  | ||||
| def alltrace(): | ||||
|     threads = {} | ||||
|     names = dict([(t.ident, t.name) for t in threading.enumerate()]) | ||||
|     for tid, stack in sys._current_frames().items(): | ||||
|         name = "{} ({:x})".format(names.get(tid), tid) | ||||
|         threads[name] = stack | ||||
|  | ||||
|     rret = [] | ||||
|     bret = [] | ||||
|     for name, stack in sorted(threads.items()): | ||||
|         ret = ["\n\n# {}".format(name)] | ||||
|         pad = None | ||||
|         for fn, lno, name, line in traceback.extract_stack(stack): | ||||
|             fn = os.sep.join(fn.split(os.sep)[-3:]) | ||||
|             ret.append('File: "{}", line {}, in {}'.format(fn, lno, name)) | ||||
|             if line: | ||||
|                 ret.append("  " + str(line.strip())) | ||||
|                 if "self.not_empty.wait()" in line: | ||||
|                     pad = " " * 4 | ||||
|  | ||||
|         if pad: | ||||
|             bret += [ret[0]] + [pad + x for x in ret[1:]] | ||||
|         else: | ||||
|             rret += ret | ||||
|  | ||||
|     return "\n".join(rret + bret) | ||||
|  | ||||
|  | ||||
| def start_stackmon(arg_str, nid): | ||||
|     suffix = "-{}".format(nid) if nid else "" | ||||
|     fp, f = arg_str.rsplit(",", 1) | ||||
|     f = int(f) | ||||
|     t = threading.Thread( | ||||
|         target=stackmon, | ||||
|         args=(fp, f, suffix), | ||||
|         name="stackmon" + suffix, | ||||
|     ) | ||||
|     t.daemon = True | ||||
|     t.start() | ||||
|  | ||||
|  | ||||
| def stackmon(fp, ival, suffix): | ||||
|     ctr = 0 | ||||
|     while True: | ||||
|         ctr += 1 | ||||
|         time.sleep(ival) | ||||
|         st = "{}, {}\n{}".format(ctr, time.time(), alltrace()) | ||||
|         with open(fp + suffix, "wb") as f: | ||||
|             f.write(st.encode("utf-8", "replace")) | ||||
|  | ||||
|  | ||||
| def start_log_thrs(logger, ival, nid): | ||||
|     ival = int(ival) | ||||
|     tname = lname = "log-thrs" | ||||
|     if nid: | ||||
|         tname = "logthr-n{}-i{:x}".format(nid, os.getpid()) | ||||
|         lname = tname[3:] | ||||
|  | ||||
|     t = threading.Thread( | ||||
|         target=log_thrs, | ||||
|         args=(logger, ival, lname), | ||||
|         name=tname, | ||||
|     ) | ||||
|     t.daemon = True | ||||
|     t.start() | ||||
|  | ||||
|  | ||||
| def log_thrs(log, ival, name): | ||||
|     while True: | ||||
|         time.sleep(ival) | ||||
|         tv = [x.name for x in threading.enumerate()] | ||||
|         tv = [ | ||||
|             x.split("-")[0] | ||||
|             if x.startswith("httpconn-") or x.startswith("thumb-") | ||||
|             else "listen" | ||||
|             if "-listen-" in x | ||||
|             else x | ||||
|             for x in tv | ||||
|             if not x.startswith("pydevd.") | ||||
|         ] | ||||
|         tv = ["{}\033[36m{}".format(v, k) for k, v in sorted(Counter(tv).items())] | ||||
|         log(name, "\033[0m \033[33m".join(tv), 3) | ||||
|  | ||||
|  | ||||
| def min_ex(): | ||||
|     et, ev, tb = sys.exc_info() | ||||
|     tb = traceback.extract_tb(tb) | ||||
|     fmt = "{} @ {} <{}>: {}" | ||||
|     ex = [fmt.format(fp.split(os.sep)[-1], ln, fun, txt) for fp, ln, fun, txt in tb] | ||||
|     ex.append("[{}] {}".format(et.__name__, ev)) | ||||
|     return "\n".join(ex[-8:]) | ||||
|  | ||||
|  | ||||
| @contextlib.contextmanager | ||||
| def ren_open(fname, *args, **kwargs): | ||||
|     fdir = kwargs.pop("fdir", None) | ||||
|     suffix = kwargs.pop("suffix", None) | ||||
|  | ||||
|     if fname == os.devnull: | ||||
|         with open(fname, *args, **kwargs) as f: | ||||
|             yield {"orz": [f, fname]} | ||||
|             return | ||||
|  | ||||
|     if suffix: | ||||
|         ext = fname.split(".")[-1] | ||||
|         if len(ext) < 7: | ||||
|             suffix += "." + ext | ||||
|  | ||||
|     orig_name = fname | ||||
|     bname = fname | ||||
|     ext = "" | ||||
|     while True: | ||||
|         ofs = bname.rfind(".") | ||||
|         if ofs < 0 or ofs < len(bname) - 7: | ||||
|             # doesn't look like an extension anymore | ||||
|             break | ||||
|  | ||||
|         ext = bname[ofs:] + ext | ||||
|         bname = bname[:ofs] | ||||
|  | ||||
|     b64 = "" | ||||
|     while True: | ||||
|         try: | ||||
|             if fdir: | ||||
|                 fpath = os.path.join(fdir, fname) | ||||
|             else: | ||||
|                 fpath = fname | ||||
|  | ||||
|             if suffix and os.path.exists(fsenc(fpath)): | ||||
|                 fpath += suffix | ||||
|                 fname += suffix | ||||
|                 ext += suffix | ||||
|  | ||||
|             with open(fsenc(fpath), *args, **kwargs) as f: | ||||
|                 if b64: | ||||
|                     fp2 = "fn-trunc.{}.txt".format(b64) | ||||
|                     fp2 = os.path.join(fdir, fp2) | ||||
|                     with open(fsenc(fp2), "wb") as f2: | ||||
|                         f2.write(orig_name.encode("utf-8")) | ||||
|  | ||||
|                 yield {"orz": [f, fname]} | ||||
|                 return | ||||
|  | ||||
|         except OSError as ex_: | ||||
|             ex = ex_ | ||||
|             if ex.errno not in [36, 63] and (not WINDOWS or ex.errno != 22): | ||||
|                 raise | ||||
|  | ||||
|         if not b64: | ||||
|             b64 = (bname + ext).encode("utf-8", "replace") | ||||
|             b64 = hashlib.sha512(b64).digest()[:12] | ||||
|             b64 = base64.urlsafe_b64encode(b64).decode("utf-8") | ||||
|  | ||||
|         badlen = len(fname) | ||||
|         while len(fname) >= badlen: | ||||
|             if len(bname) < 8: | ||||
|                 raise ex | ||||
|  | ||||
|             if len(bname) > len(ext): | ||||
|                 # drop the last letter of the filename | ||||
|                 bname = bname[:-1] | ||||
|             else: | ||||
|                 try: | ||||
|                     # drop the leftmost sub-extension | ||||
|                     _, ext = ext.split(".", 1) | ||||
|                 except: | ||||
|                     # okay do the first letter then | ||||
|                     ext = "." + ext[2:] | ||||
|  | ||||
|             fname = "{}~{}{}".format(bname, b64, ext) | ||||
|  | ||||
|  | ||||
| class MultipartParser(object): | ||||
|     def __init__(self, log_func, sr, http_headers): | ||||
|         self.sr = sr | ||||
| @@ -309,18 +644,7 @@ def get_boundary(headers): | ||||
| def read_header(sr): | ||||
|     ret = b"" | ||||
|     while True: | ||||
|         if ret.endswith(b"\r\n\r\n"): | ||||
|             break | ||||
|         elif ret.endswith(b"\r\n\r"): | ||||
|             n = 1 | ||||
|         elif ret.endswith(b"\r\n"): | ||||
|             n = 2 | ||||
|         elif ret.endswith(b"\r"): | ||||
|             n = 3 | ||||
|         else: | ||||
|             n = 4 | ||||
|  | ||||
|         buf = sr.recv(n) | ||||
|         buf = sr.recv(1024) | ||||
|         if not buf: | ||||
|             if not ret: | ||||
|                 return None | ||||
| @@ -332,11 +656,62 @@ def read_header(sr): | ||||
|             ) | ||||
|  | ||||
|         ret += buf | ||||
|         ofs = ret.find(b"\r\n\r\n") | ||||
|         if ofs < 0: | ||||
|             if len(ret) > 1024 * 64: | ||||
|                 raise Pebkac(400, "header 2big") | ||||
|             else: | ||||
|                 continue | ||||
|  | ||||
|         if len(ret) > 1024 * 64: | ||||
|             raise Pebkac(400, "header 2big") | ||||
|         if len(ret) > ofs + 4: | ||||
|             sr.unrecv(ret[ofs + 4 :]) | ||||
|  | ||||
|     return ret[:-4].decode("utf-8", "surrogateescape").split("\r\n") | ||||
|         return ret[:ofs].decode("utf-8", "surrogateescape").lstrip("\r\n").split("\r\n") | ||||
|  | ||||
|  | ||||
| def humansize(sz, terse=False): | ||||
|     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(" ", "") | ||||
|  | ||||
|  | ||||
| def get_spd(nbyte, t0, t=None): | ||||
|     if t is None: | ||||
|         t = time.time() | ||||
|  | ||||
|     bps = nbyte / ((t - t0) + 0.001) | ||||
|     s1 = humansize(nbyte).replace(" ", "\033[33m").replace("iB", "") | ||||
|     s2 = humansize(bps).replace(" ", "\033[35m").replace("iB", "") | ||||
|     return "{} \033[0m{}/s\033[0m".format(s1, s2) | ||||
|  | ||||
|  | ||||
| def s2hms(s, optional_h=False): | ||||
|     s = int(s) | ||||
|     h, s = divmod(s, 3600) | ||||
|     m, s = divmod(s, 60) | ||||
|     if not h and optional_h: | ||||
|         return "{}:{:02}".format(m, s) | ||||
|  | ||||
|     return "{}:{:02}:{:02}".format(h, m, s) | ||||
|  | ||||
|  | ||||
| def uncyg(path): | ||||
|     if len(path) < 2 or not path.startswith("/"): | ||||
|         return path | ||||
|  | ||||
|     if len(path) > 2 and path[2] != "/": | ||||
|         return path | ||||
|  | ||||
|     return "{}:\\{}".format(path[1], path[3:]) | ||||
|  | ||||
|  | ||||
| def undot(path): | ||||
| @@ -355,14 +730,84 @@ def undot(path): | ||||
|     return "/".join(ret) | ||||
|  | ||||
|  | ||||
| def sanitize_fn(fn): | ||||
|     return fn.replace("\\", "/").split("/")[-1].strip() | ||||
| def sanitize_fn(fn, ok, bad): | ||||
|     if "/" not in ok: | ||||
|         fn = fn.replace("\\", "/").split("/")[-1] | ||||
|  | ||||
|     if ANYWIN: | ||||
|         remap = [ | ||||
|             ["<", "<"], | ||||
|             [">", ">"], | ||||
|             [":", ":"], | ||||
|             ['"', """], | ||||
|             ["/", "/"], | ||||
|             ["\\", "\"], | ||||
|             ["|", "|"], | ||||
|             ["?", "?"], | ||||
|             ["*", "*"], | ||||
|         ] | ||||
|         for a, b in [x for x in remap if x[0] not in ok]: | ||||
|             fn = fn.replace(a, b) | ||||
|  | ||||
|         bad.extend(["con", "prn", "aux", "nul"]) | ||||
|         for n in range(1, 10): | ||||
|             bad += "com{0} lpt{0}".format(n).split(" ") | ||||
|  | ||||
|     if fn.lower() in bad: | ||||
|         fn = "_" + fn | ||||
|  | ||||
|     return fn.strip() | ||||
|  | ||||
|  | ||||
| def absreal(fpath): | ||||
|     try: | ||||
|         return fsdec(os.path.abspath(os.path.realpath(fsenc(fpath)))) | ||||
|     except: | ||||
|         if not WINDOWS: | ||||
|             raise | ||||
|  | ||||
|         # cpython bug introduced in 3.8, still exists in 3.9.1, | ||||
|         # some win7sp1 and win10:20H2 boxes cannot realpath a | ||||
|         # networked drive letter such as b"n:" or b"n:\\" | ||||
|         return os.path.abspath(os.path.realpath(fpath)) | ||||
|  | ||||
|  | ||||
| def u8safe(txt): | ||||
|     try: | ||||
|         return txt.encode("utf-8", "xmlcharrefreplace").decode("utf-8", "replace") | ||||
|     except: | ||||
|         return txt.encode("utf-8", "replace").decode("utf-8", "replace") | ||||
|  | ||||
|  | ||||
| def exclude_dotfiles(filepaths): | ||||
|     for fpath in filepaths: | ||||
|         if not fpath.split("/")[-1].startswith("."): | ||||
|             yield fpath | ||||
|     return [x for x in filepaths if not x.split("/")[-1].startswith(".")] | ||||
|  | ||||
|  | ||||
| def http_ts(ts): | ||||
|     file_dt = datetime.utcfromtimestamp(ts) | ||||
|     return file_dt.strftime(HTTP_TS_FMT) | ||||
|  | ||||
|  | ||||
| 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 | ||||
|  | ||||
|  | ||||
| def html_bescape(s, quote=False, crlf=False): | ||||
|     """html.escape but bytestrings""" | ||||
|     s = s.replace(b"&", b"&").replace(b"<", b"<").replace(b">", b">") | ||||
|     if quote: | ||||
|         s = s.replace(b'"', b""").replace(b"'", b"'") | ||||
|     if crlf: | ||||
|         s = s.replace(b"\r", b"
").replace(b"\n", b"
") | ||||
|  | ||||
|     return s | ||||
|  | ||||
|  | ||||
| def quotep(txt): | ||||
| @@ -379,11 +824,18 @@ def quotep(txt): | ||||
| def unquotep(txt): | ||||
|     """url unquoter which deals with bytes correctly""" | ||||
|     btxt = w8enc(txt) | ||||
|     unq1 = btxt.replace(b"+", b" ") | ||||
|     unq2 = unquote(unq1) | ||||
|     # btxt = btxt.replace(b"+", b" ") | ||||
|     unq2 = unquote(btxt) | ||||
|     return w8dec(unq2) | ||||
|  | ||||
|  | ||||
| def vsplit(vpath): | ||||
|     if "/" not in vpath: | ||||
|         return "", vpath | ||||
|  | ||||
|     return vpath.rsplit("/", 1) | ||||
|  | ||||
|  | ||||
| def w8dec(txt): | ||||
|     """decodes filesystem-bytes to wtf8""" | ||||
|     if PY2: | ||||
| @@ -400,6 +852,16 @@ def w8enc(txt): | ||||
|     return txt.encode(FS_ENCODING, "surrogateescape") | ||||
|  | ||||
|  | ||||
| def w8b64dec(txt): | ||||
|     """decodes base64(filesystem-bytes) to wtf8""" | ||||
|     return w8dec(base64.urlsafe_b64decode(txt.encode("ascii"))) | ||||
|  | ||||
|  | ||||
| def w8b64enc(txt): | ||||
|     """encodes wtf8 to base64(filesystem-bytes)""" | ||||
|     return base64.urlsafe_b64encode(w8enc(txt)).decode("ascii") | ||||
|  | ||||
|  | ||||
| if PY2 and WINDOWS: | ||||
|     # moonrunes become \x3f with bytestrings, | ||||
|     # losing mojibake support is worth | ||||
| @@ -413,6 +875,43 @@ else: | ||||
|     fsdec = w8dec | ||||
|  | ||||
|  | ||||
| def s3enc(mem_cur, rd, fn): | ||||
|     ret = [] | ||||
|     for v in [rd, fn]: | ||||
|         try: | ||||
|             mem_cur.execute("select * from a where b = ?", (v,)) | ||||
|             ret.append(v) | ||||
|         except: | ||||
|             ret.append("//" + w8b64enc(v)) | ||||
|             # self.log("mojien/{} [{}] {}".format(k, v, ret[-1][2:])) | ||||
|  | ||||
|     return tuple(ret) | ||||
|  | ||||
|  | ||||
| def s3dec(rd, fn): | ||||
|     ret = [] | ||||
|     for k, v in [["d", rd], ["f", fn]]: | ||||
|         if v.startswith("//"): | ||||
|             ret.append(w8b64dec(v[2:])) | ||||
|             # self.log("mojide/{} [{}] {}".format(k, ret[-1], v[2:])) | ||||
|         else: | ||||
|             ret.append(v) | ||||
|  | ||||
|     return tuple(ret) | ||||
|  | ||||
|  | ||||
| def atomic_move(src, dst): | ||||
|     src = fsenc(src) | ||||
|     dst = fsenc(dst) | ||||
|     if not PY2: | ||||
|         os.replace(src, dst) | ||||
|     else: | ||||
|         if os.path.exists(dst): | ||||
|             os.unlink(dst) | ||||
|  | ||||
|         os.rename(src, dst) | ||||
|  | ||||
|  | ||||
| def read_socket(sr, total_size): | ||||
|     remains = total_size | ||||
|     while remains > 0: | ||||
| @@ -428,25 +927,164 @@ def read_socket(sr, total_size): | ||||
|         yield buf | ||||
|  | ||||
|  | ||||
| def hashcopy(actor, fin, fout): | ||||
|     u32_lim = int((2 ** 31) * 0.9) | ||||
| def read_socket_unbounded(sr): | ||||
|     while True: | ||||
|         buf = sr.recv(32 * 1024) | ||||
|         if not buf: | ||||
|             return | ||||
|  | ||||
|         yield buf | ||||
|  | ||||
|  | ||||
| def read_socket_chunked(sr, log=None): | ||||
|     err = "expected chunk length, got [{}] |{}| instead" | ||||
|     while True: | ||||
|         buf = b"" | ||||
|         while b"\r" not in buf: | ||||
|             rbuf = sr.recv(2) | ||||
|             if not rbuf or len(buf) > 16: | ||||
|                 err = err.format(buf.decode("utf-8", "replace"), len(buf)) | ||||
|                 raise Pebkac(400, err) | ||||
|  | ||||
|             buf += rbuf | ||||
|  | ||||
|         if not buf.endswith(b"\n"): | ||||
|             sr.recv(1) | ||||
|  | ||||
|         try: | ||||
|             chunklen = int(buf.rstrip(b"\r\n"), 16) | ||||
|         except: | ||||
|             err = err.format(buf.decode("utf-8", "replace"), len(buf)) | ||||
|             raise Pebkac(400, err) | ||||
|  | ||||
|         if chunklen == 0: | ||||
|             sr.recv(2)  # \r\n after final chunk | ||||
|             return | ||||
|  | ||||
|         if log: | ||||
|             log("receiving {} byte chunk".format(chunklen)) | ||||
|  | ||||
|         for chunk in read_socket(sr, chunklen): | ||||
|             yield chunk | ||||
|  | ||||
|         sr.recv(2)  # \r\n after each chunk too | ||||
|  | ||||
|  | ||||
| def yieldfile(fn): | ||||
|     with open(fsenc(fn), "rb", 512 * 1024) as f: | ||||
|         while True: | ||||
|             buf = f.read(64 * 1024) | ||||
|             if not buf: | ||||
|                 break | ||||
|  | ||||
|             yield buf | ||||
|  | ||||
|  | ||||
| def hashcopy(fin, fout): | ||||
|     hashobj = hashlib.sha512() | ||||
|     tlen = 0 | ||||
|     for buf in fin: | ||||
|         actor.workload += 1 | ||||
|         if actor.workload > u32_lim: | ||||
|             actor.workload = 100  # prevent overflow | ||||
|  | ||||
|         tlen += len(buf) | ||||
|         hashobj.update(buf) | ||||
|         fout.write(buf) | ||||
|  | ||||
|     digest32 = hashobj.digest()[:32] | ||||
|     digest_b64 = base64.urlsafe_b64encode(digest32).decode("utf-8").rstrip("=") | ||||
|     digest = hashobj.digest()[:33] | ||||
|     digest_b64 = base64.urlsafe_b64encode(digest).decode("utf-8") | ||||
|  | ||||
|     return tlen, hashobj.hexdigest(), digest_b64 | ||||
|  | ||||
|  | ||||
| def sendfile_py(lower, upper, f, s): | ||||
|     remains = upper - lower | ||||
|     f.seek(lower) | ||||
|     while remains > 0: | ||||
|         # time.sleep(0.01) | ||||
|         buf = f.read(min(1024 * 32, remains)) | ||||
|         if not buf: | ||||
|             return remains | ||||
|  | ||||
|         try: | ||||
|             s.sendall(buf) | ||||
|             remains -= len(buf) | ||||
|         except: | ||||
|             return remains | ||||
|  | ||||
|     return 0 | ||||
|  | ||||
|  | ||||
| def sendfile_kern(lower, upper, f, s): | ||||
|     out_fd = s.fileno() | ||||
|     in_fd = f.fileno() | ||||
|     ofs = lower | ||||
|     while ofs < upper: | ||||
|         try: | ||||
|             req = min(2 ** 30, upper - ofs) | ||||
|             select.select([], [out_fd], [], 10) | ||||
|             n = os.sendfile(out_fd, in_fd, ofs, req) | ||||
|         except Exception as ex: | ||||
|             # print("sendfile: " + repr(ex)) | ||||
|             n = 0 | ||||
|  | ||||
|         if n <= 0: | ||||
|             return upper - ofs | ||||
|  | ||||
|         ofs += n | ||||
|         # print("sendfile: ok, sent {} now, {} total, {} remains".format(n, ofs - lower, upper - ofs)) | ||||
|  | ||||
|     return 0 | ||||
|  | ||||
|  | ||||
| def statdir(logger, scandir, lstat, top): | ||||
|     if lstat and not os.supports_follow_symlinks: | ||||
|         scandir = False | ||||
|  | ||||
|     try: | ||||
|         btop = fsenc(top) | ||||
|         if scandir and hasattr(os, "scandir"): | ||||
|             src = "scandir" | ||||
|             with os.scandir(btop) as dh: | ||||
|                 for fh in dh: | ||||
|                     try: | ||||
|                         yield [fsdec(fh.name), fh.stat(follow_symlinks=not lstat)] | ||||
|                     except Exception as ex: | ||||
|                         logger(src, "[s] {} @ {}".format(repr(ex), fsdec(fh.path)), 6) | ||||
|         else: | ||||
|             src = "listdir" | ||||
|             fun = os.lstat if lstat else os.stat | ||||
|             for name in os.listdir(btop): | ||||
|                 abspath = os.path.join(btop, name) | ||||
|                 try: | ||||
|                     yield [fsdec(name), fun(abspath)] | ||||
|                 except Exception as ex: | ||||
|                     logger(src, "[s] {} @ {}".format(repr(ex), fsdec(abspath)), 6) | ||||
|  | ||||
|     except Exception as ex: | ||||
|         logger(src, "{} @ {}".format(repr(ex), top), 1) | ||||
|  | ||||
|  | ||||
| def rmdirs(logger, scandir, lstat, top): | ||||
|     if not os.path.exists(fsenc(top)) or not os.path.isdir(fsenc(top)): | ||||
|         top = os.path.dirname(top) | ||||
|      | ||||
|     dirs = statdir(logger, scandir, lstat, top) | ||||
|     dirs = [x[0] for x in dirs if stat.S_ISDIR(x[1].st_mode)] | ||||
|     dirs = [os.path.join(top, x) for x in dirs] | ||||
|     ok = [] | ||||
|     ng = [] | ||||
|     for d in dirs[::-1]: | ||||
|         a, b = rmdirs(logger, scandir, lstat, d) | ||||
|         ok += a | ||||
|         ng += b | ||||
|  | ||||
|     try: | ||||
|         os.rmdir(fsenc(top)) | ||||
|         ok.append(top) | ||||
|     except: | ||||
|         ng.append(top) | ||||
|  | ||||
|     return ok, ng | ||||
|  | ||||
|  | ||||
| def unescape_cookie(orig): | ||||
|     # mw=idk; doot=qwe%2Crty%3Basd+fgh%2Bjkl%25zxc%26vbn  # qwe,rty;asd fgh+jkl%zxc&vbn | ||||
|     ret = "" | ||||
| @@ -475,33 +1113,53 @@ def unescape_cookie(orig): | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def guess_mime(url): | ||||
|     if url.endswith(".md"): | ||||
|         return ["text/plain; charset=UTF-8"] | ||||
| def guess_mime(url, fallback="application/octet-stream"): | ||||
|     try: | ||||
|         _, ext = url.rsplit(".", 1) | ||||
|     except: | ||||
|         return fallback | ||||
|  | ||||
|     return mimetypes.guess_type(url) | ||||
|     ret = MIMES.get(ext) or mimetypes.guess_type(url)[0] or fallback | ||||
|  | ||||
|     if ";" not in ret: | ||||
|         if ret.startswith("text/") or ret.endswith("/javascript"): | ||||
|             ret += "; charset=UTF-8" | ||||
|  | ||||
|     return ret | ||||
|  | ||||
|  | ||||
| def runcmd(*argv): | ||||
| def runcmd(argv): | ||||
|     p = sp.Popen(argv, stdout=sp.PIPE, stderr=sp.PIPE) | ||||
|     stdout, stderr = p.communicate() | ||||
|     stdout = stdout.decode("utf-8") | ||||
|     stderr = stderr.decode("utf-8") | ||||
|     stdout = stdout.decode("utf-8", "replace") | ||||
|     stderr = stderr.decode("utf-8", "replace") | ||||
|     return [p.returncode, stdout, stderr] | ||||
|  | ||||
|  | ||||
| def chkcmd(*argv): | ||||
|     ok, sout, serr = runcmd(*argv) | ||||
| def chkcmd(argv): | ||||
|     ok, sout, serr = runcmd(argv) | ||||
|     if ok != 0: | ||||
|         raise Exception(serr) | ||||
|  | ||||
|     return sout, serr | ||||
|  | ||||
|  | ||||
| def mchkcmd(argv, timeout=10): | ||||
|     if PY2: | ||||
|         with open(os.devnull, "wb") as f: | ||||
|             rv = sp.call(argv, stdout=f, stderr=f) | ||||
|     else: | ||||
|         rv = sp.call(argv, stdout=sp.DEVNULL, stderr=sp.DEVNULL, timeout=timeout) | ||||
|  | ||||
|     if rv: | ||||
|         raise sp.CalledProcessError(rv, (argv[0], b"...", argv[-1])) | ||||
|  | ||||
|  | ||||
| def gzip_orig_sz(fn): | ||||
|     with open(fsenc(fn), "rb") as f: | ||||
|         f.seek(-4, 2) | ||||
|         return struct.unpack(b"I", f.read(4))[0] | ||||
|         rv = f.read(4) | ||||
|         return sunpack(b"I", rv)[0] | ||||
|  | ||||
|  | ||||
| def py_desc(): | ||||
| @@ -511,7 +1169,11 @@ def py_desc(): | ||||
|     if ofs > 0: | ||||
|         py_ver = py_ver[:ofs] | ||||
|  | ||||
|     bitness = struct.calcsize(b"P") * 8 | ||||
|     try: | ||||
|         bitness = struct.calcsize(b"P") * 8 | ||||
|     except: | ||||
|         bitness = struct.calcsize("P") * 8 | ||||
|  | ||||
|     host_os = platform.system() | ||||
|     compiler = platform.python_compiler() | ||||
|  | ||||
| @@ -523,7 +1185,26 @@ def py_desc(): | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def align_tab(lines): | ||||
|     rows = [] | ||||
|     ncols = 0 | ||||
|     for ln in lines: | ||||
|         row = [x for x in ln.split(" ") if x] | ||||
|         ncols = max(ncols, len(row)) | ||||
|         rows.append(row) | ||||
|  | ||||
|     lens = [0] * ncols | ||||
|     for row in rows: | ||||
|         for n, col in enumerate(row): | ||||
|             lens[n] = max(lens[n], len(col)) | ||||
|  | ||||
|     return ["".join(x.ljust(y + 2) for x, y in zip(row, lens)) for row in rows] | ||||
|  | ||||
|  | ||||
| class Pebkac(Exception): | ||||
|     def __init__(self, code, msg=None): | ||||
|         super(Pebkac, self).__init__(msg or HTTPCODE[code]) | ||||
|         self.code = code | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return "Pebkac({}, {})".format(self.code, repr(self.args)) | ||||
|   | ||||
							
								
								
									
										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 $< | ||||
							
								
								
									
										733
									
								
								copyparty/web/baguettebox.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										733
									
								
								copyparty/web/baguettebox.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,733 @@ | ||||
| /*! | ||||
|  * baguetteBox.js | ||||
|  * @author  feimosi | ||||
|  * @version 1.11.1-mod | ||||
|  * @url https://github.com/feimosi/baguetteBox.js | ||||
|  */ | ||||
|  | ||||
| window.baguetteBox = (function () { | ||||
|     'use strict'; | ||||
|  | ||||
|     var options = {}, | ||||
|         defaults = { | ||||
|             captions: true, | ||||
|             buttons: 'auto', | ||||
|             noScrollbars: false, | ||||
|             bodyClass: 'bbox-open', | ||||
|             titleTag: false, | ||||
|             async: false, | ||||
|             preload: 2, | ||||
|             animation: 'slideIn', | ||||
|             afterShow: null, | ||||
|             afterHide: null, | ||||
|             onChange: null, | ||||
|         }, | ||||
|         overlay, slider, btnPrev, btnNext, btnHelp, btnVmode, btnClose, | ||||
|         currentGallery = [], | ||||
|         currentIndex = 0, | ||||
|         isOverlayVisible = false, | ||||
|         touch = {},  // start-pos | ||||
|         touchFlag = false,  // busy | ||||
|         re_i = /.+\.(gif|jpe?g|png|webp)(\?|$)/i, | ||||
|         re_v = /.+\.(webm|mp4)(\?|$)/i, | ||||
|         data = {},  // all galleries | ||||
|         imagesElements = [], | ||||
|         documentLastFocus = null, | ||||
|         isFullscreen = false, | ||||
|         vmute = false, | ||||
|         vloop = false, | ||||
|         vnext = false, | ||||
|         resume_mp = false; | ||||
|  | ||||
|     var onFSC = function (e) { | ||||
|         isFullscreen = !!document.fullscreenElement; | ||||
|     }; | ||||
|  | ||||
|     var overlayClickHandler = function (e) { | ||||
|         if (e.target.id.indexOf('baguette-img') !== -1) | ||||
|             hideOverlay(); | ||||
|     }; | ||||
|  | ||||
|     var touchstartHandler = function (e) { | ||||
|         touch.count++; | ||||
|         if (touch.count > 1) | ||||
|             touch.multitouch = true; | ||||
|  | ||||
|         touch.startX = e.changedTouches[0].pageX; | ||||
|         touch.startY = e.changedTouches[0].pageY; | ||||
|     }; | ||||
|     var touchmoveHandler = function (e) { | ||||
|         if (touchFlag || touch.multitouch) | ||||
|             return; | ||||
|  | ||||
|         e.preventDefault ? e.preventDefault() : e.returnValue = false; | ||||
|         var touchEvent = e.touches[0] || e.changedTouches[0]; | ||||
|         if (touchEvent.pageX - touch.startX > 40) { | ||||
|             touchFlag = true; | ||||
|             showPreviousImage(); | ||||
|         } else if (touchEvent.pageX - touch.startX < -40) { | ||||
|             touchFlag = true; | ||||
|             showNextImage(); | ||||
|         } else if (touch.startY - touchEvent.pageY > 100) { | ||||
|             hideOverlay(); | ||||
|         } | ||||
|     }; | ||||
|     var touchendHandler = function () { | ||||
|         touch.count--; | ||||
|         if (touch.count <= 0) | ||||
|             touch.multitouch = false; | ||||
|  | ||||
|         touchFlag = false; | ||||
|     }; | ||||
|     var contextmenuHandler = function () { | ||||
|         touchendHandler(); | ||||
|     }; | ||||
|  | ||||
|     var trapFocusInsideOverlay = function (e) { | ||||
|         if (overlay.style.display === 'block' && (overlay.contains && !overlay.contains(e.target))) { | ||||
|             e.stopPropagation(); | ||||
|             btnClose.focus(); | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     function run(selector, userOptions) { | ||||
|         buildOverlay(); | ||||
|         removeFromCache(selector); | ||||
|         return bindImageClickListeners(selector, userOptions); | ||||
|     } | ||||
|  | ||||
|     function bindImageClickListeners(selector, userOptions) { | ||||
|         var galleryNodeList = QSA(selector); | ||||
|         var selectorData = { | ||||
|             galleries: [], | ||||
|             nodeList: galleryNodeList | ||||
|         }; | ||||
|         data[selector] = selectorData; | ||||
|  | ||||
|         [].forEach.call(galleryNodeList, function (galleryElement) { | ||||
|             var tagsNodeList = []; | ||||
|             if (galleryElement.tagName === 'A') | ||||
|                 tagsNodeList = [galleryElement]; | ||||
|             else | ||||
|                 tagsNodeList = galleryElement.getElementsByTagName('a'); | ||||
|  | ||||
|             tagsNodeList = [].filter.call(tagsNodeList, function (element) { | ||||
|                 if (element.className.indexOf(userOptions && userOptions.ignoreClass) === -1) | ||||
|                     return re_i.test(element.href) || re_v.test(element.href); | ||||
|             }); | ||||
|             if (!tagsNodeList.length) | ||||
|                 return; | ||||
|  | ||||
|             var gallery = []; | ||||
|             [].forEach.call(tagsNodeList, function (imageElement, imageIndex) { | ||||
|                 var imageElementClickHandler = function (e) { | ||||
|                     if (ctrl(e)) | ||||
|                         return true; | ||||
|  | ||||
|                     e.preventDefault ? e.preventDefault() : e.returnValue = false; | ||||
|                     prepareOverlay(gallery, userOptions); | ||||
|                     showOverlay(imageIndex); | ||||
|                 }; | ||||
|                 var imageItem = { | ||||
|                     eventHandler: imageElementClickHandler, | ||||
|                     imageElement: imageElement | ||||
|                 }; | ||||
|                 bind(imageElement, 'click', imageElementClickHandler); | ||||
|                 gallery.push(imageItem); | ||||
|             }); | ||||
|             selectorData.galleries.push(gallery); | ||||
|         }); | ||||
|  | ||||
|         return selectorData.galleries; | ||||
|     } | ||||
|  | ||||
|     function clearCachedData() { | ||||
|         for (var selector in data) | ||||
|             if (data.hasOwnProperty(selector)) | ||||
|                 removeFromCache(selector); | ||||
|     } | ||||
|  | ||||
|     function removeFromCache(selector) { | ||||
|         if (!data.hasOwnProperty(selector)) | ||||
|             return; | ||||
|  | ||||
|         var galleries = data[selector].galleries; | ||||
|         [].forEach.call(galleries, function (gallery) { | ||||
|             [].forEach.call(gallery, function (imageItem) { | ||||
|                 unbind(imageItem.imageElement, 'click', imageItem.eventHandler); | ||||
|             }); | ||||
|  | ||||
|             if (currentGallery === gallery) | ||||
|                 currentGallery = []; | ||||
|         }); | ||||
|  | ||||
|         delete data[selector]; | ||||
|     } | ||||
|  | ||||
|     function buildOverlay() { | ||||
|         overlay = ebi('bbox-overlay'); | ||||
|         if (!overlay) { | ||||
|             var ctr = mknod('div'); | ||||
|             ctr.innerHTML = ( | ||||
|                 '<div id="bbox-overlay" role="dialog">' + | ||||
|                 '<div id="bbox-slider"></div>' + | ||||
|                 '<button id="bbox-prev" class="bbox-btn" type="button" aria-label="Previous"><</button>' + | ||||
|                 '<button id="bbox-next" class="bbox-btn" type="button" aria-label="Next">></button>' + | ||||
|                 '<div id="bbox-btns">' + | ||||
|                 '<button id="bbox-help" type="button">?</button>' + | ||||
|                 '<button id="bbox-vmode" type="button" tt="a"></button>' + | ||||
|                 '<button id="bbox-close" type="button" aria-label="Close">X</button>' + | ||||
|                 '</div></div>' | ||||
|             ); | ||||
|             overlay = ctr.firstChild; | ||||
|             QS('body').appendChild(overlay); | ||||
|             tt.att(overlay); | ||||
|         } | ||||
|         slider = ebi('bbox-slider'); | ||||
|         btnPrev = ebi('bbox-prev'); | ||||
|         btnNext = ebi('bbox-next'); | ||||
|         btnHelp = ebi('bbox-help'); | ||||
|         btnVmode = ebi('bbox-vmode'); | ||||
|         btnClose = ebi('bbox-close'); | ||||
|         bindEvents(); | ||||
|     } | ||||
|  | ||||
|     function halp() { | ||||
|         if (ebi('bbox-halp')) | ||||
|             return; | ||||
|  | ||||
|         var list = [ | ||||
|             ['<b># hotkey</b>', '<b># operation</b>'], | ||||
|             ['escape', 'close'], | ||||
|             ['left, J', 'previous file'], | ||||
|             ['right, L', 'next file'], | ||||
|             ['home', 'first file'], | ||||
|             ['end', 'last file'], | ||||
|             ['space, P, K', 'video: play / pause'], | ||||
|             ['U', 'video: seek 10sec back'], | ||||
|             ['P', 'video: seek 10sec ahead'], | ||||
|             ['M', 'video: toggle mute'], | ||||
|             ['R', 'video: toggle loop'], | ||||
|             ['C', 'video: toggle auto-next'], | ||||
|             ['F', 'video: toggle fullscreen'], | ||||
|         ], | ||||
|             d = mknod('table'), | ||||
|             html = ['<tbody>']; | ||||
|  | ||||
|         for (var a = 0; a < list.length; a++) | ||||
|             html.push('<tr><td>' + list[a][0] + '</td><td>' + list[a][1] + '</td></tr>'); | ||||
|  | ||||
|         d.innerHTML = html.join('\n') + '</tbody>'; | ||||
|         d.setAttribute('id', 'bbox-halp'); | ||||
|         d.onclick = function () { | ||||
|             overlay.removeChild(d); | ||||
|         }; | ||||
|         overlay.appendChild(d); | ||||
|     } | ||||
|  | ||||
|     function keyDownHandler(e) { | ||||
|         if (e.ctrlKey || e.altKey || e.metaKey || e.isComposing) | ||||
|             return; | ||||
|  | ||||
|         var k = e.code + '', v = vid(); | ||||
|  | ||||
|         if (k == "ArrowLeft" || k == "KeyJ") | ||||
|             showPreviousImage(); | ||||
|         else if (k == "ArrowRight" || k == "KeyL") | ||||
|             showNextImage(); | ||||
|         else if (k == "Escape") | ||||
|             hideOverlay(); | ||||
|         else if (k == "Home") | ||||
|             showFirstImage(e); | ||||
|         else if (k == "End") | ||||
|             showLastImage(e); | ||||
|         else if (k == "Space" || k == "KeyP" || k == "KeyK") | ||||
|             playpause(); | ||||
|         else if (k == "KeyU" || k == "KeyO") | ||||
|             relseek(k == "KeyU" ? -10 : 10); | ||||
|         else if (k == "KeyM" && v) { | ||||
|             v.muted = vmute = !vmute; | ||||
|             mp_ctl(); | ||||
|         } | ||||
|         else if (k == "KeyR" && v) { | ||||
|             vloop = !vloop; | ||||
|             vnext = vnext && !vloop; | ||||
|             setVmode(); | ||||
|         } | ||||
|         else if (k == "KeyC" && v) { | ||||
|             vnext = !vnext; | ||||
|             vloop = vloop && !vnext; | ||||
|             setVmode(); | ||||
|         } | ||||
|         else if (k == "KeyF") | ||||
|             try { | ||||
|                 if (isFullscreen) | ||||
|                     document.exitFullscreen(); | ||||
|                 else | ||||
|                     v.requestFullscreen(); | ||||
|             } | ||||
|             catch (ex) { } | ||||
|     } | ||||
|  | ||||
|     function setVmode() { | ||||
|         var v = vid(); | ||||
|         ebi('bbox-vmode').style.display = v ? '' : 'none'; | ||||
|         if (!v) | ||||
|             return; | ||||
|  | ||||
|         var msg = 'When video ends, ', tts = '', lbl; | ||||
|         if (vloop) { | ||||
|             lbl = 'Loop'; | ||||
|             msg += 'repeat it'; | ||||
|             tts = '$NHotkey: R'; | ||||
|         } | ||||
|         else if (vnext) { | ||||
|             lbl = 'Cont'; | ||||
|             msg += 'continue to next'; | ||||
|             tts = '$NHotkey: C'; | ||||
|         } | ||||
|         else { | ||||
|             lbl = 'Stop'; | ||||
|             msg += 'just stop' | ||||
|         } | ||||
|         btnVmode.setAttribute('aria-label', msg); | ||||
|         btnVmode.setAttribute('tt', msg + tts); | ||||
|         btnVmode.textContent = lbl; | ||||
|  | ||||
|         v.loop = vloop | ||||
|         if (vloop && v.paused) | ||||
|             v.play(); | ||||
|     } | ||||
|  | ||||
|     function tglVmode() { | ||||
|         if (vloop) { | ||||
|             vnext = true; | ||||
|             vloop = false; | ||||
|         } | ||||
|         else if (vnext) | ||||
|             vnext = false; | ||||
|         else | ||||
|             vloop = true; | ||||
|  | ||||
|         setVmode(); | ||||
|         if (tt.en) | ||||
|             tt.show.bind(this)(); | ||||
|     } | ||||
|  | ||||
|     function keyUpHandler(e) { | ||||
|         if (e.ctrlKey || e.altKey || e.metaKey || e.isComposing) | ||||
|             return; | ||||
|  | ||||
|         var k = e.code + ''; | ||||
|  | ||||
|         if (k == "Space") | ||||
|             ev(e); | ||||
|     } | ||||
|  | ||||
|     var passiveSupp = false; | ||||
|     try { | ||||
|         var opts = { | ||||
|             get passive() { | ||||
|                 passiveSupp = true; | ||||
|                 return false; | ||||
|             } | ||||
|         }; | ||||
|         window.addEventListener('test', null, opts); | ||||
|         window.removeEventListener('test', null, opts); | ||||
|     } | ||||
|     catch (ex) { | ||||
|         passiveSupp = false; | ||||
|     } | ||||
|     var passiveEvent = passiveSupp ? { passive: false } : null; | ||||
|     var nonPassiveEvent = passiveSupp ? { passive: true } : null; | ||||
|  | ||||
|     function bindEvents() { | ||||
|         bind(overlay, 'click', overlayClickHandler); | ||||
|         bind(btnPrev, 'click', showPreviousImage); | ||||
|         bind(btnNext, 'click', showNextImage); | ||||
|         bind(btnClose, 'click', hideOverlay); | ||||
|         bind(btnVmode, 'click', tglVmode); | ||||
|         bind(btnHelp, 'click', halp); | ||||
|         bind(slider, 'contextmenu', contextmenuHandler); | ||||
|         bind(overlay, 'touchstart', touchstartHandler, nonPassiveEvent); | ||||
|         bind(overlay, 'touchmove', touchmoveHandler, passiveEvent); | ||||
|         bind(overlay, 'touchend', touchendHandler); | ||||
|         bind(document, 'focus', trapFocusInsideOverlay, true); | ||||
|     } | ||||
|  | ||||
|     function unbindEvents() { | ||||
|         unbind(overlay, 'click', overlayClickHandler); | ||||
|         unbind(btnPrev, 'click', showPreviousImage); | ||||
|         unbind(btnNext, 'click', showNextImage); | ||||
|         unbind(btnClose, 'click', hideOverlay); | ||||
|         unbind(btnVmode, 'click', tglVmode); | ||||
|         unbind(btnHelp, 'click', halp); | ||||
|         unbind(slider, 'contextmenu', contextmenuHandler); | ||||
|         unbind(overlay, 'touchstart', touchstartHandler, nonPassiveEvent); | ||||
|         unbind(overlay, 'touchmove', touchmoveHandler, passiveEvent); | ||||
|         unbind(overlay, 'touchend', touchendHandler); | ||||
|         unbind(document, 'focus', trapFocusInsideOverlay, true); | ||||
|     } | ||||
|  | ||||
|     function prepareOverlay(gallery, userOptions) { | ||||
|         if (currentGallery === gallery) | ||||
|             return; | ||||
|  | ||||
|         currentGallery = gallery; | ||||
|         setOptions(userOptions); | ||||
|         slider.innerHTML = ''; | ||||
|         imagesElements.length = 0; | ||||
|  | ||||
|         var imagesFiguresIds = []; | ||||
|         var imagesCaptionsIds = []; | ||||
|         for (var i = 0, fullImage; i < gallery.length; i++) { | ||||
|             fullImage = mknod('div'); | ||||
|             fullImage.className = 'full-image'; | ||||
|             fullImage.id = 'baguette-img-' + i; | ||||
|             imagesElements.push(fullImage); | ||||
|  | ||||
|             imagesFiguresIds.push('bbox-figure-' + i); | ||||
|             imagesCaptionsIds.push('bbox-figcaption-' + i); | ||||
|             slider.appendChild(imagesElements[i]); | ||||
|         } | ||||
|         overlay.setAttribute('aria-labelledby', imagesFiguresIds.join(' ')); | ||||
|         overlay.setAttribute('aria-describedby', imagesCaptionsIds.join(' ')); | ||||
|     } | ||||
|  | ||||
|     function setOptions(newOptions) { | ||||
|         if (!newOptions) | ||||
|             newOptions = {}; | ||||
|  | ||||
|         for (var item in defaults) { | ||||
|             options[item] = defaults[item]; | ||||
|             if (typeof newOptions[item] !== 'undefined') | ||||
|                 options[item] = newOptions[item]; | ||||
|         } | ||||
|         slider.style.transition = (options.animation === 'fadeIn' ? 'opacity .4s ease' : | ||||
|             options.animation === 'slideIn' ? '' : 'none'); | ||||
|  | ||||
|         if (options.buttons === 'auto' && ('ontouchstart' in window || currentGallery.length === 1)) | ||||
|             options.buttons = false; | ||||
|  | ||||
|         btnPrev.style.display = btnNext.style.display = (options.buttons ? '' : 'none'); | ||||
|     } | ||||
|  | ||||
|     function showOverlay(chosenImageIndex) { | ||||
|         if (options.noScrollbars) { | ||||
|             document.documentElement.style.overflowY = 'hidden'; | ||||
|             document.body.style.overflowY = 'scroll'; | ||||
|         } | ||||
|         if (overlay.style.display === 'block') | ||||
|             return; | ||||
|  | ||||
|         bind(document, 'keydown', keyDownHandler); | ||||
|         bind(document, 'keyup', keyUpHandler); | ||||
|         bind(document, 'fullscreenchange', onFSC); | ||||
|         currentIndex = chosenImageIndex; | ||||
|         touch = { | ||||
|             count: 0, | ||||
|             startX: null, | ||||
|             startY: null | ||||
|         }; | ||||
|         loadImage(currentIndex, function () { | ||||
|             preloadNext(currentIndex); | ||||
|             preloadPrev(currentIndex); | ||||
|         }); | ||||
|  | ||||
|         updateOffset(); | ||||
|         overlay.style.display = 'block'; | ||||
|         // Fade in overlay | ||||
|         setTimeout(function () { | ||||
|             overlay.className = 'visible'; | ||||
|             if (options.bodyClass && document.body.classList) | ||||
|                 document.body.classList.add(options.bodyClass); | ||||
|  | ||||
|             if (options.afterShow) | ||||
|                 options.afterShow(); | ||||
|         }, 50); | ||||
|  | ||||
|         if (options.onChange) | ||||
|             options.onChange(currentIndex, imagesElements.length); | ||||
|  | ||||
|         documentLastFocus = document.activeElement; | ||||
|         btnClose.focus(); | ||||
|         isOverlayVisible = true; | ||||
|     } | ||||
|  | ||||
|     function hideOverlay(e) { | ||||
|         ev(e); | ||||
|         playvid(false); | ||||
|         if (options.noScrollbars) { | ||||
|             document.documentElement.style.overflowY = 'auto'; | ||||
|             document.body.style.overflowY = 'auto'; | ||||
|         } | ||||
|         if (overlay.style.display === 'none') | ||||
|             return; | ||||
|  | ||||
|         unbind(document, 'keydown', keyDownHandler); | ||||
|         unbind(document, 'keyup', keyUpHandler); | ||||
|         unbind(document, 'fullscreenchange', onFSC); | ||||
|         // Fade out and hide the overlay | ||||
|         overlay.className = ''; | ||||
|         setTimeout(function () { | ||||
|             overlay.style.display = 'none'; | ||||
|             if (options.bodyClass && document.body.classList) | ||||
|                 document.body.classList.remove(options.bodyClass); | ||||
|  | ||||
|             var h = ebi('bbox-halp'); | ||||
|             if (h) | ||||
|                 h.parentNode.removeChild(h); | ||||
|  | ||||
|             if (options.afterHide) | ||||
|                 options.afterHide(); | ||||
|  | ||||
|             documentLastFocus && documentLastFocus.focus(); | ||||
|             isOverlayVisible = false; | ||||
|         }, 500); | ||||
|     } | ||||
|  | ||||
|     function loadImage(index, callback) { | ||||
|         var imageContainer = imagesElements[index]; | ||||
|         var galleryItem = currentGallery[index]; | ||||
|  | ||||
|         if (typeof imageContainer === 'undefined' || typeof galleryItem === 'undefined') | ||||
|             return;  // out-of-bounds or gallery dirty | ||||
|  | ||||
|         if (imageContainer.querySelector('img, video')) | ||||
|             // was loaded, cb and bail | ||||
|             return callback ? callback() : null; | ||||
|  | ||||
|         // maybe unloaded video | ||||
|         while (imageContainer.firstChild) | ||||
|             imageContainer.removeChild(imageContainer.firstChild); | ||||
|  | ||||
|         var imageElement = galleryItem.imageElement, | ||||
|             imageSrc = imageElement.href, | ||||
|             is_vid = re_v.test(imageSrc), | ||||
|             thumbnailElement = imageElement.querySelector('img, video'), | ||||
|             imageCaption = typeof options.captions === 'function' ? | ||||
|                 options.captions.call(currentGallery, imageElement) : | ||||
|                 imageElement.getAttribute('data-caption') || imageElement.title; | ||||
|  | ||||
|         imageSrc += imageSrc.indexOf('?') < 0 ? '?cache' : '&cache'; | ||||
|  | ||||
|         if (is_vid && index != currentIndex) | ||||
|             return;  // no preload | ||||
|  | ||||
|         var figure = mknod('figure'); | ||||
|         figure.id = 'bbox-figure-' + index; | ||||
|         figure.innerHTML = '<div class="bbox-spinner">' + | ||||
|             '<div class="bbox-double-bounce1"></div>' + | ||||
|             '<div class="bbox-double-bounce2"></div>' + | ||||
|             '</div>'; | ||||
|  | ||||
|         if (options.captions && imageCaption) { | ||||
|             var figcaption = mknod('figcaption'); | ||||
|             figcaption.id = 'bbox-figcaption-' + index; | ||||
|             figcaption.innerHTML = imageCaption; | ||||
|             figure.appendChild(figcaption); | ||||
|         } | ||||
|         imageContainer.appendChild(figure); | ||||
|  | ||||
|         var image = mknod(is_vid ? 'video' : 'img'); | ||||
|         clmod(imageContainer, 'vid', is_vid); | ||||
|  | ||||
|         image.addEventListener(is_vid ? 'loadedmetadata' : 'load', function () { | ||||
|             // Remove loader element | ||||
|             var spinner = QS('#baguette-img-' + index + ' .bbox-spinner'); | ||||
|             figure.removeChild(spinner); | ||||
|             if (!options.async && callback) | ||||
|                 callback(); | ||||
|         }); | ||||
|         image.setAttribute('src', imageSrc); | ||||
|         if (is_vid) { | ||||
|             image.setAttribute('controls', 'controls'); | ||||
|             image.onended = vidEnd; | ||||
|         } | ||||
|         image.alt = thumbnailElement ? thumbnailElement.alt || '' : ''; | ||||
|         if (options.titleTag && imageCaption) | ||||
|             image.title = imageCaption; | ||||
|  | ||||
|         figure.appendChild(image); | ||||
|  | ||||
|         if (options.async && callback) | ||||
|             callback(); | ||||
|     } | ||||
|  | ||||
|     function showNextImage(e) { | ||||
|         ev(e); | ||||
|         return show(currentIndex + 1); | ||||
|     } | ||||
|  | ||||
|     function showPreviousImage(e) { | ||||
|         ev(e); | ||||
|         return show(currentIndex - 1); | ||||
|     } | ||||
|  | ||||
|     function showFirstImage(e) { | ||||
|         if (e) | ||||
|             e.preventDefault(); | ||||
|  | ||||
|         return show(0); | ||||
|     } | ||||
|  | ||||
|     function showLastImage(e) { | ||||
|         if (e) | ||||
|             e.preventDefault(); | ||||
|  | ||||
|         return show(currentGallery.length - 1); | ||||
|     } | ||||
|  | ||||
|     function show(index, gallery) { | ||||
|         if (!isOverlayVisible && index >= 0 && index < gallery.length) { | ||||
|             prepareOverlay(gallery, options); | ||||
|             showOverlay(index); | ||||
|             return true; | ||||
|         } | ||||
|         if (index < 0) { | ||||
|             if (options.animation) | ||||
|                 bounceAnimation('left'); | ||||
|  | ||||
|             return false; | ||||
|         } | ||||
|         if (index >= imagesElements.length) { | ||||
|             if (options.animation) | ||||
|                 bounceAnimation('right'); | ||||
|  | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         var v = vid(); | ||||
|         if (v) { | ||||
|             v.src = ''; | ||||
|             v.load(); | ||||
|             v.parentNode.removeChild(v); | ||||
|         } | ||||
|  | ||||
|         currentIndex = index; | ||||
|         loadImage(currentIndex, function () { | ||||
|             preloadNext(currentIndex); | ||||
|             preloadPrev(currentIndex); | ||||
|         }); | ||||
|         updateOffset(); | ||||
|  | ||||
|         if (options.onChange) | ||||
|             options.onChange(currentIndex, imagesElements.length); | ||||
|  | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     function vid() { | ||||
|         return imagesElements[currentIndex].querySelector('video'); | ||||
|     } | ||||
|  | ||||
|     function playvid(play) { | ||||
|         if (vid()) | ||||
|             vid()[play ? 'play' : 'pause'](); | ||||
|     } | ||||
|  | ||||
|     function playpause() { | ||||
|         var v = vid(); | ||||
|         if (v) | ||||
|             v[v.paused ? "play" : "pause"](); | ||||
|     } | ||||
|  | ||||
|     function relseek(sec) { | ||||
|         if (vid()) | ||||
|             vid().currentTime += sec; | ||||
|     } | ||||
|  | ||||
|     function vidEnd() { | ||||
|         if (this == vid() && vnext) | ||||
|             showNextImage(); | ||||
|     } | ||||
|  | ||||
|     function mp_ctl() { | ||||
|         var v = vid(); | ||||
|         if (!vmute && v && mp.au && !mp.au.paused) { | ||||
|             mp.fade_out(); | ||||
|             resume_mp = true; | ||||
|         } | ||||
|         else if (resume_mp && (vmute || !v) && mp.au && mp.au.paused) { | ||||
|             mp.fade_in(); | ||||
|             resume_mp = false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function bounceAnimation(direction) { | ||||
|         slider.className = 'bounce-from-' + direction; | ||||
|         setTimeout(function () { | ||||
|             slider.className = ''; | ||||
|         }, 400); | ||||
|     } | ||||
|  | ||||
|     function updateOffset() { | ||||
|         var offset = -currentIndex * 100 + '%'; | ||||
|         if (options.animation === 'fadeIn') { | ||||
|             slider.style.opacity = 0; | ||||
|             setTimeout(function () { | ||||
|                 slider.style.transform = 'translate3d(' + offset + ',0,0)'; | ||||
|                 slider.style.opacity = 1; | ||||
|             }, 400); | ||||
|         } else { | ||||
|             slider.style.transform = 'translate3d(' + offset + ',0,0)'; | ||||
|         } | ||||
|         playvid(false); | ||||
|         var v = vid(); | ||||
|         if (v) { | ||||
|             playvid(true); | ||||
|             v.muted = vmute; | ||||
|             v.loop = vloop; | ||||
|         } | ||||
|         mp_ctl(); | ||||
|         setVmode(); | ||||
|     } | ||||
|  | ||||
|     function preloadNext(index) { | ||||
|         if (index - currentIndex >= options.preload) | ||||
|             return; | ||||
|  | ||||
|         loadImage(index + 1, function () { | ||||
|             preloadNext(index + 1); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     function preloadPrev(index) { | ||||
|         if (currentIndex - index >= options.preload) | ||||
|             return; | ||||
|  | ||||
|         loadImage(index - 1, function () { | ||||
|             preloadPrev(index - 1); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     function bind(element, event, callback, options) { | ||||
|         element.addEventListener(event, callback, options); | ||||
|     } | ||||
|  | ||||
|     function unbind(element, event, callback, options) { | ||||
|         element.removeEventListener(event, callback, options); | ||||
|     } | ||||
|  | ||||
|     function destroyPlugin() { | ||||
|         unbindEvents(); | ||||
|         clearCachedData(); | ||||
|         unbind(document, 'keydown', keyDownHandler); | ||||
|         unbind(document, 'keyup', keyUpHandler); | ||||
|         document.getElementsByTagName('body')[0].removeChild(ebi('bbox-overlay')); | ||||
|         data = {}; | ||||
|         currentGallery = []; | ||||
|         currentIndex = 0; | ||||
|     } | ||||
|  | ||||
|     return { | ||||
|         run: run, | ||||
|         show: show, | ||||
|         showNext: showNextImage, | ||||
|         showPrevious: showPreviousImage, | ||||
|         relseek: relseek, | ||||
|         playpause: playpause, | ||||
|         hide: hideOverlay, | ||||
|         destroy: destroyPlugin | ||||
|     }; | ||||
| })(); | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -2,74 +2,140 @@ | ||||
| <html lang="en"> | ||||
|  | ||||
| <head> | ||||
|     <meta charset="utf-8"> | ||||
|     <title>⇆🎉 {{ title }}</title> | ||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/browser.css{{ ts }}"> | ||||
|     {%- if can_upload %} | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/upload.css{{ ts }}"> | ||||
|     {%- endif %} | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>⇆🎉 {{ title }}</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
| 	<link rel="stylesheet" type="text/css" media="screen" href="/.cpr/browser.css?_={{ ts }}"> | ||||
| 	<link rel="stylesheet" type="text/css" media="screen" href="/.cpr/upload.css?_={{ ts }}"> | ||||
| 	{%- if css %} | ||||
| 	<link rel="stylesheet" type="text/css" 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>File Size</th> | ||||
|                 <th>Date</th> | ||||
|             </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
| 	<div id="op_search" class="opview"> | ||||
| 		{%- if have_tags_idx %} | ||||
| 		<div id="srch_form" class="tags"></div> | ||||
| 		{%- else %} | ||||
| 		<div id="srch_form"></div> | ||||
| 		{%- endif %} | ||||
| 		<div id="srch_q"></div> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="op_player" class="opview opbox opwide"></div> | ||||
|  | ||||
| 	<div id="op_bup" class="opview opbox act"> | ||||
| 		<div id="u2err"></div> | ||||
| 		<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||
| 			<input type="hidden" name="act" value="bput" /> | ||||
| 			<input type="file" name="f" multiple><br /> | ||||
| 			<input type="submit" value="start upload"> | ||||
| 		</form> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="op_mkdir" class="opview opbox act"> | ||||
| 		<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||
| 			<input type="hidden" name="act" value="mkdir" /> | ||||
| 			📂<input type="text" name="name" size="30"> | ||||
| 			<input type="submit" value="make directory"> | ||||
| 		</form> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="op_new_md" class="opview opbox"> | ||||
| 		<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||
| 			<input type="hidden" name="act" value="new_md" /> | ||||
| 			📝<input type="text" name="name" size="30"> | ||||
| 			<input type="submit" value="new markdown doc"> | ||||
| 		</form> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="op_msg" class="opview opbox act"> | ||||
| 		<form method="post" enctype="application/x-www-form-urlencoded" accept-charset="utf-8" action="{{ url_suf }}"> | ||||
| 			📟<input type="text" name="msg" size="30"> | ||||
| 			<input type="submit" value="send msg to server log"> | ||||
| 		</form> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="op_unpost" class="opview opbox"></div> | ||||
|  | ||||
| 	<div id="op_up2k" class="opview"></div> | ||||
|  | ||||
| 	<div id="op_cfg" class="opview opbox opwide"></div> | ||||
| 	 | ||||
| 	<h1 id="path"> | ||||
| 		<a href="#" id="entree" tt="show navpane (directory tree sidebar)$NHotkey: B">🌲</a> | ||||
| 		{%- for n in vpnodes %} | ||||
| 		<a href="/{{ n[0] }}">{{ n[1] }}</a> | ||||
| 		{%- endfor %} | ||||
| 	</h1> | ||||
| 	 | ||||
| 	<div id="tree"></div> | ||||
|  | ||||
| <div id="wrap"> | ||||
|  | ||||
| 	<div id="pro" class="logue">{{ logues[0] }}</div> | ||||
|  | ||||
| 	<table id="files"> | ||||
| 		<thead> | ||||
| 			<tr> | ||||
| 				<th name="lead"><span>c</span></th> | ||||
| 				<th name="href"><span>File Name</span></th> | ||||
| 				<th name="sz" sort="int"><span>Size</span></th> | ||||
| 				{%- for k in taglist %} | ||||
| 					{%- if k.startswith('.') %} | ||||
| 				<th name="tags/{{ k }}" sort="int"><span>{{ k[1:] }}</span></th> | ||||
| 					{%- else %} | ||||
| 				<th name="tags/{{ k }}"><span>{{ k[0]|upper }}{{ k[1:] }}</span></th> | ||||
| 					{%- endif %} | ||||
| 				{%- endfor %} | ||||
| 				<th name="ext"><span>T</span></th> | ||||
| 				<th name="ts"><span>Date</span></th> | ||||
| 			</tr> | ||||
| 		</thead> | ||||
| <tbody> | ||||
|  | ||||
| {%- for f in files %} | ||||
| <tr><td>{{ f[0] }}</td><td><a href="{{ f[1] }}">{{ f[2] }}</a></td><td>{{ f[3] }}</td><td>{{ f[4] }}</td></tr> | ||||
| <tr><td>{{ f.lead }}</td><td><a href="{{ f.href }}">{{ f.name|e }}</a></td><td>{{ f.sz }}</td> | ||||
| 	{%- if f.tags is defined %} | ||||
| 		{%- for k in taglist %} | ||||
| <td>{{ f.tags[k] }}</td> | ||||
| 		{%- endfor %} | ||||
| 	{%- endif %} | ||||
| <td>{{ f.ext }}</td><td>{{ f.dt }}</td></tr> | ||||
| {%- endfor %} | ||||
|  | ||||
|         </tbody> | ||||
|     </table> | ||||
|      | ||||
|     {%- if epilogue %} | ||||
|     <div id="epi" class="logue">{{ epilogue }}</div> | ||||
|     {%- endif %} | ||||
|     {%- endif %} | ||||
| 		</tbody> | ||||
| 	</table> | ||||
| 	 | ||||
| 	<div id="epi" class="logue">{{ logues[1] }}</div> | ||||
|  | ||||
|     <h2><a href="?h">control-panel</a></h2> | ||||
| 	<h2><a href="/?h">control-panel</a></h2> | ||||
|  | ||||
|     <div id="widget"> | ||||
|         <div id="wtoggle">♫</div> | ||||
|         <div id="widgeti"> | ||||
|             <div id="pctl"><a href="#" id="bprev">⏮</a><a href="#" id="bplay">▶</a><a href="#" id="bnext">⏭</a></div> | ||||
|             <canvas id="pvol" width="288" height="38"></canvas> | ||||
|             <canvas id="barpos"></canvas> | ||||
|             <canvas id="barbuf"></canvas> | ||||
|         </div> | ||||
|     </div> | ||||
|      | ||||
|     {%- if can_read %} | ||||
|     <script src="/.cpr/browser.js{{ ts }}"></script> | ||||
|     {%- endif %} | ||||
|      | ||||
|     {%- if can_upload %} | ||||
|     <script src="/.cpr/up2k.js{{ ts }}"></script> | ||||
|     {%- endif %} | ||||
| </div> | ||||
|  | ||||
| 	{%- if srv_info %} | ||||
| 	<div id="srv_info"><span>{{ srv_info }}</span></div> | ||||
| 	{%- endif %} | ||||
|  | ||||
| 	<div id="widget"></div> | ||||
|  | ||||
| 	<script> | ||||
| 		var acct = "{{ acct }}", | ||||
| 			perms = {{ perms }}, | ||||
| 			tag_order_cfg = {{ tag_order }}, | ||||
| 			have_up2k_idx = {{ have_up2k_idx|tojson }}, | ||||
| 			have_tags_idx = {{ have_tags_idx|tojson }}, | ||||
| 			have_mv = {{ have_mv|tojson }}, | ||||
| 			have_del = {{ have_del|tojson }}, | ||||
| 			have_unpost = {{ have_unpost|tojson }}, | ||||
| 			have_zip = {{ have_zip|tojson }}; | ||||
| 	</script> | ||||
| 	<script src="/.cpr/util.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/browser.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/up2k.js?_={{ ts }}"></script> | ||||
| </body> | ||||
|  | ||||
| </html> | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										60
									
								
								copyparty/web/browser2.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								copyparty/web/browser2.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|  | ||||
| <head> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>{{ title }}</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
| 	<style> | ||||
| 		html{font-family:sans-serif} | ||||
| 		td{border:1px solid #999;border-width:1px 1px 0 0;padding:0 5px} | ||||
| 		a{display:block} | ||||
| 	</style> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
| 	{%- if srv_info %} | ||||
| 	<p><span>{{ srv_info }}</span></p> | ||||
| 	{%- endif %} | ||||
|  | ||||
| 	{%- if have_b_u %} | ||||
| 	<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||
| 		<input type="hidden" name="act" value="bput" /> | ||||
| 		<input type="file" name="f" multiple /><br /> | ||||
| 		<input type="submit" value="start upload" /> | ||||
| 	</form> | ||||
| 	<br /> | ||||
| 	{%- endif %} | ||||
|  | ||||
| 	{%- if logues[0] %} | ||||
| 	<div>{{ logues[0] }}</div><br /> | ||||
| 	{%- endif %} | ||||
|  | ||||
| 	<table id="files"> | ||||
| 		<thead> | ||||
| 			<tr> | ||||
| 				<th name="lead"><span>c</span></th> | ||||
| 				<th name="href"><span>File Name</span></th> | ||||
| 				<th name="sz" sort="int"><span>Size</span></th> | ||||
| 				<th name="ts"><span>Date</span></th> | ||||
| 			</tr> | ||||
| 		</thead> | ||||
| 		<tbody> | ||||
| <tr><td></td><td><a href="../{{ url_suf }}">parent folder</a></td><td>-</td><td>-</td></tr> | ||||
|  | ||||
| {%- for f in files %} | ||||
| <tr><td>{{ f.lead }}</td><td><a href="{{ f.href }}{{ url_suf }}">{{ f.name|e }}</a></td><td>{{ f.sz }}</td><td>{{ f.dt }}</td></tr> | ||||
| {%- endfor %} | ||||
|  | ||||
| 		</tbody> | ||||
| 	</table> | ||||
| 	 | ||||
| 	{%- if logues[1] %} | ||||
| 	<div>{{ logues[1] }}</div><br /> | ||||
| 	{%- endif %} | ||||
| 	 | ||||
| 	<h2><a href="/{{ url_suf }}{{ url_suf and '&' or '?' }}h">control-panel</a></h2> | ||||
|  | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										61
									
								
								copyparty/web/dbg-audio.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								copyparty/web/dbg-audio.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| var ofun = audio_eq.apply.bind(audio_eq); | ||||
| audio_eq.apply = function () { | ||||
|     var ac1 = mp.ac; | ||||
|     ofun(); | ||||
|     var ac = mp.ac, | ||||
|         w = 2048, | ||||
|         h = 256; | ||||
|  | ||||
|     if (!audio_eq.filters.length) { | ||||
|         audio_eq.ana = null; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     var can = ebi('fft_can'); | ||||
|     if (!can) { | ||||
|         can = mknod('canvas'); | ||||
|         can.setAttribute('id', 'fft_can'); | ||||
|         can.style.cssText = 'position:absolute;left:0;bottom:5em;width:' + w + 'px;height:' + h + 'px;z-index:9001'; | ||||
|         document.body.appendChild(can); | ||||
|         can.width = w; | ||||
|         can.height = h; | ||||
|     } | ||||
|     var cc = can.getContext('2d'); | ||||
|     if (!ac) | ||||
|         return; | ||||
|  | ||||
|     var ana = ac.createAnalyser(); | ||||
|     ana.smoothingTimeConstant = 0; | ||||
|     ana.fftSize = 8192; | ||||
|  | ||||
|     audio_eq.filters[0].connect(ana); | ||||
|     audio_eq.ana = ana; | ||||
|  | ||||
|     var buf = new Uint8Array(ana.frequencyBinCount), | ||||
|         colw = can.width / buf.length; | ||||
|  | ||||
|     cc.fillStyle = '#fc0'; | ||||
|     function draw() { | ||||
|         if (ana == audio_eq.ana) | ||||
|             requestAnimationFrame(draw); | ||||
|  | ||||
|         ana.getByteFrequencyData(buf); | ||||
|  | ||||
|         cc.clearRect(0, 0, can.width, can.height); | ||||
|  | ||||
|         /*var x = 0, w = 1; | ||||
|         for (var a = 0; a < buf.length; a++) { | ||||
|             cc.fillRect(x, h - buf[a], w, h); | ||||
|             x += w; | ||||
|         }*/ | ||||
|         var mul = Math.pow(w, 4) / buf.length; | ||||
|         for (var x = 0; x < w; x++) { | ||||
|             var a = Math.floor(Math.pow(x, 4) / mul), | ||||
|                 v = buf[a]; | ||||
|  | ||||
|             cc.fillRect(x, h - v, 1, v); | ||||
|         } | ||||
|     } | ||||
|     draw(); | ||||
| }; | ||||
| audio_eq.apply(); | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 248 B | 
| @@ -1,13 +1,153 @@ | ||||
| @font-face { | ||||
| 	font-family: 'scp'; | ||||
| 	src: local('Source Code Pro Regular'), local('SourceCodePro-Regular'), url(/.cpr/deps/scp.woff2) format('woff2'); | ||||
| } | ||||
| html, body { | ||||
| 	color: #333; | ||||
| 	background: #eee; | ||||
| 	font-family: sans-serif; | ||||
| 	line-height: 1.5em; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| #tt, #toast { | ||||
| 	position: fixed; | ||||
| 	max-width: 34em; | ||||
| 	background: #222; | ||||
| 	border: 0 solid #777; | ||||
| 	box-shadow: 0 .2em .5em #222; | ||||
| 	border-radius: .4em; | ||||
| 	z-index: 9001; | ||||
| } | ||||
| #tt { | ||||
| 	overflow: hidden; | ||||
| 	margin-top: 1em; | ||||
| 	padding: 0 1.3em; | ||||
| 	height: 0; | ||||
| 	opacity: .1; | ||||
| 	transition: opacity 0.14s, height 0.14s, padding 0.14s; | ||||
| } | ||||
| #toast { | ||||
| 	top: 1.4em; | ||||
| 	right: -1em; | ||||
| 	line-height: 1.5em; | ||||
| 	padding: 1em 1.3em; | ||||
| 	border-width: .4em 0; | ||||
| 	transform: translateX(100%); | ||||
| 	transition: | ||||
| 		transform .4s cubic-bezier(.2, 1.2, .5, 1), | ||||
| 		right .4s cubic-bezier(.2, 1.2, .5, 1); | ||||
| 	text-shadow: 1px 1px 0 #000; | ||||
| 	color: #fff; | ||||
| } | ||||
| #toast pre { | ||||
| 	margin: 0; | ||||
| } | ||||
| #toastc { | ||||
| 	display: inline-block; | ||||
| 	position: absolute; | ||||
| 	overflow: hidden; | ||||
| 	left: 0; | ||||
| 	width: 0; | ||||
| 	opacity: 0; | ||||
| 	padding: .3em 0; | ||||
| 	margin: -.3em 0 0 0; | ||||
| 	line-height: 1.5em; | ||||
| 	color: #000; | ||||
| 	border: none; | ||||
| 	outline: none; | ||||
| 	text-shadow: none; | ||||
| 	border-radius: .5em 0 0 .5em; | ||||
| 	transition: left .3s, width .3s, padding .3s, opacity .3s; | ||||
| } | ||||
| #toast.vis { | ||||
| 	right: 1.3em; | ||||
| 	transform: unset; | ||||
| } | ||||
| #toast.vis #toastc { | ||||
| 	left: -2em; | ||||
| 	width: .4em; | ||||
| 	padding: .3em .8em; | ||||
| 	opacity: 1; | ||||
| } | ||||
| #toast.inf { | ||||
| 	background: #07a; | ||||
| 	border-color: #0be; | ||||
| } | ||||
| #toast.inf #toastc { | ||||
| 	background: #0be; | ||||
| } | ||||
| #toast.ok { | ||||
| 	background: #4a0; | ||||
| 	border-color: #8e4; | ||||
| } | ||||
| #toast.ok #toastc { | ||||
| 	background: #8e4; | ||||
| } | ||||
| #toast.warn { | ||||
| 	background: #970; | ||||
| 	border-color: #fc0; | ||||
| } | ||||
| #toast.warn #toastc { | ||||
| 	background: #fc0; | ||||
| } | ||||
| #toast.err { | ||||
| 	background: #900; | ||||
| 	border-color: #d06; | ||||
| } | ||||
| #toast.err #toastc { | ||||
| 	background: #d06; | ||||
| } | ||||
| #tt.b { | ||||
| 	padding: 0 2em; | ||||
| 	border-radius: .5em; | ||||
| 	box-shadow: 0 .2em 1em #000; | ||||
| } | ||||
| #tt.show { | ||||
| 	padding: 1em 1.3em; | ||||
| 	border-width: .4em 0; | ||||
| 	height: auto; | ||||
| 	opacity: 1; | ||||
| } | ||||
| #tt.show.b { | ||||
| 	padding: 1.5em 2em; | ||||
| 	border-width: .5em 0; | ||||
| } | ||||
| #tt code { | ||||
| 	background: #3c3c3c; | ||||
| 	padding: .1em .3em; | ||||
| 	border-top: 1px solid #777; | ||||
| 	border-radius: .3em; | ||||
| 	line-height: 1.7em; | ||||
| } | ||||
| #tt em { | ||||
| 	color: #f6a; | ||||
| } | ||||
| html.light #tt { | ||||
| 	background: #fff; | ||||
| 	border-color: #888 #000 #777 #000; | ||||
| } | ||||
| html.light #tt, | ||||
| html.light #toast { | ||||
| 	box-shadow: 0 .3em 1em rgba(0,0,0,0.4); | ||||
| } | ||||
| html.light #tt code { | ||||
| 	background: #060; | ||||
| 	color: #fff; | ||||
| } | ||||
| html.light #tt em { | ||||
| 	color: #d38; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| #mtw { | ||||
| 	display: none; | ||||
| } | ||||
| #mw { | ||||
| 	width: 48.5em; | ||||
| 	margin: 0 auto; | ||||
| 	margin-bottom: 6em; | ||||
| 	padding: 0 1.5em; | ||||
| } | ||||
| pre, code, a { | ||||
| 	color: #480; | ||||
| @@ -20,8 +160,8 @@ pre, code, a { | ||||
| code { | ||||
| 	font-size: .96em; | ||||
| } | ||||
| pre, code { | ||||
| 	font-family: monospace, monospace; | ||||
| pre, code, tt { | ||||
| 	font-family: 'scp', monospace, monospace; | ||||
| 	white-space: pre-wrap; | ||||
| 	word-break: break-all; | ||||
| } | ||||
| @@ -41,9 +181,12 @@ pre code { | ||||
| pre code:last-child { | ||||
| 	border-bottom: none; | ||||
| } | ||||
| pre code:before { | ||||
| pre code::before { | ||||
| 	content: counter(precode); | ||||
| 	-webkit-user-select: none; | ||||
| 	-moz-user-select: none; | ||||
| 	-ms-user-select: none; | ||||
| 	user-select: none; | ||||
| 	display: inline-block; | ||||
| 	text-align: right; | ||||
| 	font-size: .75em; | ||||
| @@ -76,31 +219,39 @@ h2 { | ||||
| 	padding-left: .4em; | ||||
| 	margin-top: 3em; | ||||
| } | ||||
| h3 { | ||||
| 	border-bottom: .1em solid #999; | ||||
| } | ||||
| h1 a, h3 a, h5 a, | ||||
| h2 a, h4 a, h6 a { | ||||
| 	color: inherit; | ||||
| 	display: block; | ||||
| 	background: none; | ||||
| 	border: none; | ||||
| 	padding: 0; | ||||
| 	margin: 0; | ||||
| } | ||||
| #m ul, | ||||
| #m ol { | ||||
| #mp ul, | ||||
| #mp ol { | ||||
| 	border-left: .3em solid #ddd; | ||||
| } | ||||
| #m>ul, | ||||
| #m>ol { | ||||
| 	border-color: #bbb; | ||||
| } | ||||
| #m ul>li { | ||||
| #mp ul>li { | ||||
| 	list-style-type: disc; | ||||
| } | ||||
| #m ul>li, | ||||
| #m ol>li { | ||||
| #mp ul>li, | ||||
| #mp ol>li { | ||||
| 	margin: .7em 0; | ||||
| } | ||||
| strong { | ||||
| 	color: #000; | ||||
| } | ||||
| p>em, | ||||
| li>em { | ||||
| li>em, | ||||
| td>em { | ||||
| 	color: #c50; | ||||
| 	padding: .1em; | ||||
| 	border-bottom: .1em solid #bbb; | ||||
| @@ -116,8 +267,9 @@ small { | ||||
| 	opacity: .8; | ||||
| } | ||||
| #toc { | ||||
| 	width: 48.5em; | ||||
| 	margin: 0 auto; | ||||
| 	margin: 0 1em; | ||||
| 	-ms-scroll-chaining: none; | ||||
| 	overscroll-behavior-y: none; | ||||
| } | ||||
| #toc ul { | ||||
| 	padding-left: 1em; | ||||
| @@ -148,7 +300,7 @@ small { | ||||
| 	z-index: 99; | ||||
| 	position: relative; | ||||
| 	display: inline-block; | ||||
| 	font-family: monospace, monospace; | ||||
| 	font-family: 'scp', monospace, monospace; | ||||
| 	font-weight: bold; | ||||
| 	font-size: 1.3em; | ||||
| 	line-height: .1em; | ||||
| @@ -162,14 +314,12 @@ small { | ||||
| } | ||||
| table { | ||||
| 	border-collapse: collapse; | ||||
| 	margin: 1em 0; | ||||
| } | ||||
| td { | ||||
| th, td { | ||||
| 	padding: .2em .5em; | ||||
| 	border: .12em solid #aaa; | ||||
| } | ||||
| th { | ||||
| 	border: .12em solid #aaa; | ||||
| } | ||||
| blink { | ||||
| 	animation: blinker .7s cubic-bezier(.9, 0, .1, 1) infinite; | ||||
| } | ||||
| @@ -181,10 +331,26 @@ blink { | ||||
| 		opacity: 1; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @media screen { | ||||
| 	html, body { | ||||
| 		margin: 0; | ||||
| 		padding: 0; | ||||
| 		outline: 0; | ||||
| 		border: none; | ||||
| 		width: 100%; | ||||
| 		height: 100%; | ||||
| 	} | ||||
| 	#mw { | ||||
| 		margin: 0 auto; | ||||
| 		right: 0; | ||||
| 	} | ||||
| 	#mp { | ||||
| 		max-width: 52em; | ||||
| 		margin-bottom: 6em; | ||||
| 		word-break: break-word; | ||||
| 		overflow-wrap: break-word; | ||||
| 		word-wrap: break-word; /*ie*/ | ||||
| 	} | ||||
| 	a { | ||||
| 		color: #fff; | ||||
| @@ -212,15 +378,17 @@ blink { | ||||
| 		padding: .5em 0; | ||||
| 	} | ||||
| 	#mn { | ||||
| 		font-weight: normal; | ||||
| 		padding: 1.3em 0 .7em 1em; | ||||
| 		font-size: 1.4em; | ||||
| 		border-bottom: 1px solid #ccc; | ||||
| 		background: #eee; | ||||
| 		z-index: 10; | ||||
| 		width: calc(100% - 1em); | ||||
| 	} | ||||
| 	#mn a { | ||||
| 		color: #444; | ||||
| 		background: none; | ||||
| 		margin: 0 0 0 -.2em; | ||||
| 		padding: 0 0 0 .4em; | ||||
| 		padding: .3em 0 .3em .4em; | ||||
| 		text-decoration: none; | ||||
| 		border: none; | ||||
| 		/* ie: */ | ||||
| @@ -233,14 +401,14 @@ blink { | ||||
| 	#mn a:last-child { | ||||
| 		padding-right: .5em; | ||||
| 	} | ||||
| 	#mn a:not(:last-child):after { | ||||
| 	#mn a:not(:last-child)::after { | ||||
| 		content: ''; | ||||
| 		width: 1.05em; | ||||
| 		height: 1.05em; | ||||
| 		margin: -.2em .3em -.2em -.4em; | ||||
| 		display: inline-block; | ||||
| 		border: 1px solid rgba(0,0,0,0.3); | ||||
| 		border-width: .05em .05em 0 0; | ||||
| 		border: 1px solid rgba(0,0,0,0.2); | ||||
| 		border-width: .2em .2em 0 0; | ||||
| 		transform: rotate(45deg); | ||||
| 	} | ||||
| 	#mn a:hover { | ||||
| @@ -248,7 +416,45 @@ blink { | ||||
| 		text-decoration: underline; | ||||
| 	} | ||||
| 	#mh { | ||||
| 		margin: 0 0 1.5em 0; | ||||
| 		padding: .4em 1em; | ||||
| 		position: relative; | ||||
| 		width: 100%; | ||||
| 		width: calc(100% - 3em); | ||||
| 		background: #eee; | ||||
| 		z-index: 9; | ||||
| 		top: 0; | ||||
| 	} | ||||
| 	#mh a { | ||||
| 		color: #444; | ||||
| 		background: none; | ||||
| 		text-decoration: underline; | ||||
| 		border: none; | ||||
| 	} | ||||
| 	#mh a:hover { | ||||
| 		color: #000; | ||||
| 		background: #ddd; | ||||
| 	} | ||||
| 	#toolsbox { | ||||
| 		overflow: hidden; | ||||
| 		display: inline-block; | ||||
| 		background: #eee; | ||||
| 		height: 1.5em; | ||||
| 		padding: 0 .2em; | ||||
| 		margin: 0 .2em; | ||||
| 		position: absolute; | ||||
| 	} | ||||
| 	#toolsbox.open { | ||||
| 		height: auto; | ||||
| 		overflow: visible; | ||||
| 		background: #eee; | ||||
| 		box-shadow: 0 .2em .2em #ccc; | ||||
| 		padding-bottom: .2em; | ||||
| 	} | ||||
| 	#toolsbox a { | ||||
| 		display: block; | ||||
| 	} | ||||
| 	#toolsbox a+a { | ||||
| 		text-decoration: none; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| @@ -270,13 +476,12 @@ blink { | ||||
| 	html.dark #toc li { | ||||
| 		border-width: 0; | ||||
| 	} | ||||
| 	html.dark #m a, | ||||
| 	html.dark #mh a { | ||||
| 	html.dark #mp a { | ||||
| 		background: #057; | ||||
| 	} | ||||
| 	html.dark #m h1 a, html.dark #m h4 a, | ||||
| 	html.dark #m h2 a, html.dark #m h5 a, | ||||
| 	html.dark #m h3 a, html.dark #m h6 a { | ||||
| 	html.dark #mp h1 a, html.dark #mp h4 a, | ||||
| 	html.dark #mp h2 a, html.dark #mp h5 a, | ||||
| 	html.dark #mp h3 a, html.dark #mp h6 a { | ||||
| 		color: inherit; | ||||
| 		background: none; | ||||
| 	} | ||||
| @@ -286,16 +491,20 @@ blink { | ||||
| 		background: #1a1a1a; | ||||
| 		border: .07em solid #333; | ||||
| 	} | ||||
| 	html.dark #m ul, | ||||
| 	html.dark #m ol { | ||||
| 	html.dark #mp ul, | ||||
| 	html.dark #mp ol { | ||||
| 		border-color: #444; | ||||
| 	} | ||||
| 	html.dark #m>ul, | ||||
| 	html.dark #m>ol { | ||||
| 		border-color: #555; | ||||
| 	} | ||||
| 	html.dark strong { | ||||
| 		color: #fff; | ||||
| 	} | ||||
| 	html.dark p>em, | ||||
| 	html.dark li>em { | ||||
| 	html.dark li>em, | ||||
| 	html.dark td>em { | ||||
| 		color: #f94; | ||||
| 		border-color: #666; | ||||
| 	} | ||||
| @@ -316,32 +525,61 @@ blink { | ||||
| 		background: #282828; | ||||
| 		border: .07em dashed #444; | ||||
| 	} | ||||
| 	html.dark #mn a:not(:last-child):after { | ||||
| 	html.dark #mn a:not(:last-child)::after { | ||||
| 		border-color: rgba(255,255,255,0.3); | ||||
| 	} | ||||
| 	html.dark #mn a { | ||||
| 		color: #ccc; | ||||
| 	} | ||||
| 	html.dark #mn { | ||||
| 		border-bottom: 1px solid #333; | ||||
| 	} | ||||
| 	html.dark #mn, | ||||
| 	html.dark #mh { | ||||
| 		background: #222; | ||||
| 	} | ||||
| 	html.dark #mh a { | ||||
| 		color: #ccc; | ||||
| 		background: none; | ||||
| 	} | ||||
| 	html.dark #mh a:hover { | ||||
| 		background: #333; | ||||
| 		color: #fff; | ||||
| 	} | ||||
| 	html.dark #toolsbox { | ||||
| 		background: #222; | ||||
| 	} | ||||
| 	html.dark #toolsbox.open { | ||||
| 		box-shadow: 0 .2em .2em #069; | ||||
| 		border-radius: 0 0 .4em .4em; | ||||
| 	} | ||||
| } | ||||
| @media screen and (min-width: 64em) { | ||||
|  | ||||
| @media screen and (min-width: 66em) { | ||||
| 	#mw { | ||||
| 		margin-left: 14em; | ||||
| 		margin-left: calc(100% - 50em); | ||||
| 		position: fixed; | ||||
| 		overflow-y: auto; | ||||
| 		left: 14em; | ||||
| 		left: calc(100% - 55em); | ||||
| 		max-width: none; | ||||
| 		bottom: 0; | ||||
| 		scrollbar-color: #eb0 #f7f7f7; | ||||
| 	} | ||||
| 	#toc { | ||||
| 		width: 13em; | ||||
| 		width: calc(100% - 52.3em); | ||||
| 		width: calc(100% - 55.3em); | ||||
| 		max-width: 30em; | ||||
| 		background: #eee; | ||||
| 		position: fixed; | ||||
| 		overflow-y: auto; | ||||
| 		top: 0; | ||||
| 		left: 0; | ||||
| 		height: 100%; | ||||
| 		overflow-y: auto; | ||||
| 		bottom: 0; | ||||
| 		padding: 0; | ||||
| 		margin: 0; | ||||
| 		box-shadow: 0 0 1em #ccc; | ||||
| 		scrollbar-color: #eb0 #f7f7f7; | ||||
| 		xscrollbar-width: thin; | ||||
| 		box-shadow: 0 0 1em rgba(0,0,0,0.1); | ||||
| 		border-top: 1px solid #d7d7d7; | ||||
| 	} | ||||
| 	#toc li { | ||||
| 		border-left: .3em solid #ccc; | ||||
| @@ -361,37 +599,132 @@ blink { | ||||
| 	 | ||||
| 	html.dark #toc { | ||||
| 		background: #282828; | ||||
| 		border-top: 1px solid #2c2c2c; | ||||
| 		box-shadow: 0 0 1em #181818; | ||||
| 	} | ||||
| 	html.dark #toc, | ||||
| 	html.dark #mw { | ||||
| 		scrollbar-color: #b80 #282828; | ||||
| 	} | ||||
| 	html.dark #toc::-webkit-scrollbar-track { | ||||
| 		background: #282828; | ||||
| 	} | ||||
| 	html.dark #toc::-webkit-scrollbar { | ||||
| 		background: #282828; | ||||
| 		width: .8em; | ||||
| 	} | ||||
| 	html.dark #toc::-webkit-scrollbar-thumb { | ||||
| 		background: #b80; | ||||
| 	} | ||||
| } | ||||
| @media screen and (min-width: 84em) { | ||||
| @media screen and (min-width: 85.5em) { | ||||
| 	#toc { width: 30em } | ||||
| 	#mw { margin-left: 32em } | ||||
| 	#mw { left: 30.5em } | ||||
| } | ||||
| @media print { | ||||
| 	@page { | ||||
| 		size: A4; | ||||
| 		padding: 0; | ||||
| 		margin: .5in .6in; | ||||
| 		mso-header-margin: .6in; | ||||
| 		mso-footer-margin: .6in; | ||||
| 		mso-paper-source: 0; | ||||
| 	} | ||||
| 	a { | ||||
| 		color: #079; | ||||
| 		text-decoration: none; | ||||
| 		border-bottom: .07em solid #4ac; | ||||
| 		padding: 0 .3em; | ||||
| 	} | ||||
| 	#toc { | ||||
| 		margin: 0 !important; | ||||
| 	} | ||||
| 	#toc>ul { | ||||
| 		border-left: .1em solid #84c4dd; | ||||
| 	} | ||||
| 	#mn, #mh { | ||||
| 		display: none; | ||||
| 	} | ||||
| } | ||||
| 	html, body, #toc, #mw { | ||||
| 		margin: 0 !important; | ||||
| 		word-break: break-word; | ||||
| 		width: 52em; | ||||
| 	} | ||||
| 	#toc { | ||||
| 		margin-left: 1em !important; | ||||
| 	} | ||||
| 	#toc a { | ||||
| 		color: #000 !important; | ||||
| 	} | ||||
| 	#toc a::after { | ||||
| 		/* hopefully supported by browsers eventually */ | ||||
| 		content: leader('.') target-counter(attr(href), page); | ||||
| 	} | ||||
| 	a[ctr]::before { | ||||
| 		content: attr(ctr) '. '; | ||||
| 	} | ||||
| 	h1 { | ||||
| 		margin: 2em 0; | ||||
| 	} | ||||
| 	h2 { | ||||
| 		margin: 2em 0 0 0; | ||||
| 	} | ||||
| 	h1, h2, h3 { | ||||
| 		page-break-inside: avoid; | ||||
| 	} | ||||
| 	h1::after, | ||||
| 	h2::after, | ||||
| 	h3::after { | ||||
| 		content: 'orz'; | ||||
| 		color: transparent; | ||||
| 		display: block; | ||||
| 		line-height: 1em; | ||||
| 		padding: 4em 0 0 0; | ||||
| 		margin: 0 0 -5em 0; | ||||
| 	} | ||||
| 	p { | ||||
| 		page-break-inside: avoid; | ||||
| 	} | ||||
| 	table { | ||||
| 		page-break-inside: auto; | ||||
| 	} | ||||
| 	tr { | ||||
| 		page-break-inside: avoid; | ||||
| 		page-break-after: auto; | ||||
| 	} | ||||
| 	thead { | ||||
| 		display: table-header-group; | ||||
| 	} | ||||
| 	tfoot { | ||||
| 		display: table-footer-group; | ||||
| 	} | ||||
| 	#mp a.vis::after { | ||||
| 		content: ' (' attr(href) ')'; | ||||
| 		border-bottom: 1px solid #bbb; | ||||
| 		color: #444; | ||||
| 	} | ||||
| 	blockquote { | ||||
| 		border-color: #555; | ||||
| 	} | ||||
| 	code { | ||||
| 		border-color: #bbb; | ||||
| 	} | ||||
| 	pre, pre code { | ||||
| 		border-color: #999; | ||||
| 	} | ||||
| 	pre code::before { | ||||
| 		color: #058; | ||||
| 	} | ||||
|  | ||||
| /* | ||||
| *[data-ln]:before { | ||||
| 	content: attr(data-ln); | ||||
| 	font-size: .8em; | ||||
| 	margin: 0 .4em; | ||||
| 	color: #f0c; | ||||
|  | ||||
| 	 | ||||
| 	html.dark a { | ||||
| 		color: #000; | ||||
| 	} | ||||
| 	html.dark pre, | ||||
| 	html.dark code { | ||||
| 		color: #240; | ||||
| 	} | ||||
| 	html.dark p>em, | ||||
| 	html.dark li>em, | ||||
| 	html.dark td>em { | ||||
| 		color: #940; | ||||
| 	} | ||||
| } | ||||
| */ | ||||
| @@ -3,45 +3,153 @@ | ||||
| 	<title>📝🎉 {{ title }}</title> <!-- 📜 --> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=0.7"> | ||||
| 	<link href="/.cpr/md.css" rel="stylesheet"> | ||||
| 	<link href="/.cpr/md.css?_={{ ts }}" rel="stylesheet"> | ||||
| 	{%- if edit %} | ||||
| 	<link href="/.cpr/md2.css?_={{ ts }}" rel="stylesheet"> | ||||
| 	{%- endif %} | ||||
| </head> | ||||
| <body> | ||||
| 	<div id="mn"></div> | ||||
| 	<div id="mn">navbar</div> | ||||
| 	<div id="mh"> | ||||
| 		<a id="lightswitch" href="#">go dark</a> | ||||
| 		<a id="navtoggle" href="#">hide nav</a> | ||||
| 		{%- if edit %} | ||||
| 			<a id="save" href="?edit" tt="Hotkey: ctrl-s">save</a> | ||||
| 			<a id="sbs" href="#" tt="editor and preview side by side">sbs</a> | ||||
| 			<a id="nsbs" href="#" tt="switch between editor and preview$NHotkey: ctrl-e">editor</a> | ||||
| 			<div id="toolsbox"> | ||||
| 				<a id="tools" href="#">tools</a> | ||||
| 				<a id="fmt_table" href="#">prettify table (ctrl-k)</a> | ||||
| 				<a id="iter_uni" href="#">non-ascii: iterate (ctrl-u)</a> | ||||
| 				<a id="mark_uni" href="#">non-ascii: markup</a> | ||||
| 				<a id="cfg_uni" href="#">non-ascii: whitelist</a> | ||||
| 				<a id="help" href="#">help</a> | ||||
| 			</div> | ||||
| 		{%- else %} | ||||
| 			<a href="?edit" tt="good: higher performance$Ngood: same document width as viewer$Nbad: assumes you know markdown">edit (basic)</a> | ||||
| 			<a href="?edit2" tt="not in-house so probably less buggy">edit (fancy)</a> | ||||
| 			<a href="?raw">view raw</a> | ||||
| 		{%- endif %} | ||||
| 	</div> | ||||
| 	<div id="toc"></div> | ||||
| 	<div id="mtw"> | ||||
| 		<textarea id="mt" autocomplete="off">{{ md }}</textarea> | ||||
| 	</div> | ||||
| 	<div id="mw"> | ||||
| 		<div id="mh"> | ||||
| 			<a id="lightswitch" href="#">go dark</a> // | ||||
| 			<a id="edit" href="?edit">edit this</a> | ||||
| 		</div> | ||||
| 		<div id="ml"> | ||||
| 			<div style="text-align:center;margin:5em 0"> | ||||
| 				<div style="font-size:2em;margin:1em 0">Loading</div> | ||||
| 				if you're still reading this, check that javascript is allowed | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div id="m"> | ||||
| 			<textarea id="mt" style="display:none">{{ md }}</textarea> | ||||
| 		</div> | ||||
| 		<div id="mp"></div> | ||||
| 	</div> | ||||
| 	 | ||||
| 	{%- if edit %} | ||||
| 	<div id="helpbox"> | ||||
| 		<textarea autocomplete="off"> | ||||
|  | ||||
| write markdown (most html is 🙆 too) | ||||
|  | ||||
| ## hotkey list | ||||
| * `Ctrl-S` to save | ||||
| * `Ctrl-E` to toggle mode | ||||
| * `Ctrl-K` to prettyprint a table | ||||
| * `Ctrl-U` to iterate non-ascii chars | ||||
| * `Ctrl-H` / `Ctrl-Shift-H` to create a header | ||||
| * `TAB` / `Shift-TAB` to indent/dedent a selection | ||||
|  | ||||
| ## toolbar | ||||
| 1. toggle dark mode | ||||
| 2. show/hide navigation bar | ||||
| 3. save changes on server | ||||
| 4. side-by-side editing | ||||
| 5. toggle editor/preview | ||||
| 6. this thing :^) | ||||
|  | ||||
| ## markdown | ||||
| ||| | ||||
| |--|--| | ||||
| |`**bold**`|**bold**| | ||||
| |`_italic_`|_italic_| | ||||
| |`~~strike~~`|~~strike~~| | ||||
| |`` `code` ``|`code`| | ||||
| |`[](#hotkey-list)`|[](#hotkey-list)| | ||||
| |`[](/foo/bar.md#header)`|[](/foo/bar.md#header)| | ||||
| |`<blink>💯</blink>`|<blink>💯</blink>| | ||||
|  | ||||
| ## tables | ||||
|     |left-aligned|centered|right-aligned | ||||
|     | ---------- | :----: | ----------: | ||||
|     |one         |two     |three | ||||
|  | ||||
| |left-aligned|centered|right-aligned | ||||
| | ---------- | :----: | ----------: | ||||
| |one         |two     |three | ||||
|  | ||||
| ## lists | ||||
| 	* one | ||||
| 	* two | ||||
| 	1. one | ||||
| 	1. two | ||||
| * one | ||||
| * two | ||||
| 1. one | ||||
| 1. two | ||||
|  | ||||
| ## headers | ||||
| 	# level 1 | ||||
| 	## level 2 | ||||
| 	### level 3 | ||||
|  | ||||
| ## quote | ||||
| 	> hello | ||||
| > hello | ||||
|  | ||||
| ## codeblock | ||||
| 		four spaces (no tab pls) | ||||
|  | ||||
| ## code in lists | ||||
| 	* foo | ||||
| 	  bar | ||||
|           six spaces total | ||||
| * foo | ||||
|   bar | ||||
|       six spaces total | ||||
| . | ||||
| 		</textarea> | ||||
| 	</div> | ||||
| 	{%- endif %} | ||||
| 	 | ||||
| 	<script> | ||||
|  | ||||
| var link_md_as_html = false;  // TODO (does nothing) | ||||
| var last_modified = {{ lastmod }}; | ||||
| var md_opt = { | ||||
| 	link_md_as_html: false, | ||||
| 	allow_plugins: {{ md_plug }}, | ||||
| 	modpoll_freq: {{ md_chk_rate }} | ||||
| }; | ||||
|  | ||||
| (function () { | ||||
|     var btn = document.getElementById("lightswitch"); | ||||
|     var toggle = function () { | ||||
|         var dark = !document.documentElement.getAttribute("class"); | ||||
|         document.documentElement.setAttribute("class", dark ? "dark" : ""); | ||||
|         btn.innerHTML = "go " + (dark ? "light" : "dark"); | ||||
|         if (window.localStorage) | ||||
|             localStorage.setItem('darkmode', dark ? 1 : 0); | ||||
|     }; | ||||
|     btn.onclick = toggle; | ||||
|     if (window.localStorage && localStorage.getItem('darkmode') == 1) | ||||
| 		toggle(); | ||||
|     var l = localStorage, | ||||
| 		drk = l.getItem('lightmode') != 1, | ||||
| 		btn = document.getElementById("lightswitch"), | ||||
| 		f = function (e) { | ||||
| if (e) { e.preventDefault(); drk = !drk; } | ||||
| document.documentElement.setAttribute("class", drk? "dark":"light"); | ||||
| btn.innerHTML = "go " + (drk ? "light":"dark"); | ||||
| l.setItem('lightmode', drk? 0:1); | ||||
|     	}; | ||||
| 	 | ||||
| 	btn.onclick = f; | ||||
| 	f(); | ||||
| })(); | ||||
|  | ||||
| 	</script> | ||||
| 	<script src="/.cpr/deps/marked.full.js"></script> | ||||
| 	<script src="/.cpr/md.js"></script> | ||||
|     <script src="/.cpr/util.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/deps/marked.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/md.js?_={{ ts }}"></script> | ||||
| 	{%- if edit %} | ||||
| 	<script src="/.cpr/md2.js?_={{ ts }}"></script> | ||||
| 	{%- endif %} | ||||
| </body></html> | ||||
|   | ||||
| @@ -1,17 +1,60 @@ | ||||
| /*var conv = new showdown.Converter(); | ||||
| conv.setFlavor('github'); | ||||
| conv.setOption('tasklists', 0); | ||||
| var mhtml = conv.makeHtml(dom_md.value); | ||||
| */ | ||||
| "use strict"; | ||||
|  | ||||
| var dom_toc = document.getElementById('toc'); | ||||
| var dom_wrap = document.getElementById('mw'); | ||||
| var dom_head = document.getElementById('mh'); | ||||
| var dom_nav = document.getElementById('mn'); | ||||
| var dom_doc = document.getElementById('m'); | ||||
| var dom_md = document.getElementById('mt'); | ||||
| var dom_toc = ebi('toc'); | ||||
| var dom_wrap = ebi('mw'); | ||||
| var dom_hbar = ebi('mh'); | ||||
| var dom_nav = ebi('mn'); | ||||
| var dom_pre = ebi('mp'); | ||||
| var dom_src = ebi('mt'); | ||||
| var dom_navtgl = ebi('navtoggle'); | ||||
|  | ||||
| // add toolbar buttons | ||||
|  | ||||
| // chrome 49 needs this | ||||
| var chromedbg = function () { console.log(arguments); } | ||||
|  | ||||
| // null-logger | ||||
| var dbg = function () { }; | ||||
|  | ||||
| // replace dbg with the real deal here or in the console: | ||||
| // dbg = chromedbg | ||||
| // dbg = console.log | ||||
|  | ||||
|  | ||||
| // plugins | ||||
| var md_plug = {}; | ||||
|  | ||||
|  | ||||
| function hesc(txt) { | ||||
|     return txt.replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">"); | ||||
| } | ||||
|  | ||||
|  | ||||
| function cls(dom, name, add) { | ||||
|     var re = new RegExp('(^| )' + name + '( |$)'); | ||||
|     var lst = (dom.getAttribute('class') + '').replace(re, "$1$2").replace(/  /, ""); | ||||
|     dom.setAttribute('class', lst + (add ? ' ' + name : '')); | ||||
| } | ||||
|  | ||||
|  | ||||
| function statify(obj) { | ||||
|     return JSON.parse(JSON.stringify(obj)); | ||||
| } | ||||
|  | ||||
|  | ||||
| // dodge browser issues | ||||
| (function () { | ||||
|     var ua = navigator.userAgent; | ||||
|     if (ua.indexOf(') Gecko/') !== -1 && /Linux| Mac /.exec(ua)) { | ||||
|         // necessary on ff-68.7 at least | ||||
|         var s = mknod('style'); | ||||
|         s.innerHTML = '@page { margin: .5in .6in .8in .6in; }'; | ||||
|         console.log(s.innerHTML); | ||||
|         document.head.appendChild(s); | ||||
|     } | ||||
| })(); | ||||
|  | ||||
|  | ||||
| // add navbar | ||||
| (function () { | ||||
|     var n = document.location + ''; | ||||
|     n = n.substr(n.indexOf('//') + 2).split('?')[0].split('/'); | ||||
| @@ -22,25 +65,220 @@ var dom_md = document.getElementById('mt'); | ||||
|         if (a > 0) | ||||
|             loc.push(n[a]); | ||||
|  | ||||
|         nav.push('<a href="/' + loc.join('/') + '">' + n[a] + '</a>'); | ||||
|         var dec = hesc(uricom_dec(n[a])[0]); | ||||
|  | ||||
|         nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>'); | ||||
|     } | ||||
|     dom_nav.innerHTML = nav.join(''); | ||||
| })(); | ||||
|  | ||||
| function convert_markdown(md_text) { | ||||
|     marked.setOptions({ | ||||
|  | ||||
| // faster than replacing the entire html (chrome 1.8x, firefox 1.6x) | ||||
| function copydom(src, dst, lv) { | ||||
|     var sc = src.childNodes, | ||||
|         dc = dst.childNodes; | ||||
|  | ||||
|     if (sc.length !== dc.length) { | ||||
|         dbg("replace L%d (%d/%d) |%d|", | ||||
|             lv, sc.length, dc.length, src.innerHTML.length); | ||||
|  | ||||
|         dst.innerHTML = src.innerHTML; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     var rpl = []; | ||||
|     for (var a = sc.length - 1; a >= 0; a--) { | ||||
|         var st = sc[a].tagName, | ||||
|             dt = dc[a].tagName; | ||||
|  | ||||
|         if (st !== dt) { | ||||
|             dbg("replace L%d (%d/%d) type %s/%s", lv, a, sc.length, st, dt); | ||||
|             rpl.push(a); | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         var sa = sc[a].attributes || [], | ||||
|             da = dc[a].attributes || []; | ||||
|  | ||||
|         if (sa.length !== da.length) { | ||||
|             dbg("replace L%d (%d/%d) attr# %d/%d", | ||||
|                 lv, a, sc.length, sa.length, da.length); | ||||
|  | ||||
|             rpl.push(a); | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         var dirty = false; | ||||
|         for (var b = sa.length - 1; b >= 0; b--) { | ||||
|             var name = sa[b].name, | ||||
|                 sv = sa[b].value, | ||||
|                 dv = dc[a].getAttribute(name); | ||||
|  | ||||
|             if (name == "data-ln" && sv !== dv) { | ||||
|                 dc[a].setAttribute(name, sv); | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             if (sv !== dv) { | ||||
|                 dbg("replace L%d (%d/%d) attr %s [%s] [%s]", | ||||
|                     lv, a, sc.length, name, sv, dv); | ||||
|  | ||||
|                 dirty = true; | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         if (dirty) | ||||
|             rpl.push(a); | ||||
|     } | ||||
|  | ||||
|     // TODO pure guessing | ||||
|     if (rpl.length > sc.length / 3) { | ||||
|         dbg("replace L%d fully, %s (%d/%d) |%d|", | ||||
|             lv, rpl.length, sc.length, src.innerHTML.length); | ||||
|  | ||||
|         dst.innerHTML = src.innerHTML; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // repl is reversed; build top-down | ||||
|     var nbytes = 0; | ||||
|     for (var a = rpl.length - 1; a >= 0; a--) { | ||||
|         var html = sc[rpl[a]].outerHTML; | ||||
|         dc[rpl[a]].outerHTML = html; | ||||
|         nbytes += html.length; | ||||
|     } | ||||
|     if (nbytes > 0) | ||||
|         dbg("replaced %d bytes L%d", nbytes, lv); | ||||
|  | ||||
|     for (var a = 0; a < sc.length; a++) | ||||
|         copydom(sc[a], dc[a], lv + 1); | ||||
|  | ||||
|     if (src.innerHTML !== dst.innerHTML) { | ||||
|         dbg("setting %d bytes L%d", src.innerHTML.length, lv); | ||||
|         dst.innerHTML = src.innerHTML; | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| function md_plug_err(ex, js) { | ||||
|     var errbox = ebi('md_errbox'); | ||||
|     if (errbox) | ||||
|         errbox.parentNode.removeChild(errbox); | ||||
|  | ||||
|     if (!ex) | ||||
|         return; | ||||
|  | ||||
|     var msg = (ex + '').split('\n')[0]; | ||||
|     var ln = ex.lineNumber; | ||||
|     var o = null; | ||||
|     if (ln) { | ||||
|         msg = "Line " + ln + ", " + msg; | ||||
|         var lns = js.split('\n'); | ||||
|         if (ln < lns.length) { | ||||
|             o = mknod('span'); | ||||
|             o.style.cssText = "color:#ac2;font-size:.9em;font-family:'scp',monospace,monospace;display:block"; | ||||
|             o.textContent = lns[ln - 1]; | ||||
|         } | ||||
|     } | ||||
|     errbox = mknod('div'); | ||||
|     errbox.setAttribute('id', 'md_errbox'); | ||||
|     errbox.style.cssText = 'position:absolute;top:0;left:0;padding:1em .5em;background:#2b2b2b;color:#fc5' | ||||
|     errbox.textContent = msg; | ||||
|     errbox.onclick = function () { | ||||
|         alert('' + ex.stack); | ||||
|     }; | ||||
|     if (o) { | ||||
|         errbox.appendChild(o); | ||||
|         errbox.style.padding = '.25em .5em'; | ||||
|     } | ||||
|     dom_nav.appendChild(errbox); | ||||
|  | ||||
|     try { | ||||
|         console.trace(); | ||||
|     } | ||||
|     catch (ex2) { } | ||||
| } | ||||
|  | ||||
|  | ||||
| function load_plug(md_text, plug_type) { | ||||
|     if (!md_opt.allow_plugins) | ||||
|         return md_text; | ||||
|  | ||||
|     var find = '\n```copyparty_' + plug_type + '\n'; | ||||
|     var ofs = md_text.indexOf(find); | ||||
|     if (ofs === -1) | ||||
|         return md_text; | ||||
|  | ||||
|     var ofs2 = md_text.indexOf('\n```', ofs + 1); | ||||
|     if (ofs2 == -1) | ||||
|         return md_text; | ||||
|  | ||||
|     var js = md_text.slice(ofs + find.length, ofs2 + 1); | ||||
|     var md = md_text.slice(0, ofs + 1) + md_text.slice(ofs2 + 4); | ||||
|  | ||||
|     var old_plug = md_plug[plug_type]; | ||||
|     if (!old_plug || old_plug[1] != js) { | ||||
|         js = 'const x = { ' + js + ' }; x;'; | ||||
|         try { | ||||
|             var x = eval(js); | ||||
|         } | ||||
|         catch (ex) { | ||||
|             md_plug[plug_type] = null; | ||||
|             md_plug_err(ex, js); | ||||
|             return md; | ||||
|         } | ||||
|         if (x['ctor']) { | ||||
|             x['ctor'](); | ||||
|             delete x['ctor']; | ||||
|         } | ||||
|         md_plug[plug_type] = [x, js]; | ||||
|     } | ||||
|  | ||||
|     return md; | ||||
| } | ||||
|  | ||||
|  | ||||
| function convert_markdown(md_text, dest_dom) { | ||||
|     md_text = md_text.replace(/\r/g, ''); | ||||
|  | ||||
|     md_plug_err(null); | ||||
|     md_text = load_plug(md_text, 'pre'); | ||||
|     md_text = load_plug(md_text, 'post'); | ||||
|  | ||||
|     var marked_opts = { | ||||
|         //headerPrefix: 'h-', | ||||
|         breaks: true, | ||||
|         gfm: true | ||||
|     }); | ||||
|     var html = marked(md_text); | ||||
|     dom_doc.innerHTML = html; | ||||
|     }; | ||||
|  | ||||
|     var loader = document.getElementById('ml'); | ||||
|     loader.parentNode.removeChild(loader); | ||||
|     var ext = md_plug['pre']; | ||||
|     if (ext) | ||||
|         Object.assign(marked_opts, ext[0]); | ||||
|  | ||||
|     try { | ||||
|         var md_html = marked(md_text, marked_opts); | ||||
|     } | ||||
|     catch (ex) { | ||||
|         if (ext) | ||||
|             md_plug_err(ex, ext[1]); | ||||
|  | ||||
|         throw ex; | ||||
|     } | ||||
|     var md_dom = new DOMParser().parseFromString(md_html, "text/html").body; | ||||
|  | ||||
|     var nodes = md_dom.getElementsByTagName('a'); | ||||
|     for (var a = nodes.length - 1; a >= 0; a--) { | ||||
|         var href = nodes[a].getAttribute('href'); | ||||
|         var txt = nodes[a].textContent; | ||||
|  | ||||
|         if (!txt) | ||||
|             nodes[a].textContent = href; | ||||
|         else if (href !== txt) | ||||
|             nodes[a].setAttribute('class', 'vis'); | ||||
|     } | ||||
|  | ||||
|     // todo-lists (should probably be a marked extension) | ||||
|     var nodes = dom_doc.getElementsByTagName('input'); | ||||
|     nodes = md_dom.getElementsByTagName('input'); | ||||
|     for (var a = nodes.length - 1; a >= 0; a--) { | ||||
|         var dom_box = nodes[a]; | ||||
|         if (dom_box.getAttribute('type') !== 'checkbox') | ||||
| @@ -59,34 +297,94 @@ function convert_markdown(md_text) { | ||||
|             '<span class="todo_' + clas + '">' + char + '</span>' + | ||||
|             html.substr(html.indexOf('>') + 1); | ||||
|     } | ||||
|  | ||||
|     // separate <code> for each line in <pre> | ||||
|     nodes = md_dom.getElementsByTagName('pre'); | ||||
|     for (var a = nodes.length - 1; a >= 0; a--) { | ||||
|         var el = nodes[a]; | ||||
|  | ||||
|         var is_precode = | ||||
|             el.tagName == 'PRE' && | ||||
|             el.childNodes.length === 1 && | ||||
|             el.childNodes[0].tagName == 'CODE'; | ||||
|  | ||||
|         if (!is_precode) | ||||
|             continue; | ||||
|  | ||||
|         var nline = parseInt(el.getAttribute('data-ln')) + 1; | ||||
|         var lines = el.innerHTML.replace(/\n<\/code>$/i, '</code>').split(/\n/g); | ||||
|         for (var b = 0; b < lines.length - 1; b++) | ||||
|             lines[b] += '</code>\n<code data-ln="' + (nline + b) + '">'; | ||||
|  | ||||
|         el.innerHTML = lines.join(''); | ||||
|     } | ||||
|  | ||||
|     // self-link headers | ||||
|     var id_seen = {}, | ||||
|         dyn = md_dom.getElementsByTagName('*'); | ||||
|  | ||||
|     nodes = []; | ||||
|     for (var a = 0, aa = dyn.length; a < aa; a++) | ||||
|         if (/^[Hh]([1-6])/.exec(dyn[a].tagName) !== null) | ||||
|             nodes.push(dyn[a]); | ||||
|  | ||||
|     for (var a = 0; a < nodes.length; a++) { | ||||
|         el = nodes[a]; | ||||
|         var id = el.getAttribute('id'), | ||||
|             orig_id = id; | ||||
|  | ||||
|         if (id_seen[id]) { | ||||
|             for (var n = 1; n < 4096; n++) { | ||||
|                 id = orig_id + '-' + n; | ||||
|                 if (!id_seen[id]) | ||||
|                     break; | ||||
|             } | ||||
|             el.setAttribute('id', id); | ||||
|         } | ||||
|         id_seen[id] = 1; | ||||
|         el.innerHTML = '<a href="#' + id + '">' + el.innerHTML + '</a>'; | ||||
|     } | ||||
|  | ||||
|     ext = md_plug['post']; | ||||
|     if (ext && ext[0].render) | ||||
|         try { | ||||
|             ext[0].render(md_dom); | ||||
|         } | ||||
|         catch (ex) { | ||||
|             md_plug_err(ex, ext[1]); | ||||
|         } | ||||
|  | ||||
|     copydom(md_dom, dest_dom, 0); | ||||
|  | ||||
|     if (ext && ext[0].render2) | ||||
|         try { | ||||
|             ext[0].render2(dest_dom); | ||||
|         } | ||||
|         catch (ex) { | ||||
|             md_plug_err(ex, ext[1]); | ||||
|         } | ||||
| } | ||||
|  | ||||
|  | ||||
| function init_toc() { | ||||
|     var loader = ebi('ml'); | ||||
|     loader.parentNode.removeChild(loader); | ||||
|  | ||||
|     var anchors = [];  // list of toc entries, complex objects | ||||
|     var anchor = null; // current toc node | ||||
|     var id_seen = {};  // taken IDs | ||||
|     var html = [];     // generated toc html | ||||
|     var lv = 0;        // current indentation level in the toc html | ||||
|     var re = new RegExp('^[Hh]([1-3])'); | ||||
|     var ctr = [0, 0, 0, 0, 0, 0]; | ||||
|  | ||||
|     var manip_nodes_dyn = dom_doc.getElementsByTagName('*'); | ||||
|     var manip_nodes_dyn = dom_pre.getElementsByTagName('*'); | ||||
|     var manip_nodes = []; | ||||
|     for (var a = 0, aa = manip_nodes_dyn.length; a < aa; a++) | ||||
|         manip_nodes.push(manip_nodes_dyn[a]); | ||||
|  | ||||
|     for (var a = 0, aa = manip_nodes.length; a < aa; a++) { | ||||
|         var elm = manip_nodes[a]; | ||||
|         var m = re.exec(elm.tagName); | ||||
|  | ||||
|         var is_header = | ||||
|             m !== null; | ||||
|  | ||||
|         var is_precode = | ||||
|             !is_header && | ||||
|             elm.tagName == 'PRE' && | ||||
|             elm.childNodes.length === 1 && | ||||
|             elm.childNodes[0].tagName == 'CODE'; | ||||
|  | ||||
|         var m = /^[Hh]([1-6])/.exec(elm.tagName); | ||||
|         var is_header = m !== null; | ||||
|         if (is_header) { | ||||
|             var nlv = m[1]; | ||||
|             while (lv < nlv) { | ||||
| @@ -97,24 +395,18 @@ function init_toc() { | ||||
|                 html.push('</ul>'); | ||||
|                 lv--; | ||||
|             } | ||||
|             ctr[lv - 1]++; | ||||
|             for (var b = lv; b < 6; b++) | ||||
|                 ctr[b] = 0; | ||||
|  | ||||
|             var orig_id = elm.getAttribute('id'); | ||||
|             var id = orig_id; | ||||
|             if (id_seen[id]) { | ||||
|                 for (var n = 1; n < 4096; n++) { | ||||
|                     id = orig_id + '-' + n; | ||||
|                     if (!id_seen[id]) | ||||
|                         break; | ||||
|                 } | ||||
|                 elm.setAttribute('id', id); | ||||
|             } | ||||
|             id_seen[id] = 1; | ||||
|             elm.childNodes[0].setAttribute('ctr', ctr.slice(0, lv).join('.')); | ||||
|  | ||||
|             var ahref = '<a href="#' + id + '">' + | ||||
|                 elm.innerHTML + '</a>'; | ||||
|             var elm2 = elm.cloneNode(true); | ||||
|             elm2.childNodes[0].textContent = elm.textContent; | ||||
|             while (elm2.childNodes.length > 1) | ||||
|                 elm2.removeChild(elm2.childNodes[1]); | ||||
|  | ||||
|             html.push('<li>' + ahref + '</li>'); | ||||
|             elm.innerHTML = ahref; | ||||
|             html.push('<li>' + elm2.innerHTML + '</li>'); | ||||
|  | ||||
|             if (anchor != null) | ||||
|                 anchors.push(anchor); | ||||
| @@ -125,17 +417,6 @@ function init_toc() { | ||||
|                 y: null | ||||
|             }; | ||||
|         } | ||||
|         else if (is_precode) { | ||||
|             // not actually toc-related (sorry), | ||||
|             // split <pre><code /></pre> into one <code> per line | ||||
|             var nline = parseInt(elm.getAttribute('data-ln')) + 1; | ||||
|             var lines = elm.innerHTML.replace(/\r?\n<\/code>$/i, '</code>').split(/\r?\n/g); | ||||
|             for (var b = 0; b < lines.length - 1; b++) | ||||
|                 lines[b] += '</code>\n<code data-ln="' + (nline + b) + '">'; | ||||
|  | ||||
|             elm.innerHTML = lines.join(''); | ||||
|         } | ||||
|  | ||||
|         if (!is_header && anchor) | ||||
|             anchor.kids.push(elm); | ||||
|     } | ||||
| @@ -207,41 +488,48 @@ function init_toc() { | ||||
|  | ||||
|  | ||||
| // "main" :p | ||||
| convert_markdown(dom_md.value); | ||||
| convert_markdown(dom_src.value, dom_pre); | ||||
| var toc = init_toc(); | ||||
|  | ||||
|  | ||||
| // scroll handler | ||||
| (function () { | ||||
|     var timer_active = false; | ||||
|     var final = null; | ||||
| var redraw = (function () { | ||||
|     var sbs = false; | ||||
|     function onresize() { | ||||
|         sbs = window.matchMedia('(min-width: 64em)').matches; | ||||
|         var y = (dom_hbar.offsetTop + dom_hbar.offsetHeight) + 'px'; | ||||
|         if (sbs) { | ||||
|             dom_toc.style.top = y; | ||||
|             dom_wrap.style.top = y; | ||||
|             dom_toc.style.marginTop = '0'; | ||||
|         } | ||||
|         onscroll(); | ||||
|     } | ||||
|  | ||||
|     function onscroll() { | ||||
|         clearTimeout(final); | ||||
|         timer_active = false; | ||||
|         toc.refresh(); | ||||
|  | ||||
|         var y = 0; | ||||
|         if (window.matchMedia('(min-width: 64em)').matches) | ||||
|             y = parseInt(dom_nav.offsetHeight) - window.scrollY; | ||||
|  | ||||
|         dom_toc.style.marginTop = y < 0 ? 0 : y + "px"; | ||||
|     } | ||||
|     onscroll(); | ||||
|  | ||||
|     function ev_onscroll() { | ||||
|         // long timeout: scroll ended | ||||
|         clearTimeout(final); | ||||
|         final = setTimeout(onscroll, 100); | ||||
|     window.onresize = onresize; | ||||
|     window.onscroll = onscroll; | ||||
|     dom_wrap.onscroll = onscroll; | ||||
|  | ||||
|         // short timeout: continuous updates | ||||
|         if (timer_active) | ||||
|             return; | ||||
|  | ||||
|         timer_active = true; | ||||
|         setTimeout(onscroll, 10); | ||||
|     }; | ||||
|  | ||||
|     window.onscroll = ev_onscroll; | ||||
|     window.onresize = ev_onscroll; | ||||
|     onresize(); | ||||
|     return onresize; | ||||
| })(); | ||||
|  | ||||
|  | ||||
| dom_navtgl.onclick = function () { | ||||
|     var hidden = dom_navtgl.innerHTML == 'hide nav'; | ||||
|     dom_navtgl.innerHTML = hidden ? 'show nav' : 'hide nav'; | ||||
|     dom_nav.style.display = hidden ? 'none' : 'block'; | ||||
|  | ||||
|     swrite('hidenav', hidden ? 1 : 0); | ||||
|     redraw(); | ||||
| }; | ||||
|  | ||||
| if (sread('hidenav') == 1) | ||||
|     dom_navtgl.onclick(); | ||||
|  | ||||
| if (window['tt']) | ||||
|     tt.init(); | ||||
|   | ||||
							
								
								
									
										110
									
								
								copyparty/web/md2.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								copyparty/web/md2.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,110 @@ | ||||
| #toc { | ||||
| 	display: none; | ||||
| } | ||||
| #mtw { | ||||
| 	display: block; | ||||
| 	position: fixed; | ||||
| 	left: .5em; | ||||
| 	bottom: 0; | ||||
| 	width: calc(100% - 56em); | ||||
| } | ||||
| #mw { | ||||
| 	left: calc(100% - 55em); | ||||
| 	overflow-y: auto; | ||||
| 	position: fixed; | ||||
| 	bottom: 0; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* single-screen */ | ||||
| #mtw.preview, | ||||
| #mw.editor { | ||||
| 	opacity: 0; | ||||
| 	z-index: 1; | ||||
| } | ||||
| #mw.preview, | ||||
| #mtw.editor { | ||||
| 	z-index: 5; | ||||
| } | ||||
| #mtw.single, | ||||
| #mw.single { | ||||
| 	margin: 0; | ||||
| 	left: 1em; | ||||
| 	left: max(1em, calc((100% - 56em) / 2)); | ||||
| } | ||||
| #mtw.single { | ||||
| 	width: 55em; | ||||
| 	width: min(55em, calc(100% - 2em)); | ||||
| } | ||||
|  | ||||
|  | ||||
| #mp { | ||||
| 	position: relative; | ||||
| } | ||||
| #mt, #mtr { | ||||
| 	width: 100%; | ||||
| 	height: calc(100% - 1px); | ||||
| 	color: #444; | ||||
| 	background: #f7f7f7; | ||||
| 	border: 1px solid #999; | ||||
| 	outline: none; | ||||
| 	padding: 0; | ||||
| 	margin: 0; | ||||
| 	font-family: 'consolas', monospace, monospace; | ||||
| 	white-space: pre-wrap; | ||||
| 	word-break: break-word; | ||||
| 	overflow-wrap: break-word; | ||||
| 	word-wrap: break-word; /*ie*/ | ||||
| 	overflow-y: scroll; | ||||
| 	line-height: 1.3em; | ||||
| 	font-size: .9em; | ||||
| 	position: relative; | ||||
| 	scrollbar-color: #eb0 #f7f7f7; | ||||
| } | ||||
| html.dark #mt { | ||||
| 	color: #eee; | ||||
| 	background: #222; | ||||
| 	border: 1px solid #777; | ||||
| 	scrollbar-color: #b80 #282828; | ||||
| } | ||||
| #mtr { | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| } | ||||
| #save.force-save { | ||||
| 	color: #400; | ||||
| 	background: #f97; | ||||
| 	border-radius: .15em; | ||||
| } | ||||
| html.dark #save.force-save { | ||||
| 	color: #fca; | ||||
| 	background: #720; | ||||
| } | ||||
| #save.disabled { | ||||
| 	opacity: .4; | ||||
| } | ||||
| #helpbox { | ||||
| 	background: #f7f7f7; | ||||
| 	border-radius: .4em; | ||||
| 	z-index: 9001; | ||||
| 	display: none; | ||||
| 	position: fixed; | ||||
| 	padding: 2em; | ||||
| 	top: 4em; | ||||
| 	overflow-y: auto; | ||||
| 	box-shadow: 0 .5em 2em #777; | ||||
| 	height: calc(100% - 12em); | ||||
| 	left: calc(50% - 15em); | ||||
| 	right: 0; | ||||
| 	width: 30em; | ||||
| } | ||||
| #helpclose { | ||||
| 	display: block; | ||||
| } | ||||
| html.dark #helpbox { | ||||
| 	box-shadow: 0 .5em 2em #444; | ||||
| 	background: #222; | ||||
| 	border: 1px solid #079; | ||||
| 	border-width: 1px 0; | ||||
| } | ||||
							
								
								
									
										1101
									
								
								copyparty/web/md2.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1101
									
								
								copyparty/web/md2.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -8,69 +8,58 @@ html .editor-toolbar>i.separator { border-left: 1px solid #ccc; } | ||||
| html .editor-toolbar.disabled-for-preview>button:not(.no-disable) { opacity: .35 } | ||||
|  | ||||
| html { | ||||
|     line-height: 1.5em; | ||||
| 	line-height: 1.5em; | ||||
| } | ||||
| html, body { | ||||
|     margin: 0; | ||||
|     padding: 0; | ||||
|     min-height: 100%; | ||||
|     font-family: sans-serif; | ||||
|     background: #f7f7f7; | ||||
|     color: #333; | ||||
| 	margin: 0; | ||||
| 	padding: 0; | ||||
| 	min-height: 100%; | ||||
| 	font-family: sans-serif; | ||||
| 	background: #f7f7f7; | ||||
| 	color: #333; | ||||
| } | ||||
| #mn { | ||||
|     font-weight: normal; | ||||
|     margin: 1.3em 0 .7em 1em; | ||||
|     font-size: 1.4em; | ||||
| 	font-weight: normal; | ||||
| 	margin: 1.3em 0 .7em 1em; | ||||
| } | ||||
| #mn a { | ||||
|     color: #444; | ||||
|     margin: 0 0 0 -.2em; | ||||
|     padding: 0 0 0 .4em; | ||||
|     text-decoration: none; | ||||
|     /* ie: */ | ||||
|     border-bottom: .1em solid #777\9; | ||||
|     margin-right: 1em\9; | ||||
| 	color: #444; | ||||
| 	margin: 0 0 0 -.2em; | ||||
| 	padding: 0 0 0 .4em; | ||||
| 	text-decoration: none; | ||||
| 	/* ie: */ | ||||
| 	border-bottom: .1em solid #777\9; | ||||
| 	margin-right: 1em\9; | ||||
| } | ||||
| #mn a:first-child { | ||||
|     padding-left: .5em; | ||||
| 	padding-left: .5em; | ||||
| } | ||||
| #mn a:last-child { | ||||
|     padding-right: .5em; | ||||
| 	padding-right: .5em; | ||||
| } | ||||
| #mn a:not(:last-child):after { | ||||
|     content: ''; | ||||
|     width: 1.05em; | ||||
|     height: 1.05em; | ||||
|     margin: -.2em .3em -.2em -.4em; | ||||
|     display: inline-block; | ||||
|     border: 1px solid rgba(0,0,0,0.3); | ||||
|     border-width: .05em .05em 0 0; | ||||
|     transform: rotate(45deg); | ||||
| 	content: ''; | ||||
| 	width: 1.05em; | ||||
| 	height: 1.05em; | ||||
| 	margin: -.2em .3em -.2em -.4em; | ||||
| 	display: inline-block; | ||||
| 	border: 1px solid rgba(0,0,0,0.2); | ||||
| 	border-width: .2em .2em 0 0; | ||||
| 	transform: rotate(45deg); | ||||
| } | ||||
| #mn a:hover { | ||||
|     color: #000; | ||||
|     text-decoration: underline; | ||||
| 	color: #000; | ||||
| 	text-decoration: underline; | ||||
| } | ||||
|  | ||||
| html .editor-toolbar>button.disabled { | ||||
|     opacity: .35; | ||||
|     pointer-events: none; | ||||
| 	opacity: .35; | ||||
| 	pointer-events: none; | ||||
| } | ||||
| html .editor-toolbar>button.save.force-save { | ||||
|     background: #f97; | ||||
| 	background: #f97; | ||||
| } | ||||
|  | ||||
| /* | ||||
| *[data-ln]:before { | ||||
| 	content: attr(data-ln); | ||||
| 	font-size: .8em; | ||||
| 	margin: 0 .4em; | ||||
| 	color: #f0c; | ||||
| } | ||||
| .cm-header { font-size: .4em !important } | ||||
| */ | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| @@ -102,29 +91,29 @@ html .editor-toolbar>button.save.force-save { | ||||
| 	line-height: 1.1em; | ||||
| } | ||||
| .mdo a { | ||||
|     color: #fff; | ||||
|     background: #39b; | ||||
|     text-decoration: none; | ||||
|     padding: 0 .3em; | ||||
|     border: none; | ||||
|     border-bottom: .07em solid #079; | ||||
| 	color: #fff; | ||||
| 	background: #39b; | ||||
| 	text-decoration: none; | ||||
| 	padding: 0 .3em; | ||||
| 	border: none; | ||||
| 	border-bottom: .07em solid #079; | ||||
| } | ||||
| .mdo h2 { | ||||
|     color: #fff; | ||||
|     background: #555; | ||||
|     margin-top: 2em; | ||||
|     border-bottom: .22em solid #999; | ||||
|     border-top: none; | ||||
| 	color: #fff; | ||||
| 	background: #555; | ||||
| 	margin-top: 2em; | ||||
| 	border-bottom: .22em solid #999; | ||||
| 	border-top: none; | ||||
| } | ||||
| .mdo h1 { | ||||
|     color: #fff; | ||||
|     background: #444; | ||||
|     font-weight: normal; | ||||
|     border-top: .4em solid #fb0; | ||||
|     border-bottom: .4em solid #777; | ||||
|     border-radius: 0 1em 0 1em; | ||||
|     margin: 3em 0 1em 0; | ||||
|     padding: .5em 0; | ||||
| 	color: #fff; | ||||
| 	background: #444; | ||||
| 	font-weight: normal; | ||||
| 	border-top: .4em solid #fb0; | ||||
| 	border-bottom: .4em solid #777; | ||||
| 	border-radius: 0 1em 0 1em; | ||||
| 	margin: 3em 0 1em 0; | ||||
| 	padding: .5em 0; | ||||
| } | ||||
| h1, h2 { | ||||
| 	line-height: 1.5em; | ||||
| @@ -161,8 +150,12 @@ h2 { | ||||
| .mdo ol>li { | ||||
| 	margin: .7em 0; | ||||
| } | ||||
| strong { | ||||
| 	color: #000; | ||||
| } | ||||
| p>em, | ||||
| li>em { | ||||
| li>em, | ||||
| td>em { | ||||
| 	color: #c50; | ||||
| 	padding: .1em; | ||||
| 	border-bottom: .1em solid #bbb; | ||||
| @@ -194,14 +187,14 @@ th { | ||||
|  | ||||
| /* mde support */ | ||||
| .mdo { | ||||
|     padding: 1em; | ||||
|     background: #f7f7f7; | ||||
| 	padding: 1em; | ||||
| 	background: #f7f7f7; | ||||
| } | ||||
| html.dark .mdo { | ||||
|     background: #1c1c1c; | ||||
| 	background: #1c1c1c; | ||||
| } | ||||
| .CodeMirror { | ||||
|     background: #f7f7f7; | ||||
| 	background: #f7f7f7; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -211,104 +204,108 @@ html.dark .mdo { | ||||
| /* darkmode */ | ||||
| html.dark .mdo, | ||||
| html.dark .CodeMirror { | ||||
|     border-color: #222; | ||||
| 	border-color: #222; | ||||
| } | ||||
| html.dark, | ||||
| html.dark body, | ||||
| html.dark .CodeMirror { | ||||
|     background: #222; | ||||
|     color: #ccc; | ||||
| 	background: #222; | ||||
| 	color: #ccc; | ||||
| } | ||||
| html.dark .CodeMirror-cursor { | ||||
|     border-color: #fff; | ||||
| 	border-color: #fff; | ||||
| } | ||||
| html.dark .CodeMirror-selected { | ||||
|     box-shadow: 0 0 1px #0cf inset; | ||||
| 	box-shadow: 0 0 1px #0cf inset; | ||||
| } | ||||
| html.dark .CodeMirror-selected, | ||||
| html.dark .CodeMirror-selectedtext { | ||||
|     border-radius: .1em; | ||||
|     background: #246; | ||||
|     color: #fff; | ||||
| 	border-radius: .1em; | ||||
| 	background: #246; | ||||
| 	color: #fff; | ||||
| } | ||||
| html.dark .mdo a { | ||||
|     background: #057; | ||||
| 	background: #057; | ||||
| } | ||||
| html.dark .mdo h1 a, html.dark .mdo h4 a, | ||||
| html.dark .mdo h2 a, html.dark .mdo h5 a, | ||||
| html.dark .mdo h3 a, html.dark .mdo h6 a { | ||||
|     color: inherit; | ||||
|     background: none; | ||||
| 	color: inherit; | ||||
| 	background: none; | ||||
| } | ||||
| html.dark pre, | ||||
| html.dark code { | ||||
|     color: #8c0; | ||||
|     background: #1a1a1a; | ||||
|     border: .07em solid #333; | ||||
| 	color: #8c0; | ||||
| 	background: #1a1a1a; | ||||
| 	border: .07em solid #333; | ||||
| } | ||||
| html.dark .mdo ul, | ||||
| html.dark .mdo ol { | ||||
|     border-color: #444; | ||||
| 	border-color: #444; | ||||
| } | ||||
| html.dark .mdo>ul, | ||||
| html.dark .mdo>ol { | ||||
|     border-color: #555; | ||||
| 	border-color: #555; | ||||
| } | ||||
| html.dark strong { | ||||
| 	color: #fff; | ||||
| } | ||||
| html.dark p>em, | ||||
| html.dark li>em { | ||||
|     color: #f94; | ||||
|     border-color: #666; | ||||
| html.dark li>em, | ||||
| html.dark td>em { | ||||
| 	color: #f94; | ||||
| 	border-color: #666; | ||||
| } | ||||
| html.dark h1 { | ||||
|     background: #383838; | ||||
|     border-top: .4em solid #b80; | ||||
|     border-bottom: .4em solid #4c4c4c; | ||||
| 	background: #383838; | ||||
| 	border-top: .4em solid #b80; | ||||
| 	border-bottom: .4em solid #4c4c4c; | ||||
| } | ||||
| html.dark h2 { | ||||
|     background: #444; | ||||
|     border-bottom: .22em solid #555; | ||||
| 	background: #444; | ||||
| 	border-bottom: .22em solid #555; | ||||
| } | ||||
| html.dark td, | ||||
| html.dark th { | ||||
|     border-color: #444; | ||||
| 	border-color: #444; | ||||
| } | ||||
| html.dark blockquote { | ||||
|     background: #282828; | ||||
|     border: .07em dashed #444; | ||||
| 	background: #282828; | ||||
| 	border: .07em dashed #444; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| html.dark #mn a { | ||||
|     color: #ccc; | ||||
| 	color: #ccc; | ||||
| } | ||||
| html.dark #mn a:not(:last-child):after { | ||||
|     border-color: rgba(255,255,255,0.3); | ||||
| 	border-color: rgba(255,255,255,0.3); | ||||
| } | ||||
| html.dark .editor-toolbar { | ||||
|     border-color: #2c2c2c; | ||||
|     background: #1c1c1c; | ||||
| 	border-color: #2c2c2c; | ||||
| 	background: #1c1c1c; | ||||
| } | ||||
| html.dark .editor-toolbar>i.separator { | ||||
|     border-left: 1px solid #444; | ||||
|     border-right: 1px solid #111; | ||||
| 	border-left: 1px solid #444; | ||||
| 	border-right: 1px solid #111; | ||||
| } | ||||
| html.dark .editor-toolbar>button { | ||||
|     margin-left: -1px; border: 1px solid rgba(255,255,255,0.1); | ||||
|     color: #aaa; | ||||
| 	margin-left: -1px; border: 1px solid rgba(255,255,255,0.1); | ||||
| 	color: #aaa; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| html.dark .editor-toolbar>button:hover { | ||||
|     color: #333; | ||||
| 	color: #333; | ||||
| } | ||||
| html.dark .editor-toolbar>button.active { | ||||
|     color: #333; | ||||
|     border-color: #ec1; | ||||
|     background: #c90; | ||||
| 	color: #333; | ||||
| 	border-color: #ec1; | ||||
| 	background: #c90; | ||||
| } | ||||
| html.dark .editor-toolbar::after, | ||||
| html.dark .editor-toolbar::before { | ||||
|     background: none; | ||||
| 	background: none; | ||||
| } | ||||
| @@ -3,9 +3,9 @@ | ||||
| 	<title>📝🎉 {{ title }}</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=0.7"> | ||||
| 	<link href="/.cpr/mde.css" rel="stylesheet"> | ||||
| 	<link href="/.cpr/deps/mini-fa.css" rel="stylesheet"> | ||||
| 	<link href="/.cpr/deps/easymde.css" rel="stylesheet"> | ||||
| 	<link href="/.cpr/mde.css?_={{ ts }}" rel="stylesheet"> | ||||
| 	<link href="/.cpr/deps/mini-fa.css?_={{ ts }}" rel="stylesheet"> | ||||
| 	<link href="/.cpr/deps/easymde.css?_={{ ts }}" rel="stylesheet"> | ||||
| </head> | ||||
| <body> | ||||
| 	<div id="mw"> | ||||
| @@ -17,28 +17,32 @@ | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div id="m"> | ||||
| 			<textarea id="mt" style="display:none">{{ md }}</textarea> | ||||
| 			<textarea id="mt" style="display:none" autocomplete="off">{{ md }}</textarea> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<script> | ||||
|  | ||||
| var link_md_as_html = false;  // TODO (does nothing) | ||||
| var last_modified = {{ lastmod }}; | ||||
| var md_opt = { | ||||
| 	link_md_as_html: false, | ||||
| 	allow_plugins: {{ md_plug }}, | ||||
| 	modpoll_freq: {{ md_chk_rate }} | ||||
| }; | ||||
|  | ||||
| var lightswitch = (function () { | ||||
| 	var fun = function () { | ||||
| 		var dark = !!!document.documentElement.getAttribute("class"); | ||||
| 		document.documentElement.setAttribute("class", dark ? "dark" : ""); | ||||
| 		if (window.localStorage) | ||||
| 			localStorage.setItem('darkmode', dark ? 1 : 0); | ||||
| 	}; | ||||
| 	if (window.localStorage && localStorage.getItem('darkmode') == 1) | ||||
| 		fun(); | ||||
| 	 | ||||
| 	return fun; | ||||
| 	var l = localStorage, | ||||
| 		drk = l.getItem('lightmode') != 1, | ||||
| 		f = function (e) { | ||||
| if (e) drk = !drk; | ||||
| document.documentElement.setAttribute("class", drk? "dark":"light"); | ||||
| l.setItem('lightmode', drk? 0:1); | ||||
| 		}; | ||||
| 	f(); | ||||
| 	return f; | ||||
| })(); | ||||
|  | ||||
| 	</script> | ||||
| 	<script src="/.cpr/deps/easymde.full.js"></script> | ||||
| 	<script src="/.cpr/mde.js"></script> | ||||
|     <script src="/.cpr/util.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/deps/easymde.js?_={{ ts }}"></script> | ||||
| 	<script src="/.cpr/mde.js?_={{ ts }}"></script> | ||||
| </body></html> | ||||
|   | ||||
| @@ -1,7 +1,9 @@ | ||||
| var dom_wrap = document.getElementById('mw'); | ||||
| var dom_nav = document.getElementById('mn'); | ||||
| var dom_doc = document.getElementById('m'); | ||||
| var dom_md = document.getElementById('mt'); | ||||
| "use strict"; | ||||
|  | ||||
| var dom_wrap = ebi('mw'); | ||||
| var dom_nav = ebi('mn'); | ||||
| var dom_doc = ebi('m'); | ||||
| var dom_md = ebi('mt'); | ||||
|  | ||||
| (function () { | ||||
|     var n = document.location + ''; | ||||
| @@ -13,7 +15,9 @@ var dom_md = document.getElementById('mt'); | ||||
|         if (a > 0) | ||||
|             loc.push(n[a]); | ||||
|  | ||||
|         nav.push('<a href="/' + loc.join('/') + '">' + n[a] + '</a>'); | ||||
|         var dec = uricom_dec(n[a])[0].replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">"); | ||||
|  | ||||
|         nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>'); | ||||
|     } | ||||
|     dom_nav.innerHTML = nav.join(''); | ||||
| })(); | ||||
| @@ -51,7 +55,8 @@ var mde = (function () { | ||||
|             "save": "Ctrl-S" | ||||
|         }, | ||||
|         insertTexts: ["[](", ")"], | ||||
|         tabSize: 4, | ||||
|         indentWithTabs: false, | ||||
|         tabSize: 2, | ||||
|         toolbar: tbar, | ||||
|         previewClass: 'mdo', | ||||
|         onToggleFullScreen: set_jumpto, | ||||
| @@ -60,17 +65,17 @@ var mde = (function () { | ||||
|     mde.codemirror.on("change", function () { | ||||
|         md_changed(mde); | ||||
|     }); | ||||
|     var loader = document.getElementById('ml'); | ||||
|     var loader = ebi('ml'); | ||||
|     loader.parentNode.removeChild(loader); | ||||
|     return mde; | ||||
| })(); | ||||
|  | ||||
| function set_jumpto() { | ||||
|     document.querySelector('.editor-preview-side').onclick = jumpto; | ||||
|     QS('.editor-preview-side').onclick = jumpto; | ||||
| } | ||||
|  | ||||
| function jumpto(ev) { | ||||
|     var tgt = ev.target || ev.srcElement; | ||||
|     var tgt = ev.target; | ||||
|     var ln = null; | ||||
|     while (tgt && !ln) { | ||||
|         ln = tgt.getAttribute('data-ln'); | ||||
| @@ -89,7 +94,7 @@ function md_changed(mde, on_srv) { | ||||
|         window.md_saved = mde.value(); | ||||
|  | ||||
|     var md_now = mde.value(); | ||||
|     var save_btn = document.querySelector('.editor-toolbar button.save'); | ||||
|     var save_btn = QS('.editor-toolbar button.save'); | ||||
|  | ||||
|     if (md_now == window.md_saved) | ||||
|         save_btn.classList.add('disabled'); | ||||
| @@ -100,16 +105,13 @@ function md_changed(mde, on_srv) { | ||||
| } | ||||
|  | ||||
| function save(mde) { | ||||
|     var save_btn = document.querySelector('.editor-toolbar button.save'); | ||||
|     if (save_btn.classList.contains('disabled')) { | ||||
|         alert('there is nothing to save'); | ||||
|         return; | ||||
|     } | ||||
|     var save_btn = QS('.editor-toolbar button.save'); | ||||
|     if (save_btn.classList.contains('disabled')) | ||||
|         return toast.inf(2, 'no changes'); | ||||
|  | ||||
|     var force = save_btn.classList.contains('force-save'); | ||||
|     if (force && !confirm('confirm that you wish to lose the changes made on the server since you opened this document')) { | ||||
|         alert('ok, aborted'); | ||||
|         return; | ||||
|     } | ||||
|     if (force && !confirm('confirm that you wish to lose the changes made on the server since you opened this document')) | ||||
|         return toast.inf(3, 'aborted'); | ||||
|  | ||||
|     var txt = mde.value(); | ||||
|  | ||||
| @@ -118,7 +120,7 @@ function save(mde) { | ||||
|     fd.append("lastmod", (force ? -1 : last_modified)); | ||||
|     fd.append("body", txt); | ||||
|  | ||||
|     var url = (document.location + '').split('?')[0] + '?raw'; | ||||
|     var url = (document.location + '').split('?')[0]; | ||||
|     var xhr = new XMLHttpRequest(); | ||||
|     xhr.open('POST', url, true); | ||||
|     xhr.responseType = 'text'; | ||||
| @@ -133,18 +135,15 @@ function save_cb() { | ||||
|     if (this.readyState != XMLHttpRequest.DONE) | ||||
|         return; | ||||
|  | ||||
|     if (this.status !== 200) { | ||||
|         alert('Error!  The file was NOT saved.\n\n' + this.status + ": " + (this.responseText + '').replace(/^<pre>/, "")); | ||||
|         return; | ||||
|     } | ||||
|     if (this.status !== 200) | ||||
|         return alert('Error!  The file was NOT saved.\n\n' + this.status + ": " + (this.responseText + '').replace(/^<pre>/, "")); | ||||
|  | ||||
|     var r; | ||||
|     try { | ||||
|         r = JSON.parse(this.responseText); | ||||
|     } | ||||
|     catch (ex) { | ||||
|         alert('Failed to parse reply from server:\n\n' + this.responseText); | ||||
|         return; | ||||
|         return alert('Failed to parse reply from server:\n\n' + this.responseText); | ||||
|     } | ||||
|  | ||||
|     if (!r.ok) { | ||||
| @@ -207,10 +206,10 @@ function save_chk() { | ||||
|     last_modified = this.lastmod; | ||||
|     md_changed(this.mde, true); | ||||
|  | ||||
|     var ok = document.createElement('div'); | ||||
|     var ok = mknod('div'); | ||||
|     ok.setAttribute('style', 'font-size:6em;font-family:serif;font-weight:bold;color:#cf6;background:#444;border-radius:.3em;padding:.6em 0;position:fixed;top:30%;left:calc(50% - 2em);width:4em;text-align:center;z-index:9001;transition:opacity 0.2s ease-in-out;opacity:1'); | ||||
|     ok.innerHTML = 'OK✔️'; | ||||
|     var parent = document.getElementById('m'); | ||||
|     var parent = ebi('m'); | ||||
|     document.documentElement.appendChild(ok); | ||||
|     setTimeout(function () { | ||||
|         ok.style.opacity = 0; | ||||
|   | ||||
| @@ -3,7 +3,7 @@ html,body,tr,th,td,#files,a { | ||||
| 	background: none; | ||||
| 	font-weight: inherit; | ||||
| 	font-size: inherit; | ||||
| 	padding: none; | ||||
| 	padding: 0; | ||||
| 	border: none; | ||||
| } | ||||
| html { | ||||
| @@ -20,8 +20,8 @@ body { | ||||
| 	padding-bottom: 5em; | ||||
| } | ||||
| #box { | ||||
|     padding: .5em 1em; | ||||
|     background: #2c2c2c; | ||||
| 	padding: .5em 1em; | ||||
| 	background: #2c2c2c; | ||||
| } | ||||
| pre { | ||||
| 	font-family: monospace, monospace; | ||||
|   | ||||
| @@ -6,7 +6,7 @@ | ||||
|     <title>copyparty</title> | ||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/msg.css"> | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/msg.css?_={{ ts }}"> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
|   | ||||
| @@ -13,6 +13,7 @@ h1 { | ||||
| 	border-bottom: 1px solid #ccc; | ||||
| 	margin: 2em 0 .4em 0; | ||||
| 	padding: 0 0 .2em 0; | ||||
| 	font-weight: normal; | ||||
| } | ||||
| li { | ||||
| 	margin: 1em 0; | ||||
| @@ -24,4 +25,52 @@ a { | ||||
| 	border-bottom: 1px solid #aaa; | ||||
| 	border-radius: .2em; | ||||
| 	padding: .2em .8em; | ||||
| } | ||||
| table { | ||||
| 	border-collapse: collapse; | ||||
| } | ||||
| .vols td, | ||||
| .vols th { | ||||
| 	padding: .3em .6em; | ||||
| 	text-align: left; | ||||
| } | ||||
| .num { | ||||
| 	border-right: 1px solid #bbb; | ||||
| } | ||||
| .num td { | ||||
| 	padding: .1em .7em .1em 0; | ||||
| } | ||||
| .num td:first-child { | ||||
| 	text-align: right; | ||||
| } | ||||
| .btns { | ||||
| 	margin: 1em 0; | ||||
| } | ||||
|  | ||||
|  | ||||
| html.dark, | ||||
| html.dark body, | ||||
| html.dark #wrap { | ||||
| 	background: #222; | ||||
| 	color: #ccc; | ||||
| } | ||||
| html.dark h1 { | ||||
| 	border-color: #777; | ||||
| } | ||||
| html.dark a { | ||||
| 	color: #fff; | ||||
| 	background: #057; | ||||
| 	border-color: #37a; | ||||
| } | ||||
| html.dark input { | ||||
| 	color: #fff; | ||||
| 	background: #624; | ||||
| 	border: 1px solid #c27; | ||||
| 	border-width: 1px 0 0 0; | ||||
| 	border-radius: .5em; | ||||
| 	padding: .5em .7em; | ||||
| 	margin: 0 .5em 0 0; | ||||
| } | ||||
| html.dark .num { | ||||
| 	border-color: #777; | ||||
| } | ||||
| @@ -6,26 +6,56 @@ | ||||
|     <title>copyparty</title> | ||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/splash.css"> | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/splash.css?_={{ ts }}"> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
|     <div id="wrap"> | ||||
|         <p>hello {{ this.uname }}</p> | ||||
|  | ||||
|         {%- if avol %} | ||||
|         <h1>admin panel:</h1> | ||||
|         <table><tr><td> <!-- hehehe --> | ||||
|             <table class="num"> | ||||
|                 <tr><td>scanning</td><td>{{ scanning }}</td></tr> | ||||
|                 <tr><td>hash-q</td><td>{{ hashq }}</td></tr> | ||||
|                 <tr><td>tag-q</td><td>{{ tagq }}</td></tr> | ||||
|                 <tr><td>mtp-q</td><td>{{ mtpq }}</td></tr> | ||||
|             </table> | ||||
|         </td><td> | ||||
|             <table class="vols"> | ||||
|                 <thead><tr><th>vol</th><th>action</th><th>status</th></tr></thead> | ||||
|                 <tbody> | ||||
|                     {% for mp in avol %} | ||||
|                     {%- if mp in vstate and vstate[mp] %} | ||||
|                     <tr><td><a href="{{ mp }}{{ url_suf }}">{{ mp }}</a></td><td><a href="{{ mp }}?scan">rescan</a></td><td>{{ vstate[mp] }}</td></tr> | ||||
|                     {%- endif %} | ||||
|                     {% endfor %} | ||||
|                 </tbody> | ||||
|             </table> | ||||
|         </td></tr></table> | ||||
|         <div class="btns"> | ||||
|             <a href="/?stack">dump stack</a> | ||||
|         </div> | ||||
|         {%- endif %} | ||||
|  | ||||
|         {%- if rvol %} | ||||
|         <h1>you can browse these:</h1> | ||||
|         <ul> | ||||
|             {% for mp in rvol %} | ||||
|             <li><a href="/{{ mp }}">/{{ mp }}</a></li> | ||||
|             <li><a href="{{ mp }}{{ url_suf }}">{{ mp }}</a></li> | ||||
|             {% endfor %} | ||||
|         </ul> | ||||
|         {%- endif %} | ||||
|  | ||||
|         {%- if wvol %} | ||||
|         <h1>you can upload to:</h1> | ||||
|         <ul> | ||||
|             {% for mp in wvol %} | ||||
|             <li><a href="/{{ mp }}">/{{ mp }}</a></li> | ||||
|             <li><a href="{{ mp }}{{ url_suf }}">{{ mp }}</a></li> | ||||
|             {% endfor %} | ||||
|         </ul> | ||||
|         {%- endif %} | ||||
|  | ||||
|         <h1>login for more:</h1> | ||||
|         <ul> | ||||
| @@ -36,7 +66,11 @@ | ||||
|             </form> | ||||
|         </ul> | ||||
|     </div> | ||||
|     <!-- script src="/.cpr/splash.js"></script --> | ||||
| </body> | ||||
|     <script> | ||||
|  | ||||
| if (localStorage.getItem('lightmode') != 1) | ||||
|     document.documentElement.setAttribute("class", "dark"); | ||||
|  | ||||
| </script> | ||||
| </body> | ||||
| </html> | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,92 +1,4 @@ | ||||
| .opview { | ||||
| 	display: none; | ||||
| } | ||||
| .opview.act { | ||||
| 	display: block; | ||||
| } | ||||
| #ops a { | ||||
| 	color: #fc5; | ||||
| 	font-size: 1.5em; | ||||
| 	padding: 0 .3em; | ||||
| 	margin: 0; | ||||
| 	outline: none; | ||||
| } | ||||
| #ops a.act { | ||||
| 	text-decoration: underline; | ||||
| } | ||||
| /* | ||||
| #ops a+a:after, | ||||
| #ops a:first-child:after { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #01a7e1; | ||||
| 	margin-left: .3em; | ||||
| 	position: relative; | ||||
| } | ||||
| #ops a+a:before { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #ff3f1a; | ||||
| 	margin-right: .3em; | ||||
| 	margin-left: -.3em; | ||||
| } | ||||
| #ops a:last-child:after { | ||||
| 	content: ''; | ||||
| } | ||||
| #ops a.act:before, | ||||
| #ops a.act:after { | ||||
| 	text-decoration: none !important; | ||||
| } | ||||
| */ | ||||
| #ops i { | ||||
| 	font-size: 1.5em; | ||||
| } | ||||
| #ops i:before { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #01a7e1; | ||||
| 	position: relative; | ||||
| } | ||||
| #ops i:after { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #ff3f1a; | ||||
| 	margin-left: -.35em; | ||||
| 	font-size: 1.05em; | ||||
| } | ||||
| #ops, | ||||
| .opbox { | ||||
| 	border: 1px solid #3a3a3a; | ||||
| 	box-shadow: 0 0 1em #222 inset; | ||||
| } | ||||
| #ops { | ||||
| 	display: none; | ||||
| 	background: #333; | ||||
| 	margin: 1.7em 1.5em 0 1.5em; | ||||
| 	padding: .3em .6em; | ||||
| 	border-radius: .3em; | ||||
| 	border-width: .15em 0; | ||||
| } | ||||
| .opbox { | ||||
| 	background: #2d2d2d; | ||||
| 	margin: 1.5em 0 0 0; | ||||
| 	padding: .5em; | ||||
| 	border-radius: 0 1em 1em 0; | ||||
| 	border-width: .15em .3em .3em 0; | ||||
| 	max-width: 40em; | ||||
| } | ||||
| .opbox input { | ||||
| 	margin: .5em; | ||||
| } | ||||
| .opbox input[type=text] { | ||||
| 	color: #fff; | ||||
| 	background: #383838; | ||||
| 	border: none; | ||||
| 	box-shadow: 0 0 .3em #222; | ||||
| 	border-bottom: 1px solid #fc5; | ||||
| 	border-radius: .2em; | ||||
| 	padding: .2em .3em; | ||||
| } | ||||
|  | ||||
| #op_up2k { | ||||
| 	padding: 0 1em 1em 1em; | ||||
| } | ||||
| @@ -94,6 +6,9 @@ | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| 	width: 2px; | ||||
| 	height: 2px; | ||||
| 	overflow: hidden; | ||||
| } | ||||
| #u2form input { | ||||
| 	background: #444; | ||||
| @@ -104,10 +19,10 @@ | ||||
| 	color: #f87; | ||||
| 	padding: .5em; | ||||
| } | ||||
| #u2form { | ||||
| 	width: 2px; | ||||
| 	height: 2px; | ||||
| 	overflow: hidden; | ||||
| #u2err.msg { | ||||
| 	color: #999; | ||||
| 	padding: .5em; | ||||
| 	font-size: .9em; | ||||
| } | ||||
| #u2btn { | ||||
| 	color: #eee; | ||||
| @@ -117,17 +32,32 @@ | ||||
| 	background: linear-gradient(to bottom, #367 0%, #489 50%, #38788a 51%, #367 100%); | ||||
| 	filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#489', endColorstr='#38788a', GradientType=0); | ||||
| 	text-decoration: none; | ||||
| 	line-height: 1.5em; | ||||
| 	line-height: 1.3em; | ||||
| 	border: 1px solid #222; | ||||
| 	border-radius: .4em; | ||||
| 	text-align: center; | ||||
| 	font-size: 2em; | ||||
| 	margin: 1em auto; | ||||
| 	padding: 1em 0; | ||||
| 	width: 12em; | ||||
| 	font-size: 1.5em; | ||||
| 	margin: .5em auto; | ||||
| 	padding: .8em 0; | ||||
| 	width: 16em; | ||||
| 	cursor: pointer; | ||||
| 	box-shadow: .4em .4em 0 #111; | ||||
| } | ||||
| #op_up2k.srch #u2btn { | ||||
| 	background: linear-gradient(to bottom, #ca3 0%, #fd8 50%, #fc6 51%, #b92 100%); | ||||
| 	text-shadow: 1px 1px 1px #fc6; | ||||
| 	color: #333; | ||||
| } | ||||
| #u2conf #u2btn { | ||||
| 	margin: -1.5em 0; | ||||
| 	padding: .8em 0; | ||||
| 	width: 100%; | ||||
| 	max-width: 12em; | ||||
| 	display: inline-block; | ||||
| } | ||||
| #u2conf #u2btn_cw { | ||||
| 	text-align: right; | ||||
| } | ||||
| #u2notbtn { | ||||
| 	display: none; | ||||
| 	text-align: center; | ||||
| @@ -142,6 +72,9 @@ | ||||
| 	width: calc(100% - 2em); | ||||
| 	max-width: 100em; | ||||
| } | ||||
| #op_up2k.srch #u2tab { | ||||
| 	max-width: none; | ||||
| } | ||||
| #u2tab td { | ||||
| 	border: 1px solid #ccc; | ||||
| 	border-width: 0 0px 1px 0; | ||||
| @@ -149,16 +82,62 @@ | ||||
| } | ||||
| #u2tab td:nth-child(2) { | ||||
| 	width: 5em; | ||||
| 	white-space: nowrap; | ||||
| } | ||||
| #u2tab td:nth-child(3) { | ||||
| 	width: 40%; | ||||
| } | ||||
| #u2tab tr+tr:hover td { | ||||
| #op_up2k.srch td.prog { | ||||
| 	font-family: sans-serif; | ||||
| 	font-size: 1em; | ||||
| 	width: auto; | ||||
| } | ||||
| #u2tab tbody tr:hover td { | ||||
| 	background: #222; | ||||
| } | ||||
| #u2cards { | ||||
| 	padding: 1em 0 .3em 1em; | ||||
| 	margin: 1.5em auto -2.5em auto; | ||||
| 	white-space: nowrap; | ||||
| 	text-align: center; | ||||
| 	overflow: hidden; | ||||
| } | ||||
| #u2cards.w { | ||||
| 	width: 45em; | ||||
| 	text-align: left; | ||||
| } | ||||
| #u2cards a { | ||||
| 	padding: .2em 1em; | ||||
| 	border: 1px solid #777; | ||||
| 	border-width: 0 0 1px 0; | ||||
| 	background: linear-gradient(to bottom, #333, #222); | ||||
| } | ||||
| #u2cards a:first-child { | ||||
| 	border-radius: .4em 0 0 0; | ||||
| } | ||||
| #u2cards a:last-child { | ||||
| 	border-radius: 0 .4em 0 0; | ||||
| } | ||||
| #u2cards a.act { | ||||
| 	padding-bottom: .5em; | ||||
| 	border-width: 1px 1px .1em 1px; | ||||
| 	border-radius: .3em .3em 0 0; | ||||
| 	margin-left: -1px; | ||||
| 	background: linear-gradient(to bottom, #464, #333 80%); | ||||
| 	box-shadow: 0 -.17em .67em #280; | ||||
| 	border-color: #7c5 #583 #333 #583; | ||||
| 	position: relative; | ||||
| 	color: #fd7; | ||||
| } | ||||
| #u2cards span { | ||||
| 	color: #fff; | ||||
| } | ||||
| #u2conf { | ||||
| 	margin: 1em auto; | ||||
| 	width: 26em; | ||||
| 	width: 30em; | ||||
| } | ||||
| #u2conf.has_btn { | ||||
| 	width: 48em; | ||||
| } | ||||
| #u2conf * { | ||||
| 	text-align: center; | ||||
| @@ -169,12 +148,16 @@ | ||||
| 	outline: none; | ||||
| } | ||||
| #u2conf .txtbox { | ||||
| 	width: 4em; | ||||
| 	width: 3em; | ||||
| 	color: #fff; | ||||
| 	background: #444; | ||||
| 	border: 1px solid #777; | ||||
| 	font-size: 1.2em; | ||||
| 	padding: .15em 0; | ||||
| 	height: 1.05em; | ||||
| } | ||||
| #u2conf .txtbox.err { | ||||
| 	background: #922; | ||||
| } | ||||
| #u2conf a { | ||||
| 	color: #fff; | ||||
| @@ -183,39 +166,137 @@ | ||||
| 	border-radius: .1em; | ||||
| 	font-size: 1.5em; | ||||
| 	padding: .1em 0; | ||||
| 	margin: 0 -.25em; | ||||
| 	margin: 0 -1px; | ||||
| 	width: 1.5em; | ||||
| 	height: 1em; | ||||
| 	display: inline-block; | ||||
| 	position: relative; | ||||
| 	line-height: 1em; | ||||
| 	bottom: -.08em; | ||||
| 	bottom: -0.08em; | ||||
| } | ||||
| #u2conf input+a { | ||||
| 	background: #d80; | ||||
| } | ||||
| #u2conf label { | ||||
| 	font-size: 1.6em; | ||||
| 	width: 2em; | ||||
| 	height: 1em; | ||||
| 	padding: .4em 0; | ||||
| 	display: block; | ||||
| 	border-radius: .25em; | ||||
| } | ||||
| #u2conf input[type="checkbox"] { | ||||
| 	position: relative; | ||||
| 	opacity: .02; | ||||
| 	top: 2em; | ||||
| } | ||||
| #u2conf input[type="checkbox"]+label { | ||||
| 	position: relative; | ||||
| 	background: #603; | ||||
| 	border-bottom: .2em solid #a16; | ||||
| 	box-shadow: 0 .1em .3em #a00 inset; | ||||
| } | ||||
| #u2conf input[type="checkbox"]:checked+label { | ||||
| 	background: #6a1; | ||||
| 	border-bottom: .2em solid #efa; | ||||
| 	box-shadow: 0 .1em .5em #0c0; | ||||
| } | ||||
| #u2conf input[type="checkbox"]+label:hover { | ||||
| 	box-shadow: 0 .1em .3em #fb0; | ||||
| 	border-color: #fb0; | ||||
| } | ||||
| #op_up2k.srch #u2conf td:nth-child(1)>*, | ||||
| #op_up2k.srch #u2conf td:nth-child(2)>*, | ||||
| #op_up2k.srch #u2conf td:nth-child(3)>* { | ||||
| 	background: #777; | ||||
| 	border-color: #ccc; | ||||
| 	box-shadow: none; | ||||
| 	opacity: .2; | ||||
| } | ||||
| #u2foot { | ||||
| 	color: #fff; | ||||
| 	font-style: italic; | ||||
| } | ||||
| #u2foot .warn { | ||||
| 	font-size: 1.3em; | ||||
| 	padding: .5em .8em; | ||||
| 	margin: 1em -.6em; | ||||
| 	color: #f74; | ||||
| 	background: #322; | ||||
| 	border: 1px solid #633; | ||||
| 	border-width: .1em 0; | ||||
| 	text-align: center; | ||||
| } | ||||
| #u2foot .warn span { | ||||
| 	color: #f86; | ||||
| } | ||||
| html.light #u2foot .warn { | ||||
| 	color: #b00; | ||||
| 	background: #fca; | ||||
| 	border-color: #f70; | ||||
| } | ||||
| html.light #u2foot .warn span { | ||||
| 	color: #930; | ||||
| } | ||||
| #u2foot span { | ||||
| 	color: #999; | ||||
| 	font-size: .9em; | ||||
| 	font-weight: normal; | ||||
| } | ||||
| #u2footfoot { | ||||
| 	margin-bottom: -1em; | ||||
| } | ||||
| .prog { | ||||
| 	font-family: monospace; | ||||
| 	font-family: monospace, monospace; | ||||
| } | ||||
| .prog>div { | ||||
| 	display: inline-block; | ||||
| 	position: relative; | ||||
| 	overflow: hidden; | ||||
| 	margin: 0; | ||||
| 	padding: 0; | ||||
| 	height: 1.1em; | ||||
| 	margin-bottom: -.15em; | ||||
| 	box-shadow: -1px -1px 0 inset rgba(255,255,255,0.1); | ||||
| #u2tab a>span { | ||||
| 	font-weight: bold; | ||||
| 	font-style: italic; | ||||
| 	color: #fff; | ||||
| 	padding-left: .2em; | ||||
| } | ||||
| .prog>div>div { | ||||
| 	width: 0%; | ||||
| 	position: absolute; | ||||
| 	left: 0; | ||||
| 	top: 0; | ||||
| 	bottom: 0; | ||||
| 	background: #0a0; | ||||
| #u2cleanup { | ||||
| 	float: right; | ||||
| 	margin-bottom: -.3em; | ||||
| } | ||||
| .fsearch_explain { | ||||
| 	padding-left: .7em; | ||||
| 	font-size: 1.1em; | ||||
| 	line-height: 0; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| html.light #u2btn { | ||||
| 	box-shadow: .4em .4em 0 #ccc; | ||||
| } | ||||
| html.light #u2cards span { | ||||
| 	color: #000; | ||||
| } | ||||
| html.light #u2cards a { | ||||
| 	background: linear-gradient(to bottom, #eee, #fff); | ||||
| } | ||||
| html.light #u2cards a.act { | ||||
| 	color: #037; | ||||
| 	background: inherit; | ||||
| 	box-shadow: 0 -.17em .67em #0ad; | ||||
| 	border-color: #09c #05a #eee #05a; | ||||
| } | ||||
| html.light #u2conf .txtbox { | ||||
| 	background: #fff; | ||||
| 	color: #444; | ||||
| } | ||||
| html.light #u2conf .txtbox.err { | ||||
| 	background: #f96; | ||||
| 	color: #300; | ||||
| } | ||||
| html.light #op_up2k.srch #u2btn { | ||||
| 	border-color: #a80; | ||||
| } | ||||
| html.light #u2foot { | ||||
| 	color: #000; | ||||
| } | ||||
| html.light #u2tab tbody tr:hover td { | ||||
| 	background: #fff; | ||||
| } | ||||
|   | ||||
| @@ -1,70 +0,0 @@ | ||||
|     <div id="ops"><a | ||||
|         href="#" data-dest="">---</a><i></i><a | ||||
|         href="#" data-dest="up2k">up2k</a><i></i><a | ||||
|         href="#" data-dest="bup">bup</a><i></i><a | ||||
|         href="#" data-dest="mkdir">mkdir</a><i></i><a | ||||
|         href="#" data-dest="new_md">new.md</a></div> | ||||
|  | ||||
|     <div id="op_bup" class="opview opbox act"> | ||||
|         <div id="u2err"></div> | ||||
|         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}"> | ||||
|             <input type="hidden" name="act" value="bput" /> | ||||
|             <input type="file" name="f" multiple><br /> | ||||
|             <input type="submit" value="start upload"> | ||||
|         </form> | ||||
|     </div> | ||||
|  | ||||
|     <div id="op_mkdir" class="opview opbox act"> | ||||
|         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}"> | ||||
|             <input type="hidden" name="act" value="mkdir" /> | ||||
|             <input type="text" name="name" size="30"> | ||||
|             <input type="submit" value="mkdir"> | ||||
|         </form> | ||||
|     </div> | ||||
|  | ||||
|     <div id="op_new_md" class="opview opbox"> | ||||
|         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}"> | ||||
|             <input type="hidden" name="act" value="new_md" /> | ||||
|             <input type="text" name="name" size="30"> | ||||
|             <input type="submit" value="create doc"> | ||||
|         </form> | ||||
|     </div> | ||||
|  | ||||
|     <div id="op_up2k" class="opview"> | ||||
|         <form id="u2form" method="post" enctype="multipart/form-data" onsubmit="return false;"></form> | ||||
|  | ||||
|             <table id="u2conf"> | ||||
|                 <tr> | ||||
|                     <td>parallel uploads</td> | ||||
|                 </tr> | ||||
|                 <tr> | ||||
|                     <td> | ||||
|                         <a href="#" id="nthread_sub">–</a> | ||||
|                         <input class="txtbox" id="nthread" value="2" /> | ||||
|                         <a href="#" id="nthread_add">+</a> | ||||
|                     </td> | ||||
|                     <td rowspan="2"> | ||||
|                         <input type="checkbox" id="multitask" /> | ||||
|                         <label for="multitask">hash while<br />uploading</label> | ||||
|                     </td> | ||||
|                 </tr> | ||||
|             </table> | ||||
|  | ||||
|             <div id="u2notbtn"></div> | ||||
|  | ||||
|             <div id="u2btn"> | ||||
|                 drop files here<br /> | ||||
|                 (or click me) | ||||
|             </div> | ||||
|  | ||||
|             <table id="u2tab"> | ||||
|                 <tr> | ||||
|                     <td>filename</td> | ||||
|                     <td>status</td> | ||||
|                     <td>progress</td> | ||||
|                 </tr> | ||||
|             </table> | ||||
|  | ||||
|             <p id="u2foot"></p> | ||||
|             <p>( if you don't need lastmod timestamps, resumable uploads or progress bars just use the <a href="#" id="u2nope" onclick="javascript:goto('bup');">basic uploader</a>)</p> | ||||
|     </div> | ||||
							
								
								
									
										698
									
								
								copyparty/web/util.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										698
									
								
								copyparty/web/util.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,698 @@ | ||||
| "use strict"; | ||||
|  | ||||
| if (!window['console']) | ||||
|     window['console'] = { | ||||
|         "log": function (msg) { } | ||||
|     }; | ||||
|  | ||||
|  | ||||
| var is_touch = 'ontouchstart' in window, | ||||
|     IPHONE = /iPhone|iPad|iPod/i.test(navigator.userAgent), | ||||
|     ANDROID = /android/i.test(navigator.userAgent); | ||||
|  | ||||
|  | ||||
| var ebi = document.getElementById.bind(document), | ||||
|     QS = document.querySelector.bind(document), | ||||
|     QSA = document.querySelectorAll.bind(document), | ||||
|     mknod = document.createElement.bind(document); | ||||
|  | ||||
|  | ||||
| // error handler for mobile devices | ||||
| function esc(txt) { | ||||
|     return txt.replace(/[&"<>]/g, function (c) { | ||||
|         return { | ||||
|             '&': '&', | ||||
|             '"': '"', | ||||
|             '<': '<', | ||||
|             '>': '>' | ||||
|         }[c]; | ||||
|     }); | ||||
| } | ||||
| var crashed = false, ignexd = {}; | ||||
| function vis_exh(msg, url, lineNo, columnNo, error) { | ||||
|     if ((msg + '').indexOf('ResizeObserver') !== -1) | ||||
|         return;  // chrome issue 809574 (benign, from <video>) | ||||
|  | ||||
|     var ekey = url + '\n' + lineNo + '\n' + msg; | ||||
|     if (ignexd[ekey] || crashed) | ||||
|         return; | ||||
|  | ||||
|     crashed = true; | ||||
|     window.onerror = undefined; | ||||
|     var html = ['<h1>you hit a bug!</h1><p style="font-size:1.3em;margin:0">try to <a href="#" onclick="localStorage.clear();location.reload();">reset copyparty settings</a> if you are stuck here, or <a href="#" onclick="ignex();">ignore this</a> / <a href="#" onclick="ignex(true);">ignore all</a></p><p>please send me a screenshot arigathanks gozaimuch: <code>ed/irc.rizon.net</code> or <code>ed#2644</code><br />  (and if you can, press F12 and include the "Console" tab in the screenshot too)</p><p>', | ||||
|         esc(url + ' @' + lineNo + ':' + columnNo), '<br />' + esc(String(msg)) + '</p>']; | ||||
|  | ||||
|     try { | ||||
|         if (error) { | ||||
|             var find = ['desc', 'stack', 'trace']; | ||||
|             for (var a = 0; a < find.length; a++) | ||||
|                 if (String(error[find[a]]) !== 'undefined') | ||||
|                     html.push('<h3>' + find[a] + '</h3>' + | ||||
|                         esc(String(error[find[a]])).replace(/\n/g, '<br />\n')); | ||||
|         } | ||||
|         ignexd[ekey] = true; | ||||
|         html.push('<h3>localStore</h3>' + esc(JSON.stringify(localStorage))); | ||||
|     } | ||||
|     catch (e) { } | ||||
|  | ||||
|     try { | ||||
|         var exbox = ebi('exbox'); | ||||
|         if (!exbox) { | ||||
|             exbox = mknod('div'); | ||||
|             exbox.setAttribute('id', 'exbox'); | ||||
|             document.body.appendChild(exbox); | ||||
|  | ||||
|             var s = mknod('style'); | ||||
|             s.innerHTML = '#exbox{background:#333;color:#ddd;font-family:sans-serif;font-size:0.8em;padding:0 1em 1em 1em;z-index:80386;position:fixed;top:0;left:0;right:0;bottom:0;width:100%;height:100%} #exbox h1{margin:.5em 1em 0 0;padding:0} #exbox h3{border-top:1px solid #999;margin:1em 0 0 0} #exbox a{text-decoration:underline;color:#fc0} #exbox code{color:#bf7;background:#222;padding:.1em;margin:.2em;font-size:1.1em;font-family:monospace,monospace} #exbox *{line-height:1.5em}'; | ||||
|             document.head.appendChild(s); | ||||
|         } | ||||
|         exbox.innerHTML = html.join('\n'); | ||||
|         exbox.style.display = 'block'; | ||||
|     } | ||||
|     catch (e) { | ||||
|         document.body.innerHTML = html.join('\n'); | ||||
|     } | ||||
|     throw 'fatal_err'; | ||||
| } | ||||
| function ignex(all) { | ||||
|     var o = ebi('exbox'); | ||||
|     o.style.display = 'none'; | ||||
|     o.innerHTML = ''; | ||||
|     crashed = false; | ||||
|     if (!all) | ||||
|         window.onerror = vis_exh; | ||||
| } | ||||
|  | ||||
|  | ||||
| function ctrl(e) { | ||||
|     return e && (e.ctrlKey || e.metaKey); | ||||
| } | ||||
|  | ||||
|  | ||||
| function ev(e) { | ||||
|     e = e || window.event; | ||||
|     if (!e) | ||||
|         return; | ||||
|  | ||||
|     if (e.preventDefault) | ||||
|         e.preventDefault() | ||||
|  | ||||
|     if (e.stopPropagation) | ||||
|         e.stopPropagation(); | ||||
|  | ||||
|     if (e.stopImmediatePropagation) | ||||
|         e.stopImmediatePropagation(); | ||||
|  | ||||
|     e.returnValue = false; | ||||
|     return e; | ||||
| } | ||||
|  | ||||
|  | ||||
| // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/endsWith | ||||
| if (!String.prototype.endsWith) { | ||||
|     String.prototype.endsWith = function (search, this_len) { | ||||
|         if (this_len === undefined || this_len > this.length) { | ||||
|             this_len = this.length; | ||||
|         } | ||||
|         return this.substring(this_len - search.length, this_len) === search; | ||||
|     }; | ||||
| } | ||||
| if (!String.startsWith) { | ||||
|     String.prototype.startsWith = function (s, i) { | ||||
|         i = i > 0 ? i | 0 : 0; | ||||
|         return this.substring(i, i + s.length) === s; | ||||
|     }; | ||||
| } | ||||
| if (!Element.prototype.closest) { | ||||
|     Element.prototype.closest = function (s) { | ||||
|         var el = this; | ||||
|         do { | ||||
|             if (el.msMatchesSelector(s)) return el; | ||||
|             el = el.parentElement || el.parentNode; | ||||
|         } while (el !== null && el.nodeType === 1); | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| // https://stackoverflow.com/a/950146 | ||||
| function import_js(url, cb) { | ||||
|     var head = document.head || document.getElementsByTagName('head')[0]; | ||||
|     var script = mknod('script'); | ||||
|     script.type = 'text/javascript'; | ||||
|     script.src = url; | ||||
|  | ||||
|     script.onreadystatechange = cb; | ||||
|     script.onload = cb; | ||||
|  | ||||
|     head.appendChild(script); | ||||
| } | ||||
|  | ||||
|  | ||||
| var crctab = (function () { | ||||
|     var c, tab = []; | ||||
|     for (var n = 0; n < 256; n++) { | ||||
|         c = n; | ||||
|         for (var k = 0; k < 8; k++) { | ||||
|             c = ((c & 1) ? (0xEDB88320 ^ (c >>> 1)) : (c >>> 1)); | ||||
|         } | ||||
|         tab[n] = c; | ||||
|     } | ||||
|     return tab; | ||||
| })(); | ||||
|  | ||||
|  | ||||
| function crc32(str) { | ||||
|     var crc = 0 ^ (-1); | ||||
|     for (var i = 0; i < str.length; i++) { | ||||
|         crc = (crc >>> 8) ^ crctab[(crc ^ str.charCodeAt(i)) & 0xFF]; | ||||
|     } | ||||
|     return ((crc ^ (-1)) >>> 0).toString(16); | ||||
| } | ||||
|  | ||||
|  | ||||
| function clmod(obj, cls, add) { | ||||
|     var re = new RegExp('\\s*\\b' + cls + '\\s*\\b', 'g'); | ||||
|     if (add == 't') | ||||
|         add = !re.test(obj.className); | ||||
|  | ||||
|     obj.className = obj.className.replace(re, ' ') + (add ? ' ' + cls : ''); | ||||
| } | ||||
|  | ||||
|  | ||||
| function sortfiles(nodes) { | ||||
|     var sopts = jread('fsort', [["href", 1, ""]]); | ||||
|  | ||||
|     try { | ||||
|         var is_srch = false; | ||||
|         if (nodes[0]['rp']) { | ||||
|             is_srch = true; | ||||
|             for (var b = 0, bb = nodes.length; b < bb; b++) | ||||
|                 nodes[b].ext = nodes[b].rp.split('.').pop(); | ||||
|             for (var b = 0; b < sopts.length; b++) | ||||
|                 if (sopts[b][0] == 'href') | ||||
|                     sopts[b][0] = 'rp'; | ||||
|         } | ||||
|         for (var a = sopts.length - 1; a >= 0; a--) { | ||||
|             var name = sopts[a][0], rev = sopts[a][1], typ = sopts[a][2]; | ||||
|             if (!name) | ||||
|                 continue; | ||||
|  | ||||
|             if (name == 'ts') | ||||
|                 typ = 'int'; | ||||
|  | ||||
|             if (name.indexOf('tags/') === 0) { | ||||
|                 name = name.slice(5); | ||||
|                 for (var b = 0, bb = nodes.length; b < bb; b++) | ||||
|                     nodes[b]._sv = nodes[b].tags[name]; | ||||
|             } | ||||
|             else { | ||||
|                 for (var b = 0, bb = nodes.length; b < bb; b++) { | ||||
|                     var v = nodes[b][name]; | ||||
|  | ||||
|                     if ((v + '').indexOf('<a ') === 0) | ||||
|                         v = v.split('>')[1]; | ||||
|                     else if (name == "href" && v) { | ||||
|                         if (v.slice(-1) == '/') | ||||
|                             v = '\t' + v; | ||||
|  | ||||
|                         v = uricom_dec(v)[0] | ||||
|                     } | ||||
|  | ||||
|                     nodes[b]._sv = v; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             var onodes = nodes.map(function (x) { return x; }); | ||||
|             nodes.sort(function (n1, n2) { | ||||
|                 var v1 = n1._sv, | ||||
|                     v2 = n2._sv; | ||||
|  | ||||
|                 if (v1 === undefined) { | ||||
|                     if (v2 === undefined) { | ||||
|                         return onodes.indexOf(n1) - onodes.indexOf(n2); | ||||
|                     } | ||||
|                     return -1 * rev; | ||||
|                 } | ||||
|                 if (v2 === undefined) return 1 * rev; | ||||
|  | ||||
|                 var ret = rev * (typ == 'int' ? (v1 - v2) : (v1.localeCompare(v2))); | ||||
|                 if (ret === 0) | ||||
|                     ret = onodes.indexOf(n1) - onodes.indexOf(n2); | ||||
|  | ||||
|                 return ret; | ||||
|             }); | ||||
|         } | ||||
|         for (var b = 0, bb = nodes.length; b < bb; b++) { | ||||
|             delete nodes[b]._sv; | ||||
|             if (is_srch) | ||||
|                 delete nodes[b].ext; | ||||
|         } | ||||
|     } | ||||
|     catch (ex) { | ||||
|         console.log("failed to apply sort config: " + ex); | ||||
|         console.log("resetting fsort " + sread('fsort')) | ||||
|         localStorage.removeItem('fsort'); | ||||
|     } | ||||
|     return nodes; | ||||
| } | ||||
|  | ||||
|  | ||||
| function sortTable(table, col, cb) { | ||||
|     var tb = table.tBodies[0], | ||||
|         th = table.tHead.rows[0].cells, | ||||
|         tr = Array.prototype.slice.call(tb.rows, 0), | ||||
|         i, reverse = th[col].className.indexOf('sort1') !== -1 ? -1 : 1; | ||||
|     for (var a = 0, thl = th.length; a < thl; a++) | ||||
|         th[a].className = th[a].className.replace(/ *sort-?1 */, " "); | ||||
|     th[col].className += ' sort' + reverse; | ||||
|     var stype = th[col].getAttribute('sort'); | ||||
|     try { | ||||
|         var nrules = [], rules = jread("fsort", []); | ||||
|         rules.unshift([th[col].getAttribute('name'), reverse, stype || '']); | ||||
|         for (var a = 0; a < rules.length; a++) { | ||||
|             var add = true; | ||||
|             for (var b = 0; b < a; b++) | ||||
|                 if (rules[a][0] == rules[b][0]) | ||||
|                     add = false; | ||||
|  | ||||
|             if (add) | ||||
|                 nrules.push(rules[a]); | ||||
|  | ||||
|             if (nrules.length >= 10) | ||||
|                 break; | ||||
|         } | ||||
|         jwrite("fsort", nrules); | ||||
|     } | ||||
|     catch (ex) { | ||||
|         console.log("failed to persist sort rules, resetting: " + ex); | ||||
|         jwrite("fsort", null); | ||||
|     } | ||||
|     var vl = []; | ||||
|     for (var a = 0; a < tr.length; a++) { | ||||
|         var cell = tr[a].cells[col]; | ||||
|         if (!cell) { | ||||
|             vl.push([null, a]); | ||||
|             continue; | ||||
|         } | ||||
|         var v = cell.getAttribute('sortv') || cell.textContent.trim(); | ||||
|         if (stype == 'int') { | ||||
|             v = parseInt(v.replace(/[, ]/g, '')) || 0; | ||||
|         } | ||||
|         vl.push([v, a]); | ||||
|     } | ||||
|     vl.sort(function (a, b) { | ||||
|         a = a[0]; | ||||
|         b = b[0]; | ||||
|         if (a === null) | ||||
|             return -1; | ||||
|         if (b === null) | ||||
|             return 1; | ||||
|  | ||||
|         if (stype == 'int') { | ||||
|             return reverse * (a - b); | ||||
|         } | ||||
|         return reverse * (a.localeCompare(b)); | ||||
|     }); | ||||
|     for (i = 0; i < tr.length; ++i) tb.appendChild(tr[vl[i][1]]); | ||||
|     if (cb) cb(); | ||||
| } | ||||
| function makeSortable(table, cb) { | ||||
|     var th = table.tHead, i; | ||||
|     th && (th = th.rows[0]) && (th = th.cells); | ||||
|     if (th) i = th.length; | ||||
|     else return; // if no `<thead>` then do nothing | ||||
|     while (--i >= 0) (function (i) { | ||||
|         th[i].onclick = function (e) { | ||||
|             ev(e); | ||||
|             sortTable(table, i, cb); | ||||
|         }; | ||||
|     }(i)); | ||||
| } | ||||
|  | ||||
|  | ||||
| function linksplit(rp) { | ||||
|     var ret = []; | ||||
|     var apath = '/'; | ||||
|     if (rp && rp.charAt(0) == '/') | ||||
|         rp = rp.slice(1); | ||||
|  | ||||
|     while (rp) { | ||||
|         var link = rp; | ||||
|         var ofs = rp.indexOf('/'); | ||||
|         if (ofs === -1) { | ||||
|             rp = null; | ||||
|         } | ||||
|         else { | ||||
|             link = rp.slice(0, ofs + 1); | ||||
|             rp = rp.slice(ofs + 1); | ||||
|         } | ||||
|         var vlink = esc(link), | ||||
|             elink = uricom_enc(link); | ||||
|  | ||||
|         if (link.indexOf('/') !== -1) { | ||||
|             vlink = vlink.slice(0, -1) + '<span>/</span>'; | ||||
|             elink = elink.slice(0, -3) + '/'; | ||||
|         } | ||||
|  | ||||
|         ret.push('<a href="' + apath + elink + '">' + vlink + '</a>'); | ||||
|         apath += elink; | ||||
|     } | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
|  | ||||
| function vsplit(vp) { | ||||
|     if (vp.endsWith('/')) | ||||
|         vp = vp.slice(0, -1); | ||||
|  | ||||
|     var ofs = vp.lastIndexOf('/') + 1, | ||||
|         base = vp.slice(0, ofs), | ||||
|         fn = vp.slice(ofs); | ||||
|  | ||||
|     return [base, fn]; | ||||
| } | ||||
|  | ||||
|  | ||||
| function uricom_enc(txt, do_fb_enc) { | ||||
|     try { | ||||
|         return encodeURIComponent(txt); | ||||
|     } | ||||
|     catch (ex) { | ||||
|         console.log("uce-err [" + txt + "]"); | ||||
|         if (do_fb_enc) | ||||
|             return esc(txt); | ||||
|  | ||||
|         return txt; | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| function uricom_dec(txt) { | ||||
|     try { | ||||
|         return [decodeURIComponent(txt), true]; | ||||
|     } | ||||
|     catch (ex) { | ||||
|         console.log("ucd-err [" + txt + "]"); | ||||
|         return [txt, false]; | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| function get_evpath() { | ||||
|     var ret = document.location.pathname; | ||||
|  | ||||
|     if (ret.indexOf('/') !== 0) | ||||
|         ret = '/' + ret; | ||||
|  | ||||
|     if (ret.lastIndexOf('/') !== ret.length - 1) | ||||
|         ret += '/'; | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
|  | ||||
| function get_vpath() { | ||||
|     return uricom_dec(get_evpath())[0]; | ||||
| } | ||||
|  | ||||
|  | ||||
| function get_pwd() { | ||||
|     var pwd = ('; ' + document.cookie).split('; cppwd='); | ||||
|     if (pwd.length < 2) | ||||
|         return null; | ||||
|  | ||||
|     return pwd[1].split(';')[0]; | ||||
| } | ||||
|  | ||||
|  | ||||
| function unix2iso(ts) { | ||||
|     return new Date(ts * 1000).toISOString().replace("T", " ").slice(0, -5); | ||||
| } | ||||
|  | ||||
|  | ||||
| function s2ms(s) { | ||||
|     s = Math.floor(s); | ||||
|     var m = Math.floor(s / 60); | ||||
|     return m + ":" + ("0" + (s - m * 60)).slice(-2); | ||||
| } | ||||
|  | ||||
|  | ||||
| function has(haystack, needle) { | ||||
|     for (var a = 0; a < haystack.length; a++) | ||||
|         if (haystack[a] == needle) | ||||
|             return true; | ||||
|  | ||||
|     return false; | ||||
| } | ||||
|  | ||||
|  | ||||
| function apop(arr, v) { | ||||
|     var ofs = arr.indexOf(v); | ||||
|     if (ofs !== -1) | ||||
|         arr.splice(ofs, 1); | ||||
| } | ||||
|  | ||||
|  | ||||
| function jcp(obj) { | ||||
|     return JSON.parse(JSON.stringify(obj)); | ||||
| } | ||||
|  | ||||
|  | ||||
| function sread(key) { | ||||
|     return localStorage.getItem(key); | ||||
| } | ||||
|  | ||||
| function swrite(key, val) { | ||||
|     if (val === undefined || val === null) | ||||
|         localStorage.removeItem(key); | ||||
|     else | ||||
|         localStorage.setItem(key, val); | ||||
| } | ||||
|  | ||||
| function jread(key, fb) { | ||||
|     var str = sread(key); | ||||
|     if (!str) | ||||
|         return fb; | ||||
|  | ||||
|     return JSON.parse(str); | ||||
| } | ||||
|  | ||||
| function jwrite(key, val) { | ||||
|     if (!val) | ||||
|         swrite(key); | ||||
|     else | ||||
|         swrite(key, JSON.stringify(val)); | ||||
| } | ||||
|  | ||||
| function icfg_get(name, defval) { | ||||
|     return parseInt(fcfg_get(name, defval)); | ||||
| } | ||||
|  | ||||
| function fcfg_get(name, defval) { | ||||
|     var o = ebi(name); | ||||
|  | ||||
|     var val = parseFloat(sread(name)); | ||||
|     if (isNaN(val)) | ||||
|         return parseFloat(o ? o.value : defval); | ||||
|  | ||||
|     if (o) | ||||
|         o.value = val; | ||||
|  | ||||
|     return val; | ||||
| } | ||||
|  | ||||
| function bcfg_get(name, defval) { | ||||
|     var o = ebi(name); | ||||
|     if (!o) | ||||
|         return defval; | ||||
|  | ||||
|     var val = sread(name); | ||||
|     if (val === null) | ||||
|         val = defval; | ||||
|     else | ||||
|         val = (val == '1'); | ||||
|  | ||||
|     bcfg_upd_ui(name, val); | ||||
|     return val; | ||||
| } | ||||
|  | ||||
| function bcfg_set(name, val) { | ||||
|     swrite(name, val ? '1' : '0'); | ||||
|     bcfg_upd_ui(name, val); | ||||
|     return val; | ||||
| } | ||||
|  | ||||
| function bcfg_upd_ui(name, val) { | ||||
|     var o = ebi(name); | ||||
|     if (!o) | ||||
|         return; | ||||
|  | ||||
|     if (o.getAttribute('type') == 'checkbox') | ||||
|         o.checked = val; | ||||
|     else if (o) { | ||||
|         clmod(o, 'on', val); | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| function hist_push(url) { | ||||
|     console.log("h-push " + url); | ||||
|     history.pushState(url, url, url); | ||||
| } | ||||
|  | ||||
| function hist_replace(url) { | ||||
|     console.log("h-repl " + url); | ||||
|     history.replaceState(url, url, url); | ||||
| } | ||||
|  | ||||
|  | ||||
| var tt = (function () { | ||||
|     var r = { | ||||
|         "tt": mknod("div"), | ||||
|         "en": true, | ||||
|         "el": null, | ||||
|         "skip": false | ||||
|     }; | ||||
|  | ||||
|     r.tt.setAttribute('id', 'tt'); | ||||
|     document.body.appendChild(r.tt); | ||||
|  | ||||
|     r.show = function () { | ||||
|         if (r.skip) { | ||||
|             r.skip = false; | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         var cfg = sread('tooltips'); | ||||
|         if (cfg !== null && cfg != '1') | ||||
|             return; | ||||
|  | ||||
|         var msg = this.getAttribute('tt'); | ||||
|         if (!msg) | ||||
|             return; | ||||
|  | ||||
|         r.el = this; | ||||
|         var pos = this.getBoundingClientRect(), | ||||
|             dir = this.getAttribute('ttd') || '', | ||||
|             left = pos.left < window.innerWidth / 2, | ||||
|             top = pos.top < window.innerHeight / 2, | ||||
|             big = this.className.indexOf(' ttb') !== -1; | ||||
|  | ||||
|         if (dir.indexOf('u') + 1) top = false; | ||||
|         if (dir.indexOf('d') + 1) top = true; | ||||
|         if (dir.indexOf('l') + 1) left = false; | ||||
|         if (dir.indexOf('r') + 1) left = true; | ||||
|  | ||||
|         clmod(r.tt, 'b', big); | ||||
|         r.tt.style.top = top ? pos.bottom + 'px' : 'auto'; | ||||
|         r.tt.style.bottom = top ? 'auto' : (window.innerHeight - pos.top) + 'px'; | ||||
|         r.tt.style.left = left ? pos.left + 'px' : 'auto'; | ||||
|         r.tt.style.right = left ? 'auto' : (window.innerWidth - pos.right) + 'px'; | ||||
|  | ||||
|         r.tt.innerHTML = msg.replace(/\$N/g, "<br />"); | ||||
|         r.el.addEventListener('mouseleave', r.hide); | ||||
|         clmod(r.tt, 'show', 1); | ||||
|     }; | ||||
|  | ||||
|     r.hide = function (e) { | ||||
|         ev(e); | ||||
|         clmod(r.tt, 'show'); | ||||
|         if (r.el) | ||||
|             r.el.removeEventListener('mouseleave', r.hide); | ||||
|     }; | ||||
|  | ||||
|     if (is_touch && IPHONE) { | ||||
|         var f1 = r.show, | ||||
|             f2 = r.hide; | ||||
|  | ||||
|         r.show = function () { | ||||
|             setTimeout(f1.bind(this), 301); | ||||
|         }; | ||||
|         r.hide = function () { | ||||
|             setTimeout(f2.bind(this), 301); | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     r.tt.onclick = r.hide; | ||||
|  | ||||
|     r.att = function (ctr) { | ||||
|         var _show = r.en ? r.show : null, | ||||
|             _hide = r.en ? r.hide : null, | ||||
|             o = ctr.querySelectorAll('*[tt]'); | ||||
|  | ||||
|         for (var a = o.length - 1; a >= 0; a--) { | ||||
|             o[a].onfocus = _show; | ||||
|             o[a].onblur = _hide; | ||||
|             o[a].onmouseenter = _show; | ||||
|             o[a].onmouseleave = _hide; | ||||
|         } | ||||
|         r.hide(); | ||||
|     } | ||||
|  | ||||
|     r.init = function () { | ||||
|         var ttb = ebi('tooltips'); | ||||
|         if (ttb) { | ||||
|             ttb.onclick = function (e) { | ||||
|                 ev(e); | ||||
|                 r.en = !r.en; | ||||
|                 bcfg_set('tooltips', r.en); | ||||
|                 r.init(); | ||||
|             }; | ||||
|             r.en = bcfg_get('tooltips', true) | ||||
|         } | ||||
|         r.att(document); | ||||
|     }; | ||||
|  | ||||
|     return r; | ||||
| })(); | ||||
|  | ||||
|  | ||||
| var toast = (function () { | ||||
|     var r = {}, | ||||
|         te = null, | ||||
|         visible = false, | ||||
|         obj = mknod('div'); | ||||
|  | ||||
|     obj.setAttribute('id', 'toast'); | ||||
|     document.body.appendChild(obj);; | ||||
|  | ||||
|     r.hide = function (e) { | ||||
|         ev(e); | ||||
|         clearTimeout(te); | ||||
|         clmod(obj, 'vis'); | ||||
|         r.visible = false; | ||||
|     }; | ||||
|  | ||||
|     r.show = function (cl, ms, txt) { | ||||
|         clearTimeout(te); | ||||
|         if (ms) | ||||
|             te = setTimeout(r.hide, ms * 1000); | ||||
|  | ||||
|         var html = '', hp = txt.split(/(?=<.?pre>)/i); | ||||
|         for (var a = 0; a < hp.length; a++) | ||||
|             html += hp[a].startsWith('<pre>') ? hp[a] : | ||||
|                 hp[a].replace(/<br ?.?>\n/g, '\n').replace(/\n<br ?.?>/g, '\n').replace(/\n/g, '<br />\n'); | ||||
|  | ||||
|         obj.innerHTML = '<a href="#" id="toastc">x</a>' + html; | ||||
|         obj.className = cl; | ||||
|         ms += obj.offsetWidth; | ||||
|         obj.className += ' vis'; | ||||
|         ebi('toastc').onclick = r.hide; | ||||
|         r.visible = true; | ||||
|     }; | ||||
|  | ||||
|     r.ok = function (ms, txt) { | ||||
|         r.show('ok', ms, txt); | ||||
|     }; | ||||
|     r.inf = function (ms, txt) { | ||||
|         r.show('inf', ms, txt); | ||||
|     }; | ||||
|     r.warn = function (ms, txt) { | ||||
|         r.show('warn', ms, txt); | ||||
|     }; | ||||
|     r.err = function (ms, txt) { | ||||
|         r.show('err', ms, txt); | ||||
|     }; | ||||
|  | ||||
|     return r; | ||||
| })(); | ||||
							
								
								
									
										22
									
								
								docs/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								docs/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| # example `.epilogue.html` | ||||
| save one of these as `.epilogue.html` inside a folder to customize it: | ||||
|  | ||||
| * [`minimal-up2k.html`](minimal-up2k.html) will [simplify the upload ui](https://user-images.githubusercontent.com/241032/118311195-dd6ca380-b4ef-11eb-86f3-75a3ff2e1332.png) | ||||
|  | ||||
|  | ||||
|  | ||||
| # example browser-css | ||||
| point `--css-browser` to one of these by URL: | ||||
|  | ||||
| * [`browser.css`](browser.css) changes the background | ||||
| * [`browser-icons.css`](browser-icons.css) adds filetype icons | ||||
|  | ||||
|  | ||||
|  | ||||
| # other stuff | ||||
|  | ||||
| ## [`rclone.md`](rclone.md) | ||||
| * notes on using rclone as a fuse client/server | ||||
|  | ||||
| ## [`example.conf`](example.conf) | ||||
| * example config file for `-c` which never really happened | ||||
							
								
								
									
										95
									
								
								docs/biquad.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								docs/biquad.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | ||||
| <!DOCTYPE html><html><head></head><body><script> | ||||
|  | ||||
| setTimeout(location.reload.bind(location), 700); | ||||
| document.documentElement.scrollLeft = 0; | ||||
|  | ||||
| var can = document.createElement('canvas'), | ||||
|     cc = can.getContext('2d'), | ||||
|     w = 2048, | ||||
|     h = 1024; | ||||
|  | ||||
| w = 2048; | ||||
|  | ||||
| can.width = w; | ||||
| can.height = h; | ||||
| document.body.appendChild(can); | ||||
| can.style.cssText = 'width:' + w + 'px;height:' + h + 'px'; | ||||
|  | ||||
| cc.fillStyle = '#000'; | ||||
| cc.fillRect(0, 0, w, h); | ||||
|  | ||||
| var cfg = [ // hz, q, g | ||||
|     [31.25 * 0.88, 0, 1.4],  // shelf | ||||
|     [31.25 * 1.04, 0.7, 0.96],  // peak | ||||
|     [62.5, 0.7, 1], | ||||
|     [125, 0.8, 1], | ||||
|     [250, 0.9, 1.03], | ||||
|     [500, 0.9, 1.1], | ||||
|     [1000, 0.9, 1.1], | ||||
|     [2000, 0.9, 1.105], | ||||
|     [4000, 0.88, 1.05], | ||||
|     [8000 * 1.006, 0.73, 1.24], | ||||
|     //[16000 * 1.00, 0.5, 1.75],  // peak.v1 | ||||
|     //[16000 * 1.19, 0, 1.8]  // shelf.v1 | ||||
|     [16000 * 0.89, 0.7, 1.26],  // peak | ||||
|     [16000 * 1.13, 0.82, 1.09],  // peak | ||||
|     [16000 * 1.205, 0, 1.9]  // shelf | ||||
| ]; | ||||
|  | ||||
| var freqs = new Float32Array(22000), | ||||
|     sum = new Float32Array(freqs.length), | ||||
|     ac = new AudioContext(), | ||||
|     step = w / freqs.length, | ||||
|     colors = [ | ||||
|         'rgba(255, 0, 0, 0.7)', | ||||
|         'rgba(0, 224, 0, 0.7)', | ||||
|         'rgba(0, 64, 255, 0.7)' | ||||
|     ]; | ||||
|  | ||||
| var order = []; | ||||
|  | ||||
| for (var a = 0; a < cfg.length; a += 2) | ||||
|     order.push(a); | ||||
|  | ||||
| for (var a = 1; a < cfg.length; a += 2) | ||||
|     order.push(a); | ||||
|  | ||||
| for (var ia = 0; ia < order.length; ia++) { | ||||
|     var a = order[ia], | ||||
|         fi = ac.createBiquadFilter(), | ||||
|         mag = new Float32Array(freqs.length), | ||||
|         phase = new Float32Array(freqs.length); | ||||
|  | ||||
|     for (var b = 0; b < freqs.length; b++) | ||||
|         freqs[b] = b; | ||||
|  | ||||
|     fi.type = a == 0 ? 'lowshelf' : a == cfg.length - 1 ? 'highshelf' : 'peaking'; | ||||
|     fi.frequency.value = cfg[a][0]; | ||||
|     fi.Q.value = cfg[a][1]; | ||||
|     fi.gain.value = 1; | ||||
|  | ||||
|     fi.getFrequencyResponse(freqs, mag, phase); | ||||
|     cc.fillStyle = colors[a % colors.length]; | ||||
|     for (var b = 0; b < sum.length; b++) { | ||||
|         mag[b] -= 1; | ||||
|         sum[b] += mag[b] * cfg[a][2]; | ||||
|         var y = h - (mag[b] * h * 3); | ||||
|         cc.fillRect(b * step, y, step, h - y); | ||||
|         cc.fillRect(b * step - 1, y - 1, 3, 3); | ||||
|     } | ||||
| } | ||||
|  | ||||
| var min = 999999, max = 0; | ||||
| for (var a = 0; a < sum.length; a++) { | ||||
|     min = Math.min(min, sum[a]); | ||||
|     max = Math.max(max, sum[a]); | ||||
| } | ||||
| cc.fillStyle = 'rgba(255,255,255,1)'; | ||||
| for (var a = 0; a < sum.length; a++) { | ||||
|     var v = (sum[a] - min) / (max - min); | ||||
|     cc.fillRect(a * step, 0, step, v * h / 2); | ||||
| } | ||||
|  | ||||
| cc.fillRect(0, 460, w, 1); | ||||
|  | ||||
| </script></body></html> | ||||
							
								
								
									
										66
									
								
								docs/browser-icons.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								docs/browser-icons.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
| /* put filetype icons inline with text | ||||
| #ggrid>a>span:before, | ||||
| #ggrid>a>span.dir:before { | ||||
| 	display: inline; | ||||
| 	line-height: 0; | ||||
| 	font-size: 1.7em; | ||||
| 	margin: -.7em .1em -.5em -.6em; | ||||
| } | ||||
| */ | ||||
|  | ||||
|  | ||||
| /* move folder icons top-left */ | ||||
| #ggrid>a>span.dir:before { | ||||
| 	content: initial; | ||||
| } | ||||
| #ggrid>a[href$="/"]:before { | ||||
| 	content: '📂'; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* put filetype icons top-left */ | ||||
| #ggrid>a:before { | ||||
|     display: block; | ||||
|     position: absolute; | ||||
| 	padding: .3em 0; | ||||
|     margin: -.4em; | ||||
|     text-shadow: 0 0 .1em #000; | ||||
| 	background: linear-gradient(135deg,rgba(255,255,255,0) 50%,rgba(255,255,255,0.2)); | ||||
| 	border-radius: .3em; | ||||
|     font-size: 2em; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* video */ | ||||
| #ggrid>a:is( | ||||
| [href$=".mkv"i], | ||||
| [href$=".mp4"i], | ||||
| [href$=".webm"i], | ||||
| ):before { | ||||
|     content: '📺'; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* audio */ | ||||
| #ggrid>a:is( | ||||
| [href$=".mp3"i], | ||||
| [href$=".ogg"i], | ||||
| [href$=".opus"i], | ||||
| [href$=".flac"i], | ||||
| [href$=".m4a"i], | ||||
| [href$=".aac"i], | ||||
| ):before { | ||||
|     content: '🎵'; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* image */ | ||||
| #ggrid>a:is( | ||||
| [href$=".jpg"i], | ||||
| [href$=".jpeg"i], | ||||
| [href$=".png"i], | ||||
| [href$=".gif"i], | ||||
| [href$=".webp"i], | ||||
| ):before { | ||||
|     content: '🎨'; | ||||
| } | ||||
							
								
								
									
										29
									
								
								docs/browser.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								docs/browser.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| html { | ||||
|     background: #333 url('/wp/wallhaven-mdjrqy.jpg') center / cover no-repeat fixed; | ||||
| } | ||||
| #files th { | ||||
|     background: rgba(32, 32, 32, 0.9) !important; | ||||
| } | ||||
| #ops, | ||||
| #treeul, | ||||
| #files td { | ||||
|     background: rgba(32, 32, 32, 0.3) !important; | ||||
| } | ||||
|  | ||||
|  | ||||
| html.light { | ||||
|     background: #eee url('/wp/wallhaven-dpxl6l.png') center / cover no-repeat fixed; | ||||
| } | ||||
| html.light #files th { | ||||
|     background: rgba(255, 255, 255, 0.9) !important; | ||||
| } | ||||
| html.light #ops, | ||||
| html.light #treeul, | ||||
| html.light #files td { | ||||
|     background: rgba(248, 248, 248, 0.8) !important; | ||||
| } | ||||
|  | ||||
|  | ||||
| #files * { | ||||
|     background: transparent !important; | ||||
| } | ||||
| @@ -10,19 +10,25 @@ u k:k | ||||
| # share "." (the current directory) | ||||
| # as "/" (the webroot) for the following users: | ||||
| # "r" grants read-access for anyone | ||||
| # "a ed" grants read-write to ed | ||||
| # "rw ed" grants read-write to ed | ||||
| . | ||||
| / | ||||
| r | ||||
| a ed | ||||
| rw ed | ||||
|  | ||||
| # custom permissions for the "priv" folder: | ||||
| # user "k" can see/read the contents | ||||
| # and "ed" gets read-write access | ||||
| # user "k" can only see/read the contents | ||||
| # user "ed" gets read-write access | ||||
| ./priv | ||||
| /priv | ||||
| r k | ||||
| a ed | ||||
| rw ed | ||||
|  | ||||
| # this does the same thing: | ||||
| ./priv | ||||
| /priv | ||||
| r ed k | ||||
| w ed | ||||
|  | ||||
| # share /home/ed/Music/ as /music and let anyone read it | ||||
| # (this will replace any folder called "music" in the webroot) | ||||
| @@ -32,10 +38,14 @@ r | ||||
|  | ||||
| # and a folder where anyone can upload | ||||
| # but nobody can see the contents | ||||
| # and set the e2d flag to enable the uploads database | ||||
| # and set the nodupe flag to reject duplicate uploads | ||||
| /home/ed/inc | ||||
| /dump | ||||
| w | ||||
| c e2d | ||||
| c nodupe | ||||
|  | ||||
| # this entire config file can be replaced with these arguments: | ||||
| # -u ed:123 -u k:k -v .::r:aed -v priv:priv:rk:aed -v /home/ed/Music:music:r -v /home/ed/inc:dump:w | ||||
| # -u ed:123 -u k:k -v .::r:a,ed -v priv:priv:r,k:rw,ed -v /home/ed/Music:music:r -v /home/ed/inc:dump:w:c,e2d:c,nodupe | ||||
| # but note that the config file always wins in case of conflicts | ||||
|   | ||||
							
								
								
									
										51
									
								
								docs/hls.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								docs/hls.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,51 @@ | ||||
| <!DOCTYPE html><html lang="en"><head> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>hls-test</title> | ||||
| 	<meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
| </head><body> | ||||
|  | ||||
| <video id="vid" controls></video> | ||||
| <script src="hls.light.js"></script> | ||||
| <script> | ||||
|  | ||||
| var video = document.getElementById('vid'); | ||||
| var hls = new Hls({ | ||||
| 	debug: true, | ||||
| 	autoStartLoad: false | ||||
| }); | ||||
| hls.loadSource('live/v.m3u8'); | ||||
| hls.attachMedia(video); | ||||
| hls.on(Hls.Events.MANIFEST_PARSED, function() { | ||||
| 	hls.startLoad(0); | ||||
| }); | ||||
| hls.on(Hls.Events.MEDIA_ATTACHED, function() { | ||||
| 	video.muted = true; | ||||
| 	video.play(); | ||||
| }); | ||||
|  | ||||
| /* | ||||
| general good news: | ||||
| - doesn't need fixed-length segments; ok to let x264 pick optimal keyframes and slice on those | ||||
| - hls.js polls the m3u8 for new segments, scales the duration accordingly, seeking works great | ||||
| - the sfx will grow by 66 KiB since that's how small hls.js can get, wait thats not good | ||||
|  | ||||
| # vod, creates m3u8 at the end, fixed keyframes, v bad | ||||
| ffmpeg -hide_banner -threads 0 -flags -global_header -i ..\CowboyBebopMovie-OP1.webm -vf scale=1280:-4,format=yuv420p -ac 2 -c:a libopus -b:a 128k -c:v libx264 -preset slow -crf 24 -maxrate:v 5M -bufsize:v 10M -g 120 -keyint_min 120 -sc_threshold 0 -hls_time 4 -hls_playlist_type vod -hls_segment_filename v%05d.ts v.m3u8 | ||||
|  | ||||
| # live, updates m3u8 as it goes, dynamic keyframes, streamable with hls.js | ||||
| ffmpeg -hide_banner -threads 0 -flags -global_header -i ..\..\CowboyBebopMovie-OP1.webm -vf scale=1280:-4,format=yuv420p -ac 2 -c:a libopus -b:a 128k -c:v libx264 -preset slow -crf 24 -maxrate:v 5M -bufsize:v 10M -f segment -segment_list v.m3u8 -segment_format mpegts -segment_list_flags live v%05d.ts | ||||
|  | ||||
| # fmp4 (fragmented mp4), doesn't work with hls.js, gets duratoin 149:07:51 (536871s), probably the tkhd/mdhd 0xffffffff (timebase 8000? ok) | ||||
| ffmpeg -re -hide_banner -threads 0 -flags +cgop -i ..\..\CowboyBebopMovie-OP1.webm -vf scale=1280:-4,format=yuv420p -ac 2 -c:a libopus -b:a 128k -c:v libx264 -preset slow -crf 24 -maxrate:v 5M -bufsize:v 10M -f segment -segment_list v.m3u8 -segment_format fmp4 -segment_list_flags live v%05d.mp4 | ||||
|  | ||||
| # try 2, works, uses tempfiles for m3u8 updates, good, 6% smaller | ||||
| ffmpeg -re -hide_banner -threads 0 -flags +cgop -i ..\..\CowboyBebopMovie-OP1.webm -vf scale=1280:-4,format=yuv420p -ac 2 -c:a libopus -b:a 128k -c:v libx264 -preset slow -crf 24 -maxrate:v 5M -bufsize:v 10M -f hls -hls_segment_type fmp4 -hls_list_size 0 -hls_segment_filename v%05d.mp4 v.m3u8 | ||||
|  | ||||
| more notes | ||||
| - adding -hls_flags single_file makes duration wack during playback (for both fmp4 and ts), ok once finalized and refreshed, gives no size reduction anyways | ||||
| - bebop op has good keyframe spacing for testing hls.js, in particular it hops one seg back and immediately resumes if it hits eof with the explicit hls.startLoad(0); otherwise it jumps into the middle of a seg and becomes art | ||||
| - can probably -c:v copy most of the time, is there a way to check for cgop? todo | ||||
|  | ||||
| */ | ||||
| </script> | ||||
| </body></html> | ||||
							
								
								
									
										32
									
								
								docs/minimal-up2k.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								docs/minimal-up2k.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| <!-- | ||||
|   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, #wrap>h2:last-child,  /* main tabs and navigators (tree/breadcrumbs) */ | ||||
|  | ||||
|     #u2cleanup, #u2conf tr:first-child>td[rowspan]:not(#u2btn_cw),  /* most of the config options */ | ||||
|  | ||||
|     #u2cards  /* 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: 3em} | ||||
|  | ||||
|     /* and embiggen the upload button */ | ||||
|     #u2conf #u2btn, #u2btn {padding:1.5em 0} | ||||
|  | ||||
|     /* adjust the button area a bit */ | ||||
|     #u2conf.has_btn {width: 35em !important; margin: 5em auto} | ||||
|  | ||||
| </style> | ||||
|  | ||||
| <a href="#" onclick="this.parentNode.innerHTML='';">show advanced options</a> | ||||
							
								
								
									
										242
									
								
								docs/music-analysis.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										242
									
								
								docs/music-analysis.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,242 @@ | ||||
| #!/bin/bash | ||||
| echo please dont actually run this as a scriopt | ||||
| exit 1 | ||||
|  | ||||
|  | ||||
| # dependency-heavy, not particularly good fit | ||||
| pacman -S llvm10 | ||||
| python3 -m pip install --user librosa | ||||
| git clone https://github.com/librosa/librosa.git | ||||
|  | ||||
|  | ||||
| # correct bpm for tracks with bad tags | ||||
| br=' | ||||
| /Trip Trip Trip\(Hardcore Edit\).mp3/ {v=176} | ||||
| /World!!.BIG_SOS/ {v=175} | ||||
| /\/08\..*\(BIG_SOS Bootleg\)\.mp3/ {v=175} | ||||
| /もってけ!セーラ服.Asterisk DnB/ {v=175} | ||||
| /Rondo\(Asterisk DnB Re.mp3/ {v=175} | ||||
| /Ray Nautica 175 Edit/ {v=175;x="thunk"} | ||||
| /TOKIMEKI Language.Jauz/ {v=174} | ||||
| /YUPPUN Hardcore Remix\).mp3/ {v=174;x="keeps drifting"} | ||||
| /(èâAâï.î╧ûδ|バーチャリアル.狐耶)J-Core Remix\).mp3/ {v=172;x="hard"} | ||||
| /lucky train..Freezer/ {v=170} | ||||
| /Alf zero Bootleg ReMix/ {v=170} | ||||
| /Prisoner of Love.Kacky/ {v=170} | ||||
| /火炎 .Qota/ {v=170} | ||||
| /\(hu-zin Bootleg\)\.mp3/ {v=170} | ||||
| /15. STRAIGHT BET\(Milynn Bootleg\)\.mp3/ {v=170} | ||||
| /\/13.*\(Milynn Bootleg\)\.mp3/ {v=167;x="way hard"} | ||||
| /COLOR PLANET .10SAI . nijikon Remix\)\.mp3/ {v=165} | ||||
| /11\. (朝はご飯派|Æ⌐é═é▓ö╤öh)\.mp3/ {v=162} | ||||
| /09\. Where.s the core/ {v=160} | ||||
| /PLANET\(Koushif Jersey Club Bootleg\)remaster.mp3/ {v=160;x="starts ez turns bs"} | ||||
| /kened Soul - Madeon x Angel Beats!.mp3/ {v=160} | ||||
| /Dear Moments\(Mother Harlot Bootleg\)\.mp3/ {v=150} | ||||
| /POWER.Ringos UKG/ {v=140} | ||||
| /ブルー・フィールド\(Ringos UKG Remix\).mp3/ {v=135} | ||||
| /プラチナジェット.Ringo Remix..mp3/ {v=131.2} | ||||
| /Mirrorball Love \(TKM Bootleg Mix\).mp3/ {v=130} | ||||
| /Photon Melodies \(TKM Bootleg Mix\).mp3/ {v=128} | ||||
| /Trap of Love \(TKM Bootleg Mix\).mp3/ {v=128} | ||||
| /One Step \(TKM Bootleg Mix\)\.mp3/ {v=126} | ||||
| /04 (トリカムイ岩|âgâèâJâÇâCèΓ).mp3/ {v=125} | ||||
| /Get your Wish \(NAWN REMIX\)\.mp3/ {v=95} | ||||
| /Flicker .Nitro Fun/ {v=92} | ||||
| /\/14\..*suicat Remix/ {v=85.5;x="tricky"} | ||||
| /Yanagi Nagi - Harumodoki \(EO Remix\)\.mp3/ {v=150} | ||||
| /Azure - Nicology\.mp3/ {v=128;x="off by 5 how"} | ||||
| ' | ||||
|  | ||||
|  | ||||
| # afun host, collects/grades the results | ||||
| runfun() { cores=8; touch run; rm -f /dev/shm/mres.*; t00=$(date +%s); tbc() { bc | sed -r 's/(\.[0-9]{2}).*/\1/'; }; for ((core=0; core<$cores; core++)); do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db 'select dur.w, dur.v, bpm.v from mt bpm join mt dur on bpm.w = dur.w where bpm.k = ".bpm" and dur.k = ".dur" order by dur.w' | uniq -w16 | while IFS=\| read w dur bpm; do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db "select rd, fn from up where substr(w,1,16) = '$w'" | sed -r "s/^/$bpm /"; done | grep mir/cr | tr \| / | awk '{v=$1;sub(/[^ ]+ /,"")} '"$br"' {printf "%s %s\n",v,$0}' | while read bpm fn; do [ -e run ] || break; n=$((n+1)); ncore=$((n%cores)); [ $ncore -eq $core ] || continue; t0=$(date +%s.%N); (afun || exit 1; t=$(date +%s.%N); td=$(echo "scale=3; $t - $t0" | tbc); bd=$(echo "scale=3; $bpm / $py" | tbc); printf '%4s sec, %4s orig, %6s py, %4s div, %s\n' $td $bpm $py $bd "$fn") | tee -a /dev/shm/mres.$ncore; rv=${PIPESTATUS[0]}; [ $rv -eq 0 ] || { echo "FAULT($rv): $fn"; }; done & done; wait 2>/dev/null; cat /dev/shm/mres.* | awk 'function prt(c) {printf "\033[3%sm%s\033[0m\n",c,$0} $8!="div,"{next} $5!~/^[0-9\.]+/{next} {meta=$3;det=$5;div=meta/det} div<0.7{det/=2} div>1.3{det*=2} {idet=sprintf("%.0f",det)} {idiff=idet-meta} meta>idet{idiff=meta-idet} idiff==0{n0++;prt(6);next} idiff==1{n1++;prt(3);next} idiff>10{nx++;prt(1);next} {n10++;prt(5)} END {printf "ok: %d   1off: %2s   (%3s)   10off: %2s   (%3s)   fail: %2s\n",n0,n1,n0+n1,n10,n0+n1+n10,nx}'; te=$(date +%s); echo $((te-t00)) sec spent; } | ||||
|  | ||||
|  | ||||
| # ok:   8   1off: 62   ( 70)   10off: 86   (156)   fail: 25   # 105 sec,  librosa @ 8c archvm on 3700x w10 | ||||
| # ok:   4   1off: 59   ( 63)   10off: 65   (128)   fail: 53   # using original tags (bad) | ||||
| afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -t 60 /dev/shm/$core.wav || return 1; py="$(/home/ed/src/librosa/examples/beat_tracker.py /dev/shm/$core.wav x 2>&1 | awk 'BEGIN {v=1} /^Estimated tempo: /{v=$3} END {print v}')"; } runfun | ||||
|  | ||||
|  | ||||
| # ok: 119   1off:  5   (124)   10off:  8   (132)   fail: 49   # 51 sec,  vamp-example-fixedtempo | ||||
| # ok: 109   1off:  4   (113)   10off:  9   (122)   fail: 59   # bad-tags | ||||
| afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 22050 -f f32le /dev/shm/$core.pcm || return 1; py="$(python3 -c 'import vamp; import numpy as np; f = open("/dev/shm/'$core'.pcm", "rb"); d = np.fromfile(f, dtype=np.float32); c = vamp.collect(d, 22050, "vamp-example-plugins:fixedtempo", parameters={"maxdflen":40}); print(c["list"][0]["label"].split(" ")[0])')"; }; runfun | ||||
|  | ||||
|  | ||||
| # ok: 102   1off: 61   (163)   10off: 12   (175)   fail:  6   # 61 sec,  vamp-qm-tempotracker | ||||
| # ok:  80   1off: 48   (128)   10off: 11   (139)   fail: 42   # bad-tags | ||||
| afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 22050 -f f32le /dev/shm/$core.pcm || return 1; py="$(python3 -c 'import vamp; import numpy as np; f = open("/dev/shm/'$core'.pcm", "rb"); d = np.fromfile(f, dtype=np.float32); c = vamp.collect(d, 22050, "qm-vamp-plugins:qm-tempotracker", parameters={"inputtempo":150}); v = [float(x["label"].split(" ")[0]) for x in c["list"] if x["label"]]; v = list(sorted(v))[len(v)//4:-len(v)//4]; print(round(sum(v) / len(v), 1))')"; }; runfun | ||||
|  | ||||
|  | ||||
| # ok: 133   1off: 32   (165)   10off: 12   (177)   fail:  3   # 51 sec,  vamp-beatroot | ||||
| # ok: 101   1off: 22   (123)   10off: 16   (139)   fail: 39   # bad-tags | ||||
| # note: some tracks fully fail to analyze (unlike the others which always provide a guess) | ||||
| afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 22050 -f f32le /dev/shm/$core.pcm || return 1; py="$(python3 -c 'import vamp; import numpy as np; f = open("/dev/shm/'$core'.pcm", "rb"); d = np.fromfile(f, dtype=np.float32); c = vamp.collect(d, 22050, "beatroot-vamp:beatroot"); cl=c["list"]; print(round(60*((len(cl)-1)/(float(cl[-1]["timestamp"]-cl[1]["timestamp"]))), 2))')"; }; runfun | ||||
|  | ||||
|  | ||||
| # ok: 124   1off:  9   (133)   10off: 40   (173)   fail:  8   # 231 sec,  essentia/full | ||||
| # ok: 109   1off:  8   (117)   10off: 22   (139)   fail: 42   # bad-tags | ||||
| afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 /dev/shm/$core.wav || return 1; py="$(python3 -c 'import essentia; import essentia.standard as es; fe, fef = es.MusicExtractor(lowlevelStats=["mean", "stdev"], rhythmStats=["mean", "stdev"], tonalStats=["mean", "stdev"])("/dev/shm/'$core'.wav"); print("{:.2f}".format(fe["rhythm.bpm"]))')"; }; runfun | ||||
|  | ||||
|  | ||||
| # ok: 113   1off: 18   (131)   10off: 46   (177)   fail:  4   # 134 sec,  essentia/re2013 | ||||
| # ok: 101   1off: 15   (116)   10off: 26   (142)   fail: 39   # bad-tags | ||||
| afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 /dev/shm/$core.wav || return 1; py="$(python3 -c 'from essentia.standard import *; a=MonoLoader(filename="/dev/shm/'$core'.wav")(); bpm,beats,confidence,_,intervals=RhythmExtractor2013(method="multifeature")(a); print("{:.2f}".format(bpm))')"; }; runfun | ||||
|  | ||||
|  | ||||
|  | ||||
| ######################################################################## | ||||
| ## | ||||
| ##  key detectyion | ||||
| ## | ||||
| ######################################################################## | ||||
|  | ||||
|  | ||||
|  | ||||
| # console scriptlet reusing keytabs from browser.js | ||||
| var m=''; for (var a=0; a<24; a++) m += 's/\\|(' + maps["traktor_sharps"][a].trim() + "|" + maps["rekobo_classic"][a].trim() + "|" + maps["traktor_musical"][a].trim() + "|" + maps["traktor_open"][a].trim() + ')$/|' + maps["rekobo_alnum"][a].trim() + '/;'; console.log(m); | ||||
|  | ||||
|  | ||||
| # translate to camelot | ||||
| re='s/\|(B|B|B|6d)$/|1B/;s/\|(F#|F#|Gb|7d)$/|2B/;s/\|(C#|Db|Db|8d)$/|3B/;s/\|(G#|Ab|Ab|9d)$/|4B/;s/\|(D#|Eb|Eb|10d)$/|5B/;s/\|(A#|Bb|Bb|11d)$/|6B/;s/\|(F|F|F|12d)$/|7B/;s/\|(C|C|C|1d)$/|8B/;s/\|(G|G|G|2d)$/|9B/;s/\|(D|D|D|3d)$/|10B/;s/\|(A|A|A|4d)$/|11B/;s/\|(E|E|E|5d)$/|12B/;s/\|(G#m|Abm|Abm|6m)$/|1A/;s/\|(D#m|Ebm|Ebm|7m)$/|2A/;s/\|(A#m|Bbm|Bbm|8m)$/|3A/;s/\|(Fm|Fm|Fm|9m)$/|4A/;s/\|(Cm|Cm|Cm|10m)$/|5A/;s/\|(Gm|Gm|Gm|11m)$/|6A/;s/\|(Dm|Dm|Dm|12m)$/|7A/;s/\|(Am|Am|Am|1m)$/|8A/;s/\|(Em|Em|Em|2m)$/|9A/;s/\|(Bm|Bm|Bm|3m)$/|10A/;s/\|(F#m|F#m|Gbm|4m)$/|11A/;s/\|(C#m|Dbm|Dbm|5m)$/|12A/;' | ||||
|  | ||||
|  | ||||
| # runner/wrapper | ||||
| runfun() { cores=8; touch run; tbc() { bc | sed -r 's/(\.[0-9]{2}).*/\1/'; }; for ((core=0; core<$cores; core++)); do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db 'select dur.w, dur.v, key.v from mt key join mt dur on key.w = dur.w where key.k = "key" and dur.k = ".dur" order by dur.w' | uniq -w16 | grep -vE '(Off-Key|None)$' | sed -r "s/ //g;$re" | uniq -w16 | while IFS=\| read w dur bpm; do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db "select rd, fn from up where substr(w,1,16) = '$w'" | sed -r "s/^/$bpm /"; done| grep mir/cr | tr \| / | while read key fn; do [ -e run ] || break; n=$((n+1)); ncore=$((n%cores)); [ $ncore -eq $core ] || continue; t0=$(date +%s.%N); (afun || exit 1; t=$(date +%s.%N); td=$(echo "scale=3; $t - $t0" | tbc); [ "$key" = "$py" ] && c=2 || c=5; printf '%4s sec, %4s orig, \033[3%dm%4s py,\033[0m %s\n' $td "$key" $c "$py" "$fn") || break; done & done; time wait 2>/dev/null; } | ||||
|  | ||||
|  | ||||
| # ok: 26   1off: 10   2off: 1   fail: 3   #  15 sec, keyfinder | ||||
| afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 -t 60 /dev/shm/$core.wav || break; py="$(python3 -c 'import sys; import keyfinder; print(keyfinder.key(sys.argv[1]).camelot())' "/dev/shm/$core.wav")"; }; runfun | ||||
|  | ||||
|  | ||||
| # https://github.com/MTG/essentia/raw/master/src/examples/tutorial/example_key_by_steps_streaming.py | ||||
| # https://essentia.upf.edu/reference/std_Key.html  # edma edmm braw bgate | ||||
| sed -ri 's/^(key = Key\().*/\1profileType="bgate")/' example_key_by_steps_streaming.py | ||||
| afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 -t 60 /dev/shm/$core.wav || break; py="$(python3 example_key_by_steps_streaming.py /dev/shm/$core.{wav,yml} 2>/dev/null | sed -r "s/ major//;s/ minor/m/;s/^/|/;$re;s/.//")"; }; runfun | ||||
|  | ||||
|  | ||||
|  | ||||
| ######################################################################## | ||||
| ## | ||||
| ##  misc | ||||
| ## | ||||
| ######################################################################## | ||||
|  | ||||
|  | ||||
|  | ||||
| python3 -m pip install --user vamp | ||||
|  | ||||
| import librosa | ||||
| d, r = librosa.load('/dev/shm/0.wav') | ||||
| d.dtype | ||||
| # dtype('float32') | ||||
| d.shape | ||||
| # (1323000,) | ||||
| d | ||||
| # array([-1.9614939e-08,  1.8037968e-08, -1.4106059e-08, ..., | ||||
| #         1.2024145e-01,  2.7462116e-01,  1.6202132e-01], dtype=float32) | ||||
|  | ||||
|  | ||||
|  | ||||
| import vamp | ||||
| c = vamp.collect(d, r, "vamp-example-plugins:fixedtempo") | ||||
| c | ||||
| # {'list': [{'timestamp':  0.005804988, 'duration':  9.999092971, 'label': '110.0 bpm', 'values': array([109.98116], dtype=float32)}]} | ||||
|  | ||||
|  | ||||
|  | ||||
| ffmpeg -ss 48 -i /mnt/Users/ed/Music/mir/cr-a/'I Beg You(ths Bootleg).wav' -ac 1 -ar 22050 -f f32le -t 60 /dev/shm/f32.pcm | ||||
|  | ||||
| import numpy as np | ||||
| f = open('/dev/shm/f32.pcm', 'rb') | ||||
| d = np.fromfile(f, dtype=np.float32) | ||||
| d | ||||
| array([-0.17803933, -0.27206388, -0.41586545, ..., -0.04940119, | ||||
|        -0.0267825 , -0.03564296], dtype=float32) | ||||
|  | ||||
| d = np.reshape(d, [1, -1]) | ||||
| d | ||||
| array([[-0.17803933, -0.27206388, -0.41586545, ..., -0.04940119, | ||||
|         -0.0267825 , -0.03564296]], dtype=float32) | ||||
|  | ||||
|  | ||||
|  | ||||
| import vampyhost | ||||
| print("\n".join(vampyhost.list_plugins())) | ||||
|  | ||||
| mvamp:marsyas_bextract_centroid | ||||
| mvamp:marsyas_bextract_lpcc | ||||
| mvamp:marsyas_bextract_lsp | ||||
| mvamp:marsyas_bextract_mfcc | ||||
| mvamp:marsyas_bextract_rolloff | ||||
| mvamp:marsyas_bextract_scf | ||||
| mvamp:marsyas_bextract_sfm | ||||
| mvamp:marsyas_bextract_zero_crossings | ||||
| mvamp:marsyas_ibt | ||||
| mvamp:zerocrossing | ||||
| qm-vamp-plugins:qm-adaptivespectrogram | ||||
| qm-vamp-plugins:qm-barbeattracker | ||||
| qm-vamp-plugins:qm-chromagram | ||||
| qm-vamp-plugins:qm-constantq | ||||
| qm-vamp-plugins:qm-dwt | ||||
| qm-vamp-plugins:qm-keydetector | ||||
| qm-vamp-plugins:qm-mfcc | ||||
| qm-vamp-plugins:qm-onsetdetector | ||||
| qm-vamp-plugins:qm-segmenter | ||||
| qm-vamp-plugins:qm-similarity | ||||
| qm-vamp-plugins:qm-tempotracker | ||||
| qm-vamp-plugins:qm-tonalchange | ||||
| qm-vamp-plugins:qm-transcription | ||||
| vamp-aubio:aubiomelenergy | ||||
| vamp-aubio:aubiomfcc | ||||
| vamp-aubio:aubionotes | ||||
| vamp-aubio:aubioonset | ||||
| vamp-aubio:aubiopitch | ||||
| vamp-aubio:aubiosilence | ||||
| vamp-aubio:aubiospecdesc | ||||
| vamp-aubio:aubiotempo | ||||
| vamp-example-plugins:amplitudefollower | ||||
| vamp-example-plugins:fixedtempo | ||||
| vamp-example-plugins:percussiononsets | ||||
| vamp-example-plugins:powerspectrum | ||||
| vamp-example-plugins:spectralcentroid | ||||
| vamp-example-plugins:zerocrossing | ||||
| vamp-rubberband:rubberband | ||||
|  | ||||
|  | ||||
|  | ||||
| plug = vampyhost.load_plugin("vamp-example-plugins:fixedtempo", 22050, 0) | ||||
| plug.info | ||||
| {'apiVersion': 2, 'pluginVersion': 1, 'identifier': 'fixedtempo', 'name': 'Simple Fixed Tempo Estimator', 'description': 'Study a short section of audio and estimate its tempo, assuming the tempo is constant', 'maker': 'Vamp SDK Example Plugins', 'copyright': 'Code copyright 2008 Queen Mary, University of London.  Freely redistributable (BSD license)'} | ||||
| plug = vampyhost.load_plugin("qm-vamp-plugins:qm-tempotracker", 22050, 0) | ||||
| from pprint import pprint; pprint(plug.parameters) | ||||
|  | ||||
|  | ||||
|  | ||||
| for c in plug.parameters: print("{} \033[36m{}  [\033[33m{}\033[36m] = {}\033[0m".format(c["identifier"], c["name"], "\033[36m, \033[33m".join(c["valueNames"]), c["valueNames"][int(c["defaultValue"])])) if "valueNames" in c else print("{} \033[36m{}  [\033[33m{}..{}\033[36m] = {}\033[0m".format(c["identifier"], c["name"], c["minValue"], c["maxValue"], c["defaultValue"])) | ||||
|  | ||||
|  | ||||
|  | ||||
| beatroot-vamp:beatroot | ||||
| cl=c["list"]; 60*((len(cl)-1)/(float(cl[-1]["timestamp"]-cl[1]["timestamp"]))) | ||||
|  | ||||
|  | ||||
|  | ||||
| ffmpeg -ss 48 -i /mnt/Users/ed/Music/mir/cr-a/'I Beg You(ths Bootleg).wav' -ac 1 -ar 22050 -f f32le -t 60 /dev/shm/f32.pcm | ||||
| # 128 bpm, key 5A Cm | ||||
|  | ||||
| import vamp | ||||
| import numpy as np | ||||
| f = open('/dev/shm/f32.pcm', 'rb') | ||||
| d = np.fromfile(f, dtype=np.float32) | ||||
| c = vamp.collect(d, 22050, "vamp-example-plugins:fixedtempo", parameters={"maxdflen":40}) | ||||
| c["list"][0]["label"] | ||||
| # 127.6 bpm | ||||
|  | ||||
| c = vamp.collect(d, 22050, "qm-vamp-plugins:qm-tempotracker", parameters={"inputtempo":150}) | ||||
| print("\n".join([v["label"] for v in c["list"] if v["label"]])) | ||||
| v = [float(x["label"].split(' ')[0]) for x in c["list"] if x["label"]] | ||||
| v = list(sorted(v))[len(v)//4:-len(v)//4] | ||||
| v = sum(v) / len(v) | ||||
| # 128.1 bpm | ||||
|  | ||||
							
								
								
									
										141
									
								
								docs/notes.sh
									
									
									
									
									
								
							
							
						
						
									
										141
									
								
								docs/notes.sh
									
									
									
									
									
								
							| @@ -3,6 +3,21 @@ echo not a script | ||||
| exit 1 | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## delete all partial uploads | ||||
| ##  (supports linux/macos, probably windows+msys2) | ||||
|  | ||||
| gzip -d < .hist/up2k.snap | jq -r '.[].tnam' | while IFS= read -r f; do rm -f -- "$f"; done | ||||
| gzip -d < .hist/up2k.snap | jq -r '.[].name' | while IFS= read -r f; do wc -c -- "$f" | grep -qiE '^[^0-9a-z]*0' && rm -f -- "$f"; done | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## detect partial uploads based on file contents | ||||
| ##  (in case of context loss or old copyparties) | ||||
|  | ||||
| echo; find -type f | while IFS= read -r x; do printf '\033[A\033[36m%s\033[K\033[0m\n' "$x"; tail -c$((1024*1024)) <"$x" | xxd -a | awk 'NR==1&&/^[0: ]+.{16}$/{next} NR==2&&/^\*$/{next} NR==3&&/^[0f]+: [0 ]+65 +.{16}$/{next} {e=1} END {exit e}' || continue; printf '\033[A\033[31msus:\033[33m %s \033[0m\n\n' "$x"; done | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## create a test payload | ||||
|  | ||||
| @@ -13,7 +28,7 @@ head -c $((2*1024*1024*1024)) /dev/zero | openssl enc -aes-256-ctr -pass pass:hu | ||||
| ## testing multiple parallel uploads | ||||
| ## usage:  para | tee log | ||||
|  | ||||
| para() { for s in 1 2 3 4 5 6 7 8 12 16 24 32 48 64; do echo $s; for r in {1..4}; do for ((n=0;n<s;n++)); do curl -sF "act=bput" -F "f=@garbage.file" http://127.0.0.1:1234/ 2>&1 & done; wait; echo; done; done; } | ||||
| para() { for s in 1 2 3 4 5 6 7 8 12 16 24 32 48 64; do echo $s; for r in {1..4}; do for ((n=0;n<s;n++)); do curl -sF "act=bput" -F "f=@garbage.file" http://127.0.0.1:3923/ 2>&1 & done; wait; echo; done; done; } | ||||
|  | ||||
|  | ||||
| ## | ||||
| @@ -29,20 +44,20 @@ avg() { awk 'function pr(ncsz) {if (nsmp>0) {printf "%3s %s\n", csz, sum/nsmp} c | ||||
| dirs=("$HOME/vfs/ほげ" "$HOME/vfs/ほげ/ぴよ" "$HOME/vfs/$(printf \\xed\\x91)" "$HOME/vfs/$(printf \\xed\\x91/\\xed\\x92)") | ||||
| mkdir -p "${dirs[@]}" | ||||
| for dir in "${dirs[@]}"; do for fn in ふが "$(printf \\xed\\x93)" 'qwe,rty;asd fgh+jkl%zxc&vbn <qwe>"rty'"'"'uio&asd fgh'; do echo "$dir" > "$dir/$fn.html"; done; done | ||||
|  | ||||
| # qw er+ty%20ui%%20op<as>df&gh&jk#zx'cv"bn`m=qw*er^ty?ui@op,as.df-gh_jk | ||||
|  | ||||
| ## | ||||
| ## upload mojibake | ||||
|  | ||||
| fn=$(printf '\xba\xdc\xab.cab') | ||||
| echo asdf > "$fn" | ||||
| curl --cookie cppwd=wark -sF "act=bput" -F "f=@$fn" http://127.0.0.1:1234/moji/%ED%91/ | ||||
| curl --cookie cppwd=wark -sF "act=bput" -F "f=@$fn" http://127.0.0.1:3923/moji/%ED%91/ | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## test compression | ||||
|  | ||||
| wget -S --header='Accept-Encoding: gzip' -U 'MSIE 6.0; SV1' http://127.0.0.1:1234/.cpr/deps/ogv.js -O- | md5sum; p=~ed/dev/copyparty/copyparty/web/deps/ogv.js.gz; md5sum $p; gzip -d < $p | md5sum | ||||
| wget -S --header='Accept-Encoding: gzip' -U 'MSIE 6.0; SV1' http://127.0.0.1:3923/.cpr/deps/ogv.js -O- | md5sum; p=~ed/dev/copyparty/copyparty/web/deps/ogv.js.gz; md5sum $p; gzip -d < $p | md5sum | ||||
|  | ||||
|  | ||||
| ## | ||||
| @@ -52,6 +67,62 @@ wget -S --header='Accept-Encoding: gzip' -U 'MSIE 6.0; SV1' http://127.0.0.1:123 | ||||
| shab64() { sp=$1; f="$2"; v=0; sz=$(stat -c%s "$f"); while true; do w=$((v+sp*1024*1024)); printf $(tail -c +$((v+1)) "$f" | head -c $((w-v)) | sha512sum | cut -c-64 | sed -r 's/ .*//;s/(..)/\\x\1/g') | base64 -w0 | cut -c-43 | tr '+/' '-_'; v=$w; [ $v -lt $sz ] || break; done; } | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## poll url for performance issues | ||||
|  | ||||
| command -v gdate && date() { gdate "$@"; }; while true; do t=$(date +%s.%N); (time wget http://127.0.0.1:3923/?ls -qO- | jq -C '.files[]|{sz:.sz,ta:.tags.artist,tb:.tags.".bpm"}|del(.[]|select(.==null))' | awk -F\" '/"/{t[$2]++} END {for (k in t){v=t[k];p=sprintf("%" (v+1) "s",v);gsub(/ /,"#",p);printf "\033[36m%s\033[33m%s   ",k,p}}') 2>&1 | awk -v ts=$t 'NR==1{t1=$0} NR==2{sub(/.*0m/,"");sub(/s$/,"");t2=$0;c=2; if(t2>0.3){c=3} if(t2>0.8){c=1} } END{sub(/[0-9]{6}$/,"",ts);printf "%s   \033[3%dm%s   %s\033[0m\n",ts,c,t2,t1}'; sleep 0.1 || break; done | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## js oneliners | ||||
|  | ||||
| # get all up2k search result URLs | ||||
| var t=[]; var b=document.location.href.split('#')[0].slice(0, -1); document.querySelectorAll('#u2tab .prog a').forEach((x) => {t.push(b+encodeURI(x.getAttribute("href")))}); console.log(t.join("\n")); | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## bash oneliners | ||||
|  | ||||
| # get the size and video-id of all youtube vids in folder, assuming filename ends with -id.ext, and create a copyparty search query | ||||
| find -maxdepth 1 -printf '%s %p\n' | sort -n | awk '!/-([0-9a-zA-Z_-]{11})\.(mkv|mp4|webm)$/{next} {sub(/\.[^\.]+$/,"");n=length($0);v=substr($0,n-10);print $1, v}' | tee /dev/stderr | awk 'BEGIN {p="("} {printf("%s name like *-%s.* ",p,$2);p="or"} END {print ")\n"}' | cat >&2 | ||||
|  | ||||
| # unique stacks in a stackdump | ||||
| f=a; rm -rf stacks; mkdir stacks; grep -E '^#' $f | while IFS= read -r n; do awk -v n="$n" '!$0{o=0} o; $0==n{o=1}' <$f >stacks/f; h=$(sha1sum <stacks/f | cut -c-16); mv stacks/f stacks/$h-"$n"; done ; find stacks/ | sort | uniq -cw24 | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## sqlite3 stuff | ||||
|  | ||||
| # find dupe metadata keys | ||||
| sqlite3 up2k.db 'select mt1.w, mt1.k, mt1.v, mt2.v from mt mt1 inner join mt mt2 on mt1.w = mt2.w where mt1.k = mt2.k and mt1.rowid != mt2.rowid' | ||||
|  | ||||
| # partial reindex by deleting all tags for a list of files | ||||
| time sqlite3 up2k.db 'select mt1.w from mt mt1 inner join mt mt2 on mt1.w = mt2.w where mt1.k = +mt2.k and mt1.rowid != mt2.rowid'  > warks | ||||
| cat warks | while IFS= read -r x; do sqlite3 up2k.db "delete from mt where w = '$x'"; done | ||||
|  | ||||
| # dump all dbs | ||||
| find -iname up2k.db | while IFS= read -r x; do sqlite3 "$x" 'select substr(w,1,12), rd, fn from up' | sed -r 's/\|/ \| /g' | while IFS= read -r y; do printf '%s | %s\n' "$x" "$y"; done; done | ||||
|  | ||||
| # unschedule mtp scan for all files somewhere under "enc/" | ||||
| sqlite3 -readonly up2k.db 'select substr(up.w,1,16) from up inner join mt on mt.w = substr(up.w,1,16) where rd like "enc/%" and +mt.k = "t:mtp"' > keys; awk '{printf "delete from mt where w = \"%s\" and +k = \"t:mtp\";\n", $0}' <keys | tee /dev/stderr | sqlite3 up2k.db | ||||
|  | ||||
| # compare metadata key "key" between two databases | ||||
| sqlite3 -readonly up2k.db.key-full 'select w, v from mt where k = "key" order by w' > k1; sqlite3 -readonly up2k.db 'select w, v from mt where k = "key" order by w' > k2; ok=0; ng=0; while IFS='|' read w k2; do k1="$(grep -E "^$w" k1 | sed -r 's/.*\|//')"; [ "$k1" = "$k2" ] && ok=$((ok+1)) || { ng=$((ng+1)); printf '%3s %3s  %s\n' "$k1" "$k2" "$(sqlite3 -readonly up2k.db.key-full "select * from up where substr(w,1,16) = '$w'" | sed -r 's/\|/ | /g')"; }; done < <(cat k2); echo "match $ok   diff $ng" | ||||
|  | ||||
| # actually this is much better | ||||
| sqlite3 -readonly up2k.db.key-full 'select w, v from mt where k = "key" order by w' > k1; sqlite3 -readonly up2k.db 'select mt.w, mt.v, up.rd, up.fn from mt inner join up on mt.w = substr(up.w,1,16) where mt.k = "key" order by up.rd, up.fn' > k2; ok=0; ng=0; while IFS='|' read w k2 path; do k1="$(grep -E "^$w" k1 | sed -r 's/.*\|//')"; [ "$k1" = "$k2" ] && ok=$((ok+1)) || { ng=$((ng+1)); printf '%3s %3s  %s\n' "$k1" "$k2" "$path"; }; done < <(cat k2); echo "match $ok   diff $ng" | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## media | ||||
|  | ||||
| # split track into test files | ||||
| e=6; s=10; d=~/dev/copyparty/srv/aus; n=1; p=0; e=$((e*60)); rm -rf $d; mkdir $d; while true; do ffmpeg -hide_banner -ss $p -i 'nervous_testpilot - office.mp3' -c copy -t $s $d/$(printf %04d $n).mp3; n=$((n+1)); p=$((p+s)); [ $p -gt $e ] && break; done | ||||
|  | ||||
| -v srv/aus:aus:r:ce2dsa:ce2ts:cmtp=fgsfds=bin/mtag/sleep.py | ||||
| sqlite3 .hist/up2k.db 'select * from mt where k="fgsfds" or k="t:mtp"' | tee /dev/stderr | wc -l | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## vscode | ||||
|  | ||||
| @@ -80,3 +151,65 @@ for d in /usr /var; do find $d -type f -size +30M 2>/dev/null; done | while IFS= | ||||
| # py2 on osx | ||||
| brew install python@2 | ||||
| pip install virtualenv | ||||
|  | ||||
| # readme toc | ||||
| cat README.md | awk '!/^#/{next} {lv=length($1);sub(/[^ ]+ /,"");bab=$0;gsub(/ /,"-",bab)} {printf "%" ((lv-1)*4+1) "s [%s](#%s)\n", "*",$0,bab}' | ||||
|  | ||||
| # fix firefox phantom breakpoints, | ||||
| # suggestions from bugtracker, doesnt work (debugger is not attachable) | ||||
| devtools settings >> advanced >> enable browser chrome debugging + enable remote debugging | ||||
| burger > developer >> browser toolbox  (ctrl-alt-shift-i) | ||||
| iframe btn topright >> chrome://devtools/content/debugger/index.html | ||||
| dbg.asyncStore.pendingBreakpoints = {} | ||||
|  | ||||
| # fix firefox phantom breakpoints | ||||
| about:config >> devtools.debugger.prefs-schema-version = -1 | ||||
|  | ||||
| # determine server version | ||||
| git pull; git reset --hard origin/HEAD && git log --format=format:"%H %ai %d" --decorate=full > ../revs && cat ../{util,browser,up2k}.js >../vr && cat ../revs | while read -r rev extra; do (git reset --hard $rev >/dev/null 2>/dev/null && dsz=$(cat copyparty/web/{util,browser,up2k}.js >../vg 2>/dev/null && diff -wNarU0 ../{vg,vr} | wc -c) && printf '%s %6s %s\n' "$rev" $dsz "$extra") </dev/null; done                 | ||||
|  | ||||
| # download all sfx versions | ||||
| curl https://api.github.com/repos/9001/copyparty/releases?per_page=100 | jq -r '.[] | .tag_name + " " + .name' | while read v t; do fn="copyparty $v $t.py"; [ -e $fn ] || curl https://github.com/9001/copyparty/releases/download/$v/copyparty-sfx.py -Lo "$fn"; done | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## http 206 | ||||
|  | ||||
| # az = abcdefghijklmnopqrstuvwxyz | ||||
|  | ||||
| printf '%s\r\n' 'GET /az HTTP/1.1' 'Host: ocv.me' 'Range: bytes=5-10' '' | ncat ocv.me 80  | ||||
| # Content-Range: bytes 5-10/26 | ||||
| # Content-Length: 6 | ||||
| # fghijk | ||||
|  | ||||
| Range: bytes=0-1    "ab" Content-Range: bytes 0-1/26 | ||||
| Range: bytes=24-24  "y"  Content-Range: bytes 24-24/26 | ||||
| Range: bytes=24-25  "yz" Content-Range: bytes 24-25/26 | ||||
| Range: bytes=24-    "yz" Content-Range: bytes 24-25/26 | ||||
| Range: bytes=25-29  "z"  Content-Range: bytes 25-25/26 | ||||
| Range: bytes=26-         Content-Range: bytes */26 | ||||
|   HTTP/1.1 416 Requested Range Not Satisfiable | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## md perf | ||||
|  | ||||
| var tsh = []; | ||||
| function convert_markdown(md_text, dest_dom) { | ||||
|     tsh.push(Date.now()); | ||||
|     while (tsh.length > 10) | ||||
|         tsh.shift(); | ||||
|     if (tsh.length > 1) { | ||||
|         var end = tsh.slice(-2); | ||||
|         console.log("render", end.pop() - end.pop(), (tsh[tsh.length - 1] - tsh[0]) / (tsh.length - 1)); | ||||
|     } | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## tmpfiles.d meme | ||||
|  | ||||
| mk() { rm -rf /tmp/foo; sudo -u ed bash -c 'mkdir /tmp/foo; echo hi > /tmp/foo/bar'; } | ||||
| mk && t0="$(date)" && while true; do date -s "$(date '+ 1 hour')"; systemd-tmpfiles --clean; ls -1 /tmp | grep foo || break; done; echo "$t0" | ||||
| mk && sudo -u ed flock /tmp/foo sleep 40 & sleep 1; ps aux | grep -E 'sleep 40$' && t0="$(date)" && for n in {1..40}; do date -s "$(date '+ 1 day')"; systemd-tmpfiles --clean; ls -1 /tmp | grep foo || break; done; echo "$t0" | ||||
| mk && t0="$(date)" && for n in {1..40}; do date -s "$(date '+ 1 day')"; systemd-tmpfiles --clean; ls -1 /tmp | grep foo || break; tar -cf/dev/null /tmp/foo; done; echo "$t0" | ||||
|  | ||||
|   | ||||
							
								
								
									
										82
									
								
								docs/nuitka.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								docs/nuitka.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | ||||
| # recipe for building an exe with nuitka (extreme jank edition) | ||||
| # | ||||
| # NOTE: win7 and win10 builds both work on win10 but | ||||
| #   on win7 they immediately c0000005 in kernelbase.dll | ||||
| # | ||||
| # first install python-3.6.8-amd64.exe | ||||
| #   [x] add to path | ||||
| # | ||||
| # copypaste the rest of this file into cmd | ||||
|  | ||||
| rem from pypi | ||||
| cd \users\ed\downloads | ||||
| python -m pip install --user Nuitka-0.6.14.7.tar.gz | ||||
|  | ||||
| rem https://github.com/brechtsanders/winlibs_mingw/releases/download/10.2.0-11.0.0-8.0.0-r5/winlibs-x86_64-posix-seh-gcc-10.2.0-llvm-11.0.0-mingw-w64-8.0.0-r5.zip | ||||
| mkdir C:\Users\ed\AppData\Local\Nuitka\ | ||||
| mkdir C:\Users\ed\AppData\Local\Nuitka\Nuitka\ | ||||
| mkdir C:\Users\ed\AppData\Local\Nuitka\Nuitka\gcc\ | ||||
| mkdir C:\Users\ed\AppData\Local\Nuitka\Nuitka\gcc\x86_64\ | ||||
| mkdir C:\Users\ed\AppData\Local\Nuitka\Nuitka\gcc\x86_64\10.2.0-11.0.0-8.0.0-r5\ | ||||
| copy c:\users\ed\downloads\winlibs-x86_64-posix-seh-gcc-10.2.0-llvm-11.0.0-mingw-w64-8.0.0-r5.zip C:\Users\ed\AppData\Local\Nuitka\Nuitka\gcc\x86_64\10.2.0-11.0.0-8.0.0-r5\winlibs-x86_64-posix-seh-gcc-10.2.0-llvm-11.0.0-mingw-w64-8.0.0-r5.zip | ||||
|  | ||||
| rem https://github.com/ccache/ccache/releases/download/v3.7.12/ccache-3.7.12-windows-32.zip | ||||
| mkdir C:\Users\ed\AppData\Local\Nuitka\Nuitka\ccache\ | ||||
| mkdir C:\Users\ed\AppData\Local\Nuitka\Nuitka\ccache\v3.7.12\ | ||||
| copy c:\users\ed\downloads\ccache-3.7.12-windows-32.zip C:\Users\ed\AppData\Local\Nuitka\Nuitka\ccache\v3.7.12\ccache-3.7.12-windows-32.zip | ||||
|  | ||||
| rem https://dependencywalker.com/depends22_x64.zip | ||||
| mkdir C:\Users\ed\AppData\Local\Nuitka\Nuitka\depends\ | ||||
| mkdir C:\Users\ed\AppData\Local\Nuitka\Nuitka\depends\x86_64\ | ||||
| copy c:\users\ed\downloads\depends22_x64.zip C:\Users\ed\AppData\Local\Nuitka\Nuitka\depends\x86_64\depends22_x64.zip | ||||
|  | ||||
| cd \ | ||||
| rd /s /q %appdata%\..\local\temp\pe-copyparty | ||||
| cd \users\ed\downloads | ||||
| python copyparty-sfx.py -h | ||||
| cd %appdata%\..\local\temp\pe-copyparty\copyparty | ||||
|  | ||||
| python | ||||
| import os, re | ||||
| os.rename('../dep-j2/jinja2', '../jinja2') | ||||
| os.rename('../dep-j2/markupsafe', '../markupsafe') | ||||
|  | ||||
| print("# nuitka dies if .__init__.stuff is imported") | ||||
| with open('__init__.py','r',encoding='utf-8') as f: | ||||
|  t1 = f.read() | ||||
|  | ||||
| with open('util.py','r',encoding='utf-8') as f: | ||||
|  t2 = f.read().split('\n')[3:] | ||||
|  | ||||
| t2 = [x for x in t2 if 'from .__init__' not in x] | ||||
| t = t1 + '\n'.join(t2) | ||||
| with open('__init__.py','w',encoding='utf-8') as f: | ||||
|  f.write('\n') | ||||
|  | ||||
| with open('util.py','w',encoding='utf-8') as f: | ||||
|  f.write(t) | ||||
|  | ||||
| print("# local-imports fail, prefix module names") | ||||
| ptn = re.compile(r'^( *from )(\.[^ ]+ import .*)') | ||||
| for d, _, fs in os.walk('.'): | ||||
|  for f in fs: | ||||
|   fp = os.path.join(d, f) | ||||
|   if not fp.endswith('.py'): | ||||
|    continue | ||||
|   t = '' | ||||
|   with open(fp,'r',encoding='utf-8') as f: | ||||
|    for ln in [x.rstrip('\r\n') for x in f]: | ||||
|     m = ptn.match(ln) | ||||
|     if not m: | ||||
|      t += ln + '\n' | ||||
|      continue | ||||
|     p1, p2 = m.groups() | ||||
|     t += "{}copyparty{}\n".format(p1, p2).replace("__init__", "util") | ||||
|   with open(fp,'w',encoding='utf-8') as f: | ||||
|    f.write(t) | ||||
|  | ||||
| exit() | ||||
|  | ||||
| cd .. | ||||
|  | ||||
| rd /s /q bout & python -m nuitka --standalone --onefile --windows-onefile-tempdir --python-flag=no_site --assume-yes-for-downloads --include-data-dir=copyparty\web=copyparty\web --include-data-dir=copyparty\res=copyparty\res --run --output-dir=bout --mingw64 --include-package=markupsafe --include-package=jinja2 copyparty | ||||
							
								
								
									
										35
									
								
								docs/pretend-youre-qnap.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								docs/pretend-youre-qnap.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py | ||||
| index 2d3c1ad..e1e85a0 100644 | ||||
| --- a/copyparty/httpcli.py | ||||
| +++ b/copyparty/httpcli.py | ||||
| @@ -864,6 +864,30 @@ class HttpCli(object): | ||||
|          # | ||||
|          # send reply | ||||
|   | ||||
| +        try: | ||||
| +            fakefn = self.conn.hsrv.fakefn | ||||
| +            fakectr = self.conn.hsrv.fakectr | ||||
| +            fakedata = self.conn.hsrv.fakedata | ||||
| +        except: | ||||
| +            fakefn = b'' | ||||
| +            fakectr = 0 | ||||
| +            fakedata = b'' | ||||
| +         | ||||
| +        self.log('\n{} {}\n{}'.format(fakefn, fakectr, open_args[0])) | ||||
| +        if fakefn == open_args[0] and fakectr > 0: | ||||
| +            self.reply(fakedata, mime=guess_mime(req_path)[0]) | ||||
| +            self.conn.hsrv.fakectr = fakectr - 1 | ||||
| +        else: | ||||
| +            with open_func(*open_args) as f: | ||||
| +                fakedata = f.read() | ||||
| +             | ||||
| +            self.conn.hsrv.fakefn = open_args[0] | ||||
| +            self.conn.hsrv.fakedata = fakedata | ||||
| +            self.conn.hsrv.fakectr = 15 | ||||
| +            self.reply(fakedata, mime=guess_mime(req_path)[0]) | ||||
| +         | ||||
| +        return True | ||||
| + | ||||
|          self.out_headers["Accept-Ranges"] = "bytes" | ||||
|          self.send_headers( | ||||
|              length=upper - lower, | ||||
							
								
								
									
										62
									
								
								docs/rclone.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								docs/rclone.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | ||||
| # using rclone to mount a remote copyparty server as a local filesystem | ||||
|  | ||||
| speed estimates with server and client on the same win10 machine: | ||||
| * `1070 MiB/s` with rclone as both server and client | ||||
| * `570 MiB/s` with rclone-client and `copyparty -ed -j16` as server | ||||
| * `220 MiB/s` with rclone-client and `copyparty -ed` as server | ||||
| * `100 MiB/s` with [../bin/copyparty-fuse.py](../bin/copyparty-fuse.py) as client | ||||
|  | ||||
| when server is on another machine (1gbit LAN), | ||||
| * `75 MiB/s` with [../bin/copyparty-fuse.py](../bin/copyparty-fuse.py) as client | ||||
| * `92 MiB/s` with rclone-client and `copyparty -ed` as server | ||||
| * `103 MiB/s` (connection max) with `copyparty -ed -j16` and all the others | ||||
|  | ||||
|  | ||||
| # creating the config file | ||||
|  | ||||
| if you want to use password auth, add `headers = Cookie,cppwd=fgsfds` below | ||||
|  | ||||
|  | ||||
| ### on windows clients: | ||||
| ``` | ||||
| ( | ||||
| echo [cpp] | ||||
| echo type = http | ||||
| echo url = http://127.0.0.1:3923/ | ||||
| ) > %userprofile%\.config\rclone\rclone.conf | ||||
| ``` | ||||
|  | ||||
| also install the windows dependencies: [winfsp](https://github.com/billziss-gh/winfsp/releases/latest) | ||||
|  | ||||
|  | ||||
| ### on unix clients: | ||||
| ``` | ||||
| cat > ~/.config/rclone/rclone.conf <<'EOF' | ||||
| [cpp] | ||||
| type = http | ||||
| url = http://127.0.0.1:3923/ | ||||
| EOF | ||||
| ``` | ||||
|  | ||||
|  | ||||
| # mounting the copyparty server locally | ||||
| ``` | ||||
| rclone.exe mount --vfs-cache-max-age 5s --attr-timeout 5s --dir-cache-time 5s cpp: Z: | ||||
| ``` | ||||
|  | ||||
|  | ||||
| # use rclone as server too, replacing copyparty | ||||
|  | ||||
| feels out of place but is too good not to mention | ||||
|  | ||||
| ``` | ||||
| rclone.exe serve http --read-only . | ||||
| ``` | ||||
|  | ||||
| * `webdav` gives write-access but `http` is twice as fast | ||||
| * `ftp` is buggy, avoid | ||||
|  | ||||
|  | ||||
| # bugs | ||||
|  | ||||
| * rclone-client throws an exception if you try to read an empty file (should return zero bytes) | ||||
							
								
								
									
										32
									
								
								docs/tcp-debug.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								docs/tcp-debug.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| (cd ~/dev/copyparty && strace -Tttyyvfs 256 -o strace.strace python3 -um copyparty -i 127.0.0.1 --http-only --stackmon /dev/shm/cpps,10 ) 2>&1 | tee /dev/stderr > ~/log-copyparty-$(date +%Y-%m%d-%H%M%S).txt | ||||
|  | ||||
| 14/Jun/2021:16:34:02 1623688447.212405 death | ||||
| 14/Jun/2021:16:35:02 1623688502.420860 back | ||||
|  | ||||
| tcpdump -nni lo -w /home/ed/lo.pcap | ||||
|  | ||||
| # 16:35:25.324662 IP 127.0.0.1.48632 > 127.0.0.1.3920: Flags [F.], seq 849, ack 544, win 359, options [nop,nop,TS val 809396796 ecr 809396796], length 0 | ||||
|  | ||||
| tcpdump -nnr /home/ed/lo.pcap | awk '/ > 127.0.0.1.3920: /{sub(/ > .*/,"");sub(/.*\./,"");print}' | sort -n | uniq | while IFS= read -r port; do echo; tcpdump -nnr /home/ed/lo.pcap 2>/dev/null | grep -E "\.$port( > |: F)" | sed -r 's/ > .*, /, /'; done | grep -E '^16:35:0.*length [^0]' -C50 | ||||
|  | ||||
| 16:34:02.441732 IP 127.0.0.1.48638, length 0 | ||||
| 16:34:02.441738 IP 127.0.0.1.3920, length 0 | ||||
| 16:34:02.441744 IP 127.0.0.1.48638, length 0 | ||||
| 16:34:02.441756 IP 127.0.0.1.48638, length 791 | ||||
| 16:34:02.441759 IP 127.0.0.1.3920, length 0 | ||||
| 16:35:02.445529 IP 127.0.0.1.48638, length 0 | ||||
| 16:35:02.489194 IP 127.0.0.1.3920, length 0 | ||||
| 16:35:02.515595 IP 127.0.0.1.3920, length 216 | ||||
| 16:35:02.515600 IP 127.0.0.1.48638, length 0 | ||||
|  | ||||
| grep 48638 "$(find ~ -maxdepth 1 -name log-copyparty-\*.txt | sort | tail -n 1)" | ||||
|  | ||||
| 1623688502.510380 48638 rh | ||||
| 1623688502.511291 48638 Unrecv direct ... | ||||
| 1623688502.511827 48638 rh = 791 | ||||
| 16:35:02.518 127.0.0.1 48638       shut(8): [Errno 107] Socket not connected | ||||
| Exception in thread httpsrv-0.1-48638: | ||||
|  | ||||
| grep 48638 ~/dev/copyparty/strace.strace | ||||
| 14561 16:35:02.506310 <... accept4 resumed> {sa_family=AF_INET, sin_port=htons(48638), sin_addr=inet_addr("127.0.0.1")}, [16], SOCK_CLOEXEC) = 8<TCP:[127.0.0.1:3920->127.0.0.1:48638]> <0.000012> | ||||
| 15230 16:35:02.510725 write(1<pipe:[256639555]>, "1623688502.510380 48638 rh\n", 27 <unfinished ...> | ||||
							
								
								
									
										10
									
								
								docs/unirange.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								docs/unirange.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| v = "U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD" | ||||
| for v in v.split(","): | ||||
|     if "+" in v: | ||||
|         v = v.split("+")[1] | ||||
|     if "-" in v: | ||||
|         lo, hi = v.split("-") | ||||
|     else: | ||||
|         lo = hi = v | ||||
|     for v in range(int(lo, 16), int(hi, 16) + 1): | ||||
|         print("{:4x} [{}]".format(v, chr(v))) | ||||
							
								
								
									
										145
									
								
								scripts/copyparty-repack.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										145
									
								
								scripts/copyparty-repack.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| #!/bin/bash | ||||
| repacker=1 | ||||
| set -e | ||||
|  | ||||
| # -- download latest copyparty (source.tgz and sfx), | ||||
| # -- build minimal sfx versions, | ||||
| # -- create a .tar.gz bundle | ||||
| # | ||||
| # convenient for deploying updates to inconvenient locations | ||||
| #  (and those are usually linux so bash is good inaff) | ||||
| #   (but that said this even has macos support) | ||||
| # | ||||
| # bundle will look like: | ||||
| # -rwxr-xr-x  0 ed ed  183808 Nov 19 00:43 copyparty | ||||
| # -rw-r--r--  0 ed ed  491318 Nov 19 00:40 copyparty-extras/copyparty-0.5.4.tar.gz | ||||
| # -rwxr-xr-x  0 ed ed   30254 Nov 17 23:58 copyparty-extras/copyparty-fuse.py | ||||
| # -rwxr-xr-x  0 ed ed  481403 Nov 19 00:40 copyparty-extras/sfx-full/copyparty-sfx.sh | ||||
| # -rwxr-xr-x  0 ed ed  506043 Nov 19 00:40 copyparty-extras/sfx-full/copyparty-sfx.py | ||||
| # -rwxr-xr-x  0 ed ed  167699 Nov 19 00:43 copyparty-extras/sfx-lite/copyparty-sfx.sh | ||||
| # -rwxr-xr-x  0 ed ed  183808 Nov 19 00:43 copyparty-extras/sfx-lite/copyparty-sfx.py | ||||
|  | ||||
|  | ||||
| command -v gnutar && tar() { gnutar "$@"; } | ||||
| command -v gtar && tar() { gtar "$@"; } | ||||
| command -v gsed && sed() { gsed "$@"; } | ||||
| td="$(mktemp -d)" | ||||
| od="$(pwd)" | ||||
| cd "$td" | ||||
| pwd | ||||
|  | ||||
|  | ||||
| dl_text() { | ||||
| 	command -v curl >/dev/null && exec curl "$@" | ||||
| 	exec wget -O- "$@" | ||||
| } | ||||
| dl_files() { | ||||
| 	command -v curl >/dev/null && exec curl -L --remote-name-all "$@" | ||||
| 	exec wget "$@" | ||||
| } | ||||
| export -f dl_files | ||||
|  | ||||
|  | ||||
| # if cache exists, use that instead of bothering github | ||||
| cache="$od/.copyparty-repack.cache" | ||||
| [ -e "$cache" ] && | ||||
| 	tar -xf "$cache" || | ||||
| { | ||||
| 	# get download links from github | ||||
| 	dl_text https://api.github.com/repos/9001/copyparty/releases/latest | | ||||
| 	( | ||||
| 		# prefer jq if available | ||||
| 		jq -r '.assets[]|select(.name|test("-sfx|tar.gz")).browser_download_url' || | ||||
|  | ||||
| 		# fallback to awk (sorry) | ||||
| 		awk -F\" '/"browser_download_url".*(\.tar\.gz|-sfx\.)/ {print$4}' | ||||
| 	) | | ||||
| 	tee /dev/stderr | | ||||
| 	tr -d '\r' | tr '\n' '\0' | | ||||
| 	xargs -0 bash -c 'dl_files "$@"' _ | ||||
|  | ||||
| 	tar -czf "$cache" * | ||||
| } | ||||
|  | ||||
|  | ||||
| # move src into copyparty-extras/, | ||||
| # move sfx into copyparty-extras/sfx-full/ | ||||
| mkdir -p copyparty-extras/sfx-{full,lite} | ||||
| mv copyparty-sfx.* copyparty-extras/sfx-full/ | ||||
| mv copyparty-*.tar.gz copyparty-extras/ | ||||
|  | ||||
|  | ||||
| # unpack the source code | ||||
| ( cd copyparty-extras/ | ||||
| tar -xf *.tar.gz | ||||
| ) | ||||
|  | ||||
|  | ||||
| # use repacker from release if that is newer | ||||
| p_other=copyparty-extras/copyparty-*/scripts/copyparty-repack.sh | ||||
| other=$(awk -F= 'BEGIN{v=-1} NR<10&&/^repacker=/{v=$NF} END{print v}' <$p_other)  | ||||
| [ $repacker -lt $other ] && | ||||
|   cat $p_other >"$od/$0" && cd "$od" && rm -rf "$td" && exec "$0" "$@" | ||||
|  | ||||
|  | ||||
| # now drop the cache | ||||
| rm -f "$cache" | ||||
|  | ||||
|  | ||||
| # fix permissions | ||||
| chmod 755 \ | ||||
|   copyparty-extras/sfx-full/* \ | ||||
|   copyparty-extras/copyparty-*/{scripts,bin}/* | ||||
|  | ||||
|  | ||||
| # extract the sfx | ||||
| ( cd copyparty-extras/sfx-full/ | ||||
| ./copyparty-sfx.py -h | ||||
| ) | ||||
|  | ||||
|  | ||||
| repack() { | ||||
|  | ||||
| 	# do the repack | ||||
| 	(cd copyparty-extras/copyparty-*/ | ||||
| 	./scripts/make-sfx.sh $2 | ||||
| 	) | ||||
|  | ||||
| 	# put new sfx into copyparty-extras/$name/, | ||||
| 	( cd copyparty-extras/ | ||||
| 	mv copyparty-*/dist/* $1/ | ||||
| 	) | ||||
| } | ||||
|  | ||||
| repack sfx-full "re gz no-sh" | ||||
| repack sfx-lite "re no-ogv no-cm" | ||||
| repack sfx-lite "re no-ogv no-cm gz no-sh" | ||||
|  | ||||
|  | ||||
| # move fuse client into copyparty-extras/, | ||||
| # copy lite-sfx.py to ./copyparty, | ||||
| # delete extracted source code | ||||
| ( cd copyparty-extras/ | ||||
| mv copyparty-*/bin/copyparty-fuse.py . | ||||
| cp -pv sfx-lite/copyparty-sfx.py ../copyparty | ||||
| rm -rf copyparty-{0..9}*.*.*{0..9} | ||||
| ) | ||||
|  | ||||
|  | ||||
| # and include the repacker itself too | ||||
| cp -av "$od/$0" copyparty-extras/ || | ||||
| cp -av "$0" copyparty-extras/ || | ||||
| true | ||||
|  | ||||
|  | ||||
| # create the bundle | ||||
| printf '\n\n' | ||||
| fn=copyparty-$(date +%Y-%m%d-%H%M%S).tgz | ||||
| tar -czvf "$od/$fn" * | ||||
| cd "$od" | ||||
| rm -rf "$td" | ||||
|  | ||||
|  | ||||
| echo | ||||
| echo "done, here's your bundle:" | ||||
| ls -al "$fn" | ||||
| @@ -1,26 +1,32 @@ | ||||
| FROM    alpine:3.11 | ||||
| FROM    alpine:3.13 | ||||
| WORKDIR /z | ||||
| ENV     ver_asmcrypto=2821dd1dedd1196c378f5854037dda5c869313f3 \ | ||||
|         ver_markdownit=10.0.0 \ | ||||
|         ver_showdown=1.9.1 \ | ||||
|         ver_marked=1.0.0 \ | ||||
|         ver_ogvjs=1.6.1 \ | ||||
|         ver_mde=2.10.1 \ | ||||
|         ver_codemirror=5.53.2 \ | ||||
| ENV     ver_asmcrypto=5b994303a9d3e27e0915f72a10b6c2c51535a4dc \ | ||||
|         ver_hashwasm=4.7.0 \ | ||||
|         ver_marked=1.1.0 \ | ||||
|         ver_ogvjs=1.8.0 \ | ||||
|         ver_mde=2.14.0 \ | ||||
|         ver_codemirror=5.59.3 \ | ||||
|         ver_fontawesome=5.13.0 \ | ||||
|         ver_zopfli=1.0.3 | ||||
|  | ||||
|  | ||||
| # download | ||||
| RUN     apk add cmake make g++ git bash npm patch wget tar pigz brotli gzip unzip python3 python3-dev \ | ||||
| # download; | ||||
| # the scp url is latin from https://fonts.googleapis.com/css2?family=Source+Code+Pro&display=swap | ||||
| RUN     mkdir -p /z/dist/no-pk \ | ||||
|         && wget https://fonts.gstatic.com/s/sourcecodepro/v11/HI_SiYsKILxRpg3hIP6sJ7fM7PqlPevW.woff2 -O scp.woff2 \ | ||||
|         && apk add cmake make g++ git bash npm patch wget tar pigz brotli gzip unzip python3 python3-dev brotli py3-brotli \ | ||||
|         && wget https://github.com/brion/ogv.js/releases/download/$ver_ogvjs/ogvjs-$ver_ogvjs.zip -O ogvjs.zip \ | ||||
|         && wget https://github.com/asmcrypto/asmcrypto.js/archive/$ver_asmcrypto.tar.gz -O asmcrypto.tgz \ | ||||
|         && wget https://github.com/openpgpjs/asmcrypto.js/archive/$ver_asmcrypto.tar.gz -O asmcrypto.tgz \ | ||||
|         && wget https://github.com/markedjs/marked/archive/v$ver_marked.tar.gz -O marked.tgz \ | ||||
|         && wget https://github.com/Ionaru/easy-markdown-editor/archive/$ver_mde.tar.gz -O mde.tgz \ | ||||
|         && wget https://github.com/codemirror/CodeMirror/archive/$ver_codemirror.tar.gz -O codemirror.tgz \ | ||||
|         && wget https://github.com/FortAwesome/Font-Awesome/releases/download/$ver_fontawesome/fontawesome-free-$ver_fontawesome-web.zip -O fontawesome.zip \ | ||||
|         && wget https://github.com/google/zopfli/archive/zopfli-$ver_zopfli.tar.gz -O zopfli.tgz \ | ||||
|         && wget https://github.com/Daninet/hash-wasm/releases/download/v$ver_hashwasm/hash-wasm@$ver_hashwasm.zip -O hash-wasm.zip \ | ||||
|         && unzip ogvjs.zip \ | ||||
|         && (mkdir hash-wasm \ | ||||
|             && cd hash-wasm \ | ||||
|             && unzip ../hash-wasm.zip) \ | ||||
|         && (tar -xf asmcrypto.tgz \ | ||||
|             && cd asmcrypto.js-$ver_asmcrypto \ | ||||
|             && npm install ) \ | ||||
| @@ -36,23 +42,7 @@ RUN     apk add cmake make g++ git bash npm patch wget tar pigz brotli gzip unzi | ||||
|             && npm install \ | ||||
|             && npm i gulp-cli -g ) \ | ||||
|         && unzip fontawesome.zip \ | ||||
|         && tar -xf zopfli.tgz \ | ||||
|         && mkdir -p /z/dist/no-pk | ||||
|  | ||||
|  | ||||
| # uncomment if you wanna test the abandoned markdown converters | ||||
| #ENV     build_abandoned=1 | ||||
|  | ||||
|  | ||||
| RUN     [ $build_abandoned ] || exit 0; \ | ||||
|         git clone --depth 1 --branch $ver_showdown https://github.com/showdownjs/showdown/ \ | ||||
|         && wget https://github.com/markdown-it/markdown-it/archive/$ver_markdownit.tar.gz -O markdownit.tgz \ | ||||
|         && (cd showdown \ | ||||
|             && npm install \ | ||||
|             && npm i grunt -g ) \ | ||||
|         && (tar -xf markdownit.tgz \ | ||||
|             && cd markdown-it-$ver_markdownit \ | ||||
|             && npm install ) | ||||
|         && tar -xf zopfli.tgz | ||||
|  | ||||
|  | ||||
| # build fonttools (which needs zopfli) | ||||
| @@ -65,6 +55,7 @@ RUN     tar -xf zopfli.tgz \ | ||||
|             -S . \ | ||||
|         && make -C build \ | ||||
|         && make -C build install \ | ||||
|         && python3 -m ensurepip \ | ||||
|         && python3 -m pip install fonttools zopfli | ||||
|  | ||||
|  | ||||
| @@ -72,7 +63,12 @@ RUN     tar -xf zopfli.tgz \ | ||||
| RUN     cd asmcrypto.js-$ver_asmcrypto \ | ||||
|         && echo "export { Sha512 } from './hash/sha512/sha512';" > src/entry-export_all.ts \ | ||||
|         && node -r esm build.js \ | ||||
|         && mv asmcrypto.all.es5.js /z/dist/sha512.js | ||||
|         && awk '/HMAC state/{o=1}  /var HEAP/{o=0}  /function hmac_reset/{o=1}  /return \{/{o=0}  /var __extends =/{o=1}  /var Hash =/{o=0}  /hmac_|pbkdf2_/{next}  o{next}  {gsub(/IllegalStateError/,"Exception")}  {sub(/^ +/,"");sub(/^\/\/ .*/,"");sub(/;$/," ;")}  1' < asmcrypto.all.es5.js > /z/dist/sha512.ac.js | ||||
|  | ||||
|  | ||||
| # build hash-wasm | ||||
| RUN     cd hash-wasm \ | ||||
|         && mv sha512.umd.min.js /z/dist/sha512.hw.js | ||||
|  | ||||
|  | ||||
| # build ogvjs | ||||
| @@ -80,31 +76,27 @@ RUN     cd ogvjs-$ver_ogvjs \ | ||||
|         && cp -pv \ | ||||
|             ogv.js \ | ||||
|             ogv-worker-audio.js \ | ||||
|             ogv-demuxer-ogg.js \ | ||||
|             ogv-demuxer-ogg-wasm.js \ | ||||
|             ogv-demuxer-ogg-wasm.wasm \ | ||||
|             ogv-demuxer-webm.js \ | ||||
|             ogv-demuxer-webm-wasm.js \ | ||||
|             ogv-demuxer-webm-wasm.wasm \ | ||||
|             ogv-decoder-audio-opus.js \ | ||||
|             ogv-decoder-audio-opus-wasm.js \ | ||||
|             ogv-decoder-audio-opus-wasm.wasm \ | ||||
|             ogv-decoder-audio-vorbis.js \ | ||||
|             ogv-decoder-audio-vorbis-wasm.js \ | ||||
|             ogv-decoder-audio-vorbis-wasm.wasm \ | ||||
|             dynamicaudio.swf \ | ||||
|             /z/dist | ||||
|  | ||||
| #            ogv-demuxer-ogg.js \ | ||||
| #            ogv-demuxer-webm.js \ | ||||
| #            ogv-decoder-audio-opus.js \ | ||||
| #            ogv-decoder-audio-vorbis.js \ | ||||
| #            dynamicaudio.swf \ | ||||
|  | ||||
|  | ||||
| # build marked | ||||
| RUN     wget https://github.com/markedjs/marked/commit/5c166d4164791f643693478e4ac094d63d6e0c9a.patch -O marked-git-1.patch \ | ||||
|         && wget https://patch-diff.githubusercontent.com/raw/markedjs/marked/pull/1652.patch -O marked-git-2.patch | ||||
|  | ||||
| COPY    marked.patch /z/ | ||||
| COPY    marked-ln.patch /z/ | ||||
| RUN     cd marked-$ver_marked \ | ||||
|         && patch -p1 < /z/marked-git-1.patch \ | ||||
|         && patch -p1 < /z/marked-git-2.patch \ | ||||
|         && patch -p1 < /z/marked-ln.patch \ | ||||
|         && patch -p1 < /z/marked.patch \ | ||||
|         && npm run build \ | ||||
| @@ -138,57 +130,10 @@ RUN     cd easy-markdown-editor-$ver_mde \ | ||||
|         && patch -p1 < /z/easymde-ln.patch \ | ||||
|         && gulp \ | ||||
|         && cp -pv dist/easymde.min.css /z/dist/easymde.css \ | ||||
|         && cp -pv dist/easymde.min.js /z/dist/easymde.js \ | ||||
|         && sed -ri '/pipe.terser/d; /cleanCSS/d' gulpfile.js \ | ||||
|         && gulp \ | ||||
|         && cp -pv dist/easymde.min.css /z/dist/easymde.full.css \ | ||||
|         && cp -pv dist/easymde.min.js /z/dist/easymde.full.js | ||||
|         && cp -pv dist/easymde.min.js /z/dist/easymde.js | ||||
|  | ||||
|  | ||||
| # build showdown (abandoned; disabled by default) | ||||
| COPY    showdown.patch /z/ | ||||
| RUN     [ $build_abandoned ] || exit 0; \ | ||||
|         cd showdown \ | ||||
|         && rm -rf bin dist \ | ||||
| #       # remove ellipsis plugin \ | ||||
|         && rm \ | ||||
|             src/subParsers/ellipsis.js \ | ||||
|             test/cases/ellipsis* \ | ||||
| #       # remove html-to-md converter \ | ||||
|         && rm \ | ||||
|             test/node/testsuite.makemd.js \ | ||||
|             test/node/showdown.Converter.makeMarkdown.js \ | ||||
| #       # remove emojis \ | ||||
|         && rm src/subParsers/emoji.js \ | ||||
|         && awk '/^showdown.helper.emojis/ {o=1} !o; /^\}/ {o=0}' \ | ||||
|             >f <src/helpers.js \ | ||||
|         && mv f src/helpers.js \ | ||||
|         && rm -rf test/features/emojis \ | ||||
| #       # remove ghmentions \ | ||||
|         && rm test/features/ghMentions.* \ | ||||
| #       # remove option descriptions \ | ||||
|         && sed -ri '/descri(ption|be): /d' src/options.js \ | ||||
|         && patch -p1 < /z/showdown.patch | ||||
|  | ||||
| RUN     [ $build_abandoned ] || exit 0; \ | ||||
|         cd showdown \ | ||||
|         && grunt build \ | ||||
|         && sed -ri '/sourceMappingURL=showdown.min.js.map/d' dist/showdown.min.js \ | ||||
|         && mv dist/showdown.min.js /z/dist/showdown.js \ | ||||
|         && ls -al /z/dist/showdown.js | ||||
|  | ||||
|  | ||||
| # build markdownit (abandoned; disabled by default) | ||||
| COPY    markdown-it.patch /z/ | ||||
| RUN     [ $build_abandoned ] || exit 0; \ | ||||
|         cd markdown-it-$ver_markdownit \ | ||||
|         && patch -p1 < /z/markdown-it.patch \ | ||||
|         && make browserify \ | ||||
|         && cp -pv dist/markdown-it.min.js /z/dist/markdown-it.js \ | ||||
|         && cp -pv dist/markdown-it.js /z/dist/markdown-it-full.js | ||||
|  | ||||
|  | ||||
| # build fontawesome | ||||
| # build fontawesome and scp | ||||
| COPY    mini-fa.sh /z | ||||
| COPY    mini-fa.css /z | ||||
| RUN     /bin/ash /z/mini-fa.sh | ||||
| @@ -203,38 +148,6 @@ RUN     cd /z/dist \ | ||||
|         && rmdir no-pk | ||||
|  | ||||
|  | ||||
| # showdown: abandoned due to code-blocks in lists failing | ||||
| # 22770 orig | ||||
| # 12154 no-emojis | ||||
| # 12134 no-srcmap | ||||
| # 11189 no-descriptions | ||||
| # 11152 no-ellipsis | ||||
| # 10617 no-this.makeMd | ||||
| #  9569 no-extensions | ||||
| #  9537 no-extensions | ||||
| #  9410 no-mentions | ||||
|  | ||||
|  | ||||
| # markdown-it: abandoned because no header anchors (and too big) | ||||
| #       32322 107754 orig (wowee) | ||||
| # 19619 21392  71540 less entities | ||||
|  | ||||
|  | ||||
| # marked: | ||||
| # 9253 29773 orig | ||||
| # 9159 29633 no copyright (reverted) | ||||
| # 9040 29057 no sanitize | ||||
| # 8870 28631 no email-mangle | ||||
| # so really not worth it, just drop the patch when that stops working | ||||
|  | ||||
|  | ||||
| # easymde: | ||||
| # 91836 orig | ||||
| # 88635 no spellcheck | ||||
| # 88392 no urlRE | ||||
| # 85651 less bidi | ||||
| # 82855 less mode meta | ||||
|  | ||||
|  | ||||
| # d=/home/ed/dev/copyparty/scripts/deps-docker/; tar -cf ../x . && ssh root@$bip "cd $d && tar -xv >&2 && make >&2 && tar -cC ../../copyparty/web deps" <../x | (cd ../../copyparty/web/; cat > the.tgz; tar -xvf the.tgz) | ||||
| # git diff -U2 --no-index marked-1.1.0-orig/ marked-1.1.0-edit/ -U2 | sed -r '/^index /d;s`^(diff --git a/)[^/]+/(.* b/)[^/]+/`\1\2`; s`^(---|\+\+\+) ([ab]/)[^/]+/`\1 \2`' > ../dev/copyparty/scripts/deps-docker/marked-ln.patch | ||||
| # d=/home/ed/dev/copyparty/scripts/deps-docker/; tar -cf ../x . && ssh root@$bip "cd $d && tar -xv >&2 && make >&2 && tar -cC ../../copyparty/web deps" <../x | (cd ../../copyparty/web/; cat > the.tgz; tar -xvf the.tgz; rm the.tgz) | ||||
| # gzip -dkf ../dev/copyparty/copyparty/web/deps/deps/marked.full.js.gz && diff -NarU2 ../dev/copyparty/copyparty/web/deps/{,deps/}marked.full.js | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| diff -NarU2 CodeMirror-orig/mode/gfm/gfm.js CodeMirror-edit/mode/gfm/gfm.js | ||||
| --- CodeMirror-orig/mode/gfm/gfm.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/mode/gfm/gfm.js	2020-05-02 02:13:32.142131800 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/mode/gfm/gfm.js codemirror-5.59.3/mode/gfm/gfm.js | ||||
| --- codemirror-5.59.3-orig/mode/gfm/gfm.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/mode/gfm/gfm.js	2021-02-21 20:42:02.166174775 +0000 | ||||
| @@ -97,5 +97,5 @@ | ||||
|          } | ||||
|        } | ||||
| @@ -15,9 +15,9 @@ diff -NarU2 CodeMirror-orig/mode/gfm/gfm.js CodeMirror-edit/mode/gfm/gfm.js | ||||
| +      }*/ | ||||
|        stream.next(); | ||||
|        return null; | ||||
| diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js | ||||
| --- CodeMirror-orig/mode/meta.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/mode/meta.js	2020-05-02 03:56:58.852408400 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/mode/meta.js codemirror-5.59.3/mode/meta.js | ||||
| --- codemirror-5.59.3-orig/mode/meta.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/mode/meta.js	2021-02-21 20:42:54.798742821 +0000 | ||||
| @@ -13,4 +13,5 @@ | ||||
|   | ||||
|    CodeMirror.modeInfo = [ | ||||
| @@ -28,7 +28,7 @@ diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js | ||||
|      {name: "Gas", mime: "text/x-gas", mode: "gas", ext: ["s"]}, | ||||
|      {name: "Gherkin", mime: "text/x-feature", mode: "gherkin", ext: ["feature"]}, | ||||
| +    */ | ||||
|      {name: "GitHub Flavored Markdown", mime: "text/x-gfm", mode: "gfm", file: /^(readme|contributing|history).md$/i}, | ||||
|      {name: "GitHub Flavored Markdown", mime: "text/x-gfm", mode: "gfm", file: /^(readme|contributing|history)\.md$/i}, | ||||
| +    /* | ||||
|      {name: "Go", mime: "text/x-go", mode: "go", ext: ["go"]}, | ||||
|      {name: "Groovy", mime: "text/x-groovy", mode: "groovy", ext: ["groovy", "gradle"], file: /^Jenkinsfile$/}, | ||||
| @@ -56,16 +56,16 @@ diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js | ||||
| +    /* | ||||
|      {name: "XQuery", mime: "application/xquery", mode: "xquery", ext: ["xy", "xquery"]}, | ||||
|      {name: "Yacas", mime: "text/x-yacas", mode: "yacas", ext: ["ys"]}, | ||||
| @@ -171,4 +180,5 @@ | ||||
|      {name: "xu", mime: "text/x-xu", mode: "mscgen", ext: ["xu"]}, | ||||
|      {name: "msgenny", mime: "text/x-msgenny", mode: "mscgen", ext: ["msgenny"]} | ||||
| @@ -172,4 +181,5 @@ | ||||
|      {name: "msgenny", mime: "text/x-msgenny", mode: "mscgen", ext: ["msgenny"]}, | ||||
|      {name: "WebAssembly", mime: "text/webassembly", mode: "wast", ext: ["wat", "wast"]}, | ||||
| +    */ | ||||
|    ]; | ||||
|    // Ensure all modes have a mime property for backwards compatibility | ||||
| diff -NarU2 CodeMirror-orig/src/display/selection.js CodeMirror-edit/src/display/selection.js | ||||
| --- CodeMirror-orig/src/display/selection.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/display/selection.js	2020-05-02 03:27:30.144662800 +0200 | ||||
| @@ -83,29 +83,21 @@ | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/display/selection.js codemirror-5.59.3/src/display/selection.js | ||||
| --- codemirror-5.59.3-orig/src/display/selection.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/display/selection.js	2021-02-21 20:44:14.860894328 +0000 | ||||
| @@ -84,29 +84,21 @@ | ||||
|      let order = getOrder(lineObj, doc.direction) | ||||
|      iterateBidiSections(order, fromArg || 0, toArg == null ? lineLen : toArg, (from, to, dir, i) => { | ||||
| -      let ltr = dir == "ltr" | ||||
| @@ -105,24 +105,24 @@ diff -NarU2 CodeMirror-orig/src/display/selection.js CodeMirror-edit/src/display | ||||
| +          botRight = openEnd && last ? rightSide : toPos.right | ||||
|          add(topLeft, fromPos.top, topRight - topLeft, fromPos.bottom) | ||||
|          if (fromPos.bottom < toPos.top) add(leftSide, fromPos.bottom, null, toPos.top) | ||||
| diff -NarU2 CodeMirror-orig/src/input/ContentEditableInput.js CodeMirror-edit/src/input/ContentEditableInput.js | ||||
| --- CodeMirror-orig/src/input/ContentEditableInput.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/input/ContentEditableInput.js	2020-05-02 03:33:05.707995500 +0200 | ||||
| @@ -391,4 +391,5 @@ | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/input/ContentEditableInput.js codemirror-5.59.3/src/input/ContentEditableInput.js | ||||
| --- codemirror-5.59.3-orig/src/input/ContentEditableInput.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/input/ContentEditableInput.js	2021-02-21 20:44:33.273953867 +0000 | ||||
| @@ -399,4 +399,5 @@ | ||||
|    let info = mapFromLineView(view, line, pos.line) | ||||
|   | ||||
| +  /* | ||||
|    let order = getOrder(line, cm.doc.direction), side = "left" | ||||
|    if (order) { | ||||
| @@ -396,4 +397,5 @@ | ||||
| @@ -404,4 +405,5 @@ | ||||
|      side = partPos % 2 ? "right" : "left" | ||||
|    } | ||||
| +  */ | ||||
|    let result = nodeAndOffsetInLineMap(info.map, pos.ch, side) | ||||
|    result.offset = result.collapse == "right" ? result.end : result.start | ||||
| diff -NarU2 CodeMirror-orig/src/input/movement.js CodeMirror-edit/src/input/movement.js | ||||
| --- CodeMirror-orig/src/input/movement.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/input/movement.js	2020-05-02 03:31:19.710773500 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/input/movement.js codemirror-5.59.3/src/input/movement.js | ||||
| --- codemirror-5.59.3-orig/src/input/movement.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/input/movement.js	2021-02-21 20:45:12.763093671 +0000 | ||||
| @@ -15,4 +15,5 @@ | ||||
|   | ||||
|  export function endOfLine(visually, cm, lineObj, lineNo, dir) { | ||||
| @@ -146,9 +146,9 @@ diff -NarU2 CodeMirror-orig/src/input/movement.js CodeMirror-edit/src/input/move | ||||
|    return null | ||||
| +  */ | ||||
|  } | ||||
| diff -NarU2 CodeMirror-orig/src/line/line_data.js CodeMirror-edit/src/line/line_data.js | ||||
| --- CodeMirror-orig/src/line/line_data.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/line/line_data.js	2020-05-02 03:17:02.785065000 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/line/line_data.js codemirror-5.59.3/src/line/line_data.js | ||||
| --- codemirror-5.59.3-orig/src/line/line_data.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/line/line_data.js	2021-02-21 20:45:36.472549599 +0000 | ||||
| @@ -79,6 +79,6 @@ | ||||
|      // Optionally wire in some hacks into the token-rendering | ||||
|      // algorithm, to deal with browser quirks. | ||||
| @@ -158,9 +158,9 @@ diff -NarU2 CodeMirror-orig/src/line/line_data.js CodeMirror-edit/src/line/line_ | ||||
| +    //  builder.addToken = buildTokenBadBidi(builder.addToken, order) | ||||
|      builder.map = [] | ||||
|      let allowFrontierUpdate = lineView != cm.display.externalMeasured && lineNo(line) | ||||
| diff -NarU2 CodeMirror-orig/src/measurement/position_measurement.js CodeMirror-edit/src/measurement/position_measurement.js | ||||
| --- CodeMirror-orig/src/measurement/position_measurement.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/measurement/position_measurement.js	2020-05-02 03:35:20.674159600 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/measurement/position_measurement.js codemirror-5.59.3/src/measurement/position_measurement.js | ||||
| --- codemirror-5.59.3-orig/src/measurement/position_measurement.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/measurement/position_measurement.js	2021-02-21 20:50:52.372945293 +0000 | ||||
| @@ -380,5 +380,6 @@ | ||||
|      sticky = "after" | ||||
|    } | ||||
| @@ -199,9 +199,9 @@ diff -NarU2 CodeMirror-orig/src/measurement/position_measurement.js CodeMirror-e | ||||
| +*/ | ||||
|   | ||||
|  let measureText | ||||
| diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js | ||||
| --- CodeMirror-orig/src/util/bidi.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/util/bidi.js	2020-05-02 03:12:44.418649800 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/util/bidi.js codemirror-5.59.3/src/util/bidi.js | ||||
| --- codemirror-5.59.3-orig/src/util/bidi.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/util/bidi.js	2021-02-21 20:52:18.168092225 +0000 | ||||
| @@ -4,5 +4,5 @@ | ||||
|   | ||||
|  export function iterateBidiSections(order, from, to, f) { | ||||
| @@ -239,20 +239,19 @@ diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js | ||||
| +  var fun = function(str, direction) { | ||||
|      let outerType = direction == "ltr" ? "L" : "R" | ||||
|   | ||||
| @@ -204,12 +210,16 @@ | ||||
| @@ -204,5 +210,11 @@ | ||||
|      return direction == "rtl" ? order.reverse() : order | ||||
|    } | ||||
| -})() | ||||
|   | ||||
| +  return function(str, direction) { | ||||
| +    var ret = fun(str, direction); | ||||
| +    console.log("bidiOrdering inner ([%s], %s) => [%s]", str, direction, ret); | ||||
| +    return ret; | ||||
| +  } | ||||
| +})() | ||||
|  })() | ||||
| +*/ | ||||
|   | ||||
|  // Get the bidi ordering for the given line (and cache it). Returns | ||||
|  // false for lines that are fully left-to-right, and an array of | ||||
| @@ -210,6 +222,4 @@ | ||||
|  // BidiSpan objects otherwise. | ||||
|  export function getOrder(line, direction) { | ||||
| -  let order = line.order | ||||
| @@ -260,9 +259,9 @@ diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js | ||||
| -  return order | ||||
| +  return false; | ||||
|  } | ||||
| diff -NarU2 CodeMirror-orig/src/util/feature_detection.js CodeMirror-edit/src/util/feature_detection.js | ||||
| --- CodeMirror-orig/src/util/feature_detection.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/util/feature_detection.js	2020-05-02 03:16:21.085621400 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/util/feature_detection.js codemirror-5.59.3/src/util/feature_detection.js | ||||
| --- codemirror-5.59.3-orig/src/util/feature_detection.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/util/feature_detection.js	2021-02-21 20:49:22.191269270 +0000 | ||||
| @@ -25,4 +25,5 @@ | ||||
|  } | ||||
|   | ||||
|   | ||||
| @@ -1,33 +1,57 @@ | ||||
| diff -NarU2 easymde-orig/gulpfile.js easymde-mod1/gulpfile.js | ||||
| --- easymde-orig/gulpfile.js	2020-04-06 14:09:36.000000000 +0200 | ||||
| +++ easymde-mod1/gulpfile.js	2020-05-01 14:33:52.260175200 +0200 | ||||
| diff -NarU2 easy-markdown-editor-2.14.0-orig/gulpfile.js easy-markdown-editor-2.14.0/gulpfile.js | ||||
| --- easy-markdown-editor-2.14.0-orig/gulpfile.js	2021-02-14 12:11:48.000000000 +0000 | ||||
| +++ easy-markdown-editor-2.14.0/gulpfile.js	2021-02-21 20:55:37.134701007 +0000 | ||||
| @@ -25,5 +25,4 @@ | ||||
|      './node_modules/codemirror/lib/codemirror.css', | ||||
|      './src/css/*.css', | ||||
| -    './node_modules/codemirror-spell-checker/src/css/spell-checker.css', | ||||
|  ]; | ||||
|   | ||||
| diff -NarU2 easymde-orig/package.json easymde-mod1/package.json | ||||
| --- easymde-orig/package.json	2020-04-06 14:09:36.000000000 +0200 | ||||
| +++ easymde-mod1/package.json	2020-05-01 14:33:57.189975800 +0200 | ||||
| diff -NarU2 easy-markdown-editor-2.14.0-orig/package.json easy-markdown-editor-2.14.0/package.json | ||||
| --- easy-markdown-editor-2.14.0-orig/package.json	2021-02-14 12:11:48.000000000 +0000 | ||||
| +++ easy-markdown-editor-2.14.0/package.json	2021-02-21 20:55:47.761190082 +0000 | ||||
| @@ -21,5 +21,4 @@ | ||||
|      "dependencies": { | ||||
|          "codemirror": "^5.52.2", | ||||
|          "codemirror": "^5.59.2", | ||||
| -        "codemirror-spell-checker": "1.1.2", | ||||
|          "marked": "^0.8.2" | ||||
|          "marked": "^2.0.0" | ||||
|      }, | ||||
| diff -NarU2 easymde-orig/src/js/easymde.js easymde-mod1/src/js/easymde.js | ||||
| --- easymde-orig/src/js/easymde.js	2020-04-06 14:09:36.000000000 +0200 | ||||
| +++ easymde-mod1/src/js/easymde.js	2020-05-01 14:34:19.878774400 +0200 | ||||
| @@ -11,5 +11,4 @@ | ||||
| diff -NarU2 easy-markdown-editor-2.14.0-orig/src/js/easymde.js easy-markdown-editor-2.14.0/src/js/easymde.js | ||||
| --- easy-markdown-editor-2.14.0-orig/src/js/easymde.js	2021-02-14 12:11:48.000000000 +0000 | ||||
| +++ easy-markdown-editor-2.14.0/src/js/easymde.js	2021-02-21 20:57:09.143171536 +0000 | ||||
| @@ -12,5 +12,4 @@ | ||||
|  require('codemirror/mode/gfm/gfm.js'); | ||||
|  require('codemirror/mode/xml/xml.js'); | ||||
| -var CodeMirrorSpellChecker = require('codemirror-spell-checker'); | ||||
|  var marked = require('marked/lib/marked'); | ||||
|   | ||||
| @@ -1889,18 +1888,7 @@ | ||||
| @@ -1762,9 +1761,4 @@ | ||||
|          options.autosave.uniqueId = options.autosave.unique_id; | ||||
|   | ||||
| -    // If overlay mode is specified and combine is not provided, default it to true | ||||
| -    if (options.overlayMode && options.overlayMode.combine === undefined) { | ||||
| -      options.overlayMode.combine = true; | ||||
| -    } | ||||
| - | ||||
|      // Update this options | ||||
|      this.options = options; | ||||
| @@ -2003,28 +1997,7 @@ | ||||
|      var mode, backdrop; | ||||
|   | ||||
| -    // CodeMirror overlay mode | ||||
| -    if (options.overlayMode) { | ||||
| -      CodeMirror.defineMode('overlay-mode', function(config) { | ||||
| -        return CodeMirror.overlayMode(CodeMirror.getMode(config, options.spellChecker !== false ? 'spell-checker' : 'gfm'), options.overlayMode.mode, options.overlayMode.combine); | ||||
| -      }); | ||||
| - | ||||
| -      mode = 'overlay-mode'; | ||||
| -      backdrop = options.parsingConfig; | ||||
| -      backdrop.gitHubSpice = false; | ||||
| -    } else { | ||||
|          mode = options.parsingConfig; | ||||
|          mode.name = 'gfm'; | ||||
|          mode.gitHubSpice = false; | ||||
| -    } | ||||
| -    if (options.spellChecker !== false) { | ||||
| -        mode = 'spell-checker'; | ||||
| -        backdrop = options.parsingConfig; | ||||
| @@ -37,16 +61,28 @@ diff -NarU2 easymde-orig/src/js/easymde.js easymde-mod1/src/js/easymde.js | ||||
| -        CodeMirrorSpellChecker({ | ||||
| -            codeMirrorInstance: CodeMirror, | ||||
| -        }); | ||||
| -    } else { | ||||
|          mode = options.parsingConfig; | ||||
|          mode.name = 'gfm'; | ||||
|          mode.gitHubSpice = false; | ||||
| -    } | ||||
|   | ||||
|      // eslint-disable-next-line no-unused-vars | ||||
| @@ -1927,5 +1915,4 @@ | ||||
|          configureMouse: configureMouse, | ||||
|          inputStyle: (options.inputStyle != undefined) ? options.inputStyle : isMobile() ? 'contenteditable' : 'textarea', | ||||
| -        spellcheck: (options.nativeSpellcheck != undefined) ? options.nativeSpellcheck : true, | ||||
|      }); | ||||
| diff -NarU2 easy-markdown-editor-2.14.0-orig/types/easymde.d.ts easy-markdown-editor-2.14.0/types/easymde.d.ts | ||||
| --- easy-markdown-editor-2.14.0-orig/types/easymde.d.ts	2021-02-14 12:11:48.000000000 +0000 | ||||
| +++ easy-markdown-editor-2.14.0/types/easymde.d.ts	2021-02-21 20:57:42.492620979 +0000 | ||||
| @@ -160,9 +160,4 @@ | ||||
|      } | ||||
|   | ||||
| -    interface OverlayModeOptions { | ||||
| -      mode: CodeMirror.Mode<any> | ||||
| -      combine?: boolean | ||||
| -    } | ||||
| - | ||||
|      interface Options { | ||||
|          autoDownloadFontAwesome?: boolean; | ||||
| @@ -214,7 +209,5 @@ | ||||
|   | ||||
|          promptTexts?: PromptTexts; | ||||
| -        syncSideBySidePreviewScroll?: boolean; | ||||
| - | ||||
| -        overlayMode?: OverlayModeOptions | ||||
| +        syncSideBySidePreviewScroll?: boolean | ||||
|      } | ||||
|  } | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user