mirror of
				https://github.com/9001/copyparty.git
				synced 2025-10-24 16:43:55 +00:00 
			
		
		
		
	Compare commits
	
		
			304 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 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 | 
							
								
								
									
										19
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										19
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @@ -14,12 +14,11 @@ | ||||
|                 "-emp", | ||||
|                 "-e2dsa", | ||||
|                 "-e2ts", | ||||
|                 "-a", | ||||
|                 "ed:wark", | ||||
|                 "-v", | ||||
|                 "srv::r:aed:cnodupe", | ||||
|                 "-v", | ||||
|                 "dist:dist:r" | ||||
|                 "-mtp", | ||||
|                 ".bpm=f,bin/mtag/audio-bpm.py", | ||||
|                 "-aed:wark", | ||||
|                 "-vsrv::r:aed:cnodupe", | ||||
|                 "-vdist:dist:r" | ||||
|             ] | ||||
|         }, | ||||
|         { | ||||
| @@ -41,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) | ||||
							
								
								
									
										4
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							| @@ -9,9 +9,7 @@ | ||||
|         { | ||||
|             "label": "no_dbg", | ||||
|             "type": "shell", | ||||
|             "command": "${config:python.pythonPath} -m copyparty -ed -emp -e2dsa -e2ts -a ed:wark -v srv::r:aed:cnodupe -v dist:dist:r ;exit 1" | ||||
|             // -v ~/Music/mt:mt:r:cmtp=.bpm=~/dev/copyparty/bin/mtag/audio-bpm.py:cmtp=key=~/dev/copyparty/bin/mtag/audio-key.py:ce2tsr  | ||||
|             // -v ~/Music/mt:mt:r:cmtp=.bpm=~/dev/copyparty/bin/mtag/audio-bpm.py:ce2tsr | ||||
|             "command": "${config:python.pythonPath} .vscode/launch.py" | ||||
|         } | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										374
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										374
									
								
								README.md
									
									
									
									
									
								
							| @@ -9,9 +9,12 @@ | ||||
| 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.3+` | ||||
| * *resumable* uploads need `firefox 12+` / `chrome 6+` / `safari 6+` / `IE 10+` | ||||
| * 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) | ||||
|  | ||||
|  | ||||
| ## readme toc | ||||
|  | ||||
| @@ -20,21 +23,38 @@ turn your phone or raspi into a portable file server with resumable uploads/down | ||||
|     * [notes](#notes) | ||||
|     * [status](#status) | ||||
| * [bugs](#bugs) | ||||
| * [usage](#usage) | ||||
|     * [general bugs](#general-bugs) | ||||
|     * [not my bugs](#not-my-bugs) | ||||
| * [the browser](#the-browser) | ||||
|     * [tabs](#tabs) | ||||
|     * [hotkeys](#hotkeys) | ||||
|     * [tree-mode](#tree-mode) | ||||
|     * [thumbnails](#thumbnails) | ||||
|     * [zip downloads](#zip-downloads) | ||||
|     * [uploading](#uploading) | ||||
|         * [file-search](#file-search) | ||||
|     * [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) | ||||
| * [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) | ||||
| * [dev env setup](#dev-env-setup) | ||||
| * [how to release](#how-to-release) | ||||
| * [building](#building) | ||||
|     * [dev env setup](#dev-env-setup) | ||||
|     * [just the sfx](#just-the-sfx) | ||||
|     * [complete release](#complete-release) | ||||
| * [todo](#todo) | ||||
|  | ||||
|  | ||||
| @@ -42,25 +62,39 @@ turn your phone or raspi into a portable file server with resumable uploads/down | ||||
|  | ||||
| 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 let anyone access the current folder; see `-h` for help if you want accounts and volumes etc | ||||
| 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 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:afoo -a foo:bar` shares `/mnt/music` as `/music`, `r`eadable by anyone, with user `foo` as `a`dmin (read/write), password `bar` | ||||
|   * replace `:r:afoo` with `:rfoo` to only make the folder readable by `foo` and nobody else | ||||
|   * in addition to `r`ead and `a`dmin, `w`rite makes a folder write-only, so cannot list/access files in it | ||||
| * `--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 legit https) | ||||
| * [contrib/nginx/copyparty.conf](contrib/nginx/copyparty.conf) to reverse-proxy behind nginx (for better https) | ||||
|  | ||||
|  | ||||
| ## notes | ||||
|  | ||||
| * iPhone/iPad: use Firefox to download files | ||||
| * Android-Chrome: set max "parallel uploads" for 200% upload speed (android bug) | ||||
| * Android-Firefox: takes a while to select files (in order to avoid the above android-chrome issue) | ||||
| * Desktop-Firefox: may use gigabytes of RAM if your connection is great and your files are massive | ||||
| general: | ||||
| * paper-printing is affected by dark/light-mode! use lightmode for color, darkmode for grayscale | ||||
|   * because no browsers currently implement the media-query to do this properly orz | ||||
|  | ||||
| browser-specific: | ||||
| * iPhone/iPad: use Firefox to download files | ||||
| * Android-Chrome: increase "parallel uploads" for higher speed (android bug) | ||||
| * Android-Firefox: takes a while to select files (their fix for ☝️) | ||||
| * Desktop-Firefox: ~~may use gigabytes of RAM if your files are massive~~ *seems to be OK now* | ||||
| * Desktop-Firefox: may stop you from deleting folders you've uploaded until you visit `about:memory` and click `Minimize memory usage` | ||||
|  | ||||
|  | ||||
| ## status | ||||
|  | ||||
| summary: all planned features work! now please enjoy the bloatening | ||||
|  | ||||
| * backend stuff | ||||
|   * ☑ sanic multipart parser | ||||
|   * ☑ load balancer (multiprocessing) | ||||
| @@ -77,10 +111,14 @@ you may also want these, especially on servers: | ||||
|   * ☑ FUSE client (read-only) | ||||
| * browser | ||||
|   * ☑ tree-view | ||||
|   * ☑ media player | ||||
|   * ✖ thumbnails | ||||
|   * ✖ SPA (browse while uploading) | ||||
|     * currently safe using the file-tree on the left only, not folders in the file list | ||||
|   * ☑ audio player (with OS media controls) | ||||
|   * ☑ thumbnails | ||||
|     * ☑ images using Pillow | ||||
|     * ☑ videos using FFmpeg | ||||
|     * ☑ cache eviction (max-age; maybe max-size eventually) | ||||
|   * ☑ image gallery | ||||
|   * ☑ SPA (browse while uploading) | ||||
|     * if you use the file-tree on the left only, not folders in the file list | ||||
| * server indexing | ||||
|   * ☑ locate files by contents | ||||
|   * ☑ search by name/path/date/size | ||||
| @@ -89,25 +127,75 @@ you may also want these, especially on servers: | ||||
|   * ☑ viewer | ||||
|   * ☑ editor (sure why not) | ||||
|  | ||||
| summary: it works! you can use it! (but technically not even close to beta) | ||||
|  | ||||
|  | ||||
| # 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 | ||||
| * MacOS: `--th-ff-jpg` may fix thumbnails using macports-FFmpeg | ||||
|  | ||||
| ## 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 | ||||
|  | ||||
| # usage | ||||
| * 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 | ||||
|  | ||||
|  | ||||
| # the browser | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## tabs | ||||
|  | ||||
| * `[🔎]` search by size, date, path/name, mp3-tags ... see [searching](#searching) | ||||
| * `[🚀]` 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` | ||||
| * `[⚙️]` client configuration options | ||||
|  | ||||
|  | ||||
| ## hotkeys | ||||
|  | ||||
| the browser has the following hotkeys | ||||
| * `0..9` jump to 10%..90% | ||||
| * `U/O` skip 10sec back/forward | ||||
| * `J/L` prev/next song | ||||
| * `I/K` prev/next folder | ||||
| * `P` parent folder | ||||
| * `G` toggle list / grid view | ||||
| * `T` toggle thumbnails / icons | ||||
| * when playing audio: | ||||
|   * `0..9` jump to 10%..90% | ||||
|   * `U/O` skip 10sec back/forward | ||||
|   * `J/L` prev/next song | ||||
|   * `M` play/pause (also starts playing the folder) | ||||
| * in the grid view: | ||||
|   * `S` toggle multiselect | ||||
|   * `A/D` zoom | ||||
|  | ||||
|  | ||||
| ## tree-mode | ||||
|  | ||||
| by default there's a breadcrumbs path; you can replace this with a tree-browser sidebar thing by clicking the 🌲 | ||||
|  | ||||
| click `[-]` and `[+]` 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 destructive your users are | ||||
|  | ||||
| images named `folder.jpg` and `folder.png` become the thumbnail of the folder they're in | ||||
|  | ||||
|  | ||||
| ## zip downloads | ||||
| @@ -126,12 +214,80 @@ the `zip` link next to folders can produce various types of zip/tar files using | ||||
| * `zip_crc` will take longer to download since the server has to read each file twice | ||||
|   * please let me know if you find a program old enough to actually need this | ||||
|  | ||||
| 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 | ||||
|  | ||||
| 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 resume if they are interrupted (for example by a reboot) | ||||
|   * server detects any corruption; the client reuploads affected chunks | ||||
|   * the client doesn't upload anything that already exists on the server | ||||
| * 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 | ||||
|  | ||||
| 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 | ||||
|  | ||||
|  | ||||
| ## 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` | ||||
|  | ||||
|  | ||||
| # 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 (you get the URL if it does) | ||||
| * 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 | ||||
| @@ -157,7 +313,29 @@ the same arguments can be set as volume flags, in addition to `d2d` and `d2t` fo | ||||
| * `-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*` | ||||
|  | ||||
| `e2tsr` is probably always overkill, since `e2ds`/`e2dsa` would pick up any file modifications and cause `e2ts` to reindex those | ||||
| note: | ||||
| * `e2tsr` is probably always overkill, since `e2ds`/`e2dsa` would pick up any file modifications and `e2ts` would then reindex those | ||||
| * 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 networked 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 | ||||
| @@ -176,6 +354,7 @@ see the beautiful mess of a dictionary in [mtag.py](https://github.com/9001/copy | ||||
| `--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 | ||||
|  | ||||
| @@ -188,6 +367,11 @@ copyparty can invoke external programs to collect additional metadata for files | ||||
| * `-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 | ||||
|  | ||||
| @@ -195,6 +379,47 @@ copyparty can invoke external programs to collect additional metadata for files | ||||
|   `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  | | ||||
| | directory tree  |  -  |  -  | `*1` | yep  | yep   | yep  | yep | yep  | | ||||
| | up2k            |  -  |  -  | yep  | yep  | yep   | yep  | yep | yep  | | ||||
| | icons work      |  -  |  -  | 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 | | ||||
|  | ||||
| * 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** (22d13d8)  | hits a page fault, works with `?b=u`, file input not-impl, url params are multiplying | | ||||
|  | ||||
|  | ||||
| # client examples | ||||
|  | ||||
| * javascript: dump some state into a file (two separate examples) | ||||
| @@ -213,36 +438,72 @@ copyparty can invoke external programs to collect additional metadata for files | ||||
|   * 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|head -c43;} | ||||
|     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 | ||||
|  | ||||
|  | ||||
| # dependencies | ||||
|  | ||||
| * `jinja2` (is built into the SFX) | ||||
|  | ||||
| **optional,** enables music tags: | ||||
|  | ||||
| ## 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) | ||||
|  | ||||
| **optional,** will eventually enable thumbnails: | ||||
| enable image thumbnails: | ||||
| * `Pillow` (requires py2.7 or py3.5+) | ||||
|  | ||||
| enable video thumbnails: | ||||
| * `ffmpeg` and `ffprobe` somewhere in `$PATH` | ||||
|  | ||||
| enable reading HEIF pictures: | ||||
| * `pyheif-pillow-opener` (requires Linux or a C compiler) | ||||
|  | ||||
| enable reading 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 and will never be imported / evaluated by copyparty | ||||
| these are standalone programs and will never be imported / evaluated by copyparty | ||||
|  | ||||
|  | ||||
| # sfx | ||||
|  | ||||
| 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 | ||||
| * [copyparty-sfx.sh](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.sh) -- smaller, but only for linux and macos | ||||
| 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 (**use sfx.py on systemd**) and it'll unpack and run copyparty, assuming you have python installed of course | ||||
|  | ||||
| @@ -276,18 +537,45 @@ echo $? | ||||
| 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/ | ||||
| ``` | ||||
|  | ||||
| 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: | ||||
|  | ||||
| @@ -302,15 +590,25 @@ in the `scripts` folder: | ||||
|  | ||||
| roughly sorted by priority | ||||
|  | ||||
| * readme.md as epilogue | ||||
| * single sha512 across all up2k chunks? maybe | ||||
| * reduce up2k roundtrips | ||||
|   * start from a chunk index and just go | ||||
|   * terminate client on bad data | ||||
| * drop onto folders | ||||
|  | ||||
| discarded ideas | ||||
|  | ||||
| * 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 | ||||
| * support pillow-simd | ||||
|   * feels like there isn't much point | ||||
| * cache sha512 chunks on client | ||||
|   * too dangerous | ||||
| * comment field | ||||
| * ~~look into android thumbnail cache file format~~ bad idea | ||||
| * 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 | ||||
|   | ||||
| @@ -45,3 +45,19 @@ you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releas | ||||
| # [`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 | ||||
| ``` | ||||
|   | ||||
| @@ -54,6 +54,12 @@ MACOS = platform.system() == "Darwin" | ||||
| info = log = dbg = None | ||||
|  | ||||
|  | ||||
| print("{} v{} @ {}".format( | ||||
|     platform.python_implementation(), | ||||
|     ".".join([str(x) for x in sys.version_info]), | ||||
|     sys.executable)) | ||||
|  | ||||
|  | ||||
| try: | ||||
|     from fuse import FUSE, FuseOSError, Operations | ||||
| except: | ||||
|   | ||||
							
								
								
									
										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() | ||||
							
								
								
									
										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]) | ||||
| @@ -9,6 +9,16 @@ | ||||
| * 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)) | ||||
|   | ||||
| @@ -1,3 +1,8 @@ | ||||
| # when running copyparty behind a reverse-proxy, | ||||
| # make sure that copyparty allows at least as many clients as the proxy does, | ||||
| # so run copyparty with -nc 512 if your nginx has the default limits | ||||
| # (worker_processes 1, worker_connections 512) | ||||
|  | ||||
| upstream cpp { | ||||
| 	server 127.0.0.1:3923; | ||||
| 	keepalive 120; | ||||
|   | ||||
							
								
								
									
										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$" | ||||
| } | ||||
| @@ -2,6 +2,7 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import platform | ||||
| import time | ||||
| import sys | ||||
| import os | ||||
|  | ||||
| @@ -16,12 +17,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": | ||||
|   | ||||
| @@ -12,7 +12,6 @@ import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import signal | ||||
| import shutil | ||||
| import filecmp | ||||
| import locale | ||||
| @@ -24,7 +23,7 @@ from textwrap import dedent | ||||
| from .__init__ import E, WINDOWS, VT100, PY2 | ||||
| from .__version__ import S_VERSION, S_BUILD_DT, CODENAME | ||||
| from .svchub import SvcHub | ||||
| from .util import py_desc, align_tab, IMPLICATIONS | ||||
| from .util import py_desc, align_tab, IMPLICATIONS, alltrace | ||||
|  | ||||
| HAVE_SSL = True | ||||
| try: | ||||
| @@ -56,6 +55,12 @@ 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 warn(msg): | ||||
|     print("\033[1mwarning:\033[0;33m {}\033[0m\n".format(msg)) | ||||
|  | ||||
| @@ -167,7 +172,7 @@ def configure_ssl_ciphers(al): | ||||
|         sys.exit(0) | ||||
|  | ||||
|  | ||||
| def sighandler(signal=None, frame=None): | ||||
| def sighandler(sig=None, frame=None): | ||||
|     msg = [""] * 5 | ||||
|     for th in threading.enumerate(): | ||||
|         msg.append(str(th)) | ||||
| @@ -177,6 +182,160 @@ def sighandler(signal=None, frame=None): | ||||
|     print("\n".join(msg)) | ||||
|  | ||||
|  | ||||
| def stackmon(fp, ival): | ||||
|     ctr = 0 | ||||
|     while True: | ||||
|         ctr += 1 | ||||
|         time.sleep(ival) | ||||
|         st = "{}, {}\n{}".format(ctr, time.time(), alltrace()) | ||||
|         with open(fp, "wb") as f: | ||||
|             f.write(st.encode("utf-8", "replace")) | ||||
|  | ||||
|  | ||||
| def run_argparse(argv, formatter): | ||||
|     ap = argparse.ArgumentParser( | ||||
|         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:cflag:cflag:... | ||||
|                where "permset" is accesslevel followed by username (no separator) | ||||
|                and "cflag" is config flags to set on this volume | ||||
|              | ||||
|             list of cflags: | ||||
|               "cnodupe" rejects existing files (instead of symlinking them) | ||||
|               "ce2d" sets -e2d (all -e2* args can be set using ce2* cflags) | ||||
|               "cd2t" disables metadata collection, overrides -e2t* | ||||
|               "cd2d" disables all database stuff, overrides -e2* | ||||
|  | ||||
|             example:\033[35m | ||||
|               -a ed:hunter2 -v .::r:aed -v ../inc:dump:w:aed:cnodupe  \033[36m | ||||
|               mount current directory at "/" with | ||||
|                * r (read-only) for everyone | ||||
|                * a (read+write) for ed | ||||
|               mount ../inc at "/dump" with | ||||
|                * w (write-only) for everyone | ||||
|                * a (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 | ||||
|             """ | ||||
|         ), | ||||
|     ) | ||||
|     # fmt: off | ||||
|     ap.add_argument("-c", metavar="PATH", type=str, action="append", help="add config file") | ||||
|     ap.add_argument("-nc", metavar="NUM", type=int, default=64, help="max num clients") | ||||
|     ap.add_argument("-j", metavar="CORES", type=int, default=1, help="max num cpu cores") | ||||
|     ap.add_argument("-a", metavar="ACCT", type=str, action="append", help="add account, USER:PASS; example [ed:wark") | ||||
|     ap.add_argument("-v", metavar="VOL", type=str, action="append", help="add volume, SRC:DST:FLAG; example [.::r], [/mnt/nas/music:/music:r:aed") | ||||
|     ap.add_argument("-ed", action="store_true", help="enable ?dots") | ||||
|     ap.add_argument("-emp", action="store_true", help="enable markdown plugins") | ||||
|     ap.add_argument("-mcr", metavar="SEC", type=int, default=60, help="md-editor mod-chk rate") | ||||
|     ap.add_argument("--dotpart", action="store_true", help="dotfile incomplete uploads") | ||||
|     ap.add_argument("--sparse", metavar="MiB", type=int, default=4, help="up2k min.size threshold (mswin-only)") | ||||
|     ap.add_argument("--urlform", metavar="MODE", type=str, default="print,get", help="how to handle url-forms; examples: [stash], [save,get]") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('network options') | ||||
|     ap2.add_argument("-i", metavar="IP", type=str, default="0.0.0.0", help="ip to bind (comma-sep.)") | ||||
|     ap2.add_argument("-p", metavar="PORT", type=str, 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=str, help="set allowed ssl/tls versions; [help] shows available versions; default is what your python version considers safe") | ||||
|     ap2.add_argument("--ciphers", metavar="LIST", 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", 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("-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]]", help="scan all volumes; arguments USER,VOL,FLAGS; example [**,*,ln,p,r]") | ||||
|     ap2.add_argument("--salt", type=str, 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("--log-conn", action="store_true", help="print tcp-server msgs") | ||||
|     ap2.add_argument("--ihead", metavar="HEADER", action='append', help="dump incoming header") | ||||
|     ap2.add_argument("--lf-url", metavar="RE", type=str, 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") | ||||
|     ap2.add_argument("--th-maxage", metavar="SEC", type=int, default=604800, help="max folder age") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('database 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("-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("--hist", metavar="PATH", type=str, help="where to store volume state") | ||||
|     ap2.add_argument("--no-hash", action="store_true", help="disable hashing during e2ds folder scans") | ||||
|     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("-mtm", metavar="M=t,t,t", action="append", type=str, help="add/replace metadata mapping") | ||||
|     ap2.add_argument("-mte", metavar="M,M,M", type=str, 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", action="append", type=str, help="read tag M using bin") | ||||
|     ap2.add_argument("--srch-time", metavar="SEC", type=int, default=30, help="search deadline") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('appearance options') | ||||
|     ap2.add_argument("--css-browser", metavar="L", 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("--stackmon", metavar="P,S", help="write stacktrace to Path every S second") | ||||
|      | ||||
|     return ap.parse_args(args=argv[1:]) | ||||
|     # fmt: on | ||||
|  | ||||
|  | ||||
| def main(argv=None): | ||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||
|     if WINDOWS: | ||||
| @@ -206,95 +365,20 @@ def main(argv=None): | ||||
|         argv[idx] = nk | ||||
|         time.sleep(2) | ||||
|  | ||||
|     ap = argparse.ArgumentParser( | ||||
|         formatter_class=RiceFormatter, | ||||
|         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:cflag:cflag:... | ||||
|                where "permset" is accesslevel followed by username (no separator) | ||||
|                and "cflag" is config flags to set on this volume | ||||
|              | ||||
|             list of cflags: | ||||
|               "cnodupe" rejects existing files (instead of symlinking them) | ||||
|     try: | ||||
|         al = run_argparse(argv, RiceFormatter) | ||||
|     except AssertionError: | ||||
|         al = run_argparse(argv, Dodge11874) | ||||
|  | ||||
|             example:\033[35m | ||||
|               -a ed:hunter2 -v .::r:aed -v ../inc:dump:w:aed:cnodupe  \033[36m | ||||
|               mount current directory at "/" with | ||||
|                * r (read-only) for everyone | ||||
|                * a (read+write) for ed | ||||
|               mount ../inc at "/dump" with | ||||
|                * w (write-only) for everyone | ||||
|                * a (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) | ||||
|  | ||||
|             --ciphers help = available ssl/tls ciphers, | ||||
|             --ssl-ver help = available ssl/tls versions, | ||||
|               default is what python considers safe, usually >= TLS1 | ||||
|             """ | ||||
|         ), | ||||
|     ) | ||||
|     # fmt: off | ||||
|     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 (comma-sep.)") | ||||
|     ap.add_argument("-p", metavar="PORT", type=str, default="3923", help="ports to bind (comma/range)") | ||||
|     ap.add_argument("-nc", metavar="NUM", type=int, default=64, help="max num clients") | ||||
|     ap.add_argument("-j", metavar="CORES", type=int, default=1, help="max num cpu cores") | ||||
|     ap.add_argument("-a", metavar="ACCT", type=str, action="append", help="add account") | ||||
|     ap.add_argument("-v", metavar="VOL", type=str, action="append", help="add volume") | ||||
|     ap.add_argument("-q", action="store_true", help="quiet") | ||||
|     ap.add_argument("--log-conn", action="store_true", help="print tcp-server msgs") | ||||
|     ap.add_argument("-ed", action="store_true", help="enable ?dots") | ||||
|     ap.add_argument("-emp", action="store_true", help="enable markdown plugins") | ||||
|     ap.add_argument("-mcr", metavar="SEC", type=int, default=60, help="md-editor mod-chk rate") | ||||
|     ap.add_argument("-nw", action="store_true", help="disable writes (benchmark)") | ||||
|     ap.add_argument("-nih", action="store_true", help="no info hostname") | ||||
|     ap.add_argument("-nid", action="store_true", help="no info disk-usage") | ||||
|     ap.add_argument("--no-zip", action="store_true", help="disable download as zip/tar") | ||||
|     ap.add_argument("--no-sendfile", action="store_true", help="disable sendfile (for debugging)") | ||||
|     ap.add_argument("--no-scandir", action="store_true", help="disable scandir (for debugging)") | ||||
|     ap.add_argument("--urlform", metavar="MODE", type=str, default="print,get", help="how to handle url-forms") | ||||
|     ap.add_argument("--salt", type=str, default="hunter2", help="up2k file-hash salt") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('database 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("-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("-mtm", metavar="M=t,t,t", action="append", type=str, help="add/replace metadata mapping") | ||||
|     ap2.add_argument("-mte", metavar="M,M,M", type=str, help="tags to index/display (comma-sep.)", | ||||
|         default="circle,album,.tn,artist,title,.bpm,key,.dur,.q") | ||||
|     ap2.add_argument("-mtp", metavar="M=[f,]bin", action="append", type=str, help="read tag M using bin") | ||||
|     ap2.add_argument("--srch-time", metavar="SEC", type=int, default=30, help="search deadline") | ||||
|  | ||||
|     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=str, help="ssl/tls versions to allow") | ||||
|     ap2.add_argument("--ciphers", metavar="LIST", help="set allowed ciphers") | ||||
|     ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info") | ||||
|     ap2.add_argument("--ssl-log", metavar="PATH", help="log master secrets") | ||||
|      | ||||
|     al = ap.parse_args(args=argv[1:]) | ||||
|     # fmt: on | ||||
|     if al.stackmon: | ||||
|         fp, f = al.stackmon.rsplit(",", 1) | ||||
|         f = int(f) | ||||
|         t = threading.Thread( | ||||
|             target=stackmon, | ||||
|             args=(fp, f), | ||||
|         ) | ||||
|         t.daemon = True | ||||
|         t.start() | ||||
|  | ||||
|     # propagate implications | ||||
|     for k1, k2 in IMPLICATIONS: | ||||
| @@ -326,6 +410,9 @@ def main(argv=None): | ||||
|             + "  (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).run() | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| # coding: utf-8 | ||||
|  | ||||
| VERSION = (0, 10, 3) | ||||
| CODENAME = "zip it" | ||||
| BUILD_DT = (2021, 3, 29) | ||||
| VERSION = (0, 11, 22) | ||||
| CODENAME = "the grid" | ||||
| BUILD_DT = (2021, 6, 21) | ||||
|  | ||||
| S_VERSION = ".".join(map(str, VERSION)) | ||||
| S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) | ||||
|   | ||||
| @@ -5,35 +5,49 @@ import re | ||||
| import os | ||||
| import sys | ||||
| import stat | ||||
| import base64 | ||||
| import hashlib | ||||
| import threading | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS | ||||
| from .util import IMPLICATIONS, undot, Pebkac, fsdec, fsenc, statdir, nuprint | ||||
| from .__init__ import WINDOWS | ||||
| from .util import IMPLICATIONS, uncyg, undot, Pebkac, fsdec, fsenc, statdir, nuprint | ||||
|  | ||||
|  | ||||
| class VFS(object): | ||||
|     """single level in the virtual fs""" | ||||
|  | ||||
|     def __init__(self, realpath, vpath, uread=[], uwrite=[], flags={}): | ||||
|     def __init__(self, realpath, vpath, uread=[], uwrite=[], uadm=[], flags={}): | ||||
|         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.uadm = uadm  # users who are regular admins | ||||
|         self.flags = flags  # config switches | ||||
|         self.nodes = {}  # child nodes | ||||
|         self.all_vols = {vpath: self}  # flattened recursive | ||||
|         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 | ||||
|         else: | ||||
|             self.histpath = None | ||||
|             self.all_vols = None | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return "VFS({})".format( | ||||
|             ", ".join( | ||||
|                 "{}={!r}".format(k, self.__dict__[k]) | ||||
|                 for k in "realpath vpath uread uwrite flags".split() | ||||
|                 for k in "realpath vpath uread uwrite uadm flags".split() | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|     def _trk(self, vol): | ||||
|         self.all_vols[vol.vpath] = vol | ||||
|         return vol | ||||
|     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""" | ||||
| @@ -45,18 +59,19 @@ class VFS(object): | ||||
|             name, dst = dst.split("/", 1) | ||||
|             if name in self.nodes: | ||||
|                 # exists; do not manipulate permissions | ||||
|                 return self._trk(self.nodes[name].add(src, dst)) | ||||
|                 return self.nodes[name].add(src, dst) | ||||
|  | ||||
|             vn = VFS( | ||||
|                 "{}/{}".format(self.realpath, name), | ||||
|                 os.path.join(self.realpath, name) if self.realpath else None, | ||||
|                 "{}/{}".format(self.vpath, name).lstrip("/"), | ||||
|                 self.uread, | ||||
|                 self.uwrite, | ||||
|                 self.flags, | ||||
|                 self.uadm, | ||||
|                 self._copy_flags(name), | ||||
|             ) | ||||
|             self._trk(vn) | ||||
|             vn.dbv = self.dbv or self | ||||
|             self.nodes[name] = vn | ||||
|             return self._trk(vn.add(src, dst)) | ||||
|             return vn.add(src, dst) | ||||
|  | ||||
|         if dst in self.nodes: | ||||
|             # leaf exists; return as-is | ||||
| @@ -65,8 +80,26 @@ class VFS(object): | ||||
|         # leaf does not exist; create and keep permissions blank | ||||
|         vp = "{}/{}".format(self.vpath, dst).lstrip("/") | ||||
|         vn = VFS(src, vp) | ||||
|         vn.dbv = self.dbv or self | ||||
|         self.nodes[dst] = vn | ||||
|         return self._trk(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]""" | ||||
| @@ -94,6 +127,7 @@ class VFS(object): | ||||
|         ] | ||||
|  | ||||
|     def get(self, vpath, uname, will_read, will_write): | ||||
|         # type: (str, str, bool, bool) -> tuple[VFS, str] | ||||
|         """returns [vfsnode,fs_remainder] if user has the requested permissions""" | ||||
|         vn, rem = self._find(vpath) | ||||
|  | ||||
| @@ -105,15 +139,45 @@ class VFS(object): | ||||
|  | ||||
|         return vn, 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): | ||||
|         """returns the canonical path (fully-resolved absolute fs path)""" | ||||
|         rp = self.realpath | ||||
|         if rem: | ||||
|             rp += "/" + rem | ||||
|  | ||||
|         return fsdec(os.path.realpath(fsenc(rp))) | ||||
|         try: | ||||
|             return fsdec(os.path.realpath(fsenc(rp))) | ||||
|         except: | ||||
|             if not WINDOWS: | ||||
|                 raise | ||||
|  | ||||
|     def ls(self, rem, uname, scandir, lstat=False): | ||||
|             # 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:\\" | ||||
|             # | ||||
|             # requirements to trigger: | ||||
|             #  * bytestring (not unicode str) | ||||
|             #  * just the drive letter (subfolders are ok) | ||||
|             #  * networked drive (regular disks and vmhgfs are ok) | ||||
|             #  * on an enterprise network (idk, cannot repro with samba) | ||||
|             # | ||||
|             # hits the following exceptions in succession: | ||||
|             #  * access denied at L601: "path = _getfinalpathname(path)" | ||||
|             #  * "cant concat str to bytes" at L621: "return path + tail" | ||||
|             # | ||||
|             return os.path.realpath(rp) | ||||
|  | ||||
|     def ls(self, rem, uname, scandir, incl_wo=False, lstat=False): | ||||
|         # type: (str, str, bool, 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) | ||||
| @@ -121,7 +185,12 @@ class VFS(object): | ||||
|         real.sort() | ||||
|         if not rem: | ||||
|             for name, vn2 in sorted(self.nodes.items()): | ||||
|                 if uname in vn2.uread or "*" in vn2.uread: | ||||
|                 ok = uname in vn2.uread or "*" in vn2.uread | ||||
|  | ||||
|                 if not ok and incl_wo: | ||||
|                     ok = uname in vn2.uwrite or "*" in vn2.uwrite | ||||
|  | ||||
|                 if ok: | ||||
|                     virt_vis[name] = vn2 | ||||
|  | ||||
|             # no vfs nodes in the list of real inodes | ||||
| @@ -129,13 +198,21 @@ class VFS(object): | ||||
|  | ||||
|         return [abspath, real, virt_vis] | ||||
|  | ||||
|     def walk(self, rel, rem, uname, dots, scandir, lstat=False): | ||||
|     def walk(self, rel, rem, seen, uname, dots, scandir, lstat): | ||||
|         """ | ||||
|         recursively yields from ./rem; | ||||
|         rel is a unix-style user-defined vpath (not vfs-related) | ||||
|         """ | ||||
|  | ||||
|         fsroot, vfs_ls, vfs_virt = self.ls(rem, uname, scandir, lstat) | ||||
|         fsroot, vfs_ls, vfs_virt = self.ls( | ||||
|             rem, uname, scandir, incl_wo=False, lstat=lstat | ||||
|         ) | ||||
|  | ||||
|         if seen and not fsroot.startswith(seen[-1]) and fsroot in seen: | ||||
|             print("bailing from symlink loop,\n  {}\n  {}".format(seen[-1], fsroot)) | ||||
|             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)] | ||||
|  | ||||
| @@ -150,7 +227,7 @@ class VFS(object): | ||||
|  | ||||
|             wrel = (rel + "/" + rdir).lstrip("/") | ||||
|             wrem = (rem + "/" + rdir).lstrip("/") | ||||
|             for x in self.walk(wrel, wrem, uname, scandir, lstat): | ||||
|             for x in self.walk(wrel, wrem, seen, uname, dots, scandir, lstat): | ||||
|                 yield x | ||||
|  | ||||
|         for n, vfs in sorted(vfs_virt.items()): | ||||
| @@ -158,18 +235,25 @@ class VFS(object): | ||||
|                 continue | ||||
|  | ||||
|             wrel = (rel + "/" + n).lstrip("/") | ||||
|             for x in vfs.walk(wrel, "", uname, scandir, lstat): | ||||
|             for x in vfs.walk(wrel, "", seen, uname, dots, scandir, lstat): | ||||
|                 yield x | ||||
|  | ||||
|     def zipgen(self, vrem, flt, uname, dots, scandir): | ||||
|         if flt: | ||||
|             flt = {k: True for k in flt} | ||||
|  | ||||
|         for vpath, apath, files, rd, vd in self.walk("", vrem, uname, dots, scandir): | ||||
|         for vpath, apath, files, rd, vd in self.walk( | ||||
|             "", vrem, [], uname, dots, scandir, False | ||||
|         ): | ||||
|             if flt: | ||||
|                 files = [x for x in files if x[0] in flt] | ||||
|                 rd = [x for x in rd if x[0] in flt] | ||||
|                 vd = {x: y for x, y in vd.items() if x 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]])) | ||||
| @@ -196,17 +280,19 @@ class VFS(object): | ||||
|             for f in [{"vp": v, "ap": a, "st": n[1]} for v, a, n in files]: | ||||
|                 yield f | ||||
|  | ||||
|     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 user_tree(self, uname, readable, writable, admin): | ||||
|         is_readable = False | ||||
|         if uname in self.uread or "*" in self.uread: | ||||
|             readable.append(self.vpath) | ||||
|             is_readable = True | ||||
|  | ||||
|         if uname in self.uwrite or "*" in self.uwrite: | ||||
|             writable.append(self.vpath) | ||||
|             if is_readable: | ||||
|                 admin.append(self.vpath) | ||||
|  | ||||
|         for _, vn in sorted(self.nodes.items()): | ||||
|             ret.extend(vn.user_tree(uname, readable, writable)) | ||||
|  | ||||
|         return ret | ||||
|             vn.user_tree(uname, readable, writable, admin) | ||||
|  | ||||
|  | ||||
| class AuthSrv(object): | ||||
| @@ -216,6 +302,7 @@ class AuthSrv(object): | ||||
|         self.args = args | ||||
|         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]:[\\/][^:]*|[^:]*):([^:]*):(.*)$") | ||||
| @@ -226,13 +313,8 @@ class AuthSrv(object): | ||||
|         self.reload() | ||||
|  | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func("auth", msg, c) | ||||
|  | ||||
|     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()} | ||||
|         if self.log_func: | ||||
|             self.log_func("auth", msg, c) | ||||
|  | ||||
|     def laggy_iter(self, iterable): | ||||
|         """returns [value,isFinalValue]""" | ||||
| @@ -244,10 +326,12 @@ class AuthSrv(object): | ||||
|  | ||||
|         yield prev, True | ||||
|  | ||||
|     def _parse_config_file(self, fd, user, mread, mwrite, mflags, mount): | ||||
|     def _parse_config_file(self, fd, user, mread, mwrite, madm, mflags, mount): | ||||
|         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 | ||||
| @@ -274,15 +358,26 @@ class AuthSrv(object): | ||||
|                 mount[vol_dst] = vol_src | ||||
|                 mread[vol_dst] = [] | ||||
|                 mwrite[vol_dst] = [] | ||||
|                 madm[vol_dst] = [] | ||||
|                 mflags[vol_dst] = {} | ||||
|                 continue | ||||
|  | ||||
|             lvl, uname = ln.split(" ") | ||||
|             if len(ln) > 1: | ||||
|                 lvl, uname = ln.split(" ") | ||||
|             else: | ||||
|                 lvl = ln | ||||
|                 uname = "*" | ||||
|  | ||||
|             self._read_vol_str( | ||||
|                 lvl, uname, mread[vol_dst], mwrite[vol_dst], mflags[vol_dst] | ||||
|                 lvl, | ||||
|                 uname, | ||||
|                 mread[vol_dst], | ||||
|                 mwrite[vol_dst], | ||||
|                 madm[vol_dst], | ||||
|                 mflags[vol_dst], | ||||
|             ) | ||||
|  | ||||
|     def _read_vol_str(self, lvl, uname, mr, mw, mf): | ||||
|     def _read_vol_str(self, lvl, uname, mr, mw, ma, mf): | ||||
|         if lvl == "c": | ||||
|             cval = True | ||||
|             if "=" in uname: | ||||
| @@ -300,6 +395,9 @@ class AuthSrv(object): | ||||
|         if lvl in "wa": | ||||
|             mw.append(uname) | ||||
|  | ||||
|         if lvl == "a": | ||||
|             ma.append(uname) | ||||
|  | ||||
|     def _read_volflag(self, flags, name, value, is_list): | ||||
|         if name not in ["mtp"]: | ||||
|             flags[name] = value | ||||
| @@ -323,6 +421,7 @@ class AuthSrv(object): | ||||
|         user = {}  # username:password | ||||
|         mread = {}  # mountpoint:[username] | ||||
|         mwrite = {}  # mountpoint:[username] | ||||
|         madm = {}  # mountpoint:[username] | ||||
|         mflags = {}  # mountpoint:[flag] | ||||
|         mount = {}  # dst:src (mountpoint:realpath) | ||||
|  | ||||
| @@ -340,29 +439,53 @@ class AuthSrv(object): | ||||
|                     raise Exception("invalid -v argument: [{}]".format(v_str)) | ||||
|  | ||||
|                 src, dst, perms = m.groups() | ||||
|                 if WINDOWS: | ||||
|                     src = uncyg(src) | ||||
|  | ||||
|                 # print("\n".join([src, dst, perms])) | ||||
|                 src = fsdec(os.path.abspath(fsenc(src))) | ||||
|                 dst = dst.strip("/") | ||||
|                 mount[dst] = src | ||||
|                 mread[dst] = [] | ||||
|                 mwrite[dst] = [] | ||||
|                 madm[dst] = [] | ||||
|                 mflags[dst] = {} | ||||
|  | ||||
|                 perms = perms.split(":") | ||||
|                 for (lvl, uname) in [[x[0], x[1:]] for x in perms]: | ||||
|                     self._read_vol_str(lvl, uname, mread[dst], mwrite[dst], mflags[dst]) | ||||
|                     self._read_vol_str( | ||||
|                         lvl, uname, mread[dst], mwrite[dst], madm[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, mflags, mount) | ||||
|                     try: | ||||
|                         self._parse_config_file( | ||||
|                             f, user, mread, mwrite, madm, mflags, mount | ||||
|                         ) | ||||
|                     except: | ||||
|                         m = "\n\033[1;31m\nerror in config file {} on line {}:\n\033[0m" | ||||
|                         print(m.format(cfg_fn, self.line_ctr)) | ||||
|                         raise | ||||
|  | ||||
|         # case-insensitive; normalize | ||||
|         if WINDOWS: | ||||
|             cased = {} | ||||
|             for k, v in mount.items(): | ||||
|                 try: | ||||
|                     cased[k] = fsdec(os.path.realpath(fsenc(v))) | ||||
|                 except: | ||||
|                     cased[k] = v | ||||
|  | ||||
|             mount = cased | ||||
|  | ||||
|         if not mount: | ||||
|             # -h says our defaults are CWD at root and read/write for everyone | ||||
|             vfs = VFS(os.path.abspath("."), "", ["*"], ["*"]) | ||||
|         elif "" not in mount: | ||||
|             # there's volumes but no root; make root inaccessible | ||||
|             vfs = VFS(os.path.abspath("."), "") | ||||
|             vfs = VFS(None, "") | ||||
|             vfs.flags["d2d"] = True | ||||
|  | ||||
|         maxdepth = 0 | ||||
| @@ -373,13 +496,20 @@ class AuthSrv(object): | ||||
|  | ||||
|             if dst == "": | ||||
|                 # rootfs was mapped; fully replaces the default CWD vfs | ||||
|                 vfs = VFS(mount[dst], dst, mread[dst], mwrite[dst], mflags[dst]) | ||||
|                 vfs = VFS( | ||||
|                     mount[dst], dst, mread[dst], mwrite[dst], madm[dst], mflags[dst] | ||||
|                 ) | ||||
|                 continue | ||||
|  | ||||
|             v = vfs.add(mount[dst], dst) | ||||
|             v.uread = mread[dst] | ||||
|             v.uwrite = mwrite[dst] | ||||
|             v.uadm = madm[dst] | ||||
|             v.flags = mflags[dst] | ||||
|             v.dbv = None | ||||
|  | ||||
|         vfs.all_vols = {} | ||||
|         vfs.get_all_vols(vfs.all_vols) | ||||
|  | ||||
|         missing_users = {} | ||||
|         for d in [mread, mwrite]: | ||||
| @@ -396,6 +526,67 @@ class AuthSrv(object): | ||||
|             ) | ||||
|             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]) | ||||
|                     try: | ||||
|                         os.makedirs(hpath) | ||||
|                     except: | ||||
|                         pass | ||||
|  | ||||
|                     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 = os.path.realpath(vol.histpath) | ||||
|             if vol.dbv: | ||||
|                 if os.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(): | ||||
| @@ -405,6 +596,10 @@ class AuthSrv(object): | ||||
|             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 | ||||
| @@ -438,8 +633,10 @@ class AuthSrv(object): | ||||
|             # verify tags mentioned by -mt[mp] are used by -mte | ||||
|             local_mtp = {} | ||||
|             local_only_mtp = {} | ||||
|             for a in vol.flags.get("mtp", []) + vol.flags.get("mtm", []): | ||||
|                 a = a.split("=")[0] | ||||
|             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 []: | ||||
| @@ -468,8 +665,10 @@ class AuthSrv(object): | ||||
|                     self.log(m.format(vol.vpath, mtp), 1) | ||||
|                     errors = True | ||||
|  | ||||
|         for mtp in self.args.mtp or []: | ||||
|             mtp = mtp.split("=")[0] | ||||
|         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) | ||||
| @@ -478,6 +677,8 @@ class AuthSrv(object): | ||||
|         if errors: | ||||
|             sys.exit(1) | ||||
|  | ||||
|         vfs.bubble_flags() | ||||
|  | ||||
|         try: | ||||
|             v, _ = vfs.get("/", "*", False, True) | ||||
|             if self.warn_anonwrite and os.getcwd() == v.realpath: | ||||
| @@ -490,7 +691,94 @@ class AuthSrv(object): | ||||
|         with self.mutex: | ||||
|             self.vfs = vfs | ||||
|             self.user = user | ||||
|             self.iuser = self.invert(user) | ||||
|             self.iuser = {v: k for k, v in user.items()} | ||||
|  | ||||
|         # import pprint | ||||
|         # pprint.pprint({"usr": user, "rd": mread, "wr": mwrite, "mnt": mount}) | ||||
|  | ||||
|     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.user.keys()) + ["*"] | ||||
|         else: | ||||
|             users = [users] | ||||
|  | ||||
|         for u in users: | ||||
|             if u not in self.user 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}) | ||||
|         for k, v in self.vfs.all_vols.items(): | ||||
|             self.log("/{}: read({}) write({})".format(k, v.uread, v.uwrite)) | ||||
|  | ||||
|         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) | ||||
|                 except: | ||||
|                     continue | ||||
|  | ||||
|                 atop = vn.realpath | ||||
|                 g = vn.walk("", "", [], u, 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: | ||||
|                         nuprint("\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) | ||||
|   | ||||
| @@ -44,7 +44,9 @@ class BrokerMp(object): | ||||
|             proc.clients = {} | ||||
|             proc.workload = 0 | ||||
|  | ||||
|             thr = threading.Thread(target=self.collector, args=(proc,)) | ||||
|             thr = threading.Thread( | ||||
|                 target=self.collector, args=(proc,), name="mp-collector" | ||||
|             ) | ||||
|             thr.daemon = True | ||||
|             thr.start() | ||||
|  | ||||
| @@ -52,14 +54,19 @@ class BrokerMp(object): | ||||
|             proc.start() | ||||
|  | ||||
|         if not self.args.q: | ||||
|             thr = threading.Thread(target=self.debug_load_balancer) | ||||
|             thr = threading.Thread( | ||||
|                 target=self.debug_load_balancer, name="mp-dbg-loadbalancer" | ||||
|             ) | ||||
|             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: | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
| from copyparty.authsrv import AuthSrv | ||||
|  | ||||
| import sys | ||||
| import time | ||||
| @@ -27,20 +28,23 @@ class MpWorker(object): | ||||
|         self.retpend = {} | ||||
|         self.retpend_mutex = threading.Lock() | ||||
|         self.mutex = threading.Lock() | ||||
|         self.workload_thr_active = False | ||||
|         self.workload_thr_alive = False | ||||
|  | ||||
|         # we inherited signal_handler from parent, | ||||
|         # replace it with something harmless | ||||
|         if not FAKE_MP: | ||||
|             signal.signal(signal.SIGINT, 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 = HttpSrv(self, True) | ||||
|         self.httpsrv.disconnect_func = self.httpdrop | ||||
|  | ||||
|         # 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() | ||||
| @@ -64,6 +68,7 @@ class MpWorker(object): | ||||
|  | ||||
|             # self.logw("work: [{}]".format(d[0])) | ||||
|             if dest == "shutdown": | ||||
|                 self.httpsrv.shutdown() | ||||
|                 self.logw("ok bye") | ||||
|                 sys.exit(0) | ||||
|                 return | ||||
| @@ -75,13 +80,15 @@ class MpWorker(object): | ||||
|  | ||||
|                 if self.args.log_conn: | ||||
|                     self.log("%s %s" % addr, "|%sC-qpop" % ("-" * 4,), c="1;30") | ||||
|                  | ||||
|  | ||||
|                 self.httpsrv.accept(sck, addr) | ||||
|  | ||||
|                 with self.mutex: | ||||
|                     if not self.workload_thr_active: | ||||
|                     if not self.workload_thr_alive: | ||||
|                         self.workload_thr_alive = True | ||||
|                         thr = threading.Thread(target=self.thr_workload) | ||||
|                         thr = threading.Thread( | ||||
|                             target=self.thr_workload, name="mpw-workload" | ||||
|                         ) | ||||
|                         thr.daemon = True | ||||
|                         thr.start() | ||||
|  | ||||
|   | ||||
| @@ -3,6 +3,7 @@ from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import threading | ||||
|  | ||||
| from .authsrv import AuthSrv | ||||
| from .httpsrv import HttpSrv | ||||
| from .broker_util import ExceptionalQueue, try_exec | ||||
|  | ||||
| @@ -14,6 +15,7 @@ class BrokerThr(object): | ||||
|         self.hub = hub | ||||
|         self.log = hub.log | ||||
|         self.args = hub.args | ||||
|         self.asrv = hub.asrv | ||||
|  | ||||
|         self.mutex = threading.Lock() | ||||
|  | ||||
| @@ -23,6 +25,7 @@ class BrokerThr(object): | ||||
|  | ||||
|     def shutdown(self): | ||||
|         # self.log("broker", "shutting down") | ||||
|         self.httpsrv.shutdown() | ||||
|         pass | ||||
|  | ||||
|     def put(self, want_retval, dest, *args): | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,6 +1,7 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| @@ -16,6 +17,9 @@ 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): | ||||
| @@ -30,16 +34,31 @@ class HttpConn(object): | ||||
|         self.hsrv = hsrv | ||||
|  | ||||
|         self.args = hsrv.args | ||||
|         self.auth = hsrv.auth | ||||
|         self.asrv = hsrv.asrv | ||||
|         self.is_mp = hsrv.is_mp | ||||
|         self.cert_path = hsrv.cert_path | ||||
|  | ||||
|         enth = HAVE_PIL and not self.args.no_thumb | ||||
|         self.thumbcli = ThumbCli(hsrv.broker) if enth else None | ||||
|         self.ico = Ico(self.args) | ||||
|  | ||||
|         self.t0 = time.time() | ||||
|         self.stopping = False | ||||
|         self.nbyte = 0 | ||||
|         self.workload = 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 | ||||
| @@ -61,7 +80,7 @@ class HttpConn(object): | ||||
|  | ||||
|     def get_u2idx(self): | ||||
|         if not self.u2idx: | ||||
|             self.u2idx = U2idx(self.args, self.log_func) | ||||
|             self.u2idx = U2idx(self) | ||||
|  | ||||
|         return self.u2idx | ||||
|  | ||||
| @@ -153,7 +172,7 @@ 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: | ||||
| @@ -164,7 +183,12 @@ class HttpConn(object): | ||||
|         if not self.sr: | ||||
|             self.sr = Unrecv(self.s) | ||||
|  | ||||
|         while True: | ||||
|         while not self.stopping: | ||||
|             if self.is_mp: | ||||
|                 self.workload += 50 | ||||
|                 if self.workload >= 2 ** 31: | ||||
|                     self.workload = 100 | ||||
|  | ||||
|             cli = HttpCli(self) | ||||
|             if not cli.run(): | ||||
|                 return | ||||
|   | ||||
| @@ -25,8 +25,8 @@ except ImportError: | ||||
|     sys.exit(1) | ||||
|  | ||||
| from .__init__ import E, MACOS | ||||
| from .httpconn import HttpConn | ||||
| from .authsrv import AuthSrv | ||||
| from .httpconn import HttpConn | ||||
|  | ||||
|  | ||||
| class HttpSrv(object): | ||||
| @@ -35,10 +35,12 @@ class HttpSrv(object): | ||||
|     relying on MpSrv for performance (HttpSrv is just plain threads) | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, broker): | ||||
|     def __init__(self, broker, is_mp=False): | ||||
|         self.broker = broker | ||||
|         self.is_mp = is_mp | ||||
|         self.args = broker.args | ||||
|         self.log = broker.log | ||||
|         self.asrv = broker.asrv | ||||
|  | ||||
|         self.disconnect_func = None | ||||
|         self.mutex = threading.Lock() | ||||
| @@ -46,13 +48,12 @@ class HttpSrv(object): | ||||
|         self.clients = {} | ||||
|         self.workload = 0 | ||||
|         self.workload_thr_alive = False | ||||
|         self.auth = AuthSrv(self.args, self.log) | ||||
|  | ||||
|         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", "msg", "md", "mde"] | ||||
|             for x in ["splash", "browser", "browser2", "msg", "md", "mde"] | ||||
|         } | ||||
|  | ||||
|         cert_path = os.path.join(E.cfg, "cert.pem") | ||||
| @@ -66,7 +67,11 @@ class HttpSrv(object): | ||||
|         if self.args.log_conn: | ||||
|             self.log("%s %s" % addr, "|%sC-cthr" % ("-" * 5,), c="1;30") | ||||
|  | ||||
|         thr = threading.Thread(target=self.thr_client, args=(sck, addr)) | ||||
|         thr = threading.Thread( | ||||
|             target=self.thr_client, | ||||
|             args=(sck, addr), | ||||
|             name="httpsrv-{}-{}".format(addr[0].split(".", 2)[-1][-6:], addr[1]), | ||||
|         ) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
| @@ -75,7 +80,14 @@ class HttpSrv(object): | ||||
|             return len(self.clients) | ||||
|  | ||||
|     def shutdown(self): | ||||
|         self.log("ok bye") | ||||
|         clients = list(self.clients.keys()) | ||||
|         for cli in clients: | ||||
|             try: | ||||
|                 cli.shutdown() | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|         self.log("httpsrv-n", "ok bye") | ||||
|  | ||||
|     def thr_client(self, sck, addr): | ||||
|         """thread managing one tcp client""" | ||||
| @@ -84,32 +96,46 @@ 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() | ||||
|             if self.is_mp: | ||||
|                 self.workload += 50 | ||||
|                 if not self.workload_thr_alive: | ||||
|                     self.workload_thr_alive = True | ||||
|                     thr = threading.Thread( | ||||
|                         target=self.thr_workload, name="httpsrv-workload" | ||||
|                     ) | ||||
|                     thr.daemon = True | ||||
|                     thr.start() | ||||
|  | ||||
|         fno = sck.fileno() | ||||
|         try: | ||||
|             if self.args.log_conn: | ||||
|                 self.log("%s %s" % addr, "|%sC-crun" % ("-" * 6,), c="1;30") | ||||
|  | ||||
|             cli.run() | ||||
|  | ||||
|         except (OSError, socket.error) as ex: | ||||
|             if ex.errno not in [10038, 10054, 107, 57, 9]: | ||||
|                 self.log( | ||||
|                     "%s %s" % addr, | ||||
|                     "run({}): {}".format(fno, ex), | ||||
|                     c=6, | ||||
|                 ) | ||||
|  | ||||
|         finally: | ||||
|             sck = cli.s | ||||
|             if self.args.log_conn: | ||||
|                 self.log("%s %s" % addr, "|%sC-cdone" % ("-" * 7,), c="1;30") | ||||
|  | ||||
|             try: | ||||
|                 fno = sck.fileno() | ||||
|                 sck.shutdown(socket.SHUT_RDWR) | ||||
|                 sck.close() | ||||
|             except (OSError, socket.error) as ex: | ||||
|                 if not MACOS: | ||||
|                     self.log( | ||||
|                         "%s %s" % addr, | ||||
|                         "shut({}): {}".format(sck.fileno(), ex), | ||||
|                         "shut({}): {}".format(fno, ex), | ||||
|                         c="1;30", | ||||
|                     ) | ||||
|                 if ex.errno not in [10038, 10054, 107, 57, 9]: | ||||
|   | ||||
							
								
								
									
										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] | ||||
| @@ -1,19 +1,221 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import json | ||||
| import shutil | ||||
| import subprocess as sp | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS | ||||
| from .util import fsenc, fsdec, REKOBO_LKEY | ||||
| from .util import fsenc, fsdec, uncyg, REKOBO_LKEY | ||||
|  | ||||
| if not PY2: | ||||
|     unicode = str | ||||
|  | ||||
|  | ||||
| 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 os.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 | ||||
|  | ||||
|     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 fmt.get( | ||||
|                 "format_name" | ||||
|             ) in ["mp3", "ogg", "flac"]: | ||||
|                 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: | ||||
|                 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 | ||||
| @@ -35,15 +237,7 @@ class MTag(object): | ||||
|             self.get = self.get_ffprobe | ||||
|             self.prefer_mt = True | ||||
|             # about 20x slower | ||||
|             if PY2: | ||||
|                 cmd = [b"ffprobe", b"-version"] | ||||
|                 try: | ||||
|                     sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE) | ||||
|                 except: | ||||
|                     self.usable = False | ||||
|             else: | ||||
|                 if not shutil.which("ffprobe"): | ||||
|                     self.usable = False | ||||
|             self.usable = HAVE_FFPROBE | ||||
|  | ||||
|             if self.usable and WINDOWS and sys.version_info < (3, 8): | ||||
|                 self.usable = False | ||||
| @@ -52,8 +246,10 @@ class MTag(object): | ||||
|                 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" | ||||
|             self.log(msg.format(or_ffprobe, os.path.basename(sys.executable)), c=1) | ||||
|             msg = "need mutagen{} to read media tags so please run this:\n{}{} -m pip install --user mutagen\n" | ||||
|             self.log( | ||||
|                 msg.format(or_ffprobe, " " * 37, os.path.basename(sys.executable)), c=1 | ||||
|             ) | ||||
|             return | ||||
|  | ||||
|         # https://picard-docs.musicbrainz.org/downloads/MusicBrainz_Picard_Tag_Map.html | ||||
| @@ -201,7 +397,7 @@ class MTag(object): | ||||
|         import mutagen | ||||
|  | ||||
|         try: | ||||
|             md = mutagen.File(abspath, easy=True) | ||||
|             md = mutagen.File(fsenc(abspath), easy=True) | ||||
|             x = md.info.length | ||||
|         except Exception as ex: | ||||
|             return {} | ||||
| @@ -212,7 +408,7 @@ class MTag(object): | ||||
|             try: | ||||
|                 q = int(md.info.bitrate / 1024) | ||||
|             except: | ||||
|                 q = int((os.path.getsize(abspath) / dur) / 128) | ||||
|                 q = int((os.path.getsize(fsenc(abspath)) / dur) / 128) | ||||
|  | ||||
|             ret[".dur"] = [0, dur] | ||||
|             ret[".q"] = [0, q] | ||||
| @@ -222,101 +418,7 @@ class MTag(object): | ||||
|         return self.normalize_tags(ret, md) | ||||
|  | ||||
|     def get_ffprobe(self, abspath): | ||||
|         cmd = [b"ffprobe", b"-hide_banner", b"--", fsenc(abspath)] | ||||
|         p = sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE) | ||||
|         r = p.communicate() | ||||
|         txt = r[1].decode("utf-8", "replace") | ||||
|         txt = [x.rstrip("\r") for x in txt.split("\n")] | ||||
|  | ||||
|         """ | ||||
|         note: | ||||
|           tags which contain newline will be truncated on first \n, | ||||
|           ffprobe emits \n and spacepads the : to align visually | ||||
|         note: | ||||
|           the Stream ln always mentions Audio: if audio | ||||
|           the Stream ln usually has kb/s, is more accurate | ||||
|           the Duration ln always has kb/s | ||||
|           the Metadata: after Chapter may contain BPM info, | ||||
|             title : Tempo: 126.0 | ||||
|  | ||||
|         Input #0, wav, | ||||
|           Metadata: | ||||
|             date : <OK> | ||||
|           Duration: | ||||
|             Chapter # | ||||
|             Metadata: | ||||
|               title : <NG> | ||||
|  | ||||
|         Input #0, mp3, | ||||
|           Metadata: | ||||
|             album : <OK> | ||||
|           Duration: | ||||
|             Stream #0:0: Audio: | ||||
|             Stream #0:1: Video: | ||||
|             Metadata: | ||||
|               comment : <NG> | ||||
|         """ | ||||
|  | ||||
|         ptn_md_beg = re.compile("^( +)Metadata:$") | ||||
|         ptn_md_kv = re.compile("^( +)([^:]+) *: (.*)") | ||||
|         ptn_dur = re.compile("^ *Duration: ([^ ]+)(, |$)") | ||||
|         ptn_br1 = re.compile("^ *Duration: .*, bitrate: ([0-9]+) kb/s(, |$)") | ||||
|         ptn_br2 = re.compile("^ *Stream.*: Audio:.* ([0-9]+) kb/s(, |$)") | ||||
|         ptn_audio = re.compile("^ *Stream .*: Audio: ") | ||||
|         ptn_au_parent = re.compile("^ *(Input #|Stream .*: Audio: )") | ||||
|  | ||||
|         ret = {} | ||||
|         md = {} | ||||
|         in_md = False | ||||
|         is_audio = False | ||||
|         au_parent = False | ||||
|         for ln in txt: | ||||
|             m = ptn_md_kv.match(ln) | ||||
|             if m and in_md and len(m.group(1)) == in_md: | ||||
|                 _, k, v = [x.strip() for x in m.groups()] | ||||
|                 if k != "" and v != "": | ||||
|                     md[k] = [v] | ||||
|                 continue | ||||
|             else: | ||||
|                 in_md = False | ||||
|  | ||||
|             m = ptn_md_beg.match(ln) | ||||
|             if m and au_parent: | ||||
|                 in_md = len(m.group(1)) + 2 | ||||
|                 continue | ||||
|  | ||||
|             au_parent = bool(ptn_au_parent.search(ln)) | ||||
|  | ||||
|             if ptn_audio.search(ln): | ||||
|                 is_audio = True | ||||
|  | ||||
|             m = ptn_dur.search(ln) | ||||
|             if m: | ||||
|                 sec = 0 | ||||
|                 tstr = m.group(1) | ||||
|                 if tstr.lower() != "n/a": | ||||
|                     try: | ||||
|                         tf = tstr.split(",")[0].split(".")[0].split(":") | ||||
|                         for f in tf: | ||||
|                             sec *= 60 | ||||
|                             sec += int(f) | ||||
|                     except: | ||||
|                         self.log("invalid timestr from ffprobe: [{}]".format(tstr), c=3) | ||||
|  | ||||
|                 ret[".dur"] = sec | ||||
|                 m = ptn_br1.search(ln) | ||||
|                 if m: | ||||
|                     ret[".q"] = m.group(1) | ||||
|  | ||||
|             m = ptn_br2.search(ln) | ||||
|             if m: | ||||
|                 ret[".q"] = m.group(1) | ||||
|  | ||||
|         if not is_audio: | ||||
|             return {} | ||||
|  | ||||
|         ret = {k: [0, v] for k, v in ret.items()} | ||||
|  | ||||
|         ret, md = ffprobe(abspath) | ||||
|         return self.normalize_tags(ret, md) | ||||
|  | ||||
|     def get_bin(self, parsers, abspath): | ||||
| @@ -327,10 +429,10 @@ class MTag(object): | ||||
|         env["PYTHONPATH"] = pypath | ||||
|  | ||||
|         ret = {} | ||||
|         for tagname, (binpath, timeout) in parsers.items(): | ||||
|         for tagname, mp in parsers.items(): | ||||
|             try: | ||||
|                 cmd = [sys.executable, binpath, abspath] | ||||
|                 args = {"env": env, "timeout": timeout} | ||||
|                 cmd = [sys.executable, mp.bin, abspath] | ||||
|                 args = {"env": env, "timeout": mp.timeout} | ||||
|  | ||||
|                 if WINDOWS: | ||||
|                     args["creationflags"] = 0x4000 | ||||
| @@ -339,8 +441,16 @@ class MTag(object): | ||||
|  | ||||
|                 cmd = [fsenc(x) for x in cmd] | ||||
|                 v = sp.check_output(cmd, **args).strip() | ||||
|                 if v: | ||||
|                 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 | ||||
|  | ||||
|   | ||||
| @@ -1,3 +1,6 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import tarfile | ||||
| import threading | ||||
| @@ -42,7 +45,7 @@ class StreamTar(object): | ||||
|         fmt = tarfile.GNU_FORMAT | ||||
|         self.tar = tarfile.open(fileobj=self.qfile, mode="w|", format=fmt) | ||||
|  | ||||
|         w = threading.Thread(target=self._gen) | ||||
|         w = threading.Thread(target=self._gen, name="star-gen") | ||||
|         w.daemon = True | ||||
|         w.start() | ||||
|  | ||||
|   | ||||
| @@ -1,3 +1,6 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import time | ||||
| import tempfile | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import threading | ||||
| @@ -9,9 +10,11 @@ from datetime import datetime, timedelta | ||||
| import calendar | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, MACOS, VT100 | ||||
| from .util import mp | ||||
| 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): | ||||
| @@ -35,9 +38,28 @@ class SvcHub(object): | ||||
|         self.log = self._log_disabled if args.q else self._log_enabled | ||||
|  | ||||
|         # initiate all services to manage | ||||
|         self.asrv = AuthSrv(self.args, self.log, False) | ||||
|         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 | ||||
| @@ -48,7 +70,7 @@ class SvcHub(object): | ||||
|         self.broker = Broker(self) | ||||
|  | ||||
|     def run(self): | ||||
|         thr = threading.Thread(target=self.tcpsrv.run) | ||||
|         thr = threading.Thread(target=self.tcpsrv.run, name="svchub-main") | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
| @@ -63,7 +85,20 @@ class SvcHub(object): | ||||
|  | ||||
|             self.tcpsrv.shutdown() | ||||
|             self.broker.shutdown() | ||||
|             print("nailed it") | ||||
|             if self.thumbsrv: | ||||
|                 self.thumbsrv.shutdown() | ||||
|  | ||||
|                 for n in range(200):  # 10s | ||||
|                     time.sleep(0.05) | ||||
|                     if self.thumbsrv.stopped(): | ||||
|                         break | ||||
|  | ||||
|                     if n == 3: | ||||
|                         print("waiting for thumbsrv (10sec)...") | ||||
|  | ||||
|             print("nailed it", end="") | ||||
|         finally: | ||||
|             print("\033[0m") | ||||
|  | ||||
|     def _log_disabled(self, src, msg, c=0): | ||||
|         pass | ||||
|   | ||||
| @@ -1,3 +1,6 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import time | ||||
| import zlib | ||||
| @@ -87,7 +90,7 @@ def gen_hdr(h_pos, fn, sz, lastmod, utf8, crc32, pre_crc): | ||||
|     ret += struct.pack("<LL", vsz, vsz) | ||||
|  | ||||
|     # windows support (the "?" replace below too) | ||||
|     fn = sanitize_fn(fn, "/") | ||||
|     fn = sanitize_fn(fn, ok="/") | ||||
|     bfn = fn.encode("utf-8" if utf8 else "cp437", "replace").replace(b"?", b"_") | ||||
|  | ||||
|     z64_len = len(z64v) * 8 + 4 if z64v else 0 | ||||
|   | ||||
| @@ -21,6 +21,7 @@ class TcpSrv(object): | ||||
|         self.log = hub.log | ||||
|  | ||||
|         self.num_clients = Counter() | ||||
|         self.stopping = False | ||||
|  | ||||
|         ip = "127.0.0.1" | ||||
|         eps = {ip: "local only"} | ||||
| @@ -67,7 +68,7 @@ class TcpSrv(object): | ||||
|             ip, port = srv.getsockname() | ||||
|             self.log("tcpsrv", "listening @ {0}:{1}".format(ip, port)) | ||||
|  | ||||
|         while True: | ||||
|         while not self.stopping: | ||||
|             if self.args.log_conn: | ||||
|                 self.log("tcpsrv", "|%sC-ncli" % ("-" * 1,), c="1;30") | ||||
|  | ||||
| @@ -78,8 +79,18 @@ class TcpSrv(object): | ||||
|             if self.args.log_conn: | ||||
|                 self.log("tcpsrv", "|%sC-acc1" % ("-" * 2,), c="1;30") | ||||
|  | ||||
|             ready, _, _ = select.select(self.srv, [], []) | ||||
|             try: | ||||
|                 # macos throws bad-fd | ||||
|                 ready, _, _ = select.select(self.srv, [], []) | ||||
|             except: | ||||
|                 ready = [] | ||||
|                 if not self.stopping: | ||||
|                     raise | ||||
|  | ||||
|             for srv in ready: | ||||
|                 if self.stopping: | ||||
|                     break | ||||
|  | ||||
|                 sck, addr = srv.accept() | ||||
|                 sip, sport = srv.getsockname() | ||||
|                 if self.args.log_conn: | ||||
| @@ -95,6 +106,13 @@ class TcpSrv(object): | ||||
|                 self.hub.broker.put(False, "httpconn", sck, addr) | ||||
|  | ||||
|     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_ips): | ||||
|   | ||||
							
								
								
									
										55
									
								
								copyparty/th_cli.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								copyparty/th_cli.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| # 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 | ||||
|  | ||||
|  | ||||
| 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 = os.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() | ||||
							
								
								
									
										409
									
								
								copyparty/th_srv.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										409
									
								
								copyparty/th_srv.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,409 @@ | ||||
| # 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 | ||||
| from .util import fsenc, runcmd, Queue, Cooldown, BytesIO, min_ex | ||||
| from .mtag import HAVE_FFMPEG, HAVE_FFPROBE, ffprobe | ||||
|  | ||||
|  | ||||
| if not PY2: | ||||
|     unicode = str | ||||
|  | ||||
|  | ||||
| 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 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 | ||||
|     try: | ||||
|         rd, fn = rem.rsplit("/", 1) | ||||
|     except: | ||||
|         rd = "" | ||||
|         fn = 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) | ||||
|  | ||||
|         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) | ||||
|                 try: | ||||
|                     os.makedirs(thdir) | ||||
|                 except: | ||||
|                     pass | ||||
|  | ||||
|                 inf_path = os.path.join(thdir, "dir.txt") | ||||
|                 if not os.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 = os.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 os.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]: | ||||
|                 os.utime(fsenc(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 = os.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 - os.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)) | ||||
|                 os.unlink(prev_fp) | ||||
|  | ||||
|             prev_b64 = b64 | ||||
|             prev_fp = fp | ||||
|  | ||||
|         return ndirs | ||||
| @@ -7,7 +7,7 @@ import time | ||||
| import threading | ||||
| from datetime import datetime | ||||
|  | ||||
| from .util import u8safe, s3dec, html_escape, Pebkac | ||||
| from .util import s3dec, Pebkac, min_ex | ||||
| from .up2k import up2k_wark_from_hashlist | ||||
|  | ||||
|  | ||||
| @@ -19,13 +19,14 @@ except: | ||||
|  | ||||
|  | ||||
| class U2idx(object): | ||||
|     def __init__(self, args, log_func): | ||||
|         self.args = args | ||||
|         self.log_func = log_func | ||||
|         self.timeout = args.srch_time | ||||
|     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("could not load sqlite3; searchign wqill be disabled") | ||||
|             self.log("your python does not have sqlite3; searching will be disabled") | ||||
|             return | ||||
|  | ||||
|         self.cur = {} | ||||
| @@ -47,57 +48,146 @@ class U2idx(object): | ||||
|         fhash = body["hash"] | ||||
|         wark = up2k_wark_from_hashlist(self.args.salt, fsize, fhash) | ||||
|  | ||||
|         uq = "substr(w,1,16) = ? and w = ?" | ||||
|         uq = "where substr(w,1,16) = ? and w = ?" | ||||
|         uv = [wark[:16], wark] | ||||
|  | ||||
|         try: | ||||
|             return self.run_query(vols, uq, uv, {})[0] | ||||
|         except Exception as ex: | ||||
|             raise Pebkac(500, repr(ex)) | ||||
|             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 | ||||
|  | ||||
|         cur = _open(ptop) | ||||
|         if not cur: | ||||
|         histpath = self.asrv.vfs.histtab[ptop] | ||||
|         db_path = os.path.join(histpath, "up2k.db") | ||||
|         if not os.path.exists(db_path): | ||||
|             return None | ||||
|  | ||||
|         cur = sqlite3.connect(db_path, 2).cursor() | ||||
|         self.cur[ptop] = cur | ||||
|         return cur | ||||
|  | ||||
|     def search(self, vols, body): | ||||
|     def search(self, vols, uq): | ||||
|         """search by query params""" | ||||
|         if not HAVE_SQLITE3: | ||||
|             return [] | ||||
|  | ||||
|         qobj = {} | ||||
|         _conv_sz(qobj, body, "sz_min", "up.sz >= ?") | ||||
|         _conv_sz(qobj, body, "sz_max", "up.sz <= ?") | ||||
|         _conv_dt(qobj, body, "dt_min", "up.mt >= ?") | ||||
|         _conv_dt(qobj, body, "dt_max", "up.mt <= ?") | ||||
|         for seg, dk in [["path", "up.rd"], ["name", "up.fn"]]: | ||||
|             if seg in body: | ||||
|                 _conv_txt(qobj, body, seg, dk) | ||||
|         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 | ||||
|  | ||||
|         uq, uv = _sqlize(qobj) | ||||
|         while True: | ||||
|             uq = uq.strip() | ||||
|             if not uq: | ||||
|                 break | ||||
|  | ||||
|         qobj = {} | ||||
|         if "tags" in body: | ||||
|             _conv_txt(qobj, body, "tags", "mt.v") | ||||
|             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 "adv" in body: | ||||
|             _conv_adv(qobj, body, "adv") | ||||
|             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 | ||||
|  | ||||
|         try: | ||||
|             return self.run_query(vols, uq, uv, qobj) | ||||
|             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, targs): | ||||
|         self.log("qs: {} {} ,  {}".format(uq, repr(uv), repr(targs))) | ||||
|  | ||||
|     def run_query(self, vols, uq, uv): | ||||
|         done_flag = [] | ||||
|         self.active_id = "{:.6f}_{}".format( | ||||
|             time.time(), threading.current_thread().ident | ||||
| @@ -108,39 +198,19 @@ class U2idx(object): | ||||
|                 self.active_id, | ||||
|                 done_flag, | ||||
|             ), | ||||
|             name="u2idx-terminator", | ||||
|         ) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
|         if not targs: | ||||
|             if not uq: | ||||
|                 q = "select * from up" | ||||
|                 v = () | ||||
|             else: | ||||
|                 q = "select * from up where " + uq | ||||
|                 v = tuple(uv) | ||||
|         if not uq or not uv: | ||||
|             q = "select * from up" | ||||
|             v = () | ||||
|         else: | ||||
|             q = "select up.* from up" | ||||
|             keycmp = "substr(up.w,1,16)" | ||||
|             where = [] | ||||
|             v = [] | ||||
|             ctr = 0 | ||||
|             for tq, tv in sorted(targs.items()): | ||||
|                 ctr += 1 | ||||
|                 tq = tq.split("\n")[0] | ||||
|                 keycmp2 = "mt{}.w".format(ctr) | ||||
|                 q += " inner join mt mt{} on {} = {}".format(ctr, keycmp, keycmp2) | ||||
|                 keycmp = keycmp2 | ||||
|                 where.append(tq.replace("mt.", keycmp[:-1])) | ||||
|                 v.append(tv) | ||||
|             q = "select up.* from up " + uq | ||||
|             v = tuple(uv) | ||||
|  | ||||
|             if uq: | ||||
|                 where.append(uq) | ||||
|                 v.extend(uv) | ||||
|  | ||||
|             q += " where " + (" and ".join(where)) | ||||
|  | ||||
|         # self.log("q2: {} {}".format(q, repr(v))) | ||||
|         self.log("qs: {!r} {!r}".format(q, v)) | ||||
|  | ||||
|         ret = [] | ||||
|         lim = 1000 | ||||
| @@ -163,7 +233,7 @@ class U2idx(object): | ||||
|                 if rd.startswith("//") or fn.startswith("//"): | ||||
|                     rd, fn = s3dec(rd, fn) | ||||
|  | ||||
|                 rp = os.path.join(vtop, rd, fn).replace("\\", "/") | ||||
|                 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: | ||||
| @@ -178,6 +248,7 @@ class U2idx(object): | ||||
|                 hit["tags"] = tags | ||||
|  | ||||
|             ret.extend(sret) | ||||
|             # print("[{}] {}".format(ptop, sret)) | ||||
|  | ||||
|         done_flag.append(True) | ||||
|         self.active_id = None | ||||
| @@ -198,84 +269,3 @@ class U2idx(object): | ||||
|  | ||||
|         if identifier == self.active_id: | ||||
|             self.active_cur.connection.interrupt() | ||||
|  | ||||
|  | ||||
| def _open(ptop): | ||||
|     db_path = os.path.join(ptop, ".hist", "up2k.db") | ||||
|     if os.path.exists(db_path): | ||||
|         return sqlite3.connect(db_path).cursor() | ||||
|  | ||||
|  | ||||
| def _conv_sz(q, body, k, sql): | ||||
|     if k in body: | ||||
|         q[sql] = int(float(body[k]) * 1024 * 1024) | ||||
|  | ||||
|  | ||||
| def _conv_dt(q, body, k, sql): | ||||
|     if k not in body: | ||||
|         return | ||||
|  | ||||
|     v = body[k].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: | ||||
|             ts = datetime.strptime(v, fmt).timestamp() | ||||
|             break | ||||
|         except: | ||||
|             ts = None | ||||
|  | ||||
|     if ts: | ||||
|         q[sql] = ts | ||||
|  | ||||
|  | ||||
| def _conv_txt(q, body, k, sql): | ||||
|     for v in body[k].split(" "): | ||||
|         inv = "" | ||||
|         if v.startswith("-"): | ||||
|             inv = "not" | ||||
|             v = v[1:] | ||||
|  | ||||
|         if not v: | ||||
|             continue | ||||
|  | ||||
|         head = "'%'||" | ||||
|         if v.startswith("^"): | ||||
|             head = "" | ||||
|             v = v[1:] | ||||
|  | ||||
|         tail = "||'%'" | ||||
|         if v.endswith("$"): | ||||
|             tail = "" | ||||
|             v = v[:-1] | ||||
|  | ||||
|         qk = "{} {} like {}?{}".format(sql, inv, head, tail) | ||||
|         q[qk + "\n" + v] = u8safe(v) | ||||
|  | ||||
|  | ||||
| def _conv_adv(q, body, k): | ||||
|     ptn = re.compile(r"^(\.?[a-z]+) *(==?|!=|<=?|>=?) *(.*)$") | ||||
|  | ||||
|     parts = body[k].split(" ") | ||||
|     parts = [x.strip() for x in parts if x.strip()] | ||||
|  | ||||
|     for part in parts: | ||||
|         m = ptn.match(part) | ||||
|         if not m: | ||||
|             p = html_escape(part) | ||||
|             raise Pebkac(400, "invalid argument [" + p + "]") | ||||
|  | ||||
|         k, op, v = m.groups() | ||||
|         qk = "mt.k = '{}' and mt.v {} ?".format(k, op) | ||||
|         q[qk + "\n" + v] = u8safe(v) | ||||
|  | ||||
|  | ||||
| def _sqlize(qobj): | ||||
|     keys = [] | ||||
|     values = [] | ||||
|     for k, v in sorted(qobj.items()): | ||||
|         keys.append(k.split("\n")[0]) | ||||
|         values.append(v) | ||||
|  | ||||
|     return " and ".join(keys), values | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -15,8 +15,9 @@ import threading | ||||
| import mimetypes | ||||
| import contextlib | ||||
| import subprocess as sp  # nosec | ||||
| from datetime import datetime | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS | ||||
| from .__init__ import PY2, WINDOWS, ANYWIN | ||||
| from .stolen import surrogateescape | ||||
|  | ||||
| FAKE_MP = False | ||||
| @@ -34,10 +35,12 @@ 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 | ||||
|  | ||||
| surrogateescape.register_surrogateescape() | ||||
| FS_ENCODING = sys.getfilesystemencoding() | ||||
| @@ -45,10 +48,14 @@ 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", | ||||
| @@ -72,6 +79,13 @@ IMPLICATIONS = [ | ||||
| ] | ||||
|  | ||||
|  | ||||
| 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 """ | ||||
| @@ -123,6 +137,32 @@ class Counter(object): | ||||
|             self.v = absval | ||||
|  | ||||
|  | ||||
| class Cooldown(object): | ||||
|     def __init__(self, maxage): | ||||
|         self.maxage = maxage | ||||
|         self.mutex = threading.Lock() | ||||
|         self.hist = {} | ||||
|         self.oldest = 0 | ||||
|  | ||||
|     def poke(self, key): | ||||
|         with self.mutex: | ||||
|             now = time.time() | ||||
|  | ||||
|             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): | ||||
|     """ | ||||
|     undo any number of socket recv ops | ||||
| @@ -153,7 +193,7 @@ class ProgressPrinter(threading.Thread): | ||||
|     """ | ||||
|  | ||||
|     def __init__(self): | ||||
|         threading.Thread.__init__(self) | ||||
|         threading.Thread.__init__(self, name="pp") | ||||
|         self.daemon = True | ||||
|         self.msg = None | ||||
|         self.end = False | ||||
| @@ -168,6 +208,8 @@ class ProgressPrinter(threading.Thread): | ||||
|  | ||||
|             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 | ||||
| @@ -212,6 +254,45 @@ def trace(*args, **kwargs): | ||||
|     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 min_ex(): | ||||
|     et, ev, tb = sys.exc_info() | ||||
|     tb = traceback.extract_tb(tb, 2) | ||||
|     ex = [ | ||||
|         "{} @ {} <{}>: {}".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) | ||||
|  | ||||
|  | ||||
| @contextlib.contextmanager | ||||
| def ren_open(fname, *args, **kwargs): | ||||
|     fdir = kwargs.pop("fdir", None) | ||||
| @@ -222,6 +303,11 @@ def ren_open(fname, *args, **kwargs): | ||||
|             yield {"orz": [f, fname]} | ||||
|             return | ||||
|  | ||||
|     if suffix: | ||||
|         ext = fname.split(".")[-1] | ||||
|         if len(ext) < 7: | ||||
|             suffix += "." + ext | ||||
|  | ||||
|     orig_name = fname | ||||
|     bname = fname | ||||
|     ext = "" | ||||
| @@ -242,7 +328,7 @@ def ren_open(fname, *args, **kwargs): | ||||
|             else: | ||||
|                 fpath = fname | ||||
|  | ||||
|             if suffix and os.path.exists(fpath): | ||||
|             if suffix and os.path.exists(fsenc(fpath)): | ||||
|                 fpath += suffix | ||||
|                 fname += suffix | ||||
|                 ext += suffix | ||||
| @@ -265,7 +351,7 @@ def ren_open(fname, *args, **kwargs): | ||||
|         if not b64: | ||||
|             b64 = (bname + ext).encode("utf-8", "replace") | ||||
|             b64 = hashlib.sha512(b64).digest()[:12] | ||||
|             b64 = base64.urlsafe_b64encode(b64).decode("utf-8").rstrip("=") | ||||
|             b64 = base64.urlsafe_b64encode(b64).decode("utf-8") | ||||
|  | ||||
|         badlen = len(fname) | ||||
|         while len(fname) >= badlen: | ||||
| @@ -521,8 +607,10 @@ def read_header(sr): | ||||
|             else: | ||||
|                 continue | ||||
|  | ||||
|         sr.unrecv(ret[ofs + 4 :]) | ||||
|         return ret[:ofs].decode("utf-8", "surrogateescape").split("\r\n") | ||||
|         if len(ret) > ofs + 4: | ||||
|             sr.unrecv(ret[ofs + 4 :]) | ||||
|  | ||||
|         return ret[:ofs].decode("utf-8", "surrogateescape").lstrip("\r\n").split("\r\n") | ||||
|  | ||||
|  | ||||
| def humansize(sz, terse=False): | ||||
| @@ -560,6 +648,16 @@ def s2hms(s, optional_h=False): | ||||
|     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): | ||||
|     ret = [] | ||||
|     for node in path.split("/"): | ||||
| @@ -576,12 +674,12 @@ def undot(path): | ||||
|     return "/".join(ret) | ||||
|  | ||||
|  | ||||
| def sanitize_fn(fn, ok=""): | ||||
| def sanitize_fn(fn, ok="", bad=[]): | ||||
|     if "/" not in ok: | ||||
|         fn = fn.replace("\\", "/").split("/")[-1] | ||||
|  | ||||
|     if WINDOWS: | ||||
|         for bad, good in [x for x in [ | ||||
|     if ANYWIN: | ||||
|         remap = [ | ||||
|             ["<", "<"], | ||||
|             [">", ">"], | ||||
|             [":", ":"], | ||||
| @@ -591,15 +689,16 @@ def sanitize_fn(fn, ok=""): | ||||
|             ["|", "|"], | ||||
|             ["?", "?"], | ||||
|             ["*", "*"], | ||||
|         ] if x[0] not in ok]: | ||||
|             fn = fn.replace(bad, good) | ||||
|         ] | ||||
|         for a, b in [x for x in remap if x[0] not in ok]: | ||||
|             fn = fn.replace(a, b) | ||||
|  | ||||
|         bad = ["con", "prn", "aux", "nul"] | ||||
|         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 | ||||
|     if fn.lower() in bad: | ||||
|         fn = "_" + fn | ||||
|  | ||||
|     return fn.strip() | ||||
|  | ||||
| @@ -615,17 +714,29 @@ def exclude_dotfiles(filepaths): | ||||
|     return [x for x in filepaths if not x.split("/")[-1].startswith(".")] | ||||
|  | ||||
|  | ||||
| def html_escape(s, quote=False): | ||||
| 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(">", ">") | ||||
|         .replace("\r", "
") | ||||
|         .replace("\n", "
") | ||||
|     ) | ||||
|     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 | ||||
|  | ||||
| @@ -714,6 +825,8 @@ def s3dec(rd, fn): | ||||
|  | ||||
|  | ||||
| def atomic_move(src, dst): | ||||
|     src = fsenc(src) | ||||
|     dst = fsenc(dst) | ||||
|     if not PY2: | ||||
|         os.replace(src, dst) | ||||
|     else: | ||||
| @@ -792,30 +905,36 @@ def yieldfile(fn): | ||||
|  | ||||
|  | ||||
| def hashcopy(actor, fin, fout): | ||||
|     u32_lim = int((2 ** 31) * 0.9) | ||||
|     is_mp = actor.is_mp | ||||
|     hashobj = hashlib.sha512() | ||||
|     tlen = 0 | ||||
|     for buf in fin: | ||||
|         actor.workload += 1 | ||||
|         if actor.workload > u32_lim: | ||||
|             actor.workload = 100  # prevent overflow | ||||
|         if is_mp: | ||||
|             actor.workload += 1 | ||||
|             if actor.workload > 2 ** 31: | ||||
|                 actor.workload = 100 | ||||
|  | ||||
|         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): | ||||
| def sendfile_py(lower, upper, f, s, actor=None): | ||||
|     remains = upper - lower | ||||
|     f.seek(lower) | ||||
|     while remains > 0: | ||||
|         if actor: | ||||
|             actor.workload += 1 | ||||
|             if actor.workload > 2 ** 31: | ||||
|                 actor.workload = 100 | ||||
|  | ||||
|         # time.sleep(0.01) | ||||
|         buf = f.read(min(4096, remains)) | ||||
|         buf = f.read(min(1024 * 32, remains)) | ||||
|         if not buf: | ||||
|             return remains | ||||
|  | ||||
| @@ -905,18 +1024,20 @@ 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) | ||||
|     return MIMES.get(ext) or mimetypes.guess_type(url)[0] or fallback | ||||
|  | ||||
|  | ||||
| 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] | ||||
|  | ||||
|  | ||||
| @@ -928,6 +1049,17 @@ def chkcmd(*argv): | ||||
|     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) | ||||
|   | ||||
							
								
								
									
										583
									
								
								copyparty/web/baguettebox.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										583
									
								
								copyparty/web/baguettebox.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,583 @@ | ||||
| /*! | ||||
|  * 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: 'baguetteBox-open', | ||||
|             titleTag: false, | ||||
|             async: false, | ||||
|             preload: 2, | ||||
|             animation: 'slideIn', | ||||
|             afterShow: null, | ||||
|             afterHide: null, | ||||
|             onChange: null, | ||||
|         }, | ||||
|         overlay, slider, previousButton, nextButton, closeButton, | ||||
|         currentGallery = [], | ||||
|         currentIndex = 0, | ||||
|         isOverlayVisible = false, | ||||
|         touch = {},  // start-pos | ||||
|         touchFlag = false,  // busy | ||||
|         regex = /.+\.(gif|jpe?g|png|webp)/i, | ||||
|         data = {},  // all galleries | ||||
|         imagesElements = [], | ||||
|         documentLastFocus = null; | ||||
|  | ||||
|     var overlayClickHandler = function (event) { | ||||
|         if (event.target.id.indexOf('baguette-img') !== -1) { | ||||
|             hideOverlay(); | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     var touchstartHandler = function (event) { | ||||
|         touch.count++; | ||||
|         if (touch.count > 1) { | ||||
|             touch.multitouch = true; | ||||
|         } | ||||
|         touch.startX = event.changedTouches[0].pageX; | ||||
|         touch.startY = event.changedTouches[0].pageY; | ||||
|     }; | ||||
|     var touchmoveHandler = function (event) { | ||||
|         if (touchFlag || touch.multitouch) { | ||||
|             return; | ||||
|         } | ||||
|         event.preventDefault ? event.preventDefault() : event.returnValue = false; | ||||
|         var touchEvent = event.touches[0] || event.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 (event) { | ||||
|         if (overlay.style.display === 'block' && (overlay.contains && !overlay.contains(event.target))) { | ||||
|             event.stopPropagation(); | ||||
|             initFocus(); | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     function run(selector, userOptions) { | ||||
|         buildOverlay(); | ||||
|         removeFromCache(selector); | ||||
|         return bindImageClickListeners(selector, userOptions); | ||||
|     } | ||||
|  | ||||
|     function bindImageClickListeners(selector, userOptions) { | ||||
|         var galleryNodeList = document.querySelectorAll(selector); | ||||
|         var selectorData = { | ||||
|             galleries: [], | ||||
|             nodeList: galleryNodeList | ||||
|         }; | ||||
|         data[selector] = selectorData; | ||||
|  | ||||
|         [].forEach.call(galleryNodeList, function (galleryElement) { | ||||
|             if (userOptions && userOptions.filter) { | ||||
|                 regex = userOptions.filter; | ||||
|             } | ||||
|  | ||||
|             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 regex.test(element.href); | ||||
|                 } | ||||
|             }); | ||||
|             if (tagsNodeList.length === 0) { | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             var gallery = []; | ||||
|             [].forEach.call(tagsNodeList, function (imageElement, imageIndex) { | ||||
|                 var imageElementClickHandler = function (event) { | ||||
|                     if (event && event.ctrlKey) | ||||
|                         return true; | ||||
|  | ||||
|                     event.preventDefault ? event.preventDefault() : event.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('baguetteBox-overlay'); | ||||
|         if (overlay) { | ||||
|             slider = ebi('baguetteBox-slider'); | ||||
|             previousButton = ebi('previous-button'); | ||||
|             nextButton = ebi('next-button'); | ||||
|             closeButton = ebi('close-button'); | ||||
|             return; | ||||
|         } | ||||
|         overlay = mknod('div'); | ||||
|         overlay.setAttribute('role', 'dialog'); | ||||
|         overlay.id = 'baguetteBox-overlay'; | ||||
|         document.getElementsByTagName('body')[0].appendChild(overlay); | ||||
|  | ||||
|         slider = mknod('div'); | ||||
|         slider.id = 'baguetteBox-slider'; | ||||
|         overlay.appendChild(slider); | ||||
|  | ||||
|         previousButton = mknod('button'); | ||||
|         previousButton.setAttribute('type', 'button'); | ||||
|         previousButton.id = 'previous-button'; | ||||
|         previousButton.setAttribute('aria-label', 'Previous'); | ||||
|         previousButton.innerHTML = '<'; | ||||
|         overlay.appendChild(previousButton); | ||||
|  | ||||
|         nextButton = mknod('button'); | ||||
|         nextButton.setAttribute('type', 'button'); | ||||
|         nextButton.id = 'next-button'; | ||||
|         nextButton.setAttribute('aria-label', 'Next'); | ||||
|         nextButton.innerHTML = '>'; | ||||
|         overlay.appendChild(nextButton); | ||||
|  | ||||
|         closeButton = mknod('button'); | ||||
|         closeButton.setAttribute('type', 'button'); | ||||
|         closeButton.id = 'close-button'; | ||||
|         closeButton.setAttribute('aria-label', 'Close'); | ||||
|         closeButton.innerHTML = '×'; | ||||
|         overlay.appendChild(closeButton); | ||||
|  | ||||
|         previousButton.className = nextButton.className = closeButton.className = 'baguetteBox-button'; | ||||
|  | ||||
|         bindEvents(); | ||||
|     } | ||||
|  | ||||
|     function keyDownHandler(event) { | ||||
|         switch (event.keyCode) { | ||||
|             case 37: // Left | ||||
|                 showPreviousImage(); | ||||
|                 break; | ||||
|             case 39: // Right | ||||
|                 showNextImage(); | ||||
|                 break; | ||||
|             case 27: // Esc | ||||
|                 hideOverlay(); | ||||
|                 break; | ||||
|             case 36: // Home | ||||
|                 showFirstImage(event); | ||||
|                 break; | ||||
|             case 35: // End | ||||
|                 showLastImage(event); | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     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(previousButton, 'click', showPreviousImage); | ||||
|         bind(nextButton, 'click', showNextImage); | ||||
|         bind(closeButton, 'click', hideOverlay); | ||||
|         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(previousButton, 'click', showPreviousImage); | ||||
|         unbind(nextButton, 'click', showNextImage); | ||||
|         unbind(closeButton, 'click', hideOverlay); | ||||
|         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('baguetteBox-figure-' + i); | ||||
|             imagesCaptionsIds.push('baguetteBox-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; | ||||
|         } | ||||
|  | ||||
|         previousButton.style.display = nextButton.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); | ||||
|         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; | ||||
|         initFocus(); | ||||
|         isOverlayVisible = true; | ||||
|     } | ||||
|  | ||||
|     function initFocus() { | ||||
|         if (options.buttons) { | ||||
|             previousButton.focus(); | ||||
|         } else { | ||||
|             closeButton.focus(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function hideOverlay(e) { | ||||
|         ev(e); | ||||
|         if (options.noScrollbars) { | ||||
|             document.documentElement.style.overflowY = 'auto'; | ||||
|             document.body.style.overflowY = 'auto'; | ||||
|         } | ||||
|         if (overlay.style.display === 'none') { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         unbind(document, 'keydown', keyDownHandler); | ||||
|         // 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); | ||||
|             } | ||||
|             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.getElementsByTagName('img')[0]) { | ||||
|             // image is loaded, cb and bail | ||||
|             if (callback) { | ||||
|                 callback(); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         var imageElement = galleryItem.imageElement, | ||||
|             imageSrc = imageElement.href, | ||||
|             thumbnailElement = imageElement.getElementsByTagName('img')[0], | ||||
|             imageCaption = typeof options.captions === 'function' ? | ||||
|                 options.captions.call(currentGallery, imageElement) : | ||||
|                 imageElement.getAttribute('data-caption') || imageElement.title; | ||||
|  | ||||
|         var figure = mknod('figure'); | ||||
|         figure.id = 'baguetteBox-figure-' + index; | ||||
|         figure.innerHTML = '<div class="baguetteBox-spinner">' + | ||||
|             '<div class="baguetteBox-double-bounce1"></div>' + | ||||
|             '<div class="baguetteBox-double-bounce2"></div>' + | ||||
|             '</div>'; | ||||
|  | ||||
|         if (options.captions && imageCaption) { | ||||
|             var figcaption = mknod('figcaption'); | ||||
|             figcaption.id = 'baguetteBox-figcaption-' + index; | ||||
|             figcaption.innerHTML = imageCaption; | ||||
|             figure.appendChild(figcaption); | ||||
|         } | ||||
|         imageContainer.appendChild(figure); | ||||
|  | ||||
|         var image = mknod('img'); | ||||
|         image.onload = function () { | ||||
|             // Remove loader element | ||||
|             var spinner = document.querySelector('#baguette-img-' + index + ' .baguetteBox-spinner'); | ||||
|             figure.removeChild(spinner); | ||||
|             if (!options.async && callback) { | ||||
|                 callback(); | ||||
|             } | ||||
|         }; | ||||
|         image.setAttribute('src', imageSrc); | ||||
|         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(event) { | ||||
|         if (event) { | ||||
|             event.preventDefault(); | ||||
|         } | ||||
|         return show(0); | ||||
|     } | ||||
|  | ||||
|     function showLastImage(event) { | ||||
|         if (event) { | ||||
|             event.preventDefault(); | ||||
|         } | ||||
|         return show(currentGallery.length - 1); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Move the gallery to a specific index | ||||
|      * @param `index` {number} - the position of the image | ||||
|      * @param `gallery` {array} - gallery which should be opened, if omitted assumes the currently opened one | ||||
|      * @return {boolean} - true on success or false if the index is invalid | ||||
|      */ | ||||
|     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; | ||||
|         } | ||||
|  | ||||
|         currentIndex = index; | ||||
|         loadImage(currentIndex, function () { | ||||
|             preloadNext(currentIndex); | ||||
|             preloadPrev(currentIndex); | ||||
|         }); | ||||
|         updateOffset(); | ||||
|  | ||||
|         if (options.onChange) { | ||||
|             options.onChange(currentIndex, imagesElements.length); | ||||
|         } | ||||
|  | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Triggers the bounce animation | ||||
|      * @param {('left'|'right')} direction - Direction of the movement | ||||
|      */ | ||||
|     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)'; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     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); | ||||
|         document.getElementsByTagName('body')[0].removeChild(ebi('baguetteBox-overlay')); | ||||
|         data = {}; | ||||
|         currentGallery = []; | ||||
|         currentIndex = 0; | ||||
|     } | ||||
|  | ||||
|     return { | ||||
|         run: run, | ||||
|         show: show, | ||||
|         showNext: showNextImage, | ||||
|         showPrevious: showPreviousImage, | ||||
|         hide: hideOverlay, | ||||
|         destroy: destroyPlugin | ||||
|     }; | ||||
| })(); | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -2,141 +2,134 @@ | ||||
| <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 }}"> | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/upload.css{{ ts }}"> | ||||
| 	<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> | ||||
|     <div id="ops"> | ||||
|         <a href="#" data-dest="" data-desc="close submenu">---</a> | ||||
|         <a href="#" data-perm="read" data-dest="search" data-desc="search for files by attributes, path/name, music tags, or any combination of those.<br /><br /><code>foo bar</code> = must contain both foo and bar,<br /><code>foo -bar</code> = must contain foo but not bar,<br /><code>^yana .opus$</code> = must start with yana and have the opus extension">🔎</a> | ||||
|         {%- if have_up2k_idx %} | ||||
|         <a href="#" data-dest="up2k" data-desc="up2k: upload files (if you have write-access) or toggle into the search-mode and drag files onto the search button to see if they exist somewhere on the server">🚀</a> | ||||
|         {%- else %} | ||||
|         <a href="#" data-perm="write" data-dest="up2k" data-desc="up2k: upload files with resume support (close your browser and drop the same files in later)">🚀</a> | ||||
|         {%- endif %} | ||||
|         <a href="#" data-perm="write" data-dest="bup" data-desc="bup: basic uploader, even supports netscape 4.0">🎈</a> | ||||
|         <a href="#" data-perm="write" data-dest="mkdir" data-desc="mkdir: create a new directory">📂</a> | ||||
|         <a href="#" data-perm="write" data-dest="new_md" data-desc="new-md: create a new markdown document">📝</a> | ||||
|         <a href="#" data-perm="write" data-dest="msg" data-desc="msg: send a message to the server log">📟</a> | ||||
|         <a href="#" data-dest="cfg" data-desc="configuration options">⚙️</a> | ||||
|         <div id="opdesc"></div> | ||||
|     </div> | ||||
| 	<div id="ops"></div> | ||||
|  | ||||
|     <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_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> | ||||
|  | ||||
|     {%- include 'upload.html' %} | ||||
| 	<div id="op_player" class="opview opbox opwide"></div> | ||||
|  | ||||
|     <div id="op_cfg" class="opview opbox"> | ||||
|         <h3>key notation</h3> | ||||
|         <div id="key_notation"></div> | ||||
|         {%- if have_zip %} | ||||
|         <h3>folder download</h3> | ||||
|         <div id="arc_fmt"></div> | ||||
|         {%- endif %} | ||||
|         <h3>tooltips</h3> | ||||
|         <div><a id="tooltips" class="tglbtn" href="#">enable</a></div> | ||||
|     </div> | ||||
|      | ||||
|     <h1 id="path"> | ||||
|         <a href="#" id="entree">🌲</a> | ||||
|         {%- for n in vpnodes %} | ||||
|         <a href="/{{ n[0] }}">{{ n[1] }}</a> | ||||
|         {%- endfor %} | ||||
|     </h1> | ||||
|      | ||||
|     <div id="tree"> | ||||
|         <a href="#" id="detree">🍞...</a> | ||||
|         <a href="#" step="2" id="twobytwo">+</a> | ||||
|         <a href="#" step="-2" id="twig">–</a> | ||||
|         <a href="#" class="tglbtn" id="dyntree">a</a> | ||||
|         <ul id="treeul"></ul> | ||||
|         <div id="thx_ff"> </div> | ||||
|     </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="mkdir"> | ||||
| 		</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="create 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"> | ||||
| 		</form> | ||||
| 	</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 directory tree$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> | ||||
| 	<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> | ||||
| 	<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.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> | ||||
| <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> | ||||
|      | ||||
|     <div id="epi" class="logue">{{ logues[1] }}</div> | ||||
| 		</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> | ||||
|  | ||||
|     {%- if srv_info %} | ||||
|     <div id="srv_info"><span>{{ srv_info }}</span></div> | ||||
|     {%- endif %} | ||||
| 	{%- if srv_info %} | ||||
| 	<div id="srv_info"><span>{{ srv_info }}</span></div> | ||||
| 	{%- endif %} | ||||
|  | ||||
|     <div id="widget"> | ||||
|         <div id="wtoggle"> | ||||
|             <span> | ||||
|                 <a href="#" id="selall">sel.<br />all</a> | ||||
|                 <a href="#" id="selinv">sel.<br />inv.</a> | ||||
|                 <a href="#" id="selzip">zip</a> | ||||
|             </span> | ||||
|             ♫ | ||||
|         </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> | ||||
| 	<div id="widget"></div> | ||||
|  | ||||
|     <script> | ||||
|         var tag_order_cfg = {{ tag_order }}; | ||||
|     </script> | ||||
|     <script src="/.cpr/util.js{{ ts }}"></script> | ||||
|     <script src="/.cpr/browser.js{{ ts }}"></script> | ||||
|     <script src="/.cpr/up2k.js{{ ts }}"></script> | ||||
|     <script> | ||||
|         apply_perms({{ perms }}); | ||||
|     </script> | ||||
| 	<script> | ||||
| 		var perms = {{ perms }}, | ||||
| 			tag_order_cfg = {{ tag_order }}, | ||||
| 			have_up2k_idx = {{ have_up2k_idx|tojson }}, | ||||
| 			have_tags_idx = {{ have_tags_idx|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 | 
| @@ -50,6 +50,9 @@ pre code:last-child { | ||||
| 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; | ||||
| @@ -591,12 +594,3 @@ blink { | ||||
| 		color: #940; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* | ||||
| *[data-ln]:before { | ||||
| 	content: attr(data-ln); | ||||
| 	font-size: .8em; | ||||
| 	margin: 0 .4em; | ||||
| 	color: #f0c; | ||||
| } | ||||
| */ | ||||
| @@ -138,10 +138,10 @@ var md_opt = { | ||||
|         document.documentElement.setAttribute("class", dark ? "dark" : ""); | ||||
|         btn.innerHTML = "go " + (dark ? "light" : "dark"); | ||||
|         if (window.localStorage) | ||||
|             localStorage.setItem('darkmode', dark ? 1 : 0); | ||||
|             localStorage.setItem('lightmode', dark ? 0 : 1); | ||||
|     }; | ||||
|     btn.onclick = toggle; | ||||
|     if (window.localStorage && localStorage.getItem('darkmode') == 1) | ||||
|     if (window.localStorage && localStorage.getItem('lightmode') != 1) | ||||
| 		toggle(); | ||||
| })(); | ||||
|  | ||||
|   | ||||
| @@ -46,7 +46,7 @@ function statify(obj) { | ||||
|     var ua = navigator.userAgent; | ||||
|     if (ua.indexOf(') Gecko/') !== -1 && /Linux| Mac /.exec(ua)) { | ||||
|         // necessary on ff-68.7 at least | ||||
|         var s = document.createElement('style'); | ||||
|         var s = mknod('style'); | ||||
|         s.innerHTML = '@page { margin: .5in .6in .8in .6in; }'; | ||||
|         console.log(s.innerHTML); | ||||
|         document.head.appendChild(s); | ||||
| @@ -175,12 +175,12 @@ function md_plug_err(ex, js) { | ||||
|         msg = "Line " + ln + ", " + msg; | ||||
|         var lns = js.split('\n'); | ||||
|         if (ln < lns.length) { | ||||
|             o = document.createElement('span'); | ||||
|             o = mknod('span'); | ||||
|             o.style.cssText = 'color:#ac2;font-size:.9em;font-family:scp;display:block'; | ||||
|             o.textContent = lns[ln - 1]; | ||||
|         } | ||||
|     } | ||||
|     errbox = document.createElement('div'); | ||||
|     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; | ||||
|   | ||||
| @@ -1,126 +1,125 @@ | ||||
| #toc { | ||||
|     display: none; | ||||
| 	display: none; | ||||
| } | ||||
| #mtw { | ||||
|     display: block; | ||||
|     position: fixed; | ||||
|     left: .5em; | ||||
|     bottom: 0; | ||||
|     width: calc(100% - 56em); | ||||
| 	display: block; | ||||
| 	position: fixed; | ||||
| 	left: .5em; | ||||
| 	bottom: 0; | ||||
| 	width: calc(100% - 56em); | ||||
| } | ||||
| #mw { | ||||
|     left: calc(100% - 55em); | ||||
|     overflow-y: auto; | ||||
|     position: fixed; | ||||
|     bottom: 0; | ||||
| 	left: calc(100% - 55em); | ||||
| 	overflow-y: auto; | ||||
| 	position: fixed; | ||||
| 	bottom: 0; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* single-screen */ | ||||
| #mtw.preview, | ||||
| #mw.editor { | ||||
|     opacity: 0; | ||||
|     z-index: 1; | ||||
| 	opacity: 0; | ||||
| 	z-index: 1; | ||||
| } | ||||
| #mw.preview, | ||||
| #mtw.editor { | ||||
|     z-index: 5; | ||||
| 	z-index: 5; | ||||
| } | ||||
| #mtw.single, | ||||
| #mw.single { | ||||
|     margin: 0; | ||||
|     left: 1em; | ||||
|     left: max(1em, calc((100% - 56em) / 2)); | ||||
| 	margin: 0; | ||||
| 	left: 1em; | ||||
| 	left: max(1em, calc((100% - 56em) / 2)); | ||||
| } | ||||
| #mtw.single { | ||||
|     width: 55em; | ||||
|     width: min(55em, calc(100% - 2em)); | ||||
| 	width: 55em; | ||||
| 	width: min(55em, calc(100% - 2em)); | ||||
| } | ||||
|  | ||||
|  | ||||
| #mp { | ||||
|     position: relative; | ||||
| 	position: relative; | ||||
| } | ||||
| #mt, #mtr { | ||||
|     width: 100%; | ||||
|     height: calc(100% - 1px); | ||||
|     color: #444; | ||||
|     background: #f7f7f7; | ||||
|     border: 1px solid #999; | ||||
|     outline: none; | ||||
|     padding: 0; | ||||
|     margin: 0; | ||||
|     font-family: 'consolas', monospace, monospace; | ||||
|     white-space: pre-wrap; | ||||
|     word-break: break-word; | ||||
|     overflow-wrap: break-word; | ||||
|     word-wrap: break-word; /*ie*/ | ||||
|     overflow-y: scroll; | ||||
|     line-height: 1.3em; | ||||
|     font-size: .9em; | ||||
|     position: relative; | ||||
|     scrollbar-color: #eb0 #f7f7f7; | ||||
| 	width: 100%; | ||||
| 	height: calc(100% - 1px); | ||||
| 	color: #444; | ||||
| 	background: #f7f7f7; | ||||
| 	border: 1px solid #999; | ||||
| 	outline: none; | ||||
| 	padding: 0; | ||||
| 	margin: 0; | ||||
| 	font-family: '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; | ||||
| 	color: #eee; | ||||
| 	background: #222; | ||||
| 	border: 1px solid #777; | ||||
| 	scrollbar-color: #b80 #282828; | ||||
| } | ||||
| #mtr { | ||||
|     position: absolute; | ||||
|     top: 0; | ||||
|     left: 0; | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| } | ||||
| #save.force-save { | ||||
|     color: #400; | ||||
|     background: #f97; | ||||
|     border-radius: .15em; | ||||
| 	color: #400; | ||||
| 	background: #f97; | ||||
| 	border-radius: .15em; | ||||
| } | ||||
| html.dark #save.force-save { | ||||
|     color: #fca; | ||||
|     background: #720; | ||||
| 	color: #fca; | ||||
| 	background: #720; | ||||
| } | ||||
| #save.disabled { | ||||
|     opacity: .4; | ||||
| 	opacity: .4; | ||||
| } | ||||
| #helpbox, | ||||
| #toast { | ||||
|     background: #f7f7f7; | ||||
|     border-radius: .4em; | ||||
|     z-index: 9001; | ||||
| 	background: #f7f7f7; | ||||
| 	border-radius: .4em; | ||||
| 	z-index: 9001; | ||||
| } | ||||
| #helpbox { | ||||
|     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; | ||||
| 	display: none; | ||||
| 	position: fixed; | ||||
| 	padding: 2em; | ||||
| 	top: 4em; | ||||
| 	overflow-y: auto; | ||||
| 	box-shadow: 0 .5em 2em #777; | ||||
| 	height: calc(100% - 12em); | ||||
| 	left: calc(50% - 15em); | ||||
| 	right: 0; | ||||
| 	width: 30em; | ||||
| } | ||||
| #helpclose { | ||||
|     display: block; | ||||
| 	display: block; | ||||
| } | ||||
| html.dark #helpbox { | ||||
|     box-shadow: 0 .5em 2em #444; | ||||
| 	box-shadow: 0 .5em 2em #444; | ||||
| } | ||||
| html.dark #helpbox, | ||||
| html.dark #toast { | ||||
|     background: #222; | ||||
|     border: 1px solid #079; | ||||
|     border-width: 1px 0; | ||||
| 	background: #222; | ||||
| 	border: 1px solid #079; | ||||
| 	border-width: 1px 0; | ||||
| } | ||||
| #toast { | ||||
|     font-weight: bold; | ||||
|     text-align: center; | ||||
|     padding: .6em 0; | ||||
|     position: fixed; | ||||
|     z-index: 9001; | ||||
|     top: 30%; | ||||
|     transition: opacity 0.2s ease-in-out; | ||||
|     opacity: 1; | ||||
| 	font-weight: bold; | ||||
| 	text-align: center; | ||||
| 	padding: .6em 0; | ||||
| 	position: fixed; | ||||
| 	top: 30%; | ||||
| 	transition: opacity 0.2s ease-in-out; | ||||
| 	opacity: 1; | ||||
| } | ||||
|   | ||||
| @@ -16,7 +16,7 @@ var dom_sbs = ebi('sbs'); | ||||
| var dom_nsbs = ebi('nsbs'); | ||||
| var dom_tbox = ebi('toolsbox'); | ||||
| var dom_ref = (function () { | ||||
|     var d = document.createElement('div'); | ||||
|     var d = mknod('div'); | ||||
|     d.setAttribute('id', 'mtr'); | ||||
|     dom_swrap.appendChild(d); | ||||
|     d = ebi('mtr'); | ||||
| @@ -71,7 +71,7 @@ var map_src = []; | ||||
| var map_pre = []; | ||||
| function genmap(dom, oldmap) { | ||||
|     var find = nlines; | ||||
|     while (oldmap && find --> 0) { | ||||
|     while (oldmap && find-- > 0) { | ||||
|         var tmap = genmapq(dom, '*[data-ln="' + find + '"]'); | ||||
|         if (!tmap || !tmap.length) | ||||
|             continue; | ||||
| @@ -94,7 +94,7 @@ var nlines = 0; | ||||
| var draw_md = (function () { | ||||
|     var delay = 1; | ||||
|     function draw_md() { | ||||
|         var t0 = new Date().getTime(); | ||||
|         var t0 = Date.now(); | ||||
|         var src = dom_src.value; | ||||
|         convert_markdown(src, dom_pre); | ||||
|  | ||||
| @@ -110,7 +110,7 @@ var draw_md = (function () { | ||||
|  | ||||
|         cls(ebi('save'), 'disabled', src == server_md); | ||||
|  | ||||
|         var t1 = new Date().getTime(); | ||||
|         var t1 = Date.now(); | ||||
|         delay = t1 - t0 > 100 ? 25 : 1; | ||||
|     } | ||||
|  | ||||
| @@ -252,7 +252,7 @@ function Modpoll() { | ||||
|         } | ||||
|  | ||||
|         console.log('modpoll...'); | ||||
|         var url = (document.location + '').split('?')[0] + '?raw&_=' + new Date().getTime(); | ||||
|         var url = (document.location + '').split('?')[0] + '?raw&_=' + Date.now(); | ||||
|         var xhr = new XMLHttpRequest(); | ||||
|         xhr.modpoll = this; | ||||
|         xhr.open('GET', url, true); | ||||
| @@ -399,7 +399,7 @@ function save_cb() { | ||||
|  | ||||
| function run_savechk(lastmod, txt, btn, ntry) { | ||||
|     // download the saved doc from the server and compare | ||||
|     var url = (document.location + '').split('?')[0] + '?raw&_=' + new Date().getTime(); | ||||
|     var url = (document.location + '').split('?')[0] + '?raw&_=' + Date.now(); | ||||
|     var xhr = new XMLHttpRequest(); | ||||
|     xhr.open('GET', url, true); | ||||
|     xhr.responseType = 'text'; | ||||
| @@ -455,7 +455,7 @@ function toast(autoclose, style, width, msg) { | ||||
|         ok.parentNode.removeChild(ok); | ||||
|  | ||||
|     style = "width:" + width + "em;left:calc(50% - " + (width / 2) + "em);" + style; | ||||
|     ok = document.createElement('div'); | ||||
|     ok = mknod('div'); | ||||
|     ok.setAttribute('id', 'toast'); | ||||
|     ok.setAttribute('style', style); | ||||
|     ok.innerHTML = msg; | ||||
| @@ -1049,7 +1049,7 @@ action_stack = (function () { | ||||
|         var p1 = from.length, | ||||
|             p2 = to.length; | ||||
|  | ||||
|         while (p1 --> 0 && p2 --> 0) | ||||
|         while (p1-- > 0 && p2-- > 0) | ||||
|             if (from[p1] != to[p2]) | ||||
|                 break; | ||||
|  | ||||
| @@ -1142,14 +1142,3 @@ action_stack = (function () { | ||||
|         _ref: ref | ||||
|     } | ||||
| })(); | ||||
|  | ||||
| /* | ||||
| ebi('help').onclick = function () { | ||||
|     var c1 = getComputedStyle(dom_src).cssText.split(';'); | ||||
|     var c2 = getComputedStyle(dom_ref).cssText.split(';'); | ||||
|     var max = Math.min(c1.length, c2.length); | ||||
|     for (var a = 0; a < max; a++) | ||||
|         if (c1[a] !== c2[a]) | ||||
|             console.log(c1[a] + '\n' + c2[a]); | ||||
| } | ||||
| */ | ||||
|   | ||||
| @@ -8,68 +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-weight: normal; | ||||
| 	margin: 1.3em 0 .7em 1em; | ||||
| } | ||||
| #mn a { | ||||
|     color: #444; | ||||
|     margin: 0 0 0 -.2em; | ||||
|     padding: 0 0 0 .4em; | ||||
|     text-decoration: none; | ||||
|     /* ie: */ | ||||
|     border-bottom: .1em solid #777\9; | ||||
|     margin-right: 1em\9; | ||||
| 	color: #444; | ||||
| 	margin: 0 0 0 -.2em; | ||||
| 	padding: 0 0 0 .4em; | ||||
| 	text-decoration: none; | ||||
| 	/* ie: */ | ||||
| 	border-bottom: .1em solid #777\9; | ||||
| 	margin-right: 1em\9; | ||||
| } | ||||
| #mn a:first-child { | ||||
|     padding-left: .5em; | ||||
| 	padding-left: .5em; | ||||
| } | ||||
| #mn a:last-child { | ||||
|     padding-right: .5em; | ||||
| 	padding-right: .5em; | ||||
| } | ||||
| #mn a:not(:last-child):after { | ||||
|     content: ''; | ||||
|     width: 1.05em; | ||||
|     height: 1.05em; | ||||
|     margin: -.2em .3em -.2em -.4em; | ||||
|     display: inline-block; | ||||
|     border: 1px solid rgba(0,0,0,0.2); | ||||
|     border-width: .2em .2em 0 0; | ||||
|     transform: rotate(45deg); | ||||
| 	content: ''; | ||||
| 	width: 1.05em; | ||||
| 	height: 1.05em; | ||||
| 	margin: -.2em .3em -.2em -.4em; | ||||
| 	display: inline-block; | ||||
| 	border: 1px solid rgba(0,0,0,0.2); | ||||
| 	border-width: .2em .2em 0 0; | ||||
| 	transform: rotate(45deg); | ||||
| } | ||||
| #mn a:hover { | ||||
|     color: #000; | ||||
|     text-decoration: underline; | ||||
| 	color: #000; | ||||
| 	text-decoration: underline; | ||||
| } | ||||
|  | ||||
| html .editor-toolbar>button.disabled { | ||||
|     opacity: .35; | ||||
|     pointer-events: none; | ||||
| 	opacity: .35; | ||||
| 	pointer-events: none; | ||||
| } | ||||
| html .editor-toolbar>button.save.force-save { | ||||
|     background: #f97; | ||||
| 	background: #f97; | ||||
| } | ||||
|  | ||||
| /* | ||||
| *[data-ln]:before { | ||||
| 	content: attr(data-ln); | ||||
| 	font-size: .8em; | ||||
| 	margin: 0 .4em; | ||||
| 	color: #f0c; | ||||
| } | ||||
| .cm-header { font-size: .4em !important } | ||||
| */ | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| @@ -101,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; | ||||
| @@ -197,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; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -214,108 +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; | ||||
| 	color: #fff; | ||||
| } | ||||
| html.dark p>em, | ||||
| html.dark li>em, | ||||
| html.dark td>em { | ||||
|     color: #f94; | ||||
|     border-color: #666; | ||||
| 	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; | ||||
| } | ||||
| @@ -31,12 +31,12 @@ var md_opt = { | ||||
|  | ||||
| var lightswitch = (function () { | ||||
| 	var fun = function () { | ||||
| 		var dark = !!!document.documentElement.getAttribute("class"); | ||||
| 		var dark = !document.documentElement.getAttribute("class"); | ||||
| 		document.documentElement.setAttribute("class", dark ? "dark" : ""); | ||||
| 		if (window.localStorage) | ||||
| 			localStorage.setItem('darkmode', dark ? 1 : 0); | ||||
| 			localStorage.setItem('lightmode', dark ? 0 : 1); | ||||
| 	}; | ||||
| 	if (window.localStorage && localStorage.getItem('darkmode') == 1) | ||||
| 	if (window.localStorage && localStorage.getItem('lightmode') != 1) | ||||
| 		fun(); | ||||
| 	 | ||||
| 	return fun; | ||||
|   | ||||
| @@ -71,7 +71,7 @@ var mde = (function () { | ||||
| })(); | ||||
|  | ||||
| function set_jumpto() { | ||||
|     document.querySelector('.editor-preview-side').onclick = jumpto; | ||||
|     QS('.editor-preview-side').onclick = jumpto; | ||||
| } | ||||
|  | ||||
| function jumpto(ev) { | ||||
| @@ -94,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'); | ||||
| @@ -105,7 +105,7 @@ function md_changed(mde, on_srv) { | ||||
| } | ||||
|  | ||||
| function save(mde) { | ||||
|     var save_btn = document.querySelector('.editor-toolbar button.save'); | ||||
|     var save_btn = QS('.editor-toolbar button.save'); | ||||
|     if (save_btn.classList.contains('disabled')) { | ||||
|         alert('there is nothing to save'); | ||||
|         return; | ||||
| @@ -212,7 +212,7 @@ 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 = ebi('m'); | ||||
|   | ||||
| @@ -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; | ||||
|   | ||||
| @@ -26,6 +26,26 @@ a { | ||||
| 	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, | ||||
| @@ -50,4 +70,7 @@ html.dark input { | ||||
| 	border-radius: .5em; | ||||
| 	padding: .5em .7em; | ||||
| 	margin: 0 .5em 0 0; | ||||
| } | ||||
| html.dark .num { | ||||
| 	border-color: #777; | ||||
| } | ||||
| @@ -13,19 +13,49 @@ | ||||
|     <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="{{ avol[0] }}?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> | ||||
| @@ -38,7 +68,7 @@ | ||||
|     </div> | ||||
|     <script> | ||||
|  | ||||
| if (window.localStorage && localStorage.getItem('darkmode') == 1) | ||||
| if (window.localStorage && localStorage.getItem('lightmode') != 1) | ||||
|     document.documentElement.setAttribute("class", "dark"); | ||||
|  | ||||
| </script> | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -19,6 +19,11 @@ | ||||
| 	color: #f87; | ||||
| 	padding: .5em; | ||||
| } | ||||
| #u2err.msg { | ||||
| 	color: #999; | ||||
| 	padding: .5em; | ||||
| 	font-size: .9em; | ||||
| } | ||||
| #u2btn { | ||||
| 	color: #eee; | ||||
| 	background: #555; | ||||
| @@ -47,6 +52,11 @@ | ||||
| 	margin: -1.5em 0; | ||||
| 	padding: .8em 0; | ||||
| 	width: 100%; | ||||
| 	max-width: 12em; | ||||
| 	display: inline-block; | ||||
| } | ||||
| #u2conf #u2btn_cw { | ||||
| 	text-align: right; | ||||
| } | ||||
| #u2notbtn { | ||||
| 	display: none; | ||||
| @@ -72,6 +82,7 @@ | ||||
| } | ||||
| #u2tab td:nth-child(2) { | ||||
| 	width: 5em; | ||||
| 	white-space: nowrap; | ||||
| } | ||||
| #u2tab td:nth-child(3) { | ||||
| 	width: 40%; | ||||
| @@ -80,9 +91,46 @@ | ||||
| 	font-family: sans-serif; | ||||
| 	width: auto; | ||||
| } | ||||
| #u2tab tr+tr:hover td { | ||||
| #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: 30em; | ||||
| @@ -99,12 +147,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; | ||||
| @@ -113,13 +165,12 @@ | ||||
| 	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; | ||||
| @@ -130,7 +181,6 @@ | ||||
| 	height: 1em; | ||||
| 	padding: .4em 0; | ||||
| 	display: block; | ||||
| 	user-select: none; | ||||
| 	border-radius: .25em; | ||||
| } | ||||
| #u2conf input[type="checkbox"] { | ||||
| @@ -161,56 +211,20 @@ | ||||
| 	box-shadow: none; | ||||
| 	opacity: .2; | ||||
| } | ||||
| #u2cdesc { | ||||
| 	position: absolute; | ||||
| 	width: 34em; | ||||
| 	left: calc(50% - 15em); | ||||
| 	background: #222; | ||||
| 	border: 0 solid #555; | ||||
| 	text-align: center; | ||||
| 	overflow: hidden; | ||||
| 	margin: 0 -2em; | ||||
| 	height: 0; | ||||
| 	padding: 0 1em; | ||||
| 	opacity: .1; | ||||
|     transition: all 0.14s ease-in-out; | ||||
| 	border-radius: .4em; | ||||
| 	box-shadow: 0 .2em .5em #222; | ||||
| } | ||||
| #u2cdesc.show { | ||||
| 	padding: 1em; | ||||
| 	height: auto; | ||||
| 	border-width: .2em 0; | ||||
| 	opacity: 1; | ||||
| } | ||||
| #u2foot { | ||||
| 	color: #fff; | ||||
| 	font-style: italic; | ||||
| } | ||||
| #u2foot span { | ||||
| 	color: #999; | ||||
| 	font-size: .9em; | ||||
| } | ||||
| #u2footfoot { | ||||
| 	margin-bottom: -1em; | ||||
| } | ||||
| .prog { | ||||
| 	font-family: monospace; | ||||
| } | ||||
| .prog>div { | ||||
| 	display: inline-block; | ||||
| 	position: relative; | ||||
| 	overflow: hidden; | ||||
| 	margin: 0; | ||||
| 	padding: 0; | ||||
| 	height: 1.1em; | ||||
| 	margin-bottom: -.15em; | ||||
| 	box-shadow: -1px -1px 0 inset rgba(255,255,255,0.1); | ||||
| } | ||||
| .prog>div>div { | ||||
| 	width: 0%; | ||||
| 	position: absolute; | ||||
| 	left: 0; | ||||
| 	top: 0; | ||||
| 	bottom: 0; | ||||
| 	background: #0a0; | ||||
| } | ||||
| #u2tab a>span { | ||||
| 	font-weight: bold; | ||||
| 	font-style: italic; | ||||
| @@ -221,3 +235,40 @@ | ||||
| 	float: right; | ||||
| 	margin-bottom: -.3em; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| 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,92 +0,0 @@ | ||||
|  | ||||
|     <div id="op_bup" class="opview opbox act"> | ||||
|         <div id="u2err"></div> | ||||
|         <form method="post" enctype="multipart/form-data" accept-charset="utf-8"> | ||||
|             <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"> | ||||
|             <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"> | ||||
|             <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_msg" class="opview opbox"> | ||||
|         <form method="post" enctype="application/x-www-form-urlencoded" accept-charset="utf-8"> | ||||
|             <input type="text" name="msg" size="30"> | ||||
|             <input type="submit" value="send msg"> | ||||
|         </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> | ||||
|                     <td rowspan="2"> | ||||
|                         <input type="checkbox" id="multitask" /> | ||||
|                         <label for="multitask" alt="continue hashing other files while uploading">🏃</label> | ||||
|                     </td> | ||||
|                     <td rowspan="2"> | ||||
|                         <input type="checkbox" id="ask_up" /> | ||||
|                         <label for="ask_up" alt="ask for confirmation befofre upload starts">💭</label> | ||||
|                     </td> | ||||
|                     <td rowspan="2"> | ||||
|                         <input type="checkbox" id="flag_en" /> | ||||
|                         <label for="flag_en" alt="ensure only one tab is uploading at a time $N (other tabs must have this enabled too)">💤</label> | ||||
|                     </td> | ||||
|                 {%- if have_up2k_idx %} | ||||
|                     <td data-perm="read" rowspan="2"> | ||||
|                         <input type="checkbox" id="fsearch" /> | ||||
|                         <label for="fsearch" alt="don't actually upload, instead check if the files already $N exist on the server (will scan all folders you can read)">🔎</label> | ||||
|                     </td> | ||||
|                 {%- endif %} | ||||
|                     <td data-perm="read" rowspan="2" id="u2btn_cw"></td> | ||||
|                 </tr> | ||||
|                 <tr> | ||||
|                     <td> | ||||
|                         <a href="#" id="nthread_sub">–</a> | ||||
|                         <input class="txtbox" id="nthread" value="2" /> | ||||
|                         <a href="#" id="nthread_add">+</a> | ||||
|                     </td> | ||||
|                 </tr> | ||||
|             </table> | ||||
|  | ||||
|             <div id="u2cdesc"></div> | ||||
|  | ||||
|             <div id="u2notbtn"></div> | ||||
|  | ||||
|             <div id="u2btn_ct"> | ||||
|                 <div id="u2btn"> | ||||
|                     <span id="u2bm"></span><br /> | ||||
|                     drag/drop files<br /> | ||||
|                     and folders here<br /> | ||||
|                     (or click me) | ||||
|                 </div> | ||||
|             </div> | ||||
|  | ||||
|             <table id="u2tab"> | ||||
|                 <tr> | ||||
|                     <td>filename</td> | ||||
|                     <td>status</td> | ||||
|                     <td>progress<a href="#" id="u2cleanup">cleanup</a></td> | ||||
|                 </tr> | ||||
|             </table> | ||||
|  | ||||
|             <p id="u2foot"></p> | ||||
|             <p id="u2footfoot">( if you don't need lastmod timestamps, resumable uploads or progress bars just use the <a href="#" id="u2nope">basic uploader</a>)</p> | ||||
|     </div> | ||||
| @@ -1,5 +1,15 @@ | ||||
| "use strict"; | ||||
|  | ||||
| if (!window['console']) | ||||
|     window['console'] = { | ||||
|         "log": function (msg) { } | ||||
|     }; | ||||
|  | ||||
|  | ||||
| var is_touch = 'ontouchstart' in window, | ||||
|     ANDROID = /(android)/i.test(navigator.userAgent); | ||||
|  | ||||
|  | ||||
| // error handler for mobile devices | ||||
| function hcroak(msg) { | ||||
|     document.body.innerHTML = msg; | ||||
| @@ -40,9 +50,11 @@ function vis_exh(msg, url, lineNo, columnNo, error) { | ||||
| } | ||||
|  | ||||
|  | ||||
| function ebi(id) { | ||||
|     return document.getElementById(id); | ||||
| } | ||||
| var ebi = document.getElementById.bind(document), | ||||
|     QS = document.querySelector.bind(document), | ||||
|     QSA = document.querySelectorAll.bind(document), | ||||
|     mknod = document.createElement.bind(document); | ||||
|  | ||||
|  | ||||
| function ev(e) { | ||||
|     e = e || window.event; | ||||
| @@ -80,7 +92,7 @@ if (!String.startsWith) { | ||||
| // https://stackoverflow.com/a/950146 | ||||
| function import_js(url, cb) { | ||||
|     var head = document.head || document.getElementsByTagName('head')[0]; | ||||
|     var script = document.createElement('script'); | ||||
|     var script = mknod('script'); | ||||
|     script.type = 'text/javascript'; | ||||
|     script.src = url; | ||||
|  | ||||
| @@ -110,7 +122,94 @@ function crc32(str) { | ||||
|         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) { | ||||
| @@ -186,64 +285,6 @@ function makeSortable(table, cb) { | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| (function () { | ||||
|     var ops = document.querySelectorAll('#ops>a'); | ||||
|     for (var a = 0; a < ops.length; a++) { | ||||
|         ops[a].onclick = opclick; | ||||
|     } | ||||
| })(); | ||||
|  | ||||
|  | ||||
| function opclick(e) { | ||||
|     ev(e); | ||||
|  | ||||
|     var dest = this.getAttribute('data-dest'); | ||||
|     goto(dest); | ||||
|  | ||||
|     swrite('opmode', dest || null); | ||||
|  | ||||
|     var input = document.querySelector('.opview.act input:not([type="hidden"])') | ||||
|     if (input) | ||||
|         input.focus(); | ||||
| } | ||||
|  | ||||
|  | ||||
| function goto(dest) { | ||||
|     var obj = document.querySelectorAll('.opview.act'); | ||||
|     for (var a = obj.length - 1; a >= 0; a--) | ||||
|         obj[a].classList.remove('act'); | ||||
|  | ||||
|     obj = document.querySelectorAll('#ops>a'); | ||||
|     for (var a = obj.length - 1; a >= 0; a--) | ||||
|         obj[a].classList.remove('act'); | ||||
|  | ||||
|     if (dest) { | ||||
|         var ui = ebi('op_' + dest); | ||||
|         ui.classList.add('act'); | ||||
|         document.querySelector('#ops>a[data-dest=' + dest + ']').classList.add('act'); | ||||
|  | ||||
|         var fn = window['goto_' + dest]; | ||||
|         if (fn) | ||||
|             fn(); | ||||
|     } | ||||
|  | ||||
|     if (window['treectl']) | ||||
|         treectl.onscroll(); | ||||
| } | ||||
|  | ||||
|  | ||||
| (function () { | ||||
|     goto(); | ||||
|     var op = sread('opmode'); | ||||
|     if (op !== null && op !== '.') | ||||
|         try { | ||||
|             goto(op); | ||||
|         } | ||||
|         catch (ex) { } | ||||
| })(); | ||||
|  | ||||
|  | ||||
| function linksplit(rp) { | ||||
|     var ret = []; | ||||
|     var apath = '/'; | ||||
| @@ -260,12 +301,16 @@ function linksplit(rp) { | ||||
|             link = rp.slice(0, ofs + 1); | ||||
|             rp = rp.slice(ofs + 1); | ||||
|         } | ||||
|         var vlink = link; | ||||
|         if (link.indexOf('/') !== -1) | ||||
|             vlink = link.slice(0, -1) + '<span>/</span>'; | ||||
|         var vlink = esc(link), | ||||
|             elink = uricom_enc(link); | ||||
|  | ||||
|         ret.push('<a href="' + apath + link + '">' + vlink + '</a>'); | ||||
|         apath += 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; | ||||
| } | ||||
| @@ -367,11 +412,15 @@ function jwrite(key, val) { | ||||
| } | ||||
|  | ||||
| function icfg_get(name, defval) { | ||||
|     return parseInt(fcfg_get(name, defval)); | ||||
| } | ||||
|  | ||||
| function fcfg_get(name, defval) { | ||||
|     var o = ebi(name); | ||||
|  | ||||
|     var val = parseInt(sread(name)); | ||||
|     var val = parseFloat(sread(name)); | ||||
|     if (isNaN(val)) | ||||
|         return parseInt(o ? o.value : defval); | ||||
|         return parseFloat(o ? o.value : defval); | ||||
|  | ||||
|     if (o) | ||||
|         o.value = val; | ||||
| @@ -408,8 +457,7 @@ function bcfg_upd_ui(name, val) { | ||||
|     if (o.getAttribute('type') == 'checkbox') | ||||
|         o.checked = val; | ||||
|     else if (o) { | ||||
|         var fun = val ? 'add' : 'remove'; | ||||
|         o.classList[fun]('on'); | ||||
|         clmod(o, 'on', val); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -423,3 +471,67 @@ function hist_replace(url) { | ||||
|     console.log("h-repl " + url); | ||||
|     history.replaceState(url, url, url); | ||||
| } | ||||
|  | ||||
|  | ||||
| var tt = (function () { | ||||
|     var r = { | ||||
|         "tt": mknod("div"), | ||||
|         "en": true | ||||
|     }; | ||||
|  | ||||
|     r.tt.setAttribute('id', 'tt'); | ||||
|     document.body.appendChild(r.tt); | ||||
|  | ||||
|     function show() { | ||||
|         var cfg = sread('tooltips'); | ||||
|         if (cfg !== null && cfg != '1') | ||||
|             return; | ||||
|  | ||||
|         var msg = this.getAttribute('tt'); | ||||
|         if (!msg) | ||||
|             return; | ||||
|  | ||||
|         var pos = this.getBoundingClientRect(), | ||||
|             left = pos.left < window.innerWidth / 2, | ||||
|             top = pos.top < window.innerHeight / 2; | ||||
|  | ||||
|         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 />"); | ||||
|         clmod(r.tt, 'show', 1); | ||||
|     } | ||||
|  | ||||
|     function hide() { | ||||
|         clmod(r.tt, 'show'); | ||||
|     } | ||||
|  | ||||
|     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) | ||||
|         } | ||||
|  | ||||
|         var _show = r.en ? show : null, | ||||
|             _hide = r.en ? hide : null; | ||||
|  | ||||
|         var o = QSA('*[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; | ||||
|         } | ||||
|         hide(); | ||||
|     }; | ||||
|  | ||||
|     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> | ||||
							
								
								
									
										68
									
								
								docs/browser-icons.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								docs/browser-icons.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | ||||
| /* 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: '📂'; | ||||
|     display: block; | ||||
|     position: absolute; | ||||
|     margin: -.1em -.4em; | ||||
|     text-shadow: 0 0 .1em #000; | ||||
|     font-size: 2em; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* put filetype icons top-left */ | ||||
| #ggrid>a:before { | ||||
|     display: block; | ||||
|     position: absolute; | ||||
|     margin: -.1em -.4em; | ||||
|     text-shadow: 0 0 .1em #000; | ||||
|     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; | ||||
| } | ||||
| @@ -32,9 +32,13 @@ 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 | ||||
|   | ||||
							
								
								
									
										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> | ||||
| @@ -73,6 +73,23 @@ shab64() { sp=$1; f="$2"; v=0; sz=$(stat -c%s "$f"); while true; do w=$((v+sp*10 | ||||
| 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 | ||||
|  | ||||
| @@ -129,6 +146,19 @@ 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}.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}.js >../vg 2>/dev/null && diff -wNarU0 ../{vg,vr} | wc -c) && printf '%s %6s %s\n' "$rev" $dsz "$extra") </dev/null; done                 | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## http 206 | ||||
| @@ -154,7 +184,7 @@ Range: bytes=26-         Content-Range: bytes */26 | ||||
|  | ||||
| var tsh = []; | ||||
| function convert_markdown(md_text, dest_dom) { | ||||
|     tsh.push(new Date().getTime()); | ||||
|     tsh.push(Date.now()); | ||||
|     while (tsh.length > 10) | ||||
|         tsh.shift(); | ||||
|     if (tsh.length > 1) { | ||||
|   | ||||
							
								
								
									
										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 | ||||
							
								
								
									
										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 ...> | ||||
| @@ -92,20 +92,34 @@ chmod 755 \ | ||||
|   copyparty-extras/copyparty-*/{scripts,bin}/* | ||||
|  | ||||
|  | ||||
| # extract and repack the sfx with less features enabled | ||||
| # extract the sfx | ||||
| ( cd copyparty-extras/sfx-full/ | ||||
| ./copyparty-sfx.py -h | ||||
| cd ../copyparty-*/ | ||||
| ./scripts/make-sfx.sh re no-ogv no-cm | ||||
| ) | ||||
|  | ||||
|  | ||||
| # put new sfx into copyparty-extras/sfx-lite/, | ||||
| # fuse client into copyparty-extras/, | ||||
| 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-*/dist/* sfx-lite/ | ||||
| mv copyparty-*/bin/copyparty-fuse.py . | ||||
| cp -pv sfx-lite/copyparty-sfx.py ../copyparty | ||||
| rm -rf copyparty-{0..9}*.*.*{0..9} | ||||
| @@ -119,6 +133,7 @@ true | ||||
|  | ||||
|  | ||||
| # create the bundle | ||||
| printf '\n\n' | ||||
| fn=copyparty-$(date +%Y-%m%d-%H%M%S).tgz | ||||
| tar -czvf "$od/$fn" * | ||||
| cd "$od" | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| FROM    alpine:3.13 | ||||
| WORKDIR /z | ||||
| ENV     ver_asmcrypto=5b994303a9d3e27e0915f72a10b6c2c51535a4dc \ | ||||
|         ver_hashwasm=4.7.0 \ | ||||
|         ver_marked=1.1.0 \ | ||||
|         ver_ogvjs=1.8.0 \ | ||||
|         ver_mde=2.14.0 \ | ||||
| @@ -21,7 +22,11 @@ RUN     mkdir -p /z/dist/no-pk \ | ||||
|         && 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 ) \ | ||||
| @@ -58,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 | ||||
|   | ||||
							
								
								
									
										12
									
								
								scripts/install-githooks.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										12
									
								
								scripts/install-githooks.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| #!/bin/bash | ||||
| set -ex | ||||
|  | ||||
| [ -e setup.py ] || .. | ||||
| [ -e setup.py ] || { | ||||
|     echo u wot | ||||
|     exit 1 | ||||
| } | ||||
|  | ||||
| cd .git/hooks | ||||
| rm -f pre-commit | ||||
| ln -s ../../scripts/run-tests.sh pre-commit | ||||
| @@ -11,6 +11,10 @@ echo | ||||
| # `re` does a repack of an sfx which you already executed once | ||||
| #   (grabs files from the sfx-created tempdir), overrides `clean` | ||||
| # | ||||
| # `gz` creates a gzip-compressed python sfx instead of bzip2 | ||||
| # | ||||
| # `no-sh` makes just the python sfx, skips the sh/unix sfx | ||||
| # | ||||
| # `no-ogv` saves ~500k by removing the opus/vorbis audio codecs | ||||
| #   (only affects apple devices; everything else has native support) | ||||
| # | ||||
| @@ -32,6 +36,10 @@ gtar=$(command -v gtar || command -v gnutar) || true | ||||
| 	[ -e /opt/local/bin/bzip2 ] && | ||||
| 		bzip2() { /opt/local/bin/bzip2 "$@"; } | ||||
| } | ||||
|  | ||||
| gawk=$(command -v gawk || command -v gnuawk || command -v awk) | ||||
| awk() { $gawk "$@"; } | ||||
|  | ||||
| pybin=$(command -v python3 || command -v python) || { | ||||
| 	echo need python | ||||
| 	exit 1 | ||||
| @@ -45,11 +53,13 @@ pybin=$(command -v python3 || command -v python) || { | ||||
| 	exit 1 | ||||
| } | ||||
|  | ||||
| use_gz= | ||||
| do_sh=1 | ||||
| do_py=1 | ||||
| while [ ! -z "$1" ]; do | ||||
| 	[ "$1" = clean  ] && clean=1  && shift && continue | ||||
| 	[ "$1" = re     ] && repack=1 && shift && continue | ||||
| 	[ "$1" = gz     ] && use_gz=1 && shift && continue | ||||
| 	[ "$1" = no-ogv ] && no_ogv=1 && shift && continue | ||||
| 	[ "$1" = no-cm  ] && no_cm=1  && shift && continue | ||||
| 	[ "$1" = no-sh  ] && do_sh=   && shift && continue | ||||
| @@ -115,7 +125,7 @@ cd sfx | ||||
| ver= | ||||
| git describe --tags >/dev/null 2>/dev/null && { | ||||
| 	git_ver="$(git describe --tags)";  # v0.5.5-2-gb164aa0 | ||||
| 	ver="$(printf '%s\n' "$git_ver" | sed -r 's/^v//; s/-g?/./g')"; | ||||
| 	ver="$(printf '%s\n' "$git_ver" | sed -r 's/^v//')"; | ||||
| 	t_ver= | ||||
|  | ||||
| 	printf '%s\n' "$git_ver" | grep -qE '^v[0-9\.]+$' && { | ||||
| @@ -161,7 +171,7 @@ find .. -type f \( -name .DS_Store -or -name ._.DS_Store \) -delete | ||||
| find .. -type f -name ._\* | while IFS= read -r f; do cmp <(printf '\x00\x05\x16') <(head -c 3 -- "$f") && rm -f -- "$f"; done | ||||
|  | ||||
| echo use smol web deps | ||||
| rm -f copyparty/web/deps/*.full.* | ||||
| rm -f copyparty/web/deps/*.full.* copyparty/web/dbg-* copyparty/web/Makefile | ||||
|  | ||||
| # it's fine dw | ||||
| grep -lE '\.full\.(js|css)' copyparty/web/* | | ||||
| @@ -192,28 +202,68 @@ tmv "$f" | ||||
|  | ||||
| # up2k goes from 28k to 22k laff | ||||
| echo entabbening | ||||
| find | grep -E '\.(js|css|html)$' | while IFS= read -r f; do | ||||
| find | grep -E '\.css$' | while IFS= read -r f; do | ||||
| 	awk '{ | ||||
| 		sub(/^[ \t]+/,""); | ||||
| 		sub(/[ \t]+$/,""); | ||||
| 		$0=gensub(/^([a-z-]+) *: *(.*[^ ]) *;$/,"\\1:\\2;","1"); | ||||
| 		sub(/ +\{$/,"{"); | ||||
| 		gsub(/, /,",") | ||||
| 	} | ||||
| 	!/\}$/ {printf "%s",$0;next} | ||||
| 	1 | ||||
| 	' <$f | sed 's/;\}$/}/' >t | ||||
| 	tmv "$f" | ||||
| done | ||||
| find | grep -E '\.(js|html)$' | while IFS= read -r f; do | ||||
| 	unexpand -t 4 --first-only <"$f" >t | ||||
| 	tmv "$f" | ||||
| done | ||||
|  | ||||
|  | ||||
| gzres() { | ||||
| command -v pigz && | ||||
| 	pk='pigz -11 -J 34 -I 100' || | ||||
| 	pk='gzip' | ||||
|  | ||||
| echo "$pk" | ||||
| find | grep -E '\.(js|css)$' | grep -vF /deps/ | while IFS= read -r f; do | ||||
| 	echo -n . | ||||
| 	$pk "$f" | ||||
| done | ||||
| echo | ||||
| } | ||||
| gzres | ||||
|  | ||||
|  | ||||
| echo gen tarlist | ||||
| for d in copyparty dep-j2; do find $d -type f; done | | ||||
| sed -r 's/(.*)\.(.*)/\2 \1/' | LC_ALL=C sort | | ||||
| sed -r 's/([^ ]*) (.*)/\2.\1/' | grep -vE '/list1?$' > list1 | ||||
|  | ||||
| (grep -vE '\.(gz|br)$' list1; grep -E '\.(gz|br)$' list1) >list || true | ||||
|  | ||||
| echo creating tar | ||||
| args=(--owner=1000 --group=1000) | ||||
| [ "$OSTYPE" = msys ] && | ||||
| 	args=() | ||||
|  | ||||
| tar -cf tar "${args[@]}" --numeric-owner copyparty dep-j2 | ||||
| tar -cf tar "${args[@]}" --numeric-owner -T list | ||||
|  | ||||
| pc=bzip2 | ||||
| pe=bz2 | ||||
| [ $use_gz ] && pc=gzip && pe=gz | ||||
|  | ||||
| echo compressing tar | ||||
| # detect best level; bzip2 -7 is usually better than -9 | ||||
| [ $do_py ] && { for n in {2..9}; do cp tar t.$n; bzip2 -$n t.$n & done; wait; mv -v $(ls -1S t.*.bz2 | tail -n 1) tar.bz2; } | ||||
| [ $do_sh ] && { for n in {2..9}; do cp tar t.$n;  xz -ze$n t.$n & done; wait; mv -v $(ls -1S t.*.xz  | tail -n 1) tar.xz; } | ||||
| [ $do_py ] && { for n in {2..9}; do cp tar t.$n; $pc  -$n t.$n & done; wait; mv -v $(ls -1S t.*.$pe | tail -n 1) tar.bz2; } | ||||
| [ $do_sh ] && { for n in {2..9}; do cp tar t.$n; xz -ze$n t.$n & done; wait; mv -v $(ls -1S t.*.xz  | tail -n 1) tar.xz; } | ||||
| rm t.* || true | ||||
| exts=() | ||||
|  | ||||
|  | ||||
| [ $do_sh ] && { | ||||
| exts+=(sh) | ||||
| exts+=(.sh) | ||||
| echo creating unix sfx | ||||
| ( | ||||
| 	sed "s/PACK_TS/$ts/; s/PACK_HTS/$hts/; s/CPP_VER/$ver/" <../scripts/sfx.sh | | ||||
| @@ -224,17 +274,30 @@ echo creating unix sfx | ||||
|  | ||||
|  | ||||
| [ $do_py ] && { | ||||
| exts+=(py) | ||||
| echo creating generic sfx | ||||
| $pybin ../scripts/sfx.py --sfx-make tar.bz2 $ver $ts | ||||
| mv sfx.out $sfx_out.py | ||||
| chmod 755 $sfx_out.* | ||||
| 	echo creating generic sfx | ||||
|  | ||||
| 	py=../scripts/sfx.py | ||||
| 	suf= | ||||
| 	[ $use_gz ] && { | ||||
| 		sed -r 's/"r:bz2"/"r:gz"/' <$py >$py.t | ||||
| 		py=$py.t | ||||
| 		suf=-gz | ||||
| 	} | ||||
|  | ||||
| 	$pybin $py --sfx-make tar.bz2 $ver $ts | ||||
| 	mv sfx.out $sfx_out$suf.py | ||||
| 	 | ||||
| 	exts+=($suf.py) | ||||
| 	[ $use_gz ] && | ||||
| 		rm $py | ||||
| } | ||||
|  | ||||
|  | ||||
| chmod 755 $sfx_out* | ||||
|  | ||||
| printf "done:\n" | ||||
| for ext in ${exts[@]}; do | ||||
| 	printf "  %s\n" "$(realpath $sfx_out)."$ext | ||||
| 	printf "  %s\n" "$(realpath $sfx_out)"$ext | ||||
| done | ||||
|  | ||||
| # apk add bash python3 tar xz bzip2 | ||||
|   | ||||
							
								
								
									
										34
									
								
								scripts/profile.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								scripts/profile.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| import sys | ||||
|  | ||||
| sys.path.insert(0, ".") | ||||
| cmd = sys.argv[1] | ||||
|  | ||||
| if cmd == "cpp": | ||||
|     from copyparty.__main__ import main | ||||
|  | ||||
|     argv = ["__main__", "-v", "srv::r", "-v", "../../yt:yt:r"] | ||||
|     main(argv=argv) | ||||
|  | ||||
| elif cmd == "test": | ||||
|     from unittest import main | ||||
|  | ||||
|     argv = ["__main__", "discover", "-s", "tests"] | ||||
|     main(module=None, argv=argv) | ||||
|  | ||||
| else: | ||||
|     raise Exception() | ||||
|  | ||||
| # import dis; print(dis.dis(main)) | ||||
|  | ||||
|  | ||||
| # macos: | ||||
| #   option1) python3.9 -m pip install --user -U vmprof==0.4.9 | ||||
| #   option2) python3.9 -m pip install --user -U https://github.com/vmprof/vmprof-python/archive/refs/heads/master.zip | ||||
| # | ||||
| # python -m vmprof -o prof --lines ./scripts/profile.py test | ||||
|  | ||||
| # linux: ~/.local/bin/vmprofshow prof tree | grep -vF '[1m  0.' | ||||
| # macos: ~/Library/Python/3.9/bin/vmprofshow prof tree | grep -vF '[1m  0.' | ||||
| #   win: %appdata%\..\Roaming\Python\Python39\Scripts\vmprofshow.exe prof tree | ||||
							
								
								
									
										15
									
								
								scripts/run-tests.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										15
									
								
								scripts/run-tests.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| #!/bin/bash | ||||
| set -ex | ||||
|  | ||||
| pids=() | ||||
| for py in python{2,3}; do | ||||
|     nice $py -m unittest discover -s tests >/dev/null & | ||||
|     pids+=($!) | ||||
| done | ||||
|  | ||||
| python3 scripts/test/smoketest.py & | ||||
| pids+=($!) | ||||
|  | ||||
| for pid in ${pids[@]}; do | ||||
|     wait $pid | ||||
| done | ||||
							
								
								
									
										187
									
								
								scripts/sfx.py
									
									
									
									
									
								
							
							
						
						
									
										187
									
								
								scripts/sfx.py
									
									
									
									
									
								
							| @@ -2,7 +2,8 @@ | ||||
| # coding: latin-1 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os, sys, time, shutil, runpy, tarfile, hashlib, platform, tempfile, traceback | ||||
| import re, os, sys, time, shutil, signal, threading, tarfile, hashlib, platform, tempfile, traceback | ||||
| import subprocess as sp | ||||
|  | ||||
| """ | ||||
| run me with any version of python, i will unpack and run copyparty | ||||
| @@ -26,21 +27,21 @@ CKSUM = None | ||||
| STAMP = None | ||||
|  | ||||
| PY2 = sys.version_info[0] == 2 | ||||
| WINDOWS = sys.platform in ["win32", "msys"] | ||||
| sys.dont_write_bytecode = True | ||||
| me = os.path.abspath(os.path.realpath(__file__)) | ||||
| cpp = None | ||||
|  | ||||
|  | ||||
| def eprint(*args, **kwargs): | ||||
|     kwargs["file"] = sys.stderr | ||||
|     print(*args, **kwargs) | ||||
| def eprint(*a, **ka): | ||||
|     ka["file"] = sys.stderr | ||||
|     print(*a, **ka) | ||||
|  | ||||
|  | ||||
| def msg(*args, **kwargs): | ||||
|     if args: | ||||
|         args = ["[SFX]", args[0]] + list(args[1:]) | ||||
| def msg(*a, **ka): | ||||
|     if a: | ||||
|         a = ["[SFX]", a[0]] + list(a[1:]) | ||||
|  | ||||
|     eprint(*args, **kwargs) | ||||
|     eprint(*a, **ka) | ||||
|  | ||||
|  | ||||
| # skip 1 | ||||
| @@ -155,6 +156,9 @@ def encode(data, size, cksum, ver, ts): | ||||
|                 skip = True | ||||
|                 continue | ||||
|  | ||||
|             if ln.strip().startswith("# fmt: "): | ||||
|                 continue | ||||
|  | ||||
|             unpk += ln + "\n" | ||||
|  | ||||
|         for k, v in [ | ||||
| @@ -208,11 +212,11 @@ def yieldfile(fn): | ||||
|  | ||||
|  | ||||
| def hashfile(fn): | ||||
|     hasher = hashlib.md5() | ||||
|     h = hashlib.md5() | ||||
|     for block in yieldfile(fn): | ||||
|         hasher.update(block) | ||||
|         h.update(block) | ||||
|  | ||||
|     return hasher.hexdigest() | ||||
|     return h.hexdigest() | ||||
|  | ||||
|  | ||||
| def unpack(): | ||||
| @@ -221,9 +225,10 @@ def unpack(): | ||||
|     tag = "v" + str(STAMP) | ||||
|     withpid = "{}.{}".format(name, os.getpid()) | ||||
|     top = tempfile.gettempdir() | ||||
|     final = os.path.join(top, name) | ||||
|     mine = os.path.join(top, withpid) | ||||
|     tar = os.path.join(mine, "tar") | ||||
|     opj = os.path.join | ||||
|     final = opj(top, name) | ||||
|     mine = opj(top, withpid) | ||||
|     tar = opj(mine, "tar") | ||||
|  | ||||
|     try: | ||||
|         if tag in os.listdir(final): | ||||
| @@ -232,28 +237,24 @@ def unpack(): | ||||
|     except: | ||||
|         pass | ||||
|  | ||||
|     nwrite = 0 | ||||
|     sz = 0 | ||||
|     os.mkdir(mine) | ||||
|     with open(tar, "wb") as f: | ||||
|         for buf in get_payload(): | ||||
|             nwrite += len(buf) | ||||
|             sz += len(buf) | ||||
|             f.write(buf) | ||||
|  | ||||
|     if nwrite != SIZE: | ||||
|         t = "\n\n  bad file:\n    expected {} bytes, got {}\n".format(SIZE, nwrite) | ||||
|         raise Exception(t) | ||||
|  | ||||
|     cksum = hashfile(tar) | ||||
|     if cksum != CKSUM: | ||||
|         t = "\n\n  bad file:\n    {} expected,\n    {} obtained\n".format(CKSUM, cksum) | ||||
|         raise Exception(t) | ||||
|     ck = hashfile(tar) | ||||
|     if ck != CKSUM: | ||||
|         t = "\n\nexpected {} ({} byte)\nobtained {} ({} byte)\nsfx corrupt" | ||||
|         raise Exception(t.format(CKSUM, SIZE, ck, sz)) | ||||
|  | ||||
|     with tarfile.open(tar, "r:bz2") as tf: | ||||
|         tf.extractall(mine) | ||||
|  | ||||
|     os.remove(tar) | ||||
|  | ||||
|     with open(os.path.join(mine, tag), "wb") as f: | ||||
|     with open(opj(mine, tag), "wb") as f: | ||||
|         f.write(b"h\n") | ||||
|  | ||||
|     try: | ||||
| @@ -271,25 +272,25 @@ def unpack(): | ||||
|     except: | ||||
|         pass | ||||
|  | ||||
|     for fn in u8(os.listdir(top)): | ||||
|         if fn.startswith(name) and fn != withpid: | ||||
|             try: | ||||
|                 old = opj(top, fn) | ||||
|                 if time.time() - os.path.getmtime(old) > 86400: | ||||
|                     shutil.rmtree(old) | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|     try: | ||||
|         os.symlink(mine, final) | ||||
|     except: | ||||
|         try: | ||||
|             os.rename(mine, final) | ||||
|             return final | ||||
|         except: | ||||
|             msg("reloc fail,", mine) | ||||
|             return mine | ||||
|  | ||||
|     for fn in u8(os.listdir(top)): | ||||
|         if fn.startswith(name) and fn not in [name, withpid]: | ||||
|             try: | ||||
|                 old = os.path.join(top, fn) | ||||
|                 if time.time() - os.path.getmtime(old) > 10: | ||||
|                     shutil.rmtree(old) | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|     return final | ||||
|     return mine | ||||
|  | ||||
|  | ||||
| def get_payload(): | ||||
| @@ -306,46 +307,57 @@ def get_payload(): | ||||
|         if ofs < 0: | ||||
|             raise Exception("could not find archive marker") | ||||
|  | ||||
|         # start reading from the final b"\n" | ||||
|         # start at final b"\n" | ||||
|         fpos = ofs + len(ptn) - 3 | ||||
|         # msg("tar found at", fpos) | ||||
|         f.seek(fpos) | ||||
|         dpos = 0 | ||||
|         leftovers = b"" | ||||
|         rem = b"" | ||||
|         while True: | ||||
|             rbuf = f.read(1024 * 32) | ||||
|             if rbuf: | ||||
|                 buf = leftovers + rbuf | ||||
|                 buf = rem + rbuf | ||||
|                 ofs = buf.rfind(b"\n") | ||||
|                 if len(buf) <= 4: | ||||
|                     leftovers = buf | ||||
|                     rem = buf | ||||
|                     continue | ||||
|  | ||||
|                 if ofs >= len(buf) - 4: | ||||
|                     leftovers = buf[ofs:] | ||||
|                     rem = buf[ofs:] | ||||
|                     buf = buf[:ofs] | ||||
|                 else: | ||||
|                     leftovers = b"\n# " | ||||
|                     rem = b"\n# " | ||||
|             else: | ||||
|                 buf = leftovers | ||||
|                 buf = rem | ||||
|  | ||||
|             fpos += len(buf) + 1 | ||||
|             buf = ( | ||||
|                 buf.replace(b"\n# ", b"") | ||||
|                 .replace(b"\n#r", b"\r") | ||||
|                 .replace(b"\n#n", b"\n") | ||||
|             ) | ||||
|             dpos += len(buf) - 1 | ||||
|             for a, b in [[b"\n# ", b""], [b"\n#r", b"\r"], [b"\n#n", b"\n"]]: | ||||
|                 buf = buf.replace(a, b) | ||||
|  | ||||
|             dpos += len(buf) - 1 | ||||
|             yield buf | ||||
|  | ||||
|             if not rbuf: | ||||
|                 break | ||||
|  | ||||
|  | ||||
| def utime(top): | ||||
|     i = 0 | ||||
|     files = [os.path.join(dp, p) for dp, dd, df in os.walk(top) for p in dd + df] | ||||
|     while WINDOWS: | ||||
|         t = int(time.time()) | ||||
|         if i: | ||||
|             msg("utime {}, {}".format(i, t)) | ||||
|  | ||||
|         for f in files: | ||||
|             os.utime(f, (t, t)) | ||||
|  | ||||
|         i += 1 | ||||
|         time.sleep(78123) | ||||
|  | ||||
|  | ||||
| def confirm(rv): | ||||
|     msg() | ||||
|     msg(traceback.format_exc()) | ||||
|     msg("retcode", rv if rv else traceback.format_exc()) | ||||
|     msg("*** hit enter to exit ***") | ||||
|     try: | ||||
|         raw_input() if PY2 else input() | ||||
| @@ -355,37 +367,59 @@ def confirm(rv): | ||||
|     sys.exit(rv) | ||||
|  | ||||
|  | ||||
| def run(tmp, j2ver): | ||||
|     global cpp | ||||
|  | ||||
|     msg("jinja2:", j2ver or "bundled") | ||||
| def run(tmp, j2): | ||||
|     msg("jinja2:", j2 or "bundled") | ||||
|     msg("sfxdir:", tmp) | ||||
|     msg() | ||||
|  | ||||
|     # "systemd-tmpfiles-clean.timer"?? HOW do you even come up with this shit | ||||
|     # block systemd-tmpfiles-clean.timer | ||||
|     try: | ||||
|         import fcntl | ||||
|  | ||||
|         fd = os.open(tmp, os.O_RDONLY) | ||||
|         fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB) | ||||
|         tmp = os.readlink(tmp)  # can't flock a symlink, even with O_NOFOLLOW | ||||
|     except: | ||||
|         pass | ||||
|     except Exception as ex: | ||||
|         if not WINDOWS: | ||||
|             msg("\033[31mflock:", repr(ex)) | ||||
|  | ||||
|     t = threading.Thread(target=utime, args=(tmp,)) | ||||
|     t.daemon = True | ||||
|     t.start() | ||||
|  | ||||
|     ld = [tmp, os.path.join(tmp, "dep-j2")] | ||||
|     if j2ver: | ||||
|     if j2: | ||||
|         del ld[-1] | ||||
|  | ||||
|     if any([re.match(r"^-.*j[0-9]", x) for x in sys.argv]): | ||||
|         run_s(ld) | ||||
|     else: | ||||
|         run_i(ld) | ||||
|  | ||||
|  | ||||
| def run_i(ld): | ||||
|     for x in ld: | ||||
|         sys.path.insert(0, x) | ||||
|  | ||||
|     try: | ||||
|         runpy.run_module(str("copyparty"), run_name=str("__main__")) | ||||
|     except SystemExit as ex: | ||||
|         if ex.code: | ||||
|             confirm(ex.code) | ||||
|     except: | ||||
|         confirm(1) | ||||
|     from copyparty.__main__ import main as p | ||||
|  | ||||
|     p() | ||||
|  | ||||
|  | ||||
| def run_s(ld): | ||||
|     # fmt: off | ||||
|     c = "import sys,runpy;" + "".join(['sys.path.insert(0,r"' + x + '");' for x in ld]) + 'runpy.run_module("copyparty",run_name="__main__")' | ||||
|     c = [str(x) for x in [sys.executable, "-c", c] + list(sys.argv[1:])] | ||||
|     # fmt: on | ||||
|     msg("\n", c, "\n") | ||||
|     p = sp.Popen(c) | ||||
|  | ||||
|     def bye(*a): | ||||
|         p.send_signal(signal.SIGINT) | ||||
|  | ||||
|     signal.signal(signal.SIGTERM, bye) | ||||
|     p.wait() | ||||
|  | ||||
|     raise SystemExit(p.returncode) | ||||
|  | ||||
|  | ||||
| def main(): | ||||
| @@ -419,14 +453,23 @@ def main(): | ||||
|  | ||||
|     # skip 0 | ||||
|  | ||||
|     tmp = unpack() | ||||
|     tmp = os.path.realpath(unpack()) | ||||
|  | ||||
|     try: | ||||
|         from jinja2 import __version__ as j2ver | ||||
|         from jinja2 import __version__ as j2 | ||||
|     except: | ||||
|         j2ver = None | ||||
|         j2 = None | ||||
|  | ||||
|     run(tmp, j2ver) | ||||
|     try: | ||||
|         run(tmp, j2) | ||||
|     except SystemExit as ex: | ||||
|         c = ex.code | ||||
|         if c not in [0, -15]: | ||||
|             confirm(ex.code) | ||||
|     except KeyboardInterrupt: | ||||
|         pass | ||||
|     except: | ||||
|         confirm(0) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -47,7 +47,7 @@ grep -E '/(python|pypy)[0-9\.-]*$' >$dir/pys || true | ||||
| 	printf '\033[1;30mlooking for jinja2 in [%s]\033[0m\n' "$_py" >&2 | ||||
| 	$_py -c 'import jinja2' 2>/dev/null || continue | ||||
| 	printf '%s\n' "$_py" | ||||
| 	mv $dir/{,x.}jinja2 | ||||
| 	mv $dir/{,x.}dep-j2 | ||||
| 	break | ||||
| done)" | ||||
|  | ||||
|   | ||||
| @@ -17,14 +17,15 @@ __license__ = "MIT" | ||||
| __url__ = "https://github.com/9001/copyparty/" | ||||
|  | ||||
|  | ||||
| def get_spd(nbyte, nsec): | ||||
| def get_spd(nbyte, nfiles, nsec): | ||||
|     if not nsec: | ||||
|         return "0.000 MB   0.000 sec   0.000 MB/s" | ||||
|         return "0.000 MB   0 files   0.000 sec   0.000 MB/s   0.000 f/s" | ||||
|  | ||||
|     mb = nbyte / (1024 * 1024.0) | ||||
|     spd = mb / nsec | ||||
|     nspd = nfiles / nsec | ||||
|  | ||||
|     return f"{mb:.3f} MB   {nsec:.3f} sec   {spd:.3f} MB/s" | ||||
|     return f"{mb:.3f} MB   {nfiles} files   {nsec:.3f} sec   {spd:.3f} MB/s   {nspd:.3f} f/s" | ||||
|  | ||||
|  | ||||
| class Inf(object): | ||||
| @@ -36,6 +37,7 @@ class Inf(object): | ||||
|         self.mtx_reports = threading.Lock() | ||||
|  | ||||
|         self.n_byte = 0 | ||||
|         self.n_file = 0 | ||||
|         self.n_sec = 0 | ||||
|         self.n_done = 0 | ||||
|         self.t0 = t0 | ||||
| @@ -63,7 +65,8 @@ class Inf(object): | ||||
|                 continue | ||||
|  | ||||
|             msgs = msgs[-64:] | ||||
|             msgs = [f"{get_spd(self.n_byte, self.n_sec)}   {x}" for x in msgs] | ||||
|             spd = get_spd(self.n_byte, len(self.reports), self.n_sec) | ||||
|             msgs = [f"{spd}   {x}" for x in msgs] | ||||
|             print("\n".join(msgs)) | ||||
|  | ||||
|     def report(self, fn, n_byte, n_sec): | ||||
| @@ -131,8 +134,9 @@ def main(): | ||||
|  | ||||
|     num_threads = 8 | ||||
|     read_sz = 32 * 1024 | ||||
|     targs = (q, inf, read_sz) | ||||
|     for _ in range(num_threads): | ||||
|         thr = threading.Thread(target=worker, args=(q, inf, read_sz,)) | ||||
|         thr = threading.Thread(target=worker, args=targs) | ||||
|         thr.daemon = True | ||||
|         thr.start() | ||||
|  | ||||
| @@ -151,14 +155,14 @@ def main(): | ||||
|     log = inf.reports | ||||
|     log.sort() | ||||
|     for nbyte, nsec, fn in log[-64:]: | ||||
|         print(f"{get_spd(nbyte, nsec)}   {fn}") | ||||
|         spd = get_spd(nbyte, len(log), nsec) | ||||
|         print(f"{spd}   {fn}") | ||||
|  | ||||
|     print() | ||||
|     print("\n".join(inf.errors)) | ||||
|  | ||||
|     print(get_spd(inf.n_byte, t2 - t0)) | ||||
|     print(get_spd(inf.n_byte, len(log), t2 - t0)) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
|  | ||||
|   | ||||
							
								
								
									
										105
									
								
								scripts/test/race.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								scripts/test/race.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import json | ||||
| import threading | ||||
| import http.client | ||||
|  | ||||
|  | ||||
| class Conn(object): | ||||
|     def __init__(self, ip, port): | ||||
|         self.s = http.client.HTTPConnection(ip, port, timeout=260) | ||||
|         self.st = [] | ||||
|  | ||||
|     def get(self, vpath): | ||||
|         self.st = [time.time()] | ||||
|  | ||||
|         self.s.request("GET", vpath) | ||||
|         self.st.append(time.time()) | ||||
|  | ||||
|         ret = self.s.getresponse() | ||||
|         self.st.append(time.time()) | ||||
|  | ||||
|         if ret.status < 200 or ret.status >= 400: | ||||
|             raise Exception(ret.status) | ||||
|  | ||||
|         ret = ret.read() | ||||
|         self.st.append(time.time()) | ||||
|  | ||||
|         return ret | ||||
|  | ||||
|     def get_json(self, vpath): | ||||
|         ret = self.get(vpath) | ||||
|         return json.loads(ret) | ||||
|  | ||||
|  | ||||
| class CState(threading.Thread): | ||||
|     def __init__(self, cs): | ||||
|         threading.Thread.__init__(self) | ||||
|         self.daemon = True | ||||
|         self.cs = cs | ||||
|         self.start() | ||||
|  | ||||
|     def run(self): | ||||
|         colors = [5, 1, 3, 2, 7] | ||||
|         remotes = [] | ||||
|         remotes_ok = False | ||||
|         while True: | ||||
|             time.sleep(0.001) | ||||
|             if not remotes_ok: | ||||
|                 remotes = [] | ||||
|                 remotes_ok = True | ||||
|                 for conn in self.cs: | ||||
|                     try: | ||||
|                         remotes.append(conn.s.sock.getsockname()[1]) | ||||
|                     except: | ||||
|                         remotes.append("?") | ||||
|                         remotes_ok = False | ||||
|  | ||||
|             m = [] | ||||
|             for conn, remote in zip(self.cs, remotes): | ||||
|                 stage = len(conn.st) | ||||
|                 m.append(f"\033[3{colors[stage]}m{remote}") | ||||
|  | ||||
|             m = " ".join(m) | ||||
|             print(f"{m}\033[0m\n\033[A", end="") | ||||
|  | ||||
|  | ||||
| def allget(cs, urls): | ||||
|     thrs = [] | ||||
|     for c, url in zip(cs, urls): | ||||
|         t = threading.Thread(target=c.get, args=(url,)) | ||||
|         t.start() | ||||
|         thrs.append(t) | ||||
|  | ||||
|     for t in thrs: | ||||
|         t.join() | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     os.system("") | ||||
|  | ||||
|     ip, port = sys.argv[1].split(":") | ||||
|     port = int(port) | ||||
|  | ||||
|     cs = [] | ||||
|     for _ in range(64): | ||||
|         cs.append(Conn(ip, 3923)) | ||||
|  | ||||
|     CState(cs) | ||||
|  | ||||
|     urlbase = "/doujin/c95" | ||||
|     j = cs[0].get_json(f"{urlbase}?ls") | ||||
|     urls = [] | ||||
|     for d in j["dirs"]: | ||||
|         urls.append(f"{urlbase}/{d['href']}?th=w") | ||||
|  | ||||
|     for n in range(100): | ||||
|         print(n) | ||||
|         allget(cs, urls) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										209
									
								
								scripts/test/smoketest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										209
									
								
								scripts/test/smoketest.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,209 @@ | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import shlex | ||||
| import shutil | ||||
| import signal | ||||
| import tempfile | ||||
| import requests | ||||
| import threading | ||||
| import subprocess as sp | ||||
|  | ||||
|  | ||||
| CPP = [] | ||||
|  | ||||
|  | ||||
| class Cpp(object): | ||||
|     def __init__(self, args): | ||||
|         args = [sys.executable, "-m", "copyparty"] + args | ||||
|         print(" ".join([shlex.quote(x) for x in args])) | ||||
|  | ||||
|         self.ls_pre = set(list(os.listdir())) | ||||
|         self.p = sp.Popen(args) | ||||
|         # , stdout=sp.PIPE, stderr=sp.PIPE) | ||||
|  | ||||
|         self.t = threading.Thread(target=self._run) | ||||
|         self.t.daemon = True | ||||
|         self.t.start() | ||||
|  | ||||
|     def _run(self): | ||||
|         self.so, self.se = self.p.communicate() | ||||
|  | ||||
|     def stop(self, wait): | ||||
|         if wait: | ||||
|             os.kill(self.p.pid, signal.SIGINT) | ||||
|             self.t.join(timeout=2) | ||||
|         else: | ||||
|             self.p.kill()  # macos py3.8 | ||||
|  | ||||
|     def clean(self): | ||||
|         t = os.listdir() | ||||
|         for f in t: | ||||
|             if f not in self.ls_pre and f.startswith("up."): | ||||
|                 os.unlink(f) | ||||
|  | ||||
|     def await_idle(self, ub, timeout): | ||||
|         req = ["scanning</td><td>False", "hash-q</td><td>0", "tag-q</td><td>0"] | ||||
|         lim = int(timeout * 10) | ||||
|         u = ub + "?h" | ||||
|         for n in range(lim): | ||||
|             try: | ||||
|                 time.sleep(0.1) | ||||
|                 r = requests.get(u, timeout=0.1) | ||||
|                 for x in req: | ||||
|                     if x not in r.text: | ||||
|                         print("ST: {}/{} miss {}".format(n, lim, x)) | ||||
|                         raise Exception() | ||||
|                 print("ST: idle") | ||||
|                 return | ||||
|             except: | ||||
|                 pass | ||||
|  | ||||
|  | ||||
| def tc1(): | ||||
|     ub = "http://127.0.0.1:4321/" | ||||
|     td = os.path.join("srv", "smoketest") | ||||
|     try: | ||||
|         shutil.rmtree(td) | ||||
|     except: | ||||
|         if os.path.exists(td): | ||||
|             raise | ||||
|  | ||||
|     for _ in range(10): | ||||
|         try: | ||||
|             os.mkdir(td) | ||||
|         except: | ||||
|             time.sleep(0.1)  # win10 | ||||
|  | ||||
|     assert os.path.exists(td) | ||||
|  | ||||
|     vidp = os.path.join(tempfile.gettempdir(), "smoketest.h264") | ||||
|     if not os.path.exists(vidp): | ||||
|         cmd = "ffmpeg -f lavfi -i testsrc=48x32:3 -t 1 -c:v libx264 -tune animation -preset veryslow -crf 69" | ||||
|         sp.check_call(cmd.split(" ") + [vidp]) | ||||
|  | ||||
|     with open(vidp, "rb") as f: | ||||
|         ovid = f.read() | ||||
|  | ||||
|     args = [ | ||||
|         "-p4321", | ||||
|         "-e2dsa", | ||||
|         "-e2tsr", | ||||
|         "--no-mutagen", | ||||
|         "--th-ff-jpg", | ||||
|         "--hist", | ||||
|         os.path.join(td, "dbm"), | ||||
|     ] | ||||
|     pdirs = [] | ||||
|     hpaths = {} | ||||
|  | ||||
|     for d1 in ["r", "w", "a"]: | ||||
|         pdirs.append("{}/{}".format(td, d1)) | ||||
|         pdirs.append("{}/{}/j".format(td, d1)) | ||||
|         for d2 in ["r", "w", "a"]: | ||||
|             d = os.path.join(td, d1, "j", d2) | ||||
|             pdirs.append(d) | ||||
|             os.makedirs(d) | ||||
|  | ||||
|     pdirs = [x.replace("\\", "/") for x in pdirs] | ||||
|     udirs = [x.split("/", 2)[2] for x in pdirs] | ||||
|     perms = [x.rstrip("j/")[-1] for x in pdirs] | ||||
|     for pd, ud, p in zip(pdirs, udirs, perms): | ||||
|         if ud[-1] == "j": | ||||
|             continue | ||||
|  | ||||
|         hp = None | ||||
|         if pd.endswith("st/a"): | ||||
|             hp = hpaths[ud] = os.path.join(td, "db1") | ||||
|         elif pd[:-1].endswith("a/j/"): | ||||
|             hpaths[ud] = os.path.join(td, "dbm") | ||||
|             hp = None | ||||
|         else: | ||||
|             hp = "-" | ||||
|             hpaths[ud] = os.path.join(pd, ".hist") | ||||
|  | ||||
|         arg = "{}:{}:{}".format(pd, ud, p, hp) | ||||
|         if hp: | ||||
|             arg += ":chist=" + hp | ||||
|  | ||||
|         args += ["-v", arg] | ||||
|  | ||||
|     # return | ||||
|     cpp = Cpp(args) | ||||
|     CPP.append(cpp) | ||||
|     cpp.await_idle(ub, 3) | ||||
|  | ||||
|     for d in udirs: | ||||
|         vid = ovid + "\n{}".format(d).encode("utf-8") | ||||
|         try: | ||||
|             requests.post(ub + d, data={"act": "bput"}, files={"f": ("a.h264", vid)}) | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|     cpp.clean() | ||||
|  | ||||
|     # GET permission | ||||
|     for d, p in zip(udirs, perms): | ||||
|         u = "{}{}/a.h264".format(ub, d) | ||||
|         r = requests.get(u) | ||||
|         ok = bool(r) | ||||
|         if ok != (p in ["a"]): | ||||
|             raise Exception("get {} with perm {} at {}".format(ok, p, u)) | ||||
|  | ||||
|     # stat filesystem | ||||
|     for d, p in zip(pdirs, perms): | ||||
|         u = "{}/a.h264".format(d) | ||||
|         ok = os.path.exists(u) | ||||
|         if ok != (p in ["a", "w"]): | ||||
|             raise Exception("stat {} with perm {} at {}".format(ok, p, u)) | ||||
|  | ||||
|     # GET thumbnail, vreify contents | ||||
|     for d, p in zip(udirs, perms): | ||||
|         u = "{}{}/a.h264?th=j".format(ub, d) | ||||
|         r = requests.get(u) | ||||
|         ok = bool(r and r.content[:3] == b"\xff\xd8\xff") | ||||
|         if ok != (p in ["a"]): | ||||
|             raise Exception("thumb {} with perm {} at {}".format(ok, p, u)) | ||||
|  | ||||
|     # check tags | ||||
|     cpp.await_idle(ub, 5) | ||||
|     for d, p in zip(udirs, perms): | ||||
|         u = "{}{}?ls".format(ub, d) | ||||
|         r = requests.get(u) | ||||
|         j = r.json() if r else False | ||||
|         tag = None | ||||
|         if j: | ||||
|             for f in j["files"]: | ||||
|                 tag = tag or f["tags"].get("res") | ||||
|  | ||||
|         r_ok = bool(j) | ||||
|         w_ok = bool(r_ok and j.get("files")) | ||||
|  | ||||
|         if not r_ok or w_ok != (p in ["a"]): | ||||
|             raise Exception("ls {} with perm {} at {}".format(ok, p, u)) | ||||
|  | ||||
|         if (tag and p != "a") or (not tag and p == "a"): | ||||
|             raise Exception("tag {} with perm {} at {}".format(tag, p, u)) | ||||
|  | ||||
|         if tag is not None and tag != "48x32": | ||||
|             raise Exception("tag [{}] at {}".format(tag, u)) | ||||
|  | ||||
|     cpp.stop(True) | ||||
|  | ||||
|  | ||||
| def run(tc): | ||||
|     try: | ||||
|         tc() | ||||
|     finally: | ||||
|         try: | ||||
|             CPP[0].stop(False) | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     run(tc1) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										55
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										55
									
								
								setup.py
									
									
									
									
									
								
							| @@ -5,22 +5,7 @@ from __future__ import print_function | ||||
| import os | ||||
| import sys | ||||
| from shutil import rmtree | ||||
|  | ||||
| setuptools_available = True | ||||
| try: | ||||
|     # need setuptools to build wheel | ||||
|     from setuptools import setup, Command, find_packages | ||||
|  | ||||
| except ImportError: | ||||
|     # works in a pinch | ||||
|     setuptools_available = False | ||||
|     from distutils.core import setup, Command | ||||
|  | ||||
| from distutils.spawn import spawn | ||||
|  | ||||
| if "bdist_wheel" in sys.argv and not setuptools_available: | ||||
|     print("cannot build wheel without setuptools") | ||||
|     sys.exit(1) | ||||
| from setuptools import setup, Command, find_packages | ||||
|  | ||||
|  | ||||
| NAME = "copyparty" | ||||
| @@ -100,9 +85,8 @@ args = { | ||||
|     "author_email": "copyparty@ocv.me", | ||||
|     "url": "https://github.com/9001/copyparty", | ||||
|     "license": "MIT", | ||||
|     "data_files": data_files, | ||||
|     "classifiers": [ | ||||
|         "Development Status :: 3 - Alpha", | ||||
|         "Development Status :: 4 - Beta", | ||||
|         "License :: OSI Approved :: MIT License", | ||||
|         "Programming Language :: Python", | ||||
|         "Programming Language :: Python :: 2", | ||||
| @@ -120,35 +104,16 @@ args = { | ||||
|         "Environment :: Console", | ||||
|         "Environment :: No Input/Output (Daemon)", | ||||
|         "Topic :: Communications :: File Sharing", | ||||
|         "Topic :: Internet :: WWW/HTTP :: HTTP Servers", | ||||
|     ], | ||||
|     "include_package_data": True, | ||||
|     "data_files": data_files, | ||||
|     "packages": find_packages(), | ||||
|     "install_requires": ["jinja2"], | ||||
|     "extras_require": {"thumbnails": ["Pillow"], "audiotags": ["mutagen"]}, | ||||
|     "entry_points": {"console_scripts": ["copyparty = copyparty.__main__:main"]}, | ||||
|     "scripts": ["bin/copyparty-fuse.py"], | ||||
|     "cmdclass": {"clean2": clean2}, | ||||
| } | ||||
|  | ||||
|  | ||||
| if setuptools_available: | ||||
|     args.update( | ||||
|         { | ||||
|             "packages": find_packages(), | ||||
|             "install_requires": ["jinja2"], | ||||
|             "extras_require": {"thumbnails": ["Pillow"]}, | ||||
|             "include_package_data": True, | ||||
|             "entry_points": { | ||||
|                 "console_scripts": ["copyparty = copyparty.__main__:main"] | ||||
|             }, | ||||
|             "scripts": ["bin/copyparty-fuse.py"], | ||||
|         } | ||||
|     ) | ||||
| else: | ||||
|     args.update( | ||||
|         { | ||||
|             "packages": ["copyparty", "copyparty.stolen"], | ||||
|             "scripts": ["bin/copyparty-fuse.py"], | ||||
|         } | ||||
|     ) | ||||
|  | ||||
|  | ||||
| # import pprint | ||||
| # pprint.PrettyPrinter().pprint(args) | ||||
| # sys.exit(0) | ||||
|  | ||||
| setup(**args) | ||||
|   | ||||
							
								
								
									
										33
									
								
								tests/run.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										33
									
								
								tests/run.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| import sys | ||||
| import runpy | ||||
|  | ||||
| host = sys.argv[1] | ||||
| sys.argv = sys.argv[:1] + sys.argv[2:] | ||||
| sys.path.insert(0, ".") | ||||
|  | ||||
|  | ||||
| def rp(): | ||||
|     runpy.run_module("unittest", run_name="__main__") | ||||
|  | ||||
|  | ||||
| if host == "vmprof": | ||||
|     rp() | ||||
|  | ||||
| elif host == "cprofile": | ||||
|     import cProfile | ||||
|     import pstats | ||||
|  | ||||
|     log_fn = "cprofile.log" | ||||
|     cProfile.run("rp()", log_fn) | ||||
|     p = pstats.Stats(log_fn) | ||||
|     p.sort_stats(pstats.SortKey.CUMULATIVE).print_stats(64) | ||||
|  | ||||
|  | ||||
| """ | ||||
| python3.9 tests/run.py cprofile -v tests/test_httpcli.py | ||||
|  | ||||
| python3.9 -m pip install --user vmprof | ||||
| python3.9 -m vmprof --lines -o vmprof.log tests/run.py vmprof -v tests/test_httpcli.py | ||||
| """ | ||||
							
								
								
									
										210
									
								
								tests/test_httpcli.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										210
									
								
								tests/test_httpcli.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,210 @@ | ||||
| #!/usr/bin/env python | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import io | ||||
| import os | ||||
| import time | ||||
| import shutil | ||||
| import pprint | ||||
| import tarfile | ||||
| import tempfile | ||||
| import unittest | ||||
| from argparse import Namespace | ||||
|  | ||||
| from tests import util as tu | ||||
| from copyparty.authsrv import AuthSrv | ||||
| from copyparty.httpcli import HttpCli | ||||
|  | ||||
|  | ||||
| def hdr(query): | ||||
|     h = "GET /{} HTTP/1.1\r\nCookie: cppwd=o\r\nConnection: close\r\n\r\n" | ||||
|     return h.format(query).encode("utf-8") | ||||
|  | ||||
|  | ||||
| class Cfg(Namespace): | ||||
|     def __init__(self, a=[], v=[], c=None): | ||||
|         super(Cfg, self).__init__( | ||||
|             a=a, | ||||
|             v=v, | ||||
|             c=c, | ||||
|             rproxy=0, | ||||
|             ed=False, | ||||
|             no_zip=False, | ||||
|             no_scandir=False, | ||||
|             no_sendfile=True, | ||||
|             no_rescan=True, | ||||
|             ihead=False, | ||||
|             nih=True, | ||||
|             mtp=[], | ||||
|             mte="a", | ||||
|             hist=None, | ||||
|             no_hash=False, | ||||
|             css_browser=None, | ||||
|             **{k: False for k in "e2d e2ds e2dsa e2t e2ts e2tsr".split()} | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class TestHttpCli(unittest.TestCase): | ||||
|     def setUp(self): | ||||
|         self.td = tu.get_ramdisk() | ||||
|  | ||||
|     def tearDown(self): | ||||
|         os.chdir(tempfile.gettempdir()) | ||||
|         shutil.rmtree(self.td) | ||||
|  | ||||
|     def test(self): | ||||
|         td = os.path.join(self.td, "vfs") | ||||
|         os.mkdir(td) | ||||
|         os.chdir(td) | ||||
|  | ||||
|         self.dtypes = ["ra", "ro", "rx", "wa", "wo", "wx", "aa", "ao", "ax"] | ||||
|         self.can_read = ["ra", "ro", "aa", "ao"] | ||||
|         self.can_write = ["wa", "wo", "aa", "ao"] | ||||
|         self.fn = "g{:x}g".format(int(time.time() * 3)) | ||||
|  | ||||
|         allfiles = [] | ||||
|         allvols = [] | ||||
|         for top in self.dtypes: | ||||
|             allvols.append(top) | ||||
|             allfiles.append("/".join([top, self.fn])) | ||||
|             for s1 in self.dtypes: | ||||
|                 p = "/".join([top, s1]) | ||||
|                 allvols.append(p) | ||||
|                 allfiles.append(p + "/" + self.fn) | ||||
|                 allfiles.append(p + "/n/" + self.fn) | ||||
|                 for s2 in self.dtypes: | ||||
|                     p = "/".join([top, s1, "n", s2]) | ||||
|                     os.makedirs(p) | ||||
|                     allvols.append(p) | ||||
|                     allfiles.append(p + "/" + self.fn) | ||||
|  | ||||
|         for fp in allfiles: | ||||
|             with open(fp, "w") as f: | ||||
|                 f.write("ok {}\n".format(fp)) | ||||
|  | ||||
|         for top in self.dtypes: | ||||
|             vcfg = [] | ||||
|             for vol in allvols: | ||||
|                 if not vol.startswith(top): | ||||
|                     continue | ||||
|  | ||||
|                 mode = vol[-2] | ||||
|                 usr = vol[-1] | ||||
|                 if usr == "a": | ||||
|                     usr = "" | ||||
|  | ||||
|                 if "/" not in vol: | ||||
|                     vol += "/" | ||||
|  | ||||
|                 top, sub = vol.split("/", 1) | ||||
|                 vcfg.append("{0}/{1}:{1}:{2}{3}".format(top, sub, mode, usr)) | ||||
|  | ||||
|             pprint.pprint(vcfg) | ||||
|  | ||||
|             self.args = Cfg(v=vcfg, a=["o:o", "x:x"]) | ||||
|             self.asrv = AuthSrv(self.args, self.log) | ||||
|             vfiles = [x for x in allfiles if x.startswith(top)] | ||||
|             for fp in vfiles: | ||||
|                 rok, wok = self.can_rw(fp) | ||||
|                 furl = fp.split("/", 1)[1] | ||||
|                 durl = furl.rsplit("/", 1)[0] if "/" in furl else "" | ||||
|  | ||||
|                 # file download | ||||
|                 h, ret = self.curl(furl) | ||||
|                 res = "ok " + fp in ret | ||||
|                 print("[{}] {} {} = {}".format(fp, rok, wok, res)) | ||||
|                 if rok != res: | ||||
|                     print("\033[33m{}\n# {}\033[0m".format(ret, furl)) | ||||
|                     self.fail() | ||||
|  | ||||
|                 # file browser: html | ||||
|                 h, ret = self.curl(durl) | ||||
|                 res = "'{}'".format(self.fn) in ret | ||||
|                 print(res) | ||||
|                 if rok != res: | ||||
|                     print("\033[33m{}\n# {}\033[0m".format(ret, durl)) | ||||
|                     self.fail() | ||||
|  | ||||
|                 # file browser: json | ||||
|                 url = durl + "?ls" | ||||
|                 h, ret = self.curl(url) | ||||
|                 res = '"{}"'.format(self.fn) in ret | ||||
|                 print(res) | ||||
|                 if rok != res: | ||||
|                     print("\033[33m{}\n# {}\033[0m".format(ret, url)) | ||||
|                     self.fail() | ||||
|  | ||||
|                 # tar | ||||
|                 url = durl + "?tar" | ||||
|                 h, b = self.curl(url, True) | ||||
|                 # with open(os.path.join(td, "tar"), "wb") as f: | ||||
|                 #    f.write(b) | ||||
|                 try: | ||||
|                     tar = tarfile.open(fileobj=io.BytesIO(b)).getnames() | ||||
|                 except: | ||||
|                     tar = [] | ||||
|                 tar = ["/".join([y for y in [top, durl, x] if y]) for x in tar] | ||||
|                 tar = [[x] + self.can_rw(x) for x in tar] | ||||
|                 tar_ok = [x[0] for x in tar if x[1]] | ||||
|                 tar_ng = [x[0] for x in tar if not x[1]] | ||||
|                 self.assertEqual([], tar_ng) | ||||
|  | ||||
|                 if durl.split("/")[-1] in self.can_read: | ||||
|                     ref = [x for x in vfiles if self.in_dive(top + "/" + durl, x)] | ||||
|                     for f in ref: | ||||
|                         print("{}: {}".format("ok" if f in tar_ok else "NG", f)) | ||||
|                     ref.sort() | ||||
|                     tar_ok.sort() | ||||
|                     self.assertEqual(ref, tar_ok) | ||||
|  | ||||
|                 # stash | ||||
|                 h, ret = self.put(url) | ||||
|                 res = h.startswith("HTTP/1.1 200 ") | ||||
|                 self.assertEqual(res, wok) | ||||
|  | ||||
|     def can_rw(self, fp): | ||||
|         # lowest non-neutral folder declares permissions | ||||
|         expect = fp.split("/")[:-1] | ||||
|         for x in reversed(expect): | ||||
|             if x != "n": | ||||
|                 expect = x | ||||
|                 break | ||||
|  | ||||
|         return [expect in self.can_read, expect in self.can_write] | ||||
|  | ||||
|     def in_dive(self, top, fp): | ||||
|         # archiver bails at first inaccessible subvolume | ||||
|         top = top.strip("/").split("/") | ||||
|         fp = fp.split("/") | ||||
|         for f1, f2 in zip(top, fp): | ||||
|             if f1 != f2: | ||||
|                 return False | ||||
|  | ||||
|         for f in fp[len(top) :]: | ||||
|             if f == self.fn: | ||||
|                 return True | ||||
|             if f not in self.can_read and f != "n": | ||||
|                 return False | ||||
|  | ||||
|         return True | ||||
|  | ||||
|     def put(self, url): | ||||
|         buf = "PUT /{0} HTTP/1.1\r\nCookie: cppwd=o\r\nConnection: close\r\nContent-Length: {1}\r\n\r\nok {0}\n" | ||||
|         buf = buf.format(url, len(url) + 4).encode("utf-8") | ||||
|         conn = tu.VHttpConn(self.args, self.asrv, self.log, buf) | ||||
|         HttpCli(conn).run() | ||||
|         return conn.s._reply.decode("utf-8").split("\r\n\r\n", 1) | ||||
|  | ||||
|     def curl(self, url, binary=False): | ||||
|         conn = tu.VHttpConn(self.args, self.asrv, self.log, hdr(url)) | ||||
|         HttpCli(conn).run() | ||||
|         if binary: | ||||
|             h, b = conn.s._reply.split(b"\r\n\r\n", 1) | ||||
|             return [h.decode("utf-8"), b] | ||||
|  | ||||
|         return conn.s._reply.decode("utf-8").split("\r\n\r\n", 1) | ||||
|  | ||||
|     def log(self, src, msg, c=0): | ||||
|         # print(repr(msg)) | ||||
|         pass | ||||
| @@ -3,26 +3,41 @@ | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import time | ||||
| import json | ||||
| import shutil | ||||
| import tempfile | ||||
| import unittest | ||||
| import subprocess as sp  # nosec | ||||
|  | ||||
| from textwrap import dedent | ||||
| from argparse import Namespace | ||||
| from copyparty.authsrv import AuthSrv | ||||
|  | ||||
| from tests import util as tu | ||||
| from copyparty.authsrv import AuthSrv, VFS | ||||
| from copyparty import util | ||||
|  | ||||
|  | ||||
| class Cfg(Namespace): | ||||
|     def __init__(self, a=[], v=[], c=None): | ||||
|         ex = {k: False for k in "e2d e2ds e2dsa e2t e2ts e2tsr mte".split()} | ||||
|         ex = {k: False for k in "e2d e2ds e2dsa e2t e2ts e2tsr".split()} | ||||
|         ex2 = { | ||||
|             "mtp": [], | ||||
|             "mte": "a", | ||||
|             "hist": None, | ||||
|             "no_hash": False, | ||||
|             "css_browser": None, | ||||
|             "rproxy": 0, | ||||
|         } | ||||
|         ex.update(ex2) | ||||
|         super(Cfg, self).__init__(a=a, v=v, c=c, **ex) | ||||
|  | ||||
|  | ||||
| class TestVFS(unittest.TestCase): | ||||
|     def setUp(self): | ||||
|         self.td = tu.get_ramdisk() | ||||
|  | ||||
|     def tearDown(self): | ||||
|         os.chdir(tempfile.gettempdir()) | ||||
|         shutil.rmtree(self.td) | ||||
|  | ||||
|     def dump(self, vfs): | ||||
|         print(json.dumps(vfs, indent=4, sort_keys=True, default=lambda o: o.__dict__)) | ||||
|  | ||||
| @@ -39,6 +54,7 @@ class TestVFS(unittest.TestCase): | ||||
|         self.assertEqual(util.undot(query), response) | ||||
|  | ||||
|     def ls(self, vfs, vpath, uname): | ||||
|         # type: (VFS, str, str) -> tuple[str, str, str] | ||||
|         """helper for resolving and listing a folder""" | ||||
|         vn, rem = vfs.get(vpath, uname, True, False) | ||||
|         r1 = vn.ls(rem, uname, False) | ||||
| @@ -49,57 +65,11 @@ class TestVFS(unittest.TestCase): | ||||
|         real = [x[0] for x in real] | ||||
|         return fsdir, real, virt | ||||
|  | ||||
|     def runcmd(self, *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") | ||||
|         return [p.returncode, stdout, stderr] | ||||
|  | ||||
|     def chkcmd(self, *argv): | ||||
|         ok, sout, serr = self.runcmd(*argv) | ||||
|         if ok != 0: | ||||
|             raise Exception(serr) | ||||
|  | ||||
|         return sout, serr | ||||
|  | ||||
|     def get_ramdisk(self): | ||||
|         for vol in ["/dev/shm", "/Volumes/cptd"]:  # nosec (singleton test) | ||||
|             if os.path.exists(vol): | ||||
|                 return vol | ||||
|  | ||||
|         if os.path.exists("/Volumes"): | ||||
|             devname, _ = self.chkcmd("hdiutil", "attach", "-nomount", "ram://8192") | ||||
|             devname = devname.strip() | ||||
|             print("devname: [{}]".format(devname)) | ||||
|             for _ in range(10): | ||||
|                 try: | ||||
|                     _, _ = self.chkcmd( | ||||
|                         "diskutil", "eraseVolume", "HFS+", "cptd", devname | ||||
|                     ) | ||||
|                     return "/Volumes/cptd" | ||||
|                 except Exception as ex: | ||||
|                     print(repr(ex)) | ||||
|                     time.sleep(0.25) | ||||
|  | ||||
|             raise Exception("ramdisk creation failed") | ||||
|  | ||||
|         ret = os.path.join(tempfile.gettempdir(), "copyparty-test") | ||||
|         try: | ||||
|             os.mkdir(ret) | ||||
|         finally: | ||||
|             return ret | ||||
|  | ||||
|     def log(self, src, msg, c=0): | ||||
|         pass | ||||
|  | ||||
|     def test(self): | ||||
|         td = os.path.join(self.get_ramdisk(), "vfs") | ||||
|         try: | ||||
|             shutil.rmtree(td) | ||||
|         except OSError: | ||||
|             pass | ||||
|  | ||||
|         td = os.path.join(self.td, "vfs") | ||||
|         os.mkdir(td) | ||||
|         os.chdir(td) | ||||
|  | ||||
| @@ -150,13 +120,13 @@ class TestVFS(unittest.TestCase): | ||||
|         n = vfs.nodes["a"] | ||||
|         self.assertEqual(len(vfs.nodes), 1) | ||||
|         self.assertEqual(n.vpath, "a") | ||||
|         self.assertEqual(n.realpath, td + "/a") | ||||
|         self.assertEqual(n.realpath, os.path.join(td, "a")) | ||||
|         self.assertEqual(n.uread, ["*", "k"]) | ||||
|         self.assertEqual(n.uwrite, ["k"]) | ||||
|         n = n.nodes["ac"] | ||||
|         self.assertEqual(len(vfs.nodes), 1) | ||||
|         self.assertEqual(n.vpath, "a/ac") | ||||
|         self.assertEqual(n.realpath, td + "/a/ac") | ||||
|         self.assertEqual(n.realpath, os.path.join(td, "a", "ac")) | ||||
|         self.assertEqual(n.uread, ["*", "k"]) | ||||
|         self.assertEqual(n.uwrite, ["k"]) | ||||
|         n = n.nodes["acb"] | ||||
| @@ -266,7 +236,7 @@ class TestVFS(unittest.TestCase): | ||||
|         self.assertEqual(list(v1), list(v2)) | ||||
|  | ||||
|         # config file parser | ||||
|         cfg_path = os.path.join(self.get_ramdisk(), "test.cfg") | ||||
|         cfg_path = os.path.join(self.td, "test.cfg") | ||||
|         with open(cfg_path, "wb") as f: | ||||
|             f.write( | ||||
|                 dedent( | ||||
| @@ -288,7 +258,7 @@ class TestVFS(unittest.TestCase): | ||||
|         n = au.vfs | ||||
|         # root was not defined, so PWD with no access to anyone | ||||
|         self.assertEqual(n.vpath, "") | ||||
|         self.assertEqual(n.realpath, td) | ||||
|         self.assertEqual(n.realpath, None) | ||||
|         self.assertEqual(n.uread, []) | ||||
|         self.assertEqual(n.uwrite, []) | ||||
|         self.assertEqual(len(n.nodes), 1) | ||||
| @@ -299,6 +269,4 @@ class TestVFS(unittest.TestCase): | ||||
|         self.assertEqual(n.uwrite, ["asd"]) | ||||
|         self.assertEqual(len(n.nodes), 0) | ||||
|  | ||||
|         os.chdir(tempfile.gettempdir()) | ||||
|         shutil.rmtree(td) | ||||
|         os.unlink(cfg_path) | ||||
|   | ||||
							
								
								
									
										128
									
								
								tests/util.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								tests/util.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import shutil | ||||
| import jinja2 | ||||
| import tempfile | ||||
| import platform | ||||
| import subprocess as sp | ||||
|  | ||||
|  | ||||
| WINDOWS = platform.system() == "Windows" | ||||
| ANYWIN = WINDOWS or sys.platform in ["msys"] | ||||
| MACOS = platform.system() == "Darwin" | ||||
|  | ||||
| J2_ENV = jinja2.Environment(loader=jinja2.BaseLoader) | ||||
| J2_FILES = J2_ENV.from_string("{{ files|join('\n') }}") | ||||
|  | ||||
|  | ||||
| def nah(*a, **ka): | ||||
|     return False | ||||
|  | ||||
|  | ||||
| if MACOS: | ||||
|     import posixpath | ||||
|  | ||||
|     posixpath.islink = nah | ||||
|     os.path.islink = nah | ||||
|     # 25% faster; until any tests do symlink stuff | ||||
|  | ||||
|  | ||||
| from copyparty.util import Unrecv | ||||
|  | ||||
|  | ||||
| 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") | ||||
|     return [p.returncode, stdout, stderr] | ||||
|  | ||||
|  | ||||
| def chkcmd(*argv): | ||||
|     ok, sout, serr = runcmd(*argv) | ||||
|     if ok != 0: | ||||
|         raise Exception(serr) | ||||
|  | ||||
|     return sout, serr | ||||
|  | ||||
|  | ||||
| def get_ramdisk(): | ||||
|     def subdir(top): | ||||
|         ret = os.path.join(top, "cptd-{}".format(os.getpid())) | ||||
|         shutil.rmtree(ret, True) | ||||
|         os.mkdir(ret) | ||||
|         return ret | ||||
|  | ||||
|     for vol in ["/dev/shm", "/Volumes/cptd"]:  # nosec (singleton test) | ||||
|         if os.path.exists(vol): | ||||
|             return subdir(vol) | ||||
|  | ||||
|     if os.path.exists("/Volumes"): | ||||
|         # hdiutil eject /Volumes/cptd/ | ||||
|         devname, _ = chkcmd("hdiutil", "attach", "-nomount", "ram://131072") | ||||
|         devname = devname.strip() | ||||
|         print("devname: [{}]".format(devname)) | ||||
|         for _ in range(10): | ||||
|             try: | ||||
|                 _, _ = chkcmd("diskutil", "eraseVolume", "HFS+", "cptd", devname) | ||||
|                 return subdir("/Volumes/cptd") | ||||
|             except Exception as ex: | ||||
|                 print(repr(ex)) | ||||
|                 time.sleep(0.25) | ||||
|  | ||||
|         raise Exception("ramdisk creation failed") | ||||
|  | ||||
|     ret = os.path.join(tempfile.gettempdir(), "copyparty-test") | ||||
|     try: | ||||
|         os.mkdir(ret) | ||||
|     finally: | ||||
|         return subdir(ret) | ||||
|  | ||||
|  | ||||
| class NullBroker(object): | ||||
|     def put(*args): | ||||
|         pass | ||||
|  | ||||
|  | ||||
| class VSock(object): | ||||
|     def __init__(self, buf): | ||||
|         self._query = buf | ||||
|         self._reply = b"" | ||||
|         self.sendall = self.send | ||||
|  | ||||
|     def recv(self, sz): | ||||
|         ret = self._query[:sz] | ||||
|         self._query = self._query[sz:] | ||||
|         return ret | ||||
|  | ||||
|     def send(self, buf): | ||||
|         self._reply += buf | ||||
|         return len(buf) | ||||
|  | ||||
|  | ||||
| class VHttpSrv(object): | ||||
|     def __init__(self): | ||||
|         self.broker = NullBroker() | ||||
|  | ||||
|         aliases = ["splash", "browser", "browser2", "msg", "md", "mde"] | ||||
|         self.j2 = {x: J2_FILES for x in aliases} | ||||
|  | ||||
|  | ||||
| class VHttpConn(object): | ||||
|     def __init__(self, args, asrv, log, buf): | ||||
|         self.s = VSock(buf) | ||||
|         self.sr = Unrecv(self.s) | ||||
|         self.addr = ("127.0.0.1", "42069") | ||||
|         self.args = args | ||||
|         self.asrv = asrv | ||||
|         self.is_mp = False | ||||
|         self.log_func = log | ||||
|         self.log_src = "a" | ||||
|         self.lf_url = None | ||||
|         self.hsrv = VHttpSrv() | ||||
|         self.nbyte = 0 | ||||
|         self.workload = 0 | ||||
|         self.ico = None | ||||
|         self.thumbcli = None | ||||
|         self.t0 = time.time() | ||||
		Reference in New Issue
	
	Block a user