mirror of
				https://github.com/9001/copyparty.git
				synced 2025-10-31 12:03:32 +00:00 
			
		
		
		
	Compare commits
	
		
			294 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | a203e33347 | ||
|  | 3b8f697dd4 | ||
|  | 78ba16f722 | ||
|  | 0fcfe79994 | ||
|  | c0e6df4b63 | ||
|  | 322abdcb43 | ||
|  | 31100787ce | ||
|  | c57d721be4 | ||
|  | 3b5a03e977 | ||
|  | ed807ee43e | ||
|  | 073c130ae6 | ||
|  | 8810e0be13 | ||
|  | f93016ab85 | ||
|  | b19cf260c2 | ||
|  | db03e1e7eb | ||
|  | e0d975e36a | ||
|  | cfeb15259f | ||
|  | 3b3f8fc8fb | ||
|  | 88bd2c084c | ||
|  | bd367389b0 | ||
|  | 58ba71a76f | ||
|  | d03e34d55d | ||
|  | 24f239a46c | ||
|  | 2c0826f85a | ||
|  | c061461d01 | ||
|  | e7982a04fe | ||
|  | 33b91a7513 | ||
|  | 9bb1323e44 | ||
|  | e62bb807a5 | ||
|  | 3fc0d2cc4a | ||
|  | 0c786b0766 | ||
|  | 68c7528911 | ||
|  | 26e18ae800 | ||
|  | c30dc0b546 | ||
|  | f94aa46a11 | ||
|  | 403261a293 | ||
|  | c7d9cbb11f | ||
|  | 57e1c53cbb | ||
|  | 0754b553dd | ||
|  | 50661d941b | ||
|  | c5db7c1a0c | ||
|  | 2cef5365f7 | ||
|  | fbc4e94007 | ||
|  | 037ed5a2ad | ||
|  | 69dfa55705 | ||
|  | a79a5c4e3e | ||
|  | 7e80eabfe6 | ||
|  | 375b72770d | ||
|  | e2dd683def | ||
|  | 9eba50c6e4 | ||
|  | 5a579dba52 | ||
|  | e86c719575 | ||
|  | 0e87f35547 | ||
|  | b6d3d791a5 | ||
|  | c9c3302664 | ||
|  | c3e4d65b80 | ||
|  | 27a03510c5 | ||
|  | ed7727f7cb | ||
|  | 127ec10c0d | ||
|  | 5a9c0ad225 | ||
|  | 7e8daf650e | ||
|  | 0cf737b4ce | ||
|  | 74635e0113 | ||
|  | e5c4f49901 | ||
|  | e4654ee7f1 | ||
|  | e5d05c05ed | ||
|  | 73c4f99687 | ||
|  | 28c12ef3bf | ||
|  | eed82dbb54 | ||
|  | 2c4b4ab928 | ||
|  | 505a8fc6f6 | ||
|  | e4801d9b06 | ||
|  | 04f1b2cf3a | ||
|  | c06d928bb5 | ||
|  | ab09927e7b | ||
|  | 779437db67 | ||
|  | 28cbdb652e | ||
|  | 2b2415a7d8 | ||
|  | 746a8208aa | ||
|  | a2a041a98a | ||
|  | 10b436e449 | ||
|  | 4d62b34786 | ||
|  | 0546210687 | ||
|  | f8c11faada | ||
|  | 16d6e9be1f | ||
|  | aff8185f2e | ||
|  | 217d15fe81 | ||
|  | 171e93c201 | ||
|  | acc1d2e9e3 | ||
|  | 49c2f37154 | ||
|  | 69e54497aa | ||
|  | 9aa1885669 | ||
|  | 4418508513 | ||
|  | e897df3b34 | ||
|  | 8cd97ab0e7 | ||
|  | bf4949353d | ||
|  | 98a944f7cc | ||
|  | 7c10f81c92 | ||
|  | 126ecc55c3 | ||
|  | 1034a51bd2 | ||
|  | a2657887cc | ||
|  | c14b17bfaf | ||
|  | 59ebc795e7 | ||
|  | 8e128d917e | ||
|  | ea762b05e0 | ||
|  | db374b19f1 | ||
|  | ab3839ef36 | ||
|  | 9886c442f2 | ||
|  | c8d1926d52 | ||
|  | a6bd699e52 | ||
|  | 12143f2702 | ||
|  | 480705dee9 | ||
|  | 781d5094f4 | ||
|  | 5615cb94cd | ||
|  | 302302a2ac | ||
|  | 9761b4e3e9 | ||
|  | 0cf6924dca | ||
|  | 5fd81e9f90 | ||
|  | 52bf6f892b | ||
|  | f3cce232a4 | ||
|  | 53d3c8b28e | ||
|  | 83fec3cca7 | ||
|  | 3cefc99b7d | ||
|  | 3a38dcbc05 | ||
|  | 7ff08bce57 | ||
|  | fd490af434 | ||
|  | 1195b8f17e | ||
|  | 28dce13776 | ||
|  | 431f20177a | ||
|  | 87aff54d9d | ||
|  | f50462de82 | ||
|  | 9bda8c7eb6 | ||
|  | e83c63d239 | ||
|  | b38533b0cc | ||
|  | 5ccca3fbd5 | ||
|  | 9e850fc3ab | ||
|  | ffbfcd7e00 | ||
|  | 5ea7590748 | ||
|  | 290c3bc2bb | ||
|  | b12131e91c | ||
|  | 3b354447b0 | ||
|  | d09ec6feaa | ||
|  | 21405c3fda | ||
|  | 13e5c96cab | ||
|  | 426687b75e | ||
|  | c8f59fb978 | ||
|  | 871dde79a9 | ||
|  | e14d81bc6f | ||
|  | 514d046d1f | ||
|  | 4ed9528d36 | ||
|  | 625560e642 | ||
|  | 73ebd917d1 | ||
|  | cd3e0afad2 | ||
|  | d8d1f94a86 | ||
|  | 00dfd8cfd1 | ||
|  | 273de6db31 | ||
|  | c6c0eeb0ff | ||
|  | e70c74a3b5 | ||
|  | f7d939eeab | ||
|  | e815c091b9 | ||
|  | 963529b7cf | ||
|  | 638a52374d | ||
|  | d9d42b7aa2 | ||
|  | ec7e5f36a2 | ||
|  | 56110883ea | ||
|  | 7f8d7d6006 | ||
|  | 49e4fb7e12 | ||
|  | 8dbbea473f | ||
|  | 3d375d5114 | ||
|  | f3eae67d97 | ||
|  | 40c1b19235 | ||
|  | ccaf0ab159 | ||
|  | d07f147423 | ||
|  | f5cb9f92b9 | ||
|  | f991f74983 | ||
|  | 6b3295059e | ||
|  | b18a07ae6b | ||
|  | 8ab03dabda | ||
|  | 5e760e35dc | ||
|  | afbfa04514 | ||
|  | 7aace470c5 | ||
|  | b4acb24f6a | ||
|  | bcee8a4934 | ||
|  | 36b0718542 | ||
|  | 9a92bca45d | ||
|  | b07445a363 | ||
|  | a62ec0c27e | ||
|  | 57e3a2d382 | ||
|  | b61022b374 | ||
|  | a3e2b2ec87 | ||
|  | a83d3f8801 | ||
|  | 90c5f2b9d2 | ||
|  | 4885653c07 | ||
|  | 21e1cd87ca | ||
|  | 81f82e8e9f | ||
|  | c0e31851da | ||
|  | 6599c3eced | ||
|  | 5d6c61a861 | ||
|  | 1a5c66edd3 | ||
|  | deae9fe95a | ||
|  | abd65c6334 | ||
|  | 8137a99904 | ||
|  | 6f6f9c1f74 | ||
|  | 7b575f716f | ||
|  | 6ba6ea3572 | ||
|  | 9a22ad5ea3 | ||
|  | beaab9778e | ||
|  | f327bdb6b4 | ||
|  | ae180e0f5f | ||
|  | e3f1d19756 | ||
|  | 93c2bd6ef6 | ||
|  | 4d0e5ff6db | ||
|  | 0893f06919 | ||
|  | 46b6abde3f | ||
|  | 0696610dee | ||
|  | edf0d3684c | ||
|  | 7af159f5f6 | ||
|  | 7f2cb6764a | ||
|  | 96495a9bf1 | ||
|  | b2fafec5fc | ||
|  | 0850b8ae2b | ||
|  | 8a68a96c57 | ||
|  | d3aae8ed6a | ||
|  | c62ebadda8 | ||
|  | ffcee6d390 | ||
|  | de32838346 | ||
|  | b9a4e47ea2 | ||
|  | 57d994422d | ||
|  | 6ecd745323 | ||
|  | bd769f5bdb | ||
|  | 2381692aba | ||
|  | 24fdada0a0 | ||
|  | bb5169710a | ||
|  | 9cde2352f3 | ||
|  | 482dd7a938 | ||
|  | bddcc69438 | ||
|  | 19d4540630 | ||
|  | 4f5f6c81f5 | ||
|  | 7e4c1238ba | ||
|  | f7196ac773 | ||
|  | 7a7c832000 | ||
|  | 2b4ccdbebb | ||
|  | 0d16b49489 | ||
|  | 768405b691 | ||
|  | da01413b7b | ||
|  | 914e22c53e | ||
|  | 43a23bf733 | ||
|  | 92bb00c6d2 | ||
|  | b0b97a2648 | ||
|  | 2c452fe323 | ||
|  | ad73d0c77d | ||
|  | 7f9bf1c78c | ||
|  | 61a6bc3a65 | ||
|  | 46e10b0e9f | ||
|  | 8441206e26 | ||
|  | 9fdc5ee748 | ||
|  | 00ff133387 | ||
|  | 96164cb934 | ||
|  | 82fb21ae69 | ||
|  | 89d4a2b4c4 | ||
|  | fc0c7ff374 | ||
|  | 5148c4f2e9 | ||
|  | c3b59f7bcf | ||
|  | 61e148202b | ||
|  | 8a4e0739bc | ||
|  | f75c5f2fe5 | ||
|  | 81d5859588 | ||
|  | 721886bb7a | ||
|  | b23c272820 | ||
|  | cd02bfea7a | ||
|  | 6774bd88f9 | ||
|  | 1046a4f376 | ||
|  | 8081f9ddfd | ||
|  | fa656577d1 | ||
|  | b14b86990f | ||
|  | 2a6dd7b512 | ||
|  | feebdee88b | ||
|  | 99d9277f5d | ||
|  | 9af64d6156 | ||
|  | 5e3775c1af | ||
|  | 2d2e8a3da7 | ||
|  | b2a560b76f | ||
|  | 39397a489d | ||
|  | ff593a0904 | ||
|  | f12789cf44 | ||
|  | 4f8cf2fc87 | ||
|  | fda98730ac | ||
|  | 06c6ddffb6 | ||
|  | d29f0c066c | ||
|  | c9e4de3346 | ||
|  | ca0b97f72d | ||
|  | b38f20b408 | ||
|  | 05b1dbaf56 | ||
|  | b8481e32ba | 
							
								
								
									
										10
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @@ -12,12 +12,16 @@ | |||||||
|                 //"-nw", |                 //"-nw", | ||||||
|                 "-ed", |                 "-ed", | ||||||
|                 "-emp", |                 "-emp", | ||||||
|                 "-e2d", |                 "-e2dsa", | ||||||
|                 "-e2s", |                 "-e2ts", | ||||||
|  |                 "-mtp", | ||||||
|  |                 ".bpm=f,bin/mtag/audio-bpm.py", | ||||||
|                 "-a", |                 "-a", | ||||||
|                 "ed:wark", |                 "ed:wark", | ||||||
|                 "-v", |                 "-v", | ||||||
|                 "srv::r:aed:cnodupe" |                 "srv::r:aed:cnodupe", | ||||||
|  |                 "-v", | ||||||
|  |                 "dist:dist:r" | ||||||
|             ] |             ] | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|   | |||||||
							
								
								
									
										35
									
								
								.vscode/launch.py
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								.vscode/launch.py
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | |||||||
|  | # 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 os | ||||||
|  | import sys | ||||||
|  | import shlex | ||||||
|  |  | ||||||
|  | sys.path.insert(0, os.getcwd()) | ||||||
|  |  | ||||||
|  | import jstyleson | ||||||
|  | from copyparty.__main__ import main as copyparty | ||||||
|  |  | ||||||
|  | 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] | ||||||
|  | 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
									
									
								
							| @@ -8,8 +8,8 @@ | |||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|             "label": "no_dbg", |             "label": "no_dbg", | ||||||
|             "command": "${config:python.pythonPath} -m copyparty -ed -emp -e2d -e2s -a ed:wark -v srv::r:aed:cnodupe ;exit 1", |             "type": "shell", | ||||||
|             "type": "shell" |             "command": "${config:python.pythonPath} .vscode/launch.py" | ||||||
|         } |         } | ||||||
|     ] |     ] | ||||||
| } | } | ||||||
							
								
								
									
										391
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										391
									
								
								README.md
									
									
									
									
									
								
							| @@ -9,9 +9,47 @@ | |||||||
| turn your phone or raspi into a portable file server with resumable uploads/downloads using IE6 or any other browser | turn your phone or raspi into a portable file server with resumable uploads/downloads using IE6 or any other browser | ||||||
|  |  | ||||||
| * server runs on anything with `py2.7` or `py3.3+` | * 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 37+` / `safari 7+` | ||||||
| * code standard: `black` | * code standard: `black` | ||||||
|  |  | ||||||
|  | 📷 screenshots: [browser](#the-browser) // [upload](#uploading) // [md-viewer](#markdown-viewer) // [search](#searching) // [fsearch](#file-search) // [zip-DL](#zip-downloads) // [ie4](#browser-support) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## readme toc | ||||||
|  |  | ||||||
|  | * top | ||||||
|  |     * [quickstart](#quickstart) | ||||||
|  |     * [notes](#notes) | ||||||
|  |     * [status](#status) | ||||||
|  | * [bugs](#bugs) | ||||||
|  |     * [not my bugs](#not-my-bugs) | ||||||
|  | * [the browser](#the-browser) | ||||||
|  |     * [tabs](#tabs) | ||||||
|  |     * [hotkeys](#hotkeys) | ||||||
|  |     * [tree-mode](#tree-mode) | ||||||
|  |     * [zip downloads](#zip-downloads) | ||||||
|  |     * [uploading](#uploading) | ||||||
|  |         * [file-search](#file-search) | ||||||
|  |     * [markdown viewer](#markdown-viewer) | ||||||
|  |     * [other tricks](#other-tricks) | ||||||
|  | * [searching](#searching) | ||||||
|  |     * [search configuration](#search-configuration) | ||||||
|  |     * [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 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) | ||||||
|  | * [todo](#todo) | ||||||
|  |  | ||||||
|  |  | ||||||
| ## quickstart | ## quickstart | ||||||
|  |  | ||||||
| @@ -27,55 +65,350 @@ you may also want these, especially on servers: | |||||||
| ## notes | ## notes | ||||||
|  |  | ||||||
| * iPhone/iPad: use Firefox to download files | * iPhone/iPad: use Firefox to download files | ||||||
| * Android-Chrome: set max "parallel uploads" for 200% upload speed (android bug) | * Android-Chrome: increase "parallel uploads" for higher speed (android bug) | ||||||
| * Android-Firefox: takes a while to select files (in order to avoid the above android-chrome issue) | * Android-Firefox: takes a while to select files (their fix for ☝️) | ||||||
| * Desktop-Firefox: may use gigabytes of RAM if your connection is great and your files are massive | * Desktop-Firefox: ~~may use gigabytes of RAM if your files are massive~~ *seems to be OK now* | ||||||
| * paper-printing is affected by dark/light-mode! use lightmode for color, darkmode for grayscale | * 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 |   * because no browsers currently implement the media-query to do this properly orz | ||||||
|  |  | ||||||
|  |  | ||||||
| ## status | ## status | ||||||
|  |  | ||||||
| * [x] sanic multipart parser | * backend stuff | ||||||
| * [x] load balancer (multiprocessing) |   * ☑ sanic multipart parser | ||||||
| * [x] upload (plain multipart, ie6 support) |   * ☑ load balancer (multiprocessing) | ||||||
| * [x] upload (js, resumable, multithreaded) |   * ☑ volumes (mountpoints) | ||||||
| * [x] download |   * ☑ accounts | ||||||
| * [x] browser | * upload | ||||||
| * [x] media player |   * ☑ basic: plain multipart, ie6 support | ||||||
| * [ ] thumbnails |   * ☑ up2k: js, resumable, multithreaded | ||||||
| * [ ] download as zip |   * ☑ stash: simple PUT filedropper | ||||||
| * [x] volumes |   * ☑ symlink/discard existing files (content-matching) | ||||||
| * [x] accounts | * download | ||||||
| * [x] markdown viewer |   * ☑ single files in browser | ||||||
| * [x] markdown editor |   * ☑ folders as zip / tar files | ||||||
| * [x] FUSE client (read-only) |   * ☑ 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 | ||||||
|  | * server indexing | ||||||
|  |   * ☑ locate files by contents | ||||||
|  |   * ☑ search by name/path/date/size | ||||||
|  |   * ☑ search by ID3-tags etc. | ||||||
|  | * markdown | ||||||
|  |   * ☑ viewer | ||||||
|  |   * ☑ editor (sure why not) | ||||||
|  |  | ||||||
| summary: it works! you can use it! (but technically not even close to beta) | 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 | ||||||
|  | * hiding the contents at url `/d1/d2/d3` using `-v :d1/d2/d3:cd2d` has the side-effect of creating databases (for files/tags) inside folders d1 and d2, and those databases take precedence over the main db at the top of the vfs - this means all files in d2 and below will be reindexed unless you already had a vfs entry at or below d2 | ||||||
|  | * probably more, pls let me know | ||||||
|  |  | ||||||
|  | ## not my bugs | ||||||
|  |  | ||||||
|  | * 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 | ||||||
|  | * `I/K` prev/next folder | ||||||
|  | * `P` parent folder | ||||||
|  | * when playing audio: | ||||||
|  |   * `0..9` jump to 10%..90% | ||||||
|  |   * `U/O` skip 10sec back/forward | ||||||
|  |   * `J/L` prev/next song | ||||||
|  |     * `J` also starts playing the folder | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## 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 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## zip downloads | ||||||
|  |  | ||||||
|  | the `zip` link next to folders can produce various types of zip/tar files using these alternatives in the browser settings tab: | ||||||
|  |  | ||||||
|  | | name | url-suffix | description | | ||||||
|  | |--|--|--| | ||||||
|  | | `tar` | `?tar` | plain gnutar, works great with `curl \| tar -xv` | | ||||||
|  | | `zip` | `?zip=utf8` | works everywhere, glitchy filenames on win7 and older | | ||||||
|  | | `zip_dos` | `?zip` | traditional cp437 (no unicode) to fix glitchy filenames | | ||||||
|  | | `zip_crc` | `?zip=crc` | cp437 with crc32 computed early for truly ancient software | | ||||||
|  |  | ||||||
|  | * hidden files (dotfiles) are excluded unless `-ed` | ||||||
|  |   * the up2k.db is always excluded | ||||||
|  | * `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 | ||||||
|  |  | ||||||
|  | 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 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, see [file-search](#file-search) | ||||||
|  |  | ||||||
|  | path/name queries are space-separated, AND'ed together, and words are negated with a `-` prefix, so for example: | ||||||
|  | * path: `shibayan -bossa` finds all files where one of the folders contain `shibayan` but filters out any results where `bossa` exists somewhere in the path | ||||||
|  | * name: `demetori styx` gives you [good stuff](https://www.youtube.com/watch?v=zGh0g14ZJ8I&list=PL3A147BD151EE5218&index=9) | ||||||
|  |  | ||||||
|  | add `-e2ts` to also scan/index tags from music files: | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## search configuration | ||||||
|  |  | ||||||
|  | searching relies on two databases, the up2k filetree (`-e2d`) and the metadata tags (`-e2t`). Configuration can be done through arguments, volume flags, or a mix of both. | ||||||
|  |  | ||||||
|  | through arguments: | ||||||
|  | * `-e2d` enables file indexing on upload | ||||||
|  | * `-e2ds` scans writable folders on startup | ||||||
|  | * `-e2dsa` scans all mounted volumes (including readonly ones) | ||||||
|  | * `-e2t` enables metadata indexing on upload | ||||||
|  | * `-e2ts` scans for tags in all files that don't have tags yet | ||||||
|  | * `-e2tsr` deletes all existing tags, so a full reindex | ||||||
|  |  | ||||||
|  | the same arguments can be set as volume flags, in addition to `d2d` and `d2t` for disabling: | ||||||
|  | * `-v ~/music::r:ce2dsa:ce2tsr` does a full reindex of everything on startup | ||||||
|  | * `-v ~/music::r:cd2d` disables **all** indexing, even if any `-e2*` are on | ||||||
|  | * `-v ~/music::r:cd2t` disables all `-e2t*` (tags), does not affect `-e2d*` | ||||||
|  |  | ||||||
|  | `e2tsr` is probably always overkill, since `e2ds`/`e2dsa` would pick up any file modifications and cause `e2ts` to reindex those | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## metadata from audio files | ||||||
|  |  | ||||||
|  | `-mte` decides which tags to index and display in the browser (and also the display order), this can be changed per-volume: | ||||||
|  | * `-v ~/music::r:cmte=title,artist` indexes and displays *title* followed by *artist* | ||||||
|  |  | ||||||
|  | if you add/remove a tag from `mte` you will need to run with `-e2tsr` once to rebuild the database, otherwise only new files will be affected | ||||||
|  |  | ||||||
|  | `-mtm` can be used to add or redefine a metadata mapping, say you have media files with `foo` and `bar` tags and you want them to display as `qux` in the browser (preferring `foo` if both are present), then do `-mtm qux=foo,bar` and now you can `-mte artist,title,qux` | ||||||
|  |  | ||||||
|  | tags that start with a `.` such as `.bpm` and `.dur`(ation) indicate numeric value | ||||||
|  |  | ||||||
|  | see the beautiful mess of a dictionary in [mtag.py](https://github.com/9001/copyparty/blob/master/copyparty/mtag.py) for the default mappings (should cover mp3,opus,flac,m4a,wav,aif,) | ||||||
|  |  | ||||||
|  | `--no-mutagen` disables mutagen and uses ffprobe instead, which... | ||||||
|  | * is about 20x slower than mutagen | ||||||
|  | * catches a few tags that mutagen doesn't | ||||||
|  | * avoids pulling any GPL code into copyparty | ||||||
|  | * more importantly runs ffprobe on incoming files which is bad if your ffmpeg has a cve | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## file parser plugins | ||||||
|  |  | ||||||
|  | copyparty can invoke external programs to collect additional metadata for files using `mtp` (as argument or volume flag), there is a default timeout of 30sec | ||||||
|  |  | ||||||
|  | * `-mtp .bpm=~/bin/audio-bpm.py` will execute `~/bin/audio-bpm.py` with the audio file as argument 1 to provide the `.bpm` tag, if that does not exist in the audio metadata | ||||||
|  | * `-mtp key=f,t5,~/bin/audio-key.py` uses `~/bin/audio-key.py` to get the `key` tag, replacing any existing metadata tag (`f,`), aborting if it takes longer than 5sec (`t5,`) | ||||||
|  | * `-v ~/music::r:cmtp=.bpm=~/bin/audio-bpm.py:cmtp=key=f,t5,~/bin/audio-key.py` both as a per-volume config wow this is getting ugly | ||||||
|  |  | ||||||
|  | *but wait, there's more!* `-mtp` can be used for non-audio files as well using the `a` flag: `ay` only do audio files, `an` audio files are skipped, or `ad` always do it (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`) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## complete examples | ||||||
|  |  | ||||||
|  | * read-only music server with bpm and key scanning   | ||||||
|  |   `python copyparty-sfx.py -v /mnt/nas/music:/music:r -e2dsa -e2ts -mtp .bpm=f,audio-bpm.py -mtp key=f,audio-key.py` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # browser support | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | `ie` = internet-explorer, `ff` = firefox, `c` = chrome, `iOS` = iPhone/iPad, `Andr` = Android | ||||||
|  |  | ||||||
|  | | feature         | ie6 | ie9 | ie10 | ie11 | ff 52 | c 49 | iOS | Andr | | ||||||
|  | | --------------- | --- | --- | ---- | ---- | ----- | ---- | --- | ---- | | ||||||
|  | | browse files    | yep | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||||
|  | | basic uploader  | yep | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||||
|  | | make directory  | yep | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||||
|  | | send message    | yep | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||||
|  | | set sort order  |  -  | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||||
|  | | zip selection   |  -  | yep | yep  | yep  | yep   | yep  | yep | yep  | | ||||||
|  | | 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: | ||||||
|  | * 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 | # client examples | ||||||
|  |  | ||||||
| * javascript: dump some state into a file (two separate examples) | * javascript: dump some state into a file (two separate examples) | ||||||
|   * `await fetch('https://127.0.0.1:3923/', {method:"PUT", body: JSON.stringify(foo)});` |   * `await fetch('https://127.0.0.1:3923/', {method:"PUT", body: JSON.stringify(foo)});` | ||||||
|   * `var xhr = new XMLHttpRequest(); xhr.open('POST', 'https://127.0.0.1:3923/msgs?raw'); xhr.send('foo');` |   * `var xhr = new XMLHttpRequest(); xhr.open('POST', 'https://127.0.0.1:3923/msgs?raw'); xhr.send('foo');` | ||||||
|  |  | ||||||
|  | * curl/wget: upload some files (post=file, chunk=stdin) | ||||||
|  |   * `post(){ curl -b cppwd=wark http://127.0.0.1:3923/ -F act=bput -F f=@"$1";}`   | ||||||
|  |     `post movie.mkv` | ||||||
|  |   * `post(){ wget --header='Cookie: cppwd=wark' http://127.0.0.1:3923/?raw --post-file="$1" -O-;}`   | ||||||
|  |     `post movie.mkv` | ||||||
|  |   * `chunk(){ curl -b cppwd=wark http://127.0.0.1:3923/ -T-;}`   | ||||||
|  |     `chunk <movie.mkv` | ||||||
|  |  | ||||||
| * FUSE: mount a copyparty server as a local filesystem | * FUSE: mount a copyparty server as a local filesystem | ||||||
|   * cross-platform python client available in [./bin/](bin/) |   * 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) |   * [rclone](https://rclone.org/) as client can give ~5x performance, see [./docs/rclone.md](docs/rclone.md) | ||||||
|  |  | ||||||
|  | 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 <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# or chunksize >= 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 | # dependencies | ||||||
|  |  | ||||||
| * `jinja2` | * `jinja2` (is built into the SFX) | ||||||
|  |  | ||||||
| optional, will eventually enable thumbnails: | **optional,** enables 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: | ||||||
| * `Pillow` (requires py2.7 or py3.5+) | * `Pillow` (requires py2.7 or py3.5+) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## optional gpl stuff | ||||||
|  |  | ||||||
|  | some bundled tools have copyleft dependencies, see [./bin/#mtag](bin/#mtag) | ||||||
|  |  | ||||||
|  | these are standalone programs and will never be imported / evaluated by copyparty | ||||||
|  |  | ||||||
|  |  | ||||||
| # sfx | # sfx | ||||||
|  |  | ||||||
| currently there are two self-contained binaries: | 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.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 | * [copyparty-sfx.sh](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.sh) -- smaller, but only for linux and macos | ||||||
|  |  | ||||||
| @@ -127,6 +460,7 @@ pip install black bandit pylint flake8  # vscode tooling | |||||||
| in the `scripts` folder: | in the `scripts` folder: | ||||||
|  |  | ||||||
| * run `make -C deps-docker` to build all dependencies | * run `make -C deps-docker` to build all dependencies | ||||||
|  | * `git tag v1.2.3 && git push origin --tags` | ||||||
| * create github release with `make-tgz-release.sh` | * create github release with `make-tgz-release.sh` | ||||||
| * upload to pypi with `make-pypi-release.(sh|bat)` | * upload to pypi with `make-pypi-release.(sh|bat)` | ||||||
| * create sfx with `make-sfx.sh` | * create sfx with `make-sfx.sh` | ||||||
| @@ -136,15 +470,20 @@ in the `scripts` folder: | |||||||
|  |  | ||||||
| roughly sorted by priority | roughly sorted by priority | ||||||
|  |  | ||||||
|  | * separate sqlite table per tag | ||||||
|  | * audio fingerprinting | ||||||
|  | * readme.md as epilogue | ||||||
| * reduce up2k roundtrips | * reduce up2k roundtrips | ||||||
|   * start from a chunk index and just go |   * start from a chunk index and just go | ||||||
|   * terminate client on bad data |   * terminate client on bad data | ||||||
| * drop onto folders |  | ||||||
| * `os.copy_file_range` for up2k cloning | * `os.copy_file_range` for up2k cloning | ||||||
| * up2k partials ui |  | ||||||
| * support pillow-simd | * support pillow-simd | ||||||
| * cache sha512 chunks on client |  | ||||||
| * comment field |  | ||||||
| * ~~look into android thumbnail cache file format~~ bad idea |  | ||||||
| * figure out the deal with pixel3a not being connectable as hotspot | * figure out the deal with pixel3a not being connectable as hotspot | ||||||
|   * pixel3a having unpredictable 3sec latency in general :|||| |   * pixel3a having unpredictable 3sec latency in general :|||| | ||||||
|  |  | ||||||
|  | discarded ideas | ||||||
|  |  | ||||||
|  | * up2k partials ui | ||||||
|  | * cache sha512 chunks on client | ||||||
|  | * comment field | ||||||
|  | * look into android thumbnail cache file format | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # copyparty-fuse.py | # [`copyparty-fuse.py`](copyparty-fuse.py) | ||||||
| * mount a copyparty server as a local filesystem (read-only) | * mount a copyparty server as a local filesystem (read-only) | ||||||
| * **supports Windows!** -- expect `194 MiB/s` sequential read | * **supports Windows!** -- expect `194 MiB/s` sequential read | ||||||
| * **supports Linux** -- expect `117 MiB/s` sequential read | * **supports Linux** -- expect `117 MiB/s` sequential read | ||||||
| @@ -29,7 +29,7 @@ you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releas | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # copyparty-fuse🅱️.py | # [`copyparty-fuse🅱️.py`](copyparty-fuseb.py) | ||||||
| * mount a copyparty server as a local filesystem (read-only) | * mount a copyparty server as a local filesystem (read-only) | ||||||
| * does the same thing except more correct, `samba` approves | * does the same thing except more correct, `samba` approves | ||||||
| * **supports Linux** -- expect `18 MiB/s` (wait what) | * **supports Linux** -- expect `18 MiB/s` (wait what) | ||||||
| @@ -37,5 +37,11 @@ you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releas | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # copyparty-fuse-streaming.py | # [`copyparty-fuse-streaming.py`](copyparty-fuse-streaming.py) | ||||||
| * pretend this doesn't exist | * pretend this doesn't exist | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # [`mtag/`](mtag/) | ||||||
|  | * standalone programs which perform misc. file analysis | ||||||
|  | * copyparty can Popen programs like these during file indexing to collect additional metadata | ||||||
|   | |||||||
| @@ -33,6 +33,7 @@ import re | |||||||
| import os | import os | ||||||
| import sys | import sys | ||||||
| import time | import time | ||||||
|  | import json | ||||||
| import stat | import stat | ||||||
| import errno | import errno | ||||||
| import struct | import struct | ||||||
| @@ -323,7 +324,7 @@ class Gateway(object): | |||||||
|         if bad_good: |         if bad_good: | ||||||
|             path = dewin(path) |             path = dewin(path) | ||||||
|  |  | ||||||
|         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots" |         web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots&ls" | ||||||
|         r = self.sendreq("GET", web_path) |         r = self.sendreq("GET", web_path) | ||||||
|         if r.status != 200: |         if r.status != 200: | ||||||
|             self.closeconn() |             self.closeconn() | ||||||
| @@ -334,12 +335,17 @@ class Gateway(object): | |||||||
|             ) |             ) | ||||||
|             raise FuseOSError(errno.ENOENT) |             raise FuseOSError(errno.ENOENT) | ||||||
|  |  | ||||||
|         if not r.getheader("Content-Type", "").startswith("text/html"): |         ctype = r.getheader("Content-Type", "") | ||||||
|  |         if ctype == "application/json": | ||||||
|  |             parser = self.parse_jls | ||||||
|  |         elif ctype.startswith("text/html"): | ||||||
|  |             parser = self.parse_html | ||||||
|  |         else: | ||||||
|             log("listdir on file: {}".format(path)) |             log("listdir on file: {}".format(path)) | ||||||
|             raise FuseOSError(errno.ENOENT) |             raise FuseOSError(errno.ENOENT) | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             return self.parse_html(r) |             return parser(r) | ||||||
|         except: |         except: | ||||||
|             info(repr(path) + "\n" + traceback.format_exc()) |             info(repr(path) + "\n" + traceback.format_exc()) | ||||||
|             raise |             raise | ||||||
| @@ -367,6 +373,29 @@ class Gateway(object): | |||||||
|  |  | ||||||
|         return r.read() |         return r.read() | ||||||
|  |  | ||||||
|  |     def parse_jls(self, datasrc): | ||||||
|  |         rsp = b"" | ||||||
|  |         while True: | ||||||
|  |             buf = datasrc.read(1024 * 32) | ||||||
|  |             if not buf: | ||||||
|  |                 break | ||||||
|  |  | ||||||
|  |             rsp += buf | ||||||
|  |  | ||||||
|  |         rsp = json.loads(rsp.decode("utf-8")) | ||||||
|  |         ret = [] | ||||||
|  |         for is_dir, nodes in [[True, rsp["dirs"]], [False, rsp["files"]]]: | ||||||
|  |             for n in nodes: | ||||||
|  |                 fname = unquote(n["href"]).rstrip(b"/") | ||||||
|  |                 fname = fname.decode("wtf-8") | ||||||
|  |                 if bad_good: | ||||||
|  |                     fname = enwin(fname) | ||||||
|  |  | ||||||
|  |                 fun = self.stat_dir if is_dir else self.stat_file | ||||||
|  |                 ret.append([fname, fun(n["ts"], n["sz"]), 0]) | ||||||
|  |  | ||||||
|  |         return ret | ||||||
|  |  | ||||||
|     def parse_html(self, datasrc): |     def parse_html(self, datasrc): | ||||||
|         ret = [] |         ret = [] | ||||||
|         remainder = b"" |         remainder = b"" | ||||||
| @@ -818,9 +847,9 @@ class CPPF(Operations): | |||||||
|                 return cache_stat |                 return cache_stat | ||||||
|  |  | ||||||
|         fun = info |         fun = info | ||||||
|         if MACOS and path.split('/')[-1].startswith('._'): |         if MACOS and path.split("/")[-1].startswith("._"): | ||||||
|             fun = dbg |             fun = dbg | ||||||
|          |  | ||||||
|         fun("=ENOENT ({})".format(hexler(path))) |         fun("=ENOENT ({})".format(hexler(path))) | ||||||
|         raise FuseOSError(errno.ENOENT) |         raise FuseOSError(errno.ENOENT) | ||||||
|  |  | ||||||
| @@ -979,6 +1008,12 @@ def main(): | |||||||
|         log = null_log |         log = null_log | ||||||
|         dbg = null_log |         dbg = null_log | ||||||
|  |  | ||||||
|  |     if ar.a and ar.a.startswith("$"): | ||||||
|  |         fn = ar.a[1:] | ||||||
|  |         log("reading password from file [{}]".format(fn)) | ||||||
|  |         with open(fn, "rb") as f: | ||||||
|  |             ar.a = f.read().decode("utf-8").strip() | ||||||
|  |  | ||||||
|     if WINDOWS: |     if WINDOWS: | ||||||
|         os.system("rem") |         os.system("rem") | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										34
									
								
								bin/mtag/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								bin/mtag/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | |||||||
|  | standalone programs which take an audio file as argument | ||||||
|  |  | ||||||
|  | some of these rely on libraries which are not MIT-compatible | ||||||
|  |  | ||||||
|  | * [audio-bpm.py](./audio-bpm.py) detects the BPM of music using the BeatRoot Vamp Plugin; imports GPL2 | ||||||
|  | * [audio-key.py](./audio-key.py) detects the melodic key of music using the Mixxx fork of keyfinder; imports GPL3 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # dependencies | ||||||
|  |  | ||||||
|  | run [`install-deps.sh`](install-deps.sh) to build/install most dependencies required by these programs (supports windows/linux/macos) | ||||||
|  |  | ||||||
|  | *alternatively* (or preferably) use packages from your distro instead, then you'll need at least these: | ||||||
|  |  | ||||||
|  | * from distro: `numpy vamp-plugin-sdk beatroot-vamp mixxx-keyfinder ffmpeg` | ||||||
|  | * from pypy: `keyfinder vamp` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # usage from copyparty | ||||||
|  |  | ||||||
|  | `copyparty -e2dsa -e2ts -mtp key=f,audio-key.py -mtp .bpm=f,audio-bpm.py` | ||||||
|  |  | ||||||
|  | * `f,` makes the detected value replace any existing values | ||||||
|  | * the `.` in `.bpm` indicates numeric value | ||||||
|  | * assumes the python files are in the folder you're launching copyparty from, replace the filename with a relative/absolute path if that's not the case | ||||||
|  | * `mtp` modules will not run if a file has existing tags in the db, so clear out the tags with `-e2tsr` the first time you launch with new `mtp` options | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## usage with volume-flags | ||||||
|  |  | ||||||
|  | instead of affecting all volumes, you can set the options for just one volume like so: | ||||||
|  | ``` | ||||||
|  | copyparty -v /mnt/nas/music:/music:r:cmtp=key=f,audio-key.py:cmtp=.bpm=f,audio-bpm.py:ce2dsa:ce2ts | ||||||
|  | ``` | ||||||
							
								
								
									
										69
									
								
								bin/mtag/audio-bpm.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										69
									
								
								bin/mtag/audio-bpm.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,69 @@ | |||||||
|  | #!/usr/bin/env python | ||||||
|  |  | ||||||
|  | import os | ||||||
|  | import sys | ||||||
|  | import vamp | ||||||
|  | import tempfile | ||||||
|  | import numpy as np | ||||||
|  | import subprocess as sp | ||||||
|  |  | ||||||
|  | from copyparty.util import fsenc | ||||||
|  |  | ||||||
|  | """ | ||||||
|  | dep: vamp | ||||||
|  | dep: beatroot-vamp | ||||||
|  | dep: ffmpeg | ||||||
|  | """ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def det(tf): | ||||||
|  |     # fmt: off | ||||||
|  |     sp.check_call([ | ||||||
|  |         "ffmpeg", | ||||||
|  |         "-nostdin", | ||||||
|  |         "-hide_banner", | ||||||
|  |         "-v", "fatal", | ||||||
|  |         "-ss", "13", | ||||||
|  |         "-y", "-i", fsenc(sys.argv[1]), | ||||||
|  |         "-ac", "1", | ||||||
|  |         "-ar", "22050", | ||||||
|  |         "-t", "300", | ||||||
|  |         "-f", "f32le", | ||||||
|  |         tf | ||||||
|  |     ]) | ||||||
|  |     # fmt: on | ||||||
|  |  | ||||||
|  |     with open(tf, "rb") as f: | ||||||
|  |         d = np.fromfile(f, dtype=np.float32) | ||||||
|  |         try: | ||||||
|  |             # 98% accuracy on jcore | ||||||
|  |             c = vamp.collect(d, 22050, "beatroot-vamp:beatroot") | ||||||
|  |             cl = c["list"] | ||||||
|  |         except: | ||||||
|  |             # fallback; 73% accuracy | ||||||
|  |             plug = "vamp-example-plugins:fixedtempo" | ||||||
|  |             c = vamp.collect(d, 22050, plug, parameters={"maxdflen": 40}) | ||||||
|  |             print(c["list"][0]["label"].split(" ")[0]) | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         # throws if detection failed: | ||||||
|  |         bpm = float(cl[-1]["timestamp"] - cl[1]["timestamp"]) | ||||||
|  |         bpm = round(60 * ((len(cl) - 1) / bpm), 2) | ||||||
|  |         print(f"{bpm:.2f}") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def main(): | ||||||
|  |     with tempfile.NamedTemporaryFile(suffix=".pcm", delete=False) as f: | ||||||
|  |         f.write(b"h") | ||||||
|  |         tf = f.name | ||||||
|  |  | ||||||
|  |     try: | ||||||
|  |         det(tf) | ||||||
|  |     except: | ||||||
|  |         pass | ||||||
|  |     finally: | ||||||
|  |         os.unlink(tf) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == "__main__": | ||||||
|  |     main() | ||||||
							
								
								
									
										18
									
								
								bin/mtag/audio-key.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										18
									
								
								bin/mtag/audio-key.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | #!/usr/bin/env python | ||||||
|  |  | ||||||
|  | import sys | ||||||
|  | import keyfinder | ||||||
|  |  | ||||||
|  | """ | ||||||
|  | dep: github/mixxxdj/libkeyfinder | ||||||
|  | dep: pypi/keyfinder | ||||||
|  | dep: ffmpeg | ||||||
|  |  | ||||||
|  | note: cannot fsenc | ||||||
|  | """ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     print(keyfinder.key(sys.argv[1]).camelot()) | ||||||
|  | except: | ||||||
|  |     pass | ||||||
							
								
								
									
										9
									
								
								bin/mtag/file-ext.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								bin/mtag/file-ext.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | #!/usr/bin/env python | ||||||
|  |  | ||||||
|  | import sys | ||||||
|  |  | ||||||
|  | """ | ||||||
|  | example that just prints the file extension | ||||||
|  | """ | ||||||
|  |  | ||||||
|  | print(sys.argv[1].split(".")[-1]) | ||||||
							
								
								
									
										265
									
								
								bin/mtag/install-deps.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										265
									
								
								bin/mtag/install-deps.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,265 @@ | |||||||
|  | #!/bin/bash | ||||||
|  | set -e | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # install dependencies for audio-*.py | ||||||
|  | # | ||||||
|  | # linux: requires {python3,ffmpeg,fftw}-dev py3-{wheel,pip} py3-numpy{,-dev} vamp-sdk-dev patchelf | ||||||
|  | # win64: requires msys2-mingw64 environment | ||||||
|  | # macos: requires macports | ||||||
|  | # | ||||||
|  | # has the following manual dependencies, especially on mac: | ||||||
|  | #   https://www.vamp-plugins.org/pack.html | ||||||
|  | # | ||||||
|  | # installs stuff to the following locations: | ||||||
|  | #   ~/pe/ | ||||||
|  | #   whatever your python uses for --user packages | ||||||
|  | # | ||||||
|  | # does the following terrible things: | ||||||
|  | #   modifies the keyfinder python lib to load the .so in ~/pe | ||||||
|  |  | ||||||
|  |  | ||||||
|  | linux=1 | ||||||
|  |  | ||||||
|  | win= | ||||||
|  | [ ! -z "$MSYSTEM" ] || [ -e /msys2.exe ] && { | ||||||
|  | 	[ "$MSYSTEM" = MINGW64 ] || { | ||||||
|  | 		echo windows detected, msys2-mingw64 required | ||||||
|  | 		exit 1 | ||||||
|  | 	} | ||||||
|  | 	pacman -S --needed mingw-w64-x86_64-{ffmpeg,python,python-pip,vamp-plugin-sdk} | ||||||
|  | 	win=1 | ||||||
|  | 	linux= | ||||||
|  | } | ||||||
|  |  | ||||||
|  | mac= | ||||||
|  | [ $(uname -s) = Darwin ] && { | ||||||
|  | 	#pybin="$(printf '%s\n' /opt/local/bin/python* | (sed -E 's/(.*\/[^/0-9]+)([0-9]?[^/]*)$/\2 \1/' || cat) | (sort -nr || cat) | (sed -E 's/([^ ]*) (.*)/\2\1/' || cat) | grep -E '/(python|pypy)[0-9\.-]*$' | head -n 1)" | ||||||
|  | 	pybin=/opt/local/bin/python3.9 | ||||||
|  | 	[ -e "$pybin" ] || { | ||||||
|  | 		echo mac detected, python3 from macports required | ||||||
|  | 		exit 1 | ||||||
|  | 	} | ||||||
|  | 	pkgs='ffmpeg python39 py39-wheel' | ||||||
|  | 	ninst=$(port installed | awk '/^  /{print$1}' | sort | uniq | grep -E '^('"$(echo "$pkgs" | tr ' ' '|')"')$' | wc -l) | ||||||
|  | 	[ $ninst -eq 3 ] || { | ||||||
|  | 		sudo port install $pkgs | ||||||
|  | 	} | ||||||
|  | 	mac=1 | ||||||
|  | 	linux= | ||||||
|  | } | ||||||
|  |  | ||||||
|  | hash -r | ||||||
|  |  | ||||||
|  | [ $mac ] || { | ||||||
|  | 	command -v python3 && pybin=python3 || pybin=python | ||||||
|  | } | ||||||
|  |  | ||||||
|  | $pybin -m pip install --user numpy | ||||||
|  |  | ||||||
|  |  | ||||||
|  | command -v gnutar && tar() { gnutar "$@"; } | ||||||
|  | command -v gtar && tar() { gtar "$@"; } | ||||||
|  | command -v gsed && sed() { gsed "$@"; } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | need() { | ||||||
|  | 	command -v $1 >/dev/null || { | ||||||
|  | 		echo need $1 | ||||||
|  | 		exit 1 | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | need cmake | ||||||
|  | need ffmpeg | ||||||
|  | need $pybin | ||||||
|  | #need patchelf | ||||||
|  |  | ||||||
|  |  | ||||||
|  | td="$(mktemp -d)" | ||||||
|  | cln() { | ||||||
|  | 	rm -rf "$td" | ||||||
|  | } | ||||||
|  | trap cln EXIT | ||||||
|  | cd "$td" | ||||||
|  | pwd | ||||||
|  |  | ||||||
|  |  | ||||||
|  | dl_text() { | ||||||
|  | 	command -v curl >/dev/null && exec curl "$@" | ||||||
|  | 	exec wget -O- "$@" | ||||||
|  | } | ||||||
|  | dl_files() { | ||||||
|  | 	local yolo= ex= | ||||||
|  | 	[ $1 = "yolo" ] && yolo=1 && ex=k && shift | ||||||
|  | 	command -v curl >/dev/null && exec curl -${ex}JOL "$@" | ||||||
|  | 	 | ||||||
|  | 	[ $yolo ] && ex=--no-check-certificate | ||||||
|  | 	exec wget --trust-server-names $ex "$@" | ||||||
|  | } | ||||||
|  | export -f dl_files | ||||||
|  |  | ||||||
|  |  | ||||||
|  | github_tarball() { | ||||||
|  | 	dl_text "$1" | | ||||||
|  | 	tee json | | ||||||
|  | 	( | ||||||
|  | 		# prefer jq if available | ||||||
|  | 		jq -r '.tarball_url' || | ||||||
|  |  | ||||||
|  | 		# fallback to awk (sorry) | ||||||
|  | 		awk -F\" '/"tarball_url": "/ {print$4}' | ||||||
|  | 	) | | ||||||
|  | 	tee /dev/stderr | | ||||||
|  | 	tr -d '\r' | tr '\n' '\0' | | ||||||
|  | 	xargs -0 bash -c 'dl_files "$@"' _ | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | gitlab_tarball() { | ||||||
|  | 	dl_text "$1" | | ||||||
|  | 	tee json | | ||||||
|  | 	( | ||||||
|  | 		# prefer jq if available | ||||||
|  | 		jq -r '.[0].assets.sources[]|select(.format|test("tar.gz")).url' || | ||||||
|  |  | ||||||
|  | 		# fallback to abomination | ||||||
|  | 		tr \" '\n' | grep -E '\.tar\.gz$' | head -n 1 | ||||||
|  | 	) | | ||||||
|  | 	tee /dev/stderr | | ||||||
|  | 	tr -d '\r' | tr '\n' '\0' | | ||||||
|  | 	tee links | | ||||||
|  | 	xargs -0 bash -c 'dl_files "$@"' _ | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | install_keyfinder() { | ||||||
|  | 	# windows support: | ||||||
|  | 	#   use msys2 in mingw-w64 mode | ||||||
|  | 	#   pacman -S --needed mingw-w64-x86_64-{ffmpeg,python} | ||||||
|  | 	 | ||||||
|  | 	github_tarball https://api.github.com/repos/mixxxdj/libkeyfinder/releases/latest | ||||||
|  |  | ||||||
|  | 	tar -xf mixxxdj-libkeyfinder-* | ||||||
|  | 	rm -- *.tar.gz | ||||||
|  | 	cd mixxxdj-libkeyfinder* | ||||||
|  | 	 | ||||||
|  | 	h="$HOME" | ||||||
|  | 	so="lib/libkeyfinder.so" | ||||||
|  | 	memes=() | ||||||
|  |  | ||||||
|  | 	[ $win ] && | ||||||
|  | 		so="bin/libkeyfinder.dll" && | ||||||
|  | 		h="$(printf '%s\n' "$USERPROFILE" | tr '\\' '/')" && | ||||||
|  | 		memes+=(-G "MinGW Makefiles" -DBUILD_TESTING=OFF) | ||||||
|  | 	 | ||||||
|  | 	[ $mac ] && | ||||||
|  | 		so="lib/libkeyfinder.dylib" | ||||||
|  |  | ||||||
|  | 	cmake -DCMAKE_INSTALL_PREFIX="$h/pe/keyfinder" "${memes[@]}" -S . -B build | ||||||
|  | 	cmake --build build --parallel $(nproc || echo 4) | ||||||
|  | 	cmake --install build | ||||||
|  |  | ||||||
|  | 	libpath="$h/pe/keyfinder/$so" | ||||||
|  | 	[ $linux ] && [ ! -e "$libpath" ] && | ||||||
|  | 		so=lib64/libkeyfinder.so | ||||||
|  | 	 | ||||||
|  | 	libpath="$h/pe/keyfinder/$so" | ||||||
|  | 	[ -e "$libpath" ] || { | ||||||
|  | 		echo "so not found at $sop" | ||||||
|  | 		exit 1 | ||||||
|  | 	} | ||||||
|  | 	 | ||||||
|  | 	# rm -rf /Users/ed/Library/Python/3.9/lib/python/site-packages/*keyfinder* | ||||||
|  | 	CFLAGS="-I$h/pe/keyfinder/include -I/opt/local/include" \ | ||||||
|  | 	LDFLAGS="-L$h/pe/keyfinder/lib -L$h/pe/keyfinder/lib64 -L/opt/local/lib" \ | ||||||
|  | 	PKG_CONFIG_PATH=/c/msys64/mingw64/lib/pkgconfig \ | ||||||
|  | 	$pybin -m pip install --user keyfinder | ||||||
|  |  | ||||||
|  | 	pypath="$($pybin -c 'import keyfinder; print(keyfinder.__file__)')" | ||||||
|  | 	for pyso in "${pypath%/*}"/*.so; do | ||||||
|  | 		[ -e "$pyso" ] || break | ||||||
|  | 		patchelf --set-rpath "${libpath%/*}" "$pyso" || | ||||||
|  | 			echo "WARNING: patchelf failed (only fatal on musl-based distros)" | ||||||
|  | 	done | ||||||
|  | 	 | ||||||
|  | 	mv "$pypath"{,.bak} | ||||||
|  | 	( | ||||||
|  | 		printf 'import ctypes\nctypes.cdll.LoadLibrary("%s")\n' "$libpath" | ||||||
|  | 		cat "$pypath.bak" | ||||||
|  | 	) >"$pypath" | ||||||
|  |  | ||||||
|  | 	echo | ||||||
|  | 	echo libkeyfinder successfully installed to the following locations: | ||||||
|  | 	echo "  $libpath" | ||||||
|  | 	echo "  $pypath" | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | have_beatroot() { | ||||||
|  | 	$pybin -c 'import vampyhost, sys; plugs = vampyhost.list_plugins(); sys.exit(0 if "beatroot-vamp:beatroot" in plugs else 1)' | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | install_vamp() { | ||||||
|  | 	# windows support: | ||||||
|  | 	#   use msys2 in mingw-w64 mode | ||||||
|  | 	#   pacman -S --needed mingw-w64-x86_64-{ffmpeg,python,python-pip,vamp-plugin-sdk} | ||||||
|  | 	 | ||||||
|  | 	$pybin -m pip install --user vamp | ||||||
|  |  | ||||||
|  | 	have_beatroot || { | ||||||
|  | 		printf '\033[33mcould not find the vamp beatroot plugin, building from source\033[0m\n' | ||||||
|  | 		(dl_files yolo https://code.soundsoftware.ac.uk/attachments/download/885/beatroot-vamp-v1.0.tar.gz) | ||||||
|  | 		sha512sum -c <( | ||||||
|  | 			echo "1f444d1d58ccf565c0adfe99f1a1aa62789e19f5071e46857e2adfbc9d453037bc1c4dcb039b02c16240e9b97f444aaff3afb625c86aa2470233e711f55b6874  -" | ||||||
|  | 		) <beatroot-vamp-v1.0.tar.gz | ||||||
|  | 		tar -xf beatroot-vamp-v1.0.tar.gz  | ||||||
|  | 		cd beatroot-vamp-v1.0 | ||||||
|  | 		make -f Makefile.linux -j4 | ||||||
|  | 		# /home/ed/vamp /home/ed/.vamp /usr/local/lib/vamp | ||||||
|  | 		mkdir ~/vamp | ||||||
|  | 		cp -pv beatroot-vamp.* ~/vamp/ | ||||||
|  | 	} | ||||||
|  | 	 | ||||||
|  | 	have_beatroot && | ||||||
|  | 		printf '\033[32mfound the vamp beatroot plugin, nice\033[0m\n' || | ||||||
|  | 		printf '\033[31mWARNING: could not find the vamp beatroot plugin, please install it for optimal results\033[0m\n' | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # not in use because it kinda segfaults, also no windows support | ||||||
|  | install_soundtouch() { | ||||||
|  | 	gitlab_tarball https://gitlab.com/api/v4/projects/soundtouch%2Fsoundtouch/releases | ||||||
|  | 	 | ||||||
|  | 	tar -xvf soundtouch-* | ||||||
|  | 	rm -- *.tar.gz | ||||||
|  | 	cd soundtouch-* | ||||||
|  | 	 | ||||||
|  | 	# https://github.com/jrising/pysoundtouch | ||||||
|  | 	./bootstrap | ||||||
|  | 	./configure --enable-integer-samples CXXFLAGS="-fPIC" --prefix="$HOME/pe/soundtouch" | ||||||
|  | 	make -j$(nproc || echo 4) | ||||||
|  | 	make install | ||||||
|  | 	 | ||||||
|  | 	CFLAGS=-I$HOME/pe/soundtouch/include/ \ | ||||||
|  | 	LDFLAGS=-L$HOME/pe/soundtouch/lib \ | ||||||
|  | 	$pybin -m pip install --user git+https://github.com/snowxmas/pysoundtouch.git | ||||||
|  | 	 | ||||||
|  | 	pypath="$($pybin -c 'import importlib; print(importlib.util.find_spec("soundtouch").origin)')" | ||||||
|  | 	libpath="$(echo "$HOME/pe/soundtouch/lib/")" | ||||||
|  | 	patchelf --set-rpath "$libpath" "$pypath" | ||||||
|  |  | ||||||
|  | 	echo | ||||||
|  | 	echo soundtouch successfully installed to the following locations: | ||||||
|  | 	echo "  $libpath" | ||||||
|  | 	echo "  $pypath" | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [ "$1" = keyfinder ] && { install_keyfinder; exit $?; } | ||||||
|  | [ "$1" = soundtouch ] && { install_soundtouch; exit $?; } | ||||||
|  | [ "$1" = vamp ] && { install_vamp; exit $?; } | ||||||
|  |  | ||||||
|  | echo no args provided, installing keyfinder and vamp | ||||||
|  | install_keyfinder | ||||||
|  | install_vamp | ||||||
							
								
								
									
										8
									
								
								bin/mtag/sleep.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								bin/mtag/sleep.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | #!/usr/bin/env python | ||||||
|  |  | ||||||
|  | import time | ||||||
|  | import random | ||||||
|  |  | ||||||
|  | v = random.random() * 6 | ||||||
|  | time.sleep(v) | ||||||
|  | print(f"{v:.2f}") | ||||||
| @@ -12,7 +12,7 @@ | |||||||
| Description=copyparty file server | Description=copyparty file server | ||||||
|  |  | ||||||
| [Service] | [Service] | ||||||
| ExecStart=/usr/bin/python /usr/local/bin/copyparty-sfx.py -q -v /mnt::a | ExecStart=/usr/bin/python3 /usr/local/bin/copyparty-sfx.py -q -v /mnt::a | ||||||
| ExecStartPre=/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf' | ExecStartPre=/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf' | ||||||
|  |  | ||||||
| [Install] | [Install] | ||||||
|   | |||||||
| @@ -16,12 +16,17 @@ if platform.system() == "Windows": | |||||||
| VT100 = not WINDOWS or WINDOWS >= [10, 0, 14393] | VT100 = not WINDOWS or WINDOWS >= [10, 0, 14393] | ||||||
| # introduced in anniversary update | # introduced in anniversary update | ||||||
|  |  | ||||||
|  | ANYWIN = WINDOWS or sys.platform in ["msys"] | ||||||
|  |  | ||||||
| MACOS = platform.system() == "Darwin" | MACOS = platform.system() == "Darwin" | ||||||
|  |  | ||||||
|  |  | ||||||
| class EnvParams(object): | class EnvParams(object): | ||||||
|     def __init__(self): |     def __init__(self): | ||||||
|         self.mod = os.path.dirname(os.path.realpath(__file__)) |         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": |         if sys.platform == "win32": | ||||||
|             self.cfg = os.path.normpath(os.environ["APPDATA"] + "/copyparty") |             self.cfg = os.path.normpath(os.environ["APPDATA"] + "/copyparty") | ||||||
|         elif sys.platform == "darwin": |         elif sys.platform == "darwin": | ||||||
|   | |||||||
| @@ -16,12 +16,14 @@ import shutil | |||||||
| import filecmp | import filecmp | ||||||
| import locale | import locale | ||||||
| import argparse | import argparse | ||||||
|  | import threading | ||||||
|  | import traceback | ||||||
| from textwrap import dedent | from textwrap import dedent | ||||||
|  |  | ||||||
| from .__init__ import E, WINDOWS, VT100 | from .__init__ import E, WINDOWS, VT100, PY2 | ||||||
| from .__version__ import S_VERSION, S_BUILD_DT, CODENAME | from .__version__ import S_VERSION, S_BUILD_DT, CODENAME | ||||||
| from .svchub import SvcHub | from .svchub import SvcHub | ||||||
| from .util import py_desc, align_tab | from .util import py_desc, align_tab, IMPLICATIONS | ||||||
|  |  | ||||||
| HAVE_SSL = True | HAVE_SSL = True | ||||||
| try: | try: | ||||||
| @@ -53,6 +55,16 @@ class RiceFormatter(argparse.HelpFormatter): | |||||||
|         return "".join(indent + line + "\n" for line in text.splitlines()) |         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)) | ||||||
|  |  | ||||||
|  |  | ||||||
| def ensure_locale(): | def ensure_locale(): | ||||||
|     for x in [ |     for x in [ | ||||||
|         "en_US.UTF-8", |         "en_US.UTF-8", | ||||||
| @@ -160,22 +172,19 @@ def configure_ssl_ciphers(al): | |||||||
|         sys.exit(0) |         sys.exit(0) | ||||||
|  |  | ||||||
|  |  | ||||||
| def main(): | def sighandler(sig=None, frame=None): | ||||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 |     msg = [""] * 5 | ||||||
|     if WINDOWS: |     for th in threading.enumerate(): | ||||||
|         os.system("rem")  # enables colors |         msg.append(str(th)) | ||||||
|  |         msg.extend(traceback.format_stack(sys._current_frames()[th.ident])) | ||||||
|  |  | ||||||
|     desc = py_desc().replace("[", "\033[1;30m[") |     msg.append("\n") | ||||||
|  |     print("\n".join(msg)) | ||||||
|  |  | ||||||
|     f = '\033[36mcopyparty v{} "\033[35m{}\033[36m" ({})\n{}\033[0m\n' |  | ||||||
|     print(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc)) |  | ||||||
|  |  | ||||||
|     ensure_locale() |  | ||||||
|     if HAVE_SSL: |  | ||||||
|         ensure_cert() |  | ||||||
|  |  | ||||||
|  | def run_argparse(argv, formatter): | ||||||
|     ap = argparse.ArgumentParser( |     ap = argparse.ArgumentParser( | ||||||
|         formatter_class=RiceFormatter, |         formatter_class=formatter, | ||||||
|         prog="copyparty", |         prog="copyparty", | ||||||
|         description="http file sharing hub v{} ({})".format(S_VERSION, S_BUILD_DT), |         description="http file sharing hub v{} ({})".format(S_VERSION, S_BUILD_DT), | ||||||
|         epilog=dedent( |         epilog=dedent( | ||||||
| @@ -186,7 +195,10 @@ def main(): | |||||||
|                and "cflag" is config flags to set on this volume |                and "cflag" is config flags to set on this volume | ||||||
|              |              | ||||||
|             list of cflags: |             list of cflags: | ||||||
|               cnodupe rejects existing files (instead of symlinking them) |               "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 |             example:\033[35m | ||||||
|               -a ed:hunter2 -v .::r:aed -v ../inc:dump:w:aed:cnodupe  \033[36m |               -a ed:hunter2 -v .::r:aed -v ../inc:dump:w:aed:cnodupe  \033[36m | ||||||
| @@ -227,25 +239,89 @@ def main(): | |||||||
|     ap.add_argument("-q", action="store_true", help="quiet") |     ap.add_argument("-q", action="store_true", help="quiet") | ||||||
|     ap.add_argument("-ed", action="store_true", help="enable ?dots") |     ap.add_argument("-ed", action="store_true", help="enable ?dots") | ||||||
|     ap.add_argument("-emp", action="store_true", help="enable markdown plugins") |     ap.add_argument("-emp", action="store_true", help="enable markdown plugins") | ||||||
|     ap.add_argument("-e2d", action="store_true", help="enable up2k database") |  | ||||||
|     ap.add_argument("-e2s", action="store_true", help="enable up2k db-scanner") |  | ||||||
|     ap.add_argument("-mcr", metavar="SEC", type=int, default=60, help="md-editor mod-chk rate") |     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("-nw", action="store_true", help="disable writes (benchmark)") | ||||||
|     ap.add_argument("-nih", action="store_true", help="no info hostname") |     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("-nid", action="store_true", help="no info disk-usage") | ||||||
|     ap.add_argument("--no-sendfile", action="store_true", help="disable sendfile") |     ap.add_argument("--dotpart", action="store_true", help="dotfile incomplete uploads") | ||||||
|     ap.add_argument("--urlform", type=str, default="print,get", help="how to handle url-forms") |     ap.add_argument("--no-zip", action="store_true", help="disable download as zip/tar") | ||||||
|  |     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") | ||||||
|  |     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 = ap.add_argument_group('SSL/TLS options') | ||||||
|     ap2.add_argument("--http-only", action="store_true", help="disable ssl/tls") |     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("--https-only", action="store_true", help="disable plaintext") | ||||||
|     ap2.add_argument("--ssl-ver", type=str, help="ssl/tls versions to allow") |     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("--ciphers", metavar="LIST", help="set allowed ciphers") | ||||||
|     ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info") |     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.add_argument("--ssl-log", metavar="PATH", help="log master secrets") | ||||||
|     al = ap.parse_args() |  | ||||||
|  |     ap2 = ap.add_argument_group('debug options') | ||||||
|  |     ap2.add_argument("--log-conn", action="store_true", help="print tcp-server msgs") | ||||||
|  |     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("--ihead", metavar="HEADER", action='append', help="dump incoming header") | ||||||
|  |     ap2.add_argument("--lf-url", metavar="RE", type=str, default=r"^/\.cpr/", help="dont log URLs matching") | ||||||
|  |      | ||||||
|  |     return ap.parse_args(args=argv[1:]) | ||||||
|     # fmt: on |     # fmt: on | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def main(argv=None): | ||||||
|  |     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||||
|  |     if WINDOWS: | ||||||
|  |         os.system("rem")  # enables colors | ||||||
|  |  | ||||||
|  |     if argv is None: | ||||||
|  |         argv = sys.argv | ||||||
|  |  | ||||||
|  |     desc = py_desc().replace("[", "\033[1;30m[") | ||||||
|  |  | ||||||
|  |     f = '\033[36mcopyparty v{} "\033[35m{}\033[36m" ({})\n{}\033[0m\n' | ||||||
|  |     print(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc)) | ||||||
|  |  | ||||||
|  |     ensure_locale() | ||||||
|  |     if HAVE_SSL: | ||||||
|  |         ensure_cert() | ||||||
|  |  | ||||||
|  |     deprecated = [["-e2s", "-e2ds"]] | ||||||
|  |     for dk, nk in deprecated: | ||||||
|  |         try: | ||||||
|  |             idx = argv.index(dk) | ||||||
|  |         except: | ||||||
|  |             continue | ||||||
|  |  | ||||||
|  |         msg = "\033[1;31mWARNING:\033[0;1m\n  {} \033[0;33mwas replaced with\033[0;1m {} \033[0;33mand will be removed\n\033[0m" | ||||||
|  |         print(msg.format(dk, nk)) | ||||||
|  |         argv[idx] = nk | ||||||
|  |         time.sleep(2) | ||||||
|  |  | ||||||
|  |     try: | ||||||
|  |         al = run_argparse(argv, RiceFormatter) | ||||||
|  |     except AssertionError: | ||||||
|  |         al = run_argparse(argv, Dodge11874) | ||||||
|  |  | ||||||
|  |     # propagate implications | ||||||
|  |     for k1, k2 in IMPLICATIONS: | ||||||
|  |         if getattr(al, k1): | ||||||
|  |             setattr(al, k2, True) | ||||||
|  |  | ||||||
|     al.i = al.i.split(",") |     al.i = al.i.split(",") | ||||||
|     try: |     try: | ||||||
|         if "-" in al.p: |         if "-" in al.p: | ||||||
| @@ -263,7 +339,15 @@ def main(): | |||||||
|         if al.ciphers: |         if al.ciphers: | ||||||
|             configure_ssl_ciphers(al) |             configure_ssl_ciphers(al) | ||||||
|     else: |     else: | ||||||
|         print("\033[33m  ssl module does not exist; cannot enable https\033[0m\n") |         warn("ssl module does not exist; cannot enable https") | ||||||
|  |  | ||||||
|  |     if PY2 and WINDOWS and al.e2d: | ||||||
|  |         warn( | ||||||
|  |             "windows py2 cannot do unicode filenames with -e2d\n" | ||||||
|  |             + "  (if you crash with codec errors then that is why)" | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     # signal.signal(signal.SIGINT, sighandler) | ||||||
|  |  | ||||||
|     SvcHub(al).run() |     SvcHub(al).run() | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,8 +1,8 @@ | |||||||
| # coding: utf-8 | # coding: utf-8 | ||||||
|  |  | ||||||
| VERSION = (0, 7, 6) | VERSION = (0, 10, 20) | ||||||
| CODENAME = "keeping track" | CODENAME = "zip it" | ||||||
| BUILD_DT = (2021, 2, 12) | BUILD_DT = (2021, 5, 16) | ||||||
|  |  | ||||||
| S_VERSION = ".".join(map(str, VERSION)) | S_VERSION = ".".join(map(str, VERSION)) | ||||||
| S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) | S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) | ||||||
|   | |||||||
| @@ -1,12 +1,14 @@ | |||||||
| # coding: utf-8 | # coding: utf-8 | ||||||
| from __future__ import print_function, unicode_literals | from __future__ import print_function, unicode_literals | ||||||
|  |  | ||||||
| import os |  | ||||||
| import re | import re | ||||||
|  | import os | ||||||
|  | import sys | ||||||
|  | import stat | ||||||
| import threading | import threading | ||||||
|  |  | ||||||
| from .__init__ import PY2, WINDOWS | from .__init__ import PY2, WINDOWS | ||||||
| from .util import undot, Pebkac, fsdec, fsenc | from .util import IMPLICATIONS, undot, Pebkac, fsdec, fsenc, statdir, nuprint | ||||||
|  |  | ||||||
|  |  | ||||||
| class VFS(object): | class VFS(object): | ||||||
| @@ -19,6 +21,19 @@ class VFS(object): | |||||||
|         self.uwrite = uwrite  # users who can write this |         self.uwrite = uwrite  # users who can write this | ||||||
|         self.flags = flags  # config switches |         self.flags = flags  # config switches | ||||||
|         self.nodes = {}  # child nodes |         self.nodes = {}  # child nodes | ||||||
|  |         self.all_vols = {vpath: self}  # flattened recursive | ||||||
|  |  | ||||||
|  |     def __repr__(self): | ||||||
|  |         return "VFS({})".format( | ||||||
|  |             ", ".join( | ||||||
|  |                 "{}={!r}".format(k, self.__dict__[k]) | ||||||
|  |                 for k in "realpath vpath uread uwrite flags".split() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def _trk(self, vol): | ||||||
|  |         self.all_vols[vol.vpath] = vol | ||||||
|  |         return vol | ||||||
|  |  | ||||||
|     def add(self, src, dst): |     def add(self, src, dst): | ||||||
|         """get existing, or add new path to the vfs""" |         """get existing, or add new path to the vfs""" | ||||||
| @@ -30,7 +45,7 @@ class VFS(object): | |||||||
|             name, dst = dst.split("/", 1) |             name, dst = dst.split("/", 1) | ||||||
|             if name in self.nodes: |             if name in self.nodes: | ||||||
|                 # exists; do not manipulate permissions |                 # exists; do not manipulate permissions | ||||||
|                 return self.nodes[name].add(src, dst) |                 return self._trk(self.nodes[name].add(src, dst)) | ||||||
|  |  | ||||||
|             vn = VFS( |             vn = VFS( | ||||||
|                 "{}/{}".format(self.realpath, name), |                 "{}/{}".format(self.realpath, name), | ||||||
| @@ -39,8 +54,9 @@ class VFS(object): | |||||||
|                 self.uwrite, |                 self.uwrite, | ||||||
|                 self.flags, |                 self.flags, | ||||||
|             ) |             ) | ||||||
|  |             self._trk(vn) | ||||||
|             self.nodes[name] = vn |             self.nodes[name] = vn | ||||||
|             return vn.add(src, dst) |             return self._trk(vn.add(src, dst)) | ||||||
|  |  | ||||||
|         if dst in self.nodes: |         if dst in self.nodes: | ||||||
|             # leaf exists; return as-is |             # leaf exists; return as-is | ||||||
| @@ -50,7 +66,7 @@ class VFS(object): | |||||||
|         vp = "{}/{}".format(self.vpath, dst).lstrip("/") |         vp = "{}/{}".format(self.vpath, dst).lstrip("/") | ||||||
|         vn = VFS(src, vp) |         vn = VFS(src, vp) | ||||||
|         self.nodes[dst] = vn |         self.nodes[dst] = vn | ||||||
|         return vn |         return self._trk(vn) | ||||||
|  |  | ||||||
|     def _find(self, vpath): |     def _find(self, vpath): | ||||||
|         """return [vfs,remainder]""" |         """return [vfs,remainder]""" | ||||||
| @@ -95,25 +111,121 @@ class VFS(object): | |||||||
|         if rem: |         if rem: | ||||||
|             rp += "/" + 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): |             # 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, lstat=False): | ||||||
|         """return user-readable [fsdir,real,virt] items at vpath""" |         """return user-readable [fsdir,real,virt] items at vpath""" | ||||||
|         virt_vis = {}  # nodes readable by user |         virt_vis = {}  # nodes readable by user | ||||||
|         abspath = self.canonical(rem) |         abspath = self.canonical(rem) | ||||||
|         items = os.listdir(fsenc(abspath)) |         real = list(statdir(nuprint, scandir, lstat, abspath)) | ||||||
|         real = [fsdec(x) for x in items] |  | ||||||
|         real.sort() |         real.sort() | ||||||
|         if not rem: |         if not rem: | ||||||
|             for name, vn2 in sorted(self.nodes.items()): |             for name, vn2 in sorted(self.nodes.items()): | ||||||
|                 if uname in vn2.uread or "*" in vn2.uread: |                 if ( | ||||||
|  |                     uname in vn2.uread | ||||||
|  |                     or "*" in vn2.uread | ||||||
|  |                     or uname in vn2.uwrite | ||||||
|  |                     or "*" in vn2.uwrite | ||||||
|  |                 ): | ||||||
|                     virt_vis[name] = vn2 |                     virt_vis[name] = vn2 | ||||||
|  |  | ||||||
|             # no vfs nodes in the list of real inodes |             # no vfs nodes in the list of real inodes | ||||||
|             real = [x for x in real if x not in self.nodes] |             real = [x for x in real if x[0] not in self.nodes] | ||||||
|  |  | ||||||
|         return [abspath, real, virt_vis] |         return [abspath, real, virt_vis] | ||||||
|  |  | ||||||
|  |     def walk(self, rel, rem, uname, dots, scandir, lstat=False): | ||||||
|  |         """ | ||||||
|  |         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) | ||||||
|  |         rfiles = [x for x in vfs_ls if not stat.S_ISDIR(x[1].st_mode)] | ||||||
|  |         rdirs = [x for x in vfs_ls if stat.S_ISDIR(x[1].st_mode)] | ||||||
|  |  | ||||||
|  |         rfiles.sort() | ||||||
|  |         rdirs.sort() | ||||||
|  |  | ||||||
|  |         yield rel, fsroot, rfiles, rdirs, vfs_virt | ||||||
|  |  | ||||||
|  |         for rdir, _ in rdirs: | ||||||
|  |             if not dots and rdir.startswith("."): | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             wrel = (rel + "/" + rdir).lstrip("/") | ||||||
|  |             wrem = (rem + "/" + rdir).lstrip("/") | ||||||
|  |             for x in self.walk(wrel, wrem, uname, scandir, lstat): | ||||||
|  |                 yield x | ||||||
|  |  | ||||||
|  |         for n, vfs in sorted(vfs_virt.items()): | ||||||
|  |             if not dots and n.startswith("."): | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             wrel = (rel + "/" + n).lstrip("/") | ||||||
|  |             for x in vfs.walk(wrel, "", uname, 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): | ||||||
|  |             if flt: | ||||||
|  |                 files = [x for x in files if x[0] in flt] | ||||||
|  |  | ||||||
|  |                 rm = [x for x in rd if x[0] not in flt] | ||||||
|  |                 [rd.remove(x) for x in rm] | ||||||
|  |  | ||||||
|  |                 rm = [x for x in vd.keys() if x not in flt] | ||||||
|  |                 [vd.pop(x) for x in rm] | ||||||
|  |  | ||||||
|  |                 flt = None | ||||||
|  |  | ||||||
|  |             # print(repr([vpath, apath, [x[0] for x in files]])) | ||||||
|  |             fnames = [n[0] for n in files] | ||||||
|  |             vpaths = [vpath + "/" + n for n in fnames] if vpath else fnames | ||||||
|  |             apaths = [os.path.join(apath, n) for n in fnames] | ||||||
|  |             files = list(zip(vpaths, apaths, files)) | ||||||
|  |  | ||||||
|  |             if not dots: | ||||||
|  |                 # dotfile filtering based on vpath (intended visibility) | ||||||
|  |                 files = [x for x in files if "/." not in "/" + x[0]] | ||||||
|  |  | ||||||
|  |                 rm = [x for x in rd if x[0].startswith(".")] | ||||||
|  |                 for x in rm: | ||||||
|  |                     rd.remove(x) | ||||||
|  |  | ||||||
|  |                 rm = [k for k in vd.keys() if k.startswith(".")] | ||||||
|  |                 for x in rm: | ||||||
|  |                     del vd[x] | ||||||
|  |  | ||||||
|  |             # up2k filetring based on actual abspath | ||||||
|  |             files = [x for x in files if "{0}.hist{0}up2k.".format(os.sep) not in x[1]] | ||||||
|  |  | ||||||
|  |             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): |     def user_tree(self, uname, readable=False, writable=False): | ||||||
|         ret = [] |         ret = [] | ||||||
|         opt1 = readable and (uname in self.uread or "*" in self.uread) |         opt1 = readable and (uname in self.uread or "*" in self.uread) | ||||||
| @@ -134,6 +246,7 @@ class AuthSrv(object): | |||||||
|         self.args = args |         self.args = args | ||||||
|         self.log_func = log_func |         self.log_func = log_func | ||||||
|         self.warn_anonwrite = warn_anonwrite |         self.warn_anonwrite = warn_anonwrite | ||||||
|  |         self.line_ctr = 0 | ||||||
|  |  | ||||||
|         if WINDOWS: |         if WINDOWS: | ||||||
|             self.re_vol = re.compile(r"^([a-zA-Z]:[\\/][^:]*|[^:]*):([^:]*):(.*)$") |             self.re_vol = re.compile(r"^([a-zA-Z]:[\\/][^:]*|[^:]*):([^:]*):(.*)$") | ||||||
| @@ -143,14 +256,8 @@ class AuthSrv(object): | |||||||
|         self.mutex = threading.Lock() |         self.mutex = threading.Lock() | ||||||
|         self.reload() |         self.reload() | ||||||
|  |  | ||||||
|     def log(self, msg): |     def log(self, msg, c=0): | ||||||
|         self.log_func("auth", msg) |         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()} |  | ||||||
|  |  | ||||||
|     def laggy_iter(self, iterable): |     def laggy_iter(self, iterable): | ||||||
|         """returns [value,isFinalValue]""" |         """returns [value,isFinalValue]""" | ||||||
| @@ -165,7 +272,9 @@ class AuthSrv(object): | |||||||
|     def _parse_config_file(self, fd, user, mread, mwrite, mflags, mount): |     def _parse_config_file(self, fd, user, mread, mwrite, mflags, mount): | ||||||
|         vol_src = None |         vol_src = None | ||||||
|         vol_dst = None |         vol_dst = None | ||||||
|  |         self.line_ctr = 0 | ||||||
|         for ln in [x.decode("utf-8").strip() for x in fd]: |         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: |             if not ln and vol_src is not None: | ||||||
|                 vol_src = None |                 vol_src = None | ||||||
|                 vol_dst = None |                 vol_dst = None | ||||||
| @@ -195,14 +304,45 @@ class AuthSrv(object): | |||||||
|                 mflags[vol_dst] = {} |                 mflags[vol_dst] = {} | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             lvl, uname = ln.split(" ") |             if len(ln) > 1: | ||||||
|             if lvl in "ra": |                 lvl, uname = ln.split(" ") | ||||||
|                 mread[vol_dst].append(uname) |             else: | ||||||
|             if lvl in "wa": |                 lvl = ln | ||||||
|                 mwrite[vol_dst].append(uname) |                 uname = "*" | ||||||
|             if lvl == "c": |  | ||||||
|                 # config option, currently switches only |             self._read_vol_str( | ||||||
|                 mflags[vol_dst][uname] = True |                 lvl, uname, mread[vol_dst], mwrite[vol_dst], mflags[vol_dst] | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |     def _read_vol_str(self, lvl, uname, mr, mw, mf): | ||||||
|  |         if lvl == "c": | ||||||
|  |             cval = True | ||||||
|  |             if "=" in uname: | ||||||
|  |                 uname, cval = uname.split("=", 1) | ||||||
|  |  | ||||||
|  |             self._read_volflag(mf, uname, cval, False) | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         if uname == "": | ||||||
|  |             uname = "*" | ||||||
|  |  | ||||||
|  |         if lvl in "ra": | ||||||
|  |             mr.append(uname) | ||||||
|  |  | ||||||
|  |         if lvl in "wa": | ||||||
|  |             mw.append(uname) | ||||||
|  |  | ||||||
|  |     def _read_volflag(self, flags, name, value, is_list): | ||||||
|  |         if name not in ["mtp"]: | ||||||
|  |             flags[name] = value | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         if not is_list: | ||||||
|  |             value = [value] | ||||||
|  |         elif not value: | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         flags[name] = flags.get(name, []) + value | ||||||
|  |  | ||||||
|     def reload(self): |     def reload(self): | ||||||
|         """ |         """ | ||||||
| @@ -225,7 +365,7 @@ class AuthSrv(object): | |||||||
|  |  | ||||||
|         if self.args.v: |         if self.args.v: | ||||||
|             # list of src:dst:permset:permset:... |             # list of src:dst:permset:permset:... | ||||||
|             # permset is [rwa]username |             # permset is [rwa]username or [c]flag | ||||||
|             for v_str in self.args.v: |             for v_str in self.args.v: | ||||||
|                 m = self.re_vol.match(v_str) |                 m = self.re_vol.match(v_str) | ||||||
|                 if not m: |                 if not m: | ||||||
| @@ -242,28 +382,25 @@ class AuthSrv(object): | |||||||
|  |  | ||||||
|                 perms = perms.split(":") |                 perms = perms.split(":") | ||||||
|                 for (lvl, uname) in [[x[0], x[1:]] for x in perms]: |                 for (lvl, uname) in [[x[0], x[1:]] for x in perms]: | ||||||
|                     if lvl == "c": |                     self._read_vol_str(lvl, uname, mread[dst], mwrite[dst], mflags[dst]) | ||||||
|                         # config option, currently switches only |  | ||||||
|                         mflags[dst][uname] = True |  | ||||||
|                     if uname == "": |  | ||||||
|                         uname = "*" |  | ||||||
|                     if lvl in "ra": |  | ||||||
|                         mread[dst].append(uname) |  | ||||||
|                     if lvl in "wa": |  | ||||||
|                         mwrite[dst].append(uname) |  | ||||||
|  |  | ||||||
|         if self.args.c: |         if self.args.c: | ||||||
|             for cfg_fn in self.args.c: |             for cfg_fn in self.args.c: | ||||||
|                 with open(cfg_fn, "rb") as f: |                 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, 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 | ||||||
|  |  | ||||||
|         self.all_writable = [] |  | ||||||
|         if not mount: |         if not mount: | ||||||
|             # -h says our defaults are CWD at root and read/write for everyone |             # -h says our defaults are CWD at root and read/write for everyone | ||||||
|             vfs = VFS(os.path.abspath("."), "", ["*"], ["*"]) |             vfs = VFS(os.path.abspath("."), "", ["*"], ["*"]) | ||||||
|         elif "" not in mount: |         elif "" not in mount: | ||||||
|             # there's volumes but no root; make root inaccessible |             # there's volumes but no root; make root inaccessible | ||||||
|             vfs = VFS(os.path.abspath("."), "") |             vfs = VFS(os.path.abspath("."), "") | ||||||
|  |             vfs.flags["d2d"] = True | ||||||
|  |  | ||||||
|         maxdepth = 0 |         maxdepth = 0 | ||||||
|         for dst in sorted(mount.keys(), key=lambda x: (x.count("/"), len(x))): |         for dst in sorted(mount.keys(), key=lambda x: (x.count("/"), len(x))): | ||||||
| @@ -280,11 +417,6 @@ class AuthSrv(object): | |||||||
|             v.uread = mread[dst] |             v.uread = mread[dst] | ||||||
|             v.uwrite = mwrite[dst] |             v.uwrite = mwrite[dst] | ||||||
|             v.flags = mflags[dst] |             v.flags = mflags[dst] | ||||||
|             if v.uwrite: |  | ||||||
|                 self.all_writable.append(v) |  | ||||||
|  |  | ||||||
|         if vfs.uwrite and vfs not in self.all_writable: |  | ||||||
|             self.all_writable.append(vfs) |  | ||||||
|  |  | ||||||
|         missing_users = {} |         missing_users = {} | ||||||
|         for d in [mread, mwrite]: |         for d in [mread, mwrite]: | ||||||
| @@ -295,28 +427,107 @@ class AuthSrv(object): | |||||||
|  |  | ||||||
|         if missing_users: |         if missing_users: | ||||||
|             self.log( |             self.log( | ||||||
|                 "\033[31myou must -a the following users: " |                 "you must -a the following users: " | ||||||
|                 + ", ".join(k for k in sorted(missing_users)) |                 + ", ".join(k for k in sorted(missing_users)), | ||||||
|                 + "\033[0m" |                 c=1, | ||||||
|             ) |             ) | ||||||
|             raise Exception("invalid config") |             raise Exception("invalid config") | ||||||
|  |  | ||||||
|  |         all_mte = {} | ||||||
|  |         errors = False | ||||||
|  |         for vol in vfs.all_vols.values(): | ||||||
|  |             if (self.args.e2ds and vol.uwrite) or self.args.e2dsa: | ||||||
|  |                 vol.flags["e2ds"] = True | ||||||
|  |  | ||||||
|  |             if self.args.e2d or "e2ds" in vol.flags: | ||||||
|  |                 vol.flags["e2d"] = True | ||||||
|  |  | ||||||
|  |             for k in ["e2t", "e2ts", "e2tsr"]: | ||||||
|  |                 if getattr(self.args, k): | ||||||
|  |                     vol.flags[k] = True | ||||||
|  |  | ||||||
|  |             for k1, k2 in IMPLICATIONS: | ||||||
|  |                 if k1 in vol.flags: | ||||||
|  |                     vol.flags[k2] = True | ||||||
|  |  | ||||||
|  |             # default tag-list if unset | ||||||
|  |             if "mte" not in vol.flags: | ||||||
|  |                 vol.flags["mte"] = self.args.mte | ||||||
|  |  | ||||||
|  |             # append parsers from argv to volume-flags | ||||||
|  |             self._read_volflag(vol.flags, "mtp", self.args.mtp, True) | ||||||
|  |  | ||||||
|  |             # d2d drops all database features for a volume | ||||||
|  |             for grp, rm in [["d2d", "e2d"], ["d2t", "e2t"]]: | ||||||
|  |                 if not vol.flags.get(grp, False): | ||||||
|  |                     continue | ||||||
|  |  | ||||||
|  |                 vol.flags["d2t"] = True | ||||||
|  |                 vol.flags = {k: v for k, v in vol.flags.items() if not k.startswith(rm)} | ||||||
|  |  | ||||||
|  |             # mt* needs e2t so drop those too | ||||||
|  |             for grp, rm in [["e2t", "mt"]]: | ||||||
|  |                 if vol.flags.get(grp, False): | ||||||
|  |                     continue | ||||||
|  |  | ||||||
|  |                 vol.flags = {k: v for k, v in vol.flags.items() if not k.startswith(rm)} | ||||||
|  |  | ||||||
|  |             # verify tags mentioned by -mt[mp] are used by -mte | ||||||
|  |             local_mtp = {} | ||||||
|  |             local_only_mtp = {} | ||||||
|  |             for a in vol.flags.get("mtp", []) + vol.flags.get("mtm", []): | ||||||
|  |                 a = a.split("=")[0] | ||||||
|  |                 local_mtp[a] = True | ||||||
|  |                 local = True | ||||||
|  |                 for b in self.args.mtp or []: | ||||||
|  |                     b = b.split("=")[0] | ||||||
|  |                     if a == b: | ||||||
|  |                         local = False | ||||||
|  |  | ||||||
|  |                 if local: | ||||||
|  |                     local_only_mtp[a] = True | ||||||
|  |  | ||||||
|  |             local_mte = {} | ||||||
|  |             for a in vol.flags.get("mte", "").split(","): | ||||||
|  |                 local = True | ||||||
|  |                 all_mte[a] = True | ||||||
|  |                 local_mte[a] = True | ||||||
|  |                 for b in self.args.mte.split(","): | ||||||
|  |                     if not a or not b: | ||||||
|  |                         continue | ||||||
|  |  | ||||||
|  |                     if a == b: | ||||||
|  |                         local = False | ||||||
|  |  | ||||||
|  |             for mtp in local_only_mtp.keys(): | ||||||
|  |                 if mtp not in local_mte: | ||||||
|  |                     m = 'volume "/{}" defines metadata tag "{}", but doesnt use it in "-mte" (or with "cmte" in its volume-flags)' | ||||||
|  |                     self.log(m.format(vol.vpath, mtp), 1) | ||||||
|  |                     errors = True | ||||||
|  |  | ||||||
|  |         for mtp in self.args.mtp or []: | ||||||
|  |             mtp = mtp.split("=")[0] | ||||||
|  |             if mtp not in all_mte: | ||||||
|  |                 m = 'metadata tag "{}" is defined by "-mtm" or "-mtp", but is not used by "-mte" (or by any "cmte" volume-flag)' | ||||||
|  |                 self.log(m.format(mtp), 1) | ||||||
|  |                 errors = True | ||||||
|  |  | ||||||
|  |         if errors: | ||||||
|  |             sys.exit(1) | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             v, _ = vfs.get("/", "*", False, True) |             v, _ = vfs.get("/", "*", False, True) | ||||||
|             if self.warn_anonwrite and os.getcwd() == v.realpath: |             if self.warn_anonwrite and os.getcwd() == v.realpath: | ||||||
|                 self.warn_anonwrite = False |                 self.warn_anonwrite = False | ||||||
|                 self.log( |                 msg = "anyone can read/write the current directory: {}" | ||||||
|                     "\033[31manyone can read/write the current directory: {}\033[0m".format( |                 self.log(msg.format(v.realpath), c=1) | ||||||
|                         v.realpath |  | ||||||
|                     ) |  | ||||||
|                 ) |  | ||||||
|         except Pebkac: |         except Pebkac: | ||||||
|             self.warn_anonwrite = True |             self.warn_anonwrite = True | ||||||
|  |  | ||||||
|         with self.mutex: |         with self.mutex: | ||||||
|             self.vfs = vfs |             self.vfs = vfs | ||||||
|             self.user = user |             self.user = user | ||||||
|             self.iuser = self.invert(user) |             self.iuser = {v: k for k, v in user.items()} | ||||||
|  |  | ||||||
|         # import pprint |         # import pprint | ||||||
|         # pprint.pprint({"usr": user, "rd": mread, "wr": mwrite, "mnt": mount}) |         # pprint.pprint({"usr": user, "rd": mread, "wr": mwrite, "mnt": mount}) | ||||||
|   | |||||||
| @@ -51,7 +51,7 @@ class BrokerMp(object): | |||||||
|             self.procs.append(proc) |             self.procs.append(proc) | ||||||
|             proc.start() |             proc.start() | ||||||
|  |  | ||||||
|         if True: |         if not self.args.q: | ||||||
|             thr = threading.Thread(target=self.debug_load_balancer) |             thr = threading.Thread(target=self.debug_load_balancer) | ||||||
|             thr.daemon = True |             thr.daemon = True | ||||||
|             thr.start() |             thr.start() | ||||||
|   | |||||||
| @@ -49,11 +49,11 @@ class MpWorker(object): | |||||||
|         # print('k') |         # print('k') | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     def log(self, src, msg): |     def log(self, src, msg, c=0): | ||||||
|         self.q_yield.put([0, "log", [src, msg]]) |         self.q_yield.put([0, "log", [src, msg, c]]) | ||||||
|  |  | ||||||
|     def logw(self, msg): |     def logw(self, msg, c=0): | ||||||
|         self.log("mp{}".format(self.n), msg) |         self.log("mp{}".format(self.n), msg, c) | ||||||
|  |  | ||||||
|     def httpdrop(self, addr): |     def httpdrop(self, addr): | ||||||
|         self.q_yield.put([0, "httpdrop", [addr]]) |         self.q_yield.put([0, "httpdrop", [addr]]) | ||||||
| @@ -73,7 +73,9 @@ class MpWorker(object): | |||||||
|                 if PY2: |                 if PY2: | ||||||
|                     sck = pickle.loads(sck)  # nosec |                     sck = pickle.loads(sck)  # nosec | ||||||
|  |  | ||||||
|                 self.log("%s %s" % addr, "\033[1;30m|%sC-qpop\033[0m" % ("-" * 4,)) |                 if self.args.log_conn: | ||||||
|  |                     self.log("%s %s" % addr, "|%sC-qpop" % ("-" * 4,), c="1;30") | ||||||
|  |                  | ||||||
|                 self.httpsrv.accept(sck, addr) |                 self.httpsrv.accept(sck, addr) | ||||||
|  |  | ||||||
|                 with self.mutex: |                 with self.mutex: | ||||||
|   | |||||||
| @@ -28,7 +28,9 @@ class BrokerThr(object): | |||||||
|     def put(self, want_retval, dest, *args): |     def put(self, want_retval, dest, *args): | ||||||
|         if dest == "httpconn": |         if dest == "httpconn": | ||||||
|             sck, addr = args |             sck, addr = args | ||||||
|             self.log("%s %s" % addr, "\033[1;30m|%sC-qpop\033[0m" % ("-" * 4,)) |             if self.args.log_conn: | ||||||
|  |                 self.log("%s %s" % addr, "|%sC-qpop" % ("-" * 4,), c="1;30") | ||||||
|  |  | ||||||
|             self.httpsrv.accept(sck, addr) |             self.httpsrv.accept(sck, addr) | ||||||
|  |  | ||||||
|         else: |         else: | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,6 +1,7 @@ | |||||||
| # coding: utf-8 | # coding: utf-8 | ||||||
| from __future__ import print_function, unicode_literals | from __future__ import print_function, unicode_literals | ||||||
|  |  | ||||||
|  | import re | ||||||
| import os | import os | ||||||
| import sys | import sys | ||||||
| import time | import time | ||||||
| @@ -12,24 +13,10 @@ try: | |||||||
| except: | except: | ||||||
|     HAVE_SSL = False |     HAVE_SSL = False | ||||||
|  |  | ||||||
| try: |  | ||||||
|     import jinja2 |  | ||||||
| except ImportError: |  | ||||||
|     print( |  | ||||||
|         """\033[1;31m |  | ||||||
|   you do not have jinja2 installed,\033[33m |  | ||||||
|   choose one of these:\033[0m |  | ||||||
|    * apt install python-jinja2 |  | ||||||
|    * python3 -m pip install --user jinja2 |  | ||||||
|    * (try another python version, if you have one) |  | ||||||
|    * (try copyparty.sfx instead) |  | ||||||
| """ |  | ||||||
|     ) |  | ||||||
|     sys.exit(1) |  | ||||||
|  |  | ||||||
| from .__init__ import E | from .__init__ import E | ||||||
| from .util import Unrecv | from .util import Unrecv | ||||||
| from .httpcli import HttpCli | from .httpcli import HttpCli | ||||||
|  | from .u2idx import U2idx | ||||||
|  |  | ||||||
|  |  | ||||||
| class HttpConn(object): | class HttpConn(object): | ||||||
| @@ -50,17 +37,11 @@ class HttpConn(object): | |||||||
|         self.t0 = time.time() |         self.t0 = time.time() | ||||||
|         self.nbyte = 0 |         self.nbyte = 0 | ||||||
|         self.workload = 0 |         self.workload = 0 | ||||||
|  |         self.u2idx = None | ||||||
|         self.log_func = hsrv.log |         self.log_func = hsrv.log | ||||||
|  |         self.lf_url = re.compile(self.args.lf_url) if self.args.lf_url else None | ||||||
|         self.set_rproxy() |         self.set_rproxy() | ||||||
|  |  | ||||||
|         env = jinja2.Environment() |  | ||||||
|         env.loader = jinja2.FileSystemLoader(os.path.join(E.mod, "web")) |  | ||||||
|         self.tpl_mounts = env.get_template("splash.html") |  | ||||||
|         self.tpl_browser = env.get_template("browser.html") |  | ||||||
|         self.tpl_msg = env.get_template("msg.html") |  | ||||||
|         self.tpl_md = env.get_template("md.html") |  | ||||||
|         self.tpl_mde = env.get_template("mde.html") |  | ||||||
|  |  | ||||||
|     def set_rproxy(self, ip=None): |     def set_rproxy(self, ip=None): | ||||||
|         if ip is None: |         if ip is None: | ||||||
|             color = 36 |             color = 36 | ||||||
| @@ -77,8 +58,14 @@ class HttpConn(object): | |||||||
|     def respath(self, res_name): |     def respath(self, res_name): | ||||||
|         return os.path.join(E.mod, "web", res_name) |         return os.path.join(E.mod, "web", res_name) | ||||||
|  |  | ||||||
|     def log(self, msg): |     def log(self, msg, c=0): | ||||||
|         self.log_func(self.log_src, msg) |         self.log_func(self.log_src, msg, c) | ||||||
|  |  | ||||||
|  |     def get_u2idx(self): | ||||||
|  |         if not self.u2idx: | ||||||
|  |             self.u2idx = U2idx(self.args, self.log_func) | ||||||
|  |  | ||||||
|  |         return self.u2idx | ||||||
|  |  | ||||||
|     def _detect_https(self): |     def _detect_https(self): | ||||||
|         method = None |         method = None | ||||||
| @@ -102,7 +89,9 @@ class HttpConn(object): | |||||||
|                 err = "need at least 4 bytes in the first packet; got {}".format( |                 err = "need at least 4 bytes in the first packet; got {}".format( | ||||||
|                     len(method) |                     len(method) | ||||||
|                 ) |                 ) | ||||||
|                 self.log(err) |                 if method: | ||||||
|  |                     self.log(err) | ||||||
|  |  | ||||||
|                 self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8")) |                 self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8")) | ||||||
|                 return |                 return | ||||||
|  |  | ||||||
| @@ -119,7 +108,7 @@ class HttpConn(object): | |||||||
|  |  | ||||||
|         if is_https: |         if is_https: | ||||||
|             if self.sr: |             if self.sr: | ||||||
|                 self.log("\033[1;31mTODO: cannot do https in jython\033[0m") |                 self.log("TODO: cannot do https in jython", c="1;31") | ||||||
|                 return |                 return | ||||||
|  |  | ||||||
|             self.log_src = self.log_src.replace("[36m", "[35m") |             self.log_src = self.log_src.replace("[36m", "[35m") | ||||||
| @@ -141,6 +130,12 @@ class HttpConn(object): | |||||||
|                     ctx.set_ciphers(self.args.ciphers) |                     ctx.set_ciphers(self.args.ciphers) | ||||||
|  |  | ||||||
|                 self.s = ctx.wrap_socket(self.s, server_side=True) |                 self.s = ctx.wrap_socket(self.s, server_side=True) | ||||||
|  |                 msg = [ | ||||||
|  |                     "\033[1;3{:d}m{}".format(c, s) | ||||||
|  |                     for c, s in zip([0, 5, 0], self.s.cipher()) | ||||||
|  |                 ] | ||||||
|  |                 self.log(" ".join(msg) + "\033[0m") | ||||||
|  |  | ||||||
|                 if self.args.ssl_dbg and hasattr(self.s, "shared_ciphers"): |                 if self.args.ssl_dbg and hasattr(self.s, "shared_ciphers"): | ||||||
|                     overlap = [y[::-1] for y in self.s.shared_ciphers()] |                     overlap = [y[::-1] for y in self.s.shared_ciphers()] | ||||||
|                     lines = [str(x) for x in (["TLS cipher overlap:"] + overlap)] |                     lines = [str(x) for x in (["TLS cipher overlap:"] + overlap)] | ||||||
| @@ -164,7 +159,7 @@ class HttpConn(object): | |||||||
|                     pass |                     pass | ||||||
|  |  | ||||||
|                 else: |                 else: | ||||||
|                     self.log("\033[35mhandshake\033[0m " + em) |                     self.log("handshake\033[0m " + em, c=5) | ||||||
|  |  | ||||||
|                 return |                 return | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,10 +2,28 @@ | |||||||
| from __future__ import print_function, unicode_literals | from __future__ import print_function, unicode_literals | ||||||
|  |  | ||||||
| import os | import os | ||||||
|  | import sys | ||||||
| import time | import time | ||||||
| import socket | import socket | ||||||
| import threading | import threading | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     import jinja2 | ||||||
|  | except ImportError: | ||||||
|  |     print( | ||||||
|  |         """\033[1;31m | ||||||
|  |   you do not have jinja2 installed,\033[33m | ||||||
|  |   choose one of these:\033[0m | ||||||
|  |    * apt install python-jinja2 | ||||||
|  |    * {} -m pip install --user jinja2 | ||||||
|  |    * (try another python version, if you have one) | ||||||
|  |    * (try copyparty.sfx instead) | ||||||
|  | """.format( | ||||||
|  |             os.path.basename(sys.executable) | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |     sys.exit(1) | ||||||
|  |  | ||||||
| from .__init__ import E, MACOS | from .__init__ import E, MACOS | ||||||
| from .httpconn import HttpConn | from .httpconn import HttpConn | ||||||
| from .authsrv import AuthSrv | from .authsrv import AuthSrv | ||||||
| @@ -30,6 +48,13 @@ class HttpSrv(object): | |||||||
|         self.workload_thr_alive = False |         self.workload_thr_alive = False | ||||||
|         self.auth = AuthSrv(self.args, self.log) |         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", "browser2", "msg", "md", "mde"] | ||||||
|  |         } | ||||||
|  |  | ||||||
|         cert_path = os.path.join(E.cfg, "cert.pem") |         cert_path = os.path.join(E.cfg, "cert.pem") | ||||||
|         if os.path.exists(cert_path): |         if os.path.exists(cert_path): | ||||||
|             self.cert_path = cert_path |             self.cert_path = cert_path | ||||||
| @@ -38,7 +63,9 @@ class HttpSrv(object): | |||||||
|  |  | ||||||
|     def accept(self, sck, addr): |     def accept(self, sck, addr): | ||||||
|         """takes an incoming tcp connection and creates a thread to handle it""" |         """takes an incoming tcp connection and creates a thread to handle it""" | ||||||
|         self.log("%s %s" % addr, "\033[1;30m|%sC-cthr\033[0m" % ("-" * 5,)) |         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)) | ||||||
|         thr.daemon = True |         thr.daemon = True | ||||||
|         thr.start() |         thr.start() | ||||||
| @@ -66,11 +93,15 @@ class HttpSrv(object): | |||||||
|                 thr.start() |                 thr.start() | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             self.log("%s %s" % addr, "\033[1;30m|%sC-crun\033[0m" % ("-" * 6,)) |             if self.args.log_conn: | ||||||
|  |                 self.log("%s %s" % addr, "|%sC-crun" % ("-" * 6,), c="1;30") | ||||||
|  |  | ||||||
|             cli.run() |             cli.run() | ||||||
|  |  | ||||||
|         finally: |         finally: | ||||||
|             self.log("%s %s" % addr, "\033[1;30m|%sC-cdone\033[0m" % ("-" * 7,)) |             if self.args.log_conn: | ||||||
|  |                 self.log("%s %s" % addr, "|%sC-cdone" % ("-" * 7,), c="1;30") | ||||||
|  |  | ||||||
|             try: |             try: | ||||||
|                 sck.shutdown(socket.SHUT_RDWR) |                 sck.shutdown(socket.SHUT_RDWR) | ||||||
|                 sck.close() |                 sck.close() | ||||||
| @@ -78,7 +109,8 @@ class HttpSrv(object): | |||||||
|                 if not MACOS: |                 if not MACOS: | ||||||
|                     self.log( |                     self.log( | ||||||
|                         "%s %s" % addr, |                         "%s %s" % addr, | ||||||
|                         "shut_rdwr err:\n  {}\n  {}".format(repr(sck), ex), |                         "shut({}): {}".format(sck.fileno(), ex), | ||||||
|  |                         c="1;30", | ||||||
|                     ) |                     ) | ||||||
|                 if ex.errno not in [10038, 10054, 107, 57, 9]: |                 if ex.errno not in [10038, 10054, 107, 57, 9]: | ||||||
|                     # 10038 No longer considered a socket |                     # 10038 No longer considered a socket | ||||||
|   | |||||||
							
								
								
									
										347
									
								
								copyparty/mtag.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										347
									
								
								copyparty/mtag.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,347 @@ | |||||||
|  | # coding: utf-8 | ||||||
|  | from __future__ import print_function, unicode_literals | ||||||
|  |  | ||||||
|  | import re | ||||||
|  | import os | ||||||
|  | import sys | ||||||
|  | import shutil | ||||||
|  | import subprocess as sp | ||||||
|  |  | ||||||
|  | from .__init__ import PY2, WINDOWS | ||||||
|  | from .util import fsenc, fsdec, REKOBO_LKEY | ||||||
|  |  | ||||||
|  | if not PY2: | ||||||
|  |     unicode = str | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MTag(object): | ||||||
|  |     def __init__(self, log_func, args): | ||||||
|  |         self.log_func = log_func | ||||||
|  |         self.usable = True | ||||||
|  |         self.prefer_mt = False | ||||||
|  |         mappings = args.mtm | ||||||
|  |         self.backend = "ffprobe" if args.no_mutagen else "mutagen" | ||||||
|  |         or_ffprobe = " or ffprobe" | ||||||
|  |  | ||||||
|  |         if self.backend == "mutagen": | ||||||
|  |             self.get = self.get_mutagen | ||||||
|  |             try: | ||||||
|  |                 import mutagen | ||||||
|  |             except: | ||||||
|  |                 self.log("could not load mutagen, trying ffprobe instead", c=3) | ||||||
|  |                 self.backend = "ffprobe" | ||||||
|  |  | ||||||
|  |         if self.backend == "ffprobe": | ||||||
|  |             self.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 | ||||||
|  |  | ||||||
|  |             if self.usable and WINDOWS and sys.version_info < (3, 8): | ||||||
|  |                 self.usable = False | ||||||
|  |                 or_ffprobe = " or python >= 3.8" | ||||||
|  |                 msg = "found ffprobe but your python is too old; need 3.8 or newer" | ||||||
|  |                 self.log(msg, c=1) | ||||||
|  |  | ||||||
|  |         if not self.usable: | ||||||
|  |             msg = "need mutagen{} to read media tags so please run this:\n  {} -m pip install --user mutagen" | ||||||
|  |             self.log(msg.format(or_ffprobe, os.path.basename(sys.executable)), c=1) | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         # https://picard-docs.musicbrainz.org/downloads/MusicBrainz_Picard_Tag_Map.html | ||||||
|  |         tagmap = { | ||||||
|  |             "album": ["album", "talb", "\u00a9alb", "original-album", "toal"], | ||||||
|  |             "artist": [ | ||||||
|  |                 "artist", | ||||||
|  |                 "tpe1", | ||||||
|  |                 "\u00a9art", | ||||||
|  |                 "composer", | ||||||
|  |                 "performer", | ||||||
|  |                 "arranger", | ||||||
|  |                 "\u00a9wrt", | ||||||
|  |                 "tcom", | ||||||
|  |                 "tpe3", | ||||||
|  |                 "original-artist", | ||||||
|  |                 "tope", | ||||||
|  |             ], | ||||||
|  |             "title": ["title", "tit2", "\u00a9nam"], | ||||||
|  |             "circle": [ | ||||||
|  |                 "album-artist", | ||||||
|  |                 "tpe2", | ||||||
|  |                 "aart", | ||||||
|  |                 "conductor", | ||||||
|  |                 "organization", | ||||||
|  |                 "band", | ||||||
|  |             ], | ||||||
|  |             ".tn": ["tracknumber", "trck", "trkn", "track"], | ||||||
|  |             "genre": ["genre", "tcon", "\u00a9gen"], | ||||||
|  |             "date": [ | ||||||
|  |                 "original-release-date", | ||||||
|  |                 "release-date", | ||||||
|  |                 "date", | ||||||
|  |                 "tdrc", | ||||||
|  |                 "\u00a9day", | ||||||
|  |                 "original-date", | ||||||
|  |                 "original-year", | ||||||
|  |                 "tyer", | ||||||
|  |                 "tdor", | ||||||
|  |                 "tory", | ||||||
|  |                 "year", | ||||||
|  |                 "creation-time", | ||||||
|  |             ], | ||||||
|  |             ".bpm": ["bpm", "tbpm", "tmpo", "tbp"], | ||||||
|  |             "key": ["initial-key", "tkey", "key"], | ||||||
|  |             "comment": ["comment", "comm", "\u00a9cmt", "comments", "description"], | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if mappings: | ||||||
|  |             for k, v in [x.split("=") for x in mappings]: | ||||||
|  |                 tagmap[k] = v.split(",") | ||||||
|  |  | ||||||
|  |         self.tagmap = {} | ||||||
|  |         for k, vs in tagmap.items(): | ||||||
|  |             vs2 = [] | ||||||
|  |             for v in vs: | ||||||
|  |                 if "-" not in v: | ||||||
|  |                     vs2.append(v) | ||||||
|  |                     continue | ||||||
|  |  | ||||||
|  |                 vs2.append(v.replace("-", " ")) | ||||||
|  |                 vs2.append(v.replace("-", "_")) | ||||||
|  |                 vs2.append(v.replace("-", "")) | ||||||
|  |  | ||||||
|  |             self.tagmap[k] = vs2 | ||||||
|  |  | ||||||
|  |         self.rmap = { | ||||||
|  |             v: [n, k] for k, vs in self.tagmap.items() for n, v in enumerate(vs) | ||||||
|  |         } | ||||||
|  |         # self.get = self.compare | ||||||
|  |  | ||||||
|  |     def log(self, msg, c=0): | ||||||
|  |         self.log_func("mtag", msg, c) | ||||||
|  |  | ||||||
|  |     def normalize_tags(self, ret, md): | ||||||
|  |         for k, v in dict(md).items(): | ||||||
|  |             if not v: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             k = k.lower().split("::")[0].strip() | ||||||
|  |             mk = self.rmap.get(k) | ||||||
|  |             if not mk: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             pref, mk = mk | ||||||
|  |             if mk not in ret or ret[mk][0] > pref: | ||||||
|  |                 ret[mk] = [pref, v[0]] | ||||||
|  |  | ||||||
|  |         # take first value | ||||||
|  |         ret = {k: unicode(v[1]).strip() for k, v in ret.items()} | ||||||
|  |  | ||||||
|  |         # track 3/7 => track 3 | ||||||
|  |         for k, v in ret.items(): | ||||||
|  |             if k[0] == ".": | ||||||
|  |                 v = v.split("/")[0].strip().lstrip("0") | ||||||
|  |                 ret[k] = v or 0 | ||||||
|  |  | ||||||
|  |         # normalize key notation to rkeobo | ||||||
|  |         okey = ret.get("key") | ||||||
|  |         if okey: | ||||||
|  |             key = okey.replace(" ", "").replace("maj", "").replace("min", "m") | ||||||
|  |             ret["key"] = REKOBO_LKEY.get(key.lower(), okey) | ||||||
|  |  | ||||||
|  |         return ret | ||||||
|  |  | ||||||
|  |     def compare(self, abspath): | ||||||
|  |         if abspath.endswith(".au"): | ||||||
|  |             return {} | ||||||
|  |  | ||||||
|  |         print("\n" + abspath) | ||||||
|  |         r1 = self.get_mutagen(abspath) | ||||||
|  |         r2 = self.get_ffprobe(abspath) | ||||||
|  |  | ||||||
|  |         keys = {} | ||||||
|  |         for d in [r1, r2]: | ||||||
|  |             for k in d.keys(): | ||||||
|  |                 keys[k] = True | ||||||
|  |  | ||||||
|  |         diffs = [] | ||||||
|  |         l1 = [] | ||||||
|  |         l2 = [] | ||||||
|  |         for k in sorted(keys.keys()): | ||||||
|  |             if k in [".q", ".dur"]: | ||||||
|  |                 continue  # lenient | ||||||
|  |  | ||||||
|  |             v1 = r1.get(k) | ||||||
|  |             v2 = r2.get(k) | ||||||
|  |             if v1 == v2: | ||||||
|  |                 print("  ", k, v1) | ||||||
|  |             elif v1 != "0000":  # ffprobe date=0 | ||||||
|  |                 diffs.append(k) | ||||||
|  |                 print(" 1", k, v1) | ||||||
|  |                 print(" 2", k, v2) | ||||||
|  |                 if v1: | ||||||
|  |                     l1.append(k) | ||||||
|  |                 if v2: | ||||||
|  |                     l2.append(k) | ||||||
|  |  | ||||||
|  |         if diffs: | ||||||
|  |             raise Exception() | ||||||
|  |  | ||||||
|  |         return r1 | ||||||
|  |  | ||||||
|  |     def get_mutagen(self, abspath): | ||||||
|  |         import mutagen | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             md = mutagen.File(abspath, easy=True) | ||||||
|  |             x = md.info.length | ||||||
|  |         except Exception as ex: | ||||||
|  |             return {} | ||||||
|  |  | ||||||
|  |         ret = {} | ||||||
|  |         try: | ||||||
|  |             dur = int(md.info.length) | ||||||
|  |             try: | ||||||
|  |                 q = int(md.info.bitrate / 1024) | ||||||
|  |             except: | ||||||
|  |                 q = int((os.path.getsize(abspath) / dur) / 128) | ||||||
|  |  | ||||||
|  |             ret[".dur"] = [0, dur] | ||||||
|  |             ret[".q"] = [0, q] | ||||||
|  |         except: | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         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()} | ||||||
|  |  | ||||||
|  |         return self.normalize_tags(ret, md) | ||||||
|  |  | ||||||
|  |     def get_bin(self, parsers, abspath): | ||||||
|  |         pypath = os.path.abspath(os.path.dirname(os.path.dirname(__file__))) | ||||||
|  |         pypath = [str(pypath)] + [str(x) for x in sys.path if x] | ||||||
|  |         pypath = str(os.pathsep.join(pypath)) | ||||||
|  |         env = os.environ.copy() | ||||||
|  |         env["PYTHONPATH"] = pypath | ||||||
|  |  | ||||||
|  |         ret = {} | ||||||
|  |         for tagname, (binpath, timeout) in parsers.items(): | ||||||
|  |             try: | ||||||
|  |                 cmd = [sys.executable, binpath, abspath] | ||||||
|  |                 args = {"env": env, "timeout": timeout} | ||||||
|  |  | ||||||
|  |                 if WINDOWS: | ||||||
|  |                     args["creationflags"] = 0x4000 | ||||||
|  |                 else: | ||||||
|  |                     cmd = ["nice"] + cmd | ||||||
|  |  | ||||||
|  |                 cmd = [fsenc(x) for x in cmd] | ||||||
|  |                 v = sp.check_output(cmd, **args).strip() | ||||||
|  |                 if v: | ||||||
|  |                     ret[tagname] = v.decode("utf-8") | ||||||
|  |             except: | ||||||
|  |                 pass | ||||||
|  |  | ||||||
|  |         return ret | ||||||
							
								
								
									
										95
									
								
								copyparty/star.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								copyparty/star.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | |||||||
|  | import os | ||||||
|  | import tarfile | ||||||
|  | import threading | ||||||
|  |  | ||||||
|  | from .sutil import errdesc | ||||||
|  | from .util import Queue, fsenc | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class QFile(object): | ||||||
|  |     """file-like object which buffers writes into a queue""" | ||||||
|  |  | ||||||
|  |     def __init__(self): | ||||||
|  |         self.q = Queue(64) | ||||||
|  |         self.bq = [] | ||||||
|  |         self.nq = 0 | ||||||
|  |  | ||||||
|  |     def write(self, buf): | ||||||
|  |         if buf is None or self.nq >= 240 * 1024: | ||||||
|  |             self.q.put(b"".join(self.bq)) | ||||||
|  |             self.bq = [] | ||||||
|  |             self.nq = 0 | ||||||
|  |  | ||||||
|  |         if buf is None: | ||||||
|  |             self.q.put(None) | ||||||
|  |         else: | ||||||
|  |             self.bq.append(buf) | ||||||
|  |             self.nq += len(buf) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class StreamTar(object): | ||||||
|  |     """construct in-memory tar file from the given path""" | ||||||
|  |  | ||||||
|  |     def __init__(self, fgen, **kwargs): | ||||||
|  |         self.ci = 0 | ||||||
|  |         self.co = 0 | ||||||
|  |         self.qfile = QFile() | ||||||
|  |         self.fgen = fgen | ||||||
|  |         self.errf = None | ||||||
|  |  | ||||||
|  |         # python 3.8 changed to PAX_FORMAT as default, | ||||||
|  |         # waste of space and don't care about the new features | ||||||
|  |         fmt = tarfile.GNU_FORMAT | ||||||
|  |         self.tar = tarfile.open(fileobj=self.qfile, mode="w|", format=fmt) | ||||||
|  |  | ||||||
|  |         w = threading.Thread(target=self._gen) | ||||||
|  |         w.daemon = True | ||||||
|  |         w.start() | ||||||
|  |  | ||||||
|  |     def gen(self): | ||||||
|  |         while True: | ||||||
|  |             buf = self.qfile.q.get() | ||||||
|  |             if not buf: | ||||||
|  |                 break | ||||||
|  |  | ||||||
|  |             self.co += len(buf) | ||||||
|  |             yield buf | ||||||
|  |  | ||||||
|  |         yield None | ||||||
|  |         if self.errf: | ||||||
|  |             os.unlink(self.errf["ap"]) | ||||||
|  |  | ||||||
|  |     def ser(self, f): | ||||||
|  |         name = f["vp"] | ||||||
|  |         src = f["ap"] | ||||||
|  |         fsi = f["st"] | ||||||
|  |  | ||||||
|  |         inf = tarfile.TarInfo(name=name) | ||||||
|  |         inf.mode = fsi.st_mode | ||||||
|  |         inf.size = fsi.st_size | ||||||
|  |         inf.mtime = fsi.st_mtime | ||||||
|  |         inf.uid = 0 | ||||||
|  |         inf.gid = 0 | ||||||
|  |  | ||||||
|  |         self.ci += inf.size | ||||||
|  |         with open(fsenc(src), "rb", 512 * 1024) as f: | ||||||
|  |             self.tar.addfile(inf, f) | ||||||
|  |  | ||||||
|  |     def _gen(self): | ||||||
|  |         errors = [] | ||||||
|  |         for f in self.fgen: | ||||||
|  |             if "err" in f: | ||||||
|  |                 errors.append([f["vp"], f["err"]]) | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             try: | ||||||
|  |                 self.ser(f) | ||||||
|  |             except Exception as ex: | ||||||
|  |                 errors.append([f["vp"], repr(ex)]) | ||||||
|  |  | ||||||
|  |         if errors: | ||||||
|  |             self.errf = errdesc(errors) | ||||||
|  |             self.ser(self.errf) | ||||||
|  |  | ||||||
|  |         self.tar.close() | ||||||
|  |         self.qfile.write(None) | ||||||
							
								
								
									
										25
									
								
								copyparty/sutil.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								copyparty/sutil.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | import os | ||||||
|  | import time | ||||||
|  | import tempfile | ||||||
|  | from datetime import datetime | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def errdesc(errors): | ||||||
|  |     report = ["copyparty failed to add the following files to the archive:", ""] | ||||||
|  |  | ||||||
|  |     for fn, err in errors: | ||||||
|  |         report.extend([" file: {}".format(fn), "error: {}".format(err), ""]) | ||||||
|  |  | ||||||
|  |     with tempfile.NamedTemporaryFile(prefix="copyparty-", delete=False) as tf: | ||||||
|  |         tf_path = tf.name | ||||||
|  |         tf.write("\r\n".join(report).encode("utf-8", "replace")) | ||||||
|  |  | ||||||
|  |     dt = datetime.utcfromtimestamp(time.time()) | ||||||
|  |     dt = dt.strftime("%Y-%m%d-%H%M%S") | ||||||
|  |  | ||||||
|  |     os.chmod(tf_path, 0o444) | ||||||
|  |     return { | ||||||
|  |         "vp": "archive-errors-{}.txt".format(dt), | ||||||
|  |         "ap": tf_path, | ||||||
|  |         "st": os.stat(tf_path), | ||||||
|  |     } | ||||||
| @@ -9,7 +9,6 @@ from datetime import datetime, timedelta | |||||||
| import calendar | import calendar | ||||||
|  |  | ||||||
| from .__init__ import PY2, WINDOWS, MACOS, VT100 | from .__init__ import PY2, WINDOWS, MACOS, VT100 | ||||||
| from .authsrv import AuthSrv |  | ||||||
| from .tcpsrv import TcpSrv | from .tcpsrv import TcpSrv | ||||||
| from .up2k import Up2k | from .up2k import Up2k | ||||||
| from .util import mp | from .util import mp | ||||||
| @@ -39,10 +38,6 @@ class SvcHub(object): | |||||||
|         self.tcpsrv = TcpSrv(self) |         self.tcpsrv = TcpSrv(self) | ||||||
|         self.up2k = Up2k(self) |         self.up2k = Up2k(self) | ||||||
|  |  | ||||||
|         if self.args.e2d and self.args.e2s: |  | ||||||
|             auth = AuthSrv(self.args, self.log, False) |  | ||||||
|             self.up2k.build_indexes(auth.all_writable) |  | ||||||
|  |  | ||||||
|         # decide which worker impl to use |         # decide which worker impl to use | ||||||
|         if self.check_mp_enable(): |         if self.check_mp_enable(): | ||||||
|             from .broker_mp import BrokerMp as Broker |             from .broker_mp import BrokerMp as Broker | ||||||
| @@ -70,16 +65,16 @@ class SvcHub(object): | |||||||
|             self.broker.shutdown() |             self.broker.shutdown() | ||||||
|             print("nailed it") |             print("nailed it") | ||||||
|  |  | ||||||
|     def _log_disabled(self, src, msg): |     def _log_disabled(self, src, msg, c=0): | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     def _log_enabled(self, src, msg): |     def _log_enabled(self, src, msg, c=0): | ||||||
|         """handles logging from all components""" |         """handles logging from all components""" | ||||||
|         with self.log_mutex: |         with self.log_mutex: | ||||||
|             now = time.time() |             now = time.time() | ||||||
|             if now >= self.next_day: |             if now >= self.next_day: | ||||||
|                 dt = datetime.utcfromtimestamp(now) |                 dt = datetime.utcfromtimestamp(now) | ||||||
|                 print("\033[36m{}\033[0m".format(dt.strftime("%Y-%m-%d"))) |                 print("\033[36m{}\033[0m\n".format(dt.strftime("%Y-%m-%d")), end="") | ||||||
|  |  | ||||||
|                 # unix timestamp of next 00:00:00 (leap-seconds safe) |                 # unix timestamp of next 00:00:00 (leap-seconds safe) | ||||||
|                 day_now = dt.day |                 day_now = dt.day | ||||||
| @@ -89,23 +84,30 @@ class SvcHub(object): | |||||||
|                 dt = dt.replace(hour=0, minute=0, second=0) |                 dt = dt.replace(hour=0, minute=0, second=0) | ||||||
|                 self.next_day = calendar.timegm(dt.utctimetuple()) |                 self.next_day = calendar.timegm(dt.utctimetuple()) | ||||||
|  |  | ||||||
|             fmt = "\033[36m{} \033[33m{:21} \033[0m{}" |             fmt = "\033[36m{} \033[33m{:21} \033[0m{}\n" | ||||||
|             if not VT100: |             if not VT100: | ||||||
|                 fmt = "{} {:21} {}" |                 fmt = "{} {:21} {}\n" | ||||||
|                 if "\033" in msg: |                 if "\033" in msg: | ||||||
|                     msg = self.ansi_re.sub("", msg) |                     msg = self.ansi_re.sub("", msg) | ||||||
|                 if "\033" in src: |                 if "\033" in src: | ||||||
|                     src = self.ansi_re.sub("", src) |                     src = self.ansi_re.sub("", src) | ||||||
|  |             elif c: | ||||||
|  |                 if isinstance(c, int): | ||||||
|  |                     msg = "\033[3{}m{}".format(c, msg) | ||||||
|  |                 elif "\033" not in c: | ||||||
|  |                     msg = "\033[{}m{}\033[0m".format(c, msg) | ||||||
|  |                 else: | ||||||
|  |                     msg = "{}{}\033[0m".format(c, msg) | ||||||
|  |  | ||||||
|             ts = datetime.utcfromtimestamp(now).strftime("%H:%M:%S.%f")[:-3] |             ts = datetime.utcfromtimestamp(now).strftime("%H:%M:%S.%f")[:-3] | ||||||
|             msg = fmt.format(ts, src, msg) |             msg = fmt.format(ts, src, msg) | ||||||
|             try: |             try: | ||||||
|                 print(msg) |                 print(msg, end="") | ||||||
|             except UnicodeEncodeError: |             except UnicodeEncodeError: | ||||||
|                 try: |                 try: | ||||||
|                     print(msg.encode("utf-8", "replace").decode()) |                     print(msg.encode("utf-8", "replace").decode(), end="") | ||||||
|                 except: |                 except: | ||||||
|                     print(msg.encode("ascii", "replace").decode()) |                     print(msg.encode("ascii", "replace").decode(), end="") | ||||||
|  |  | ||||||
|     def check_mp_support(self): |     def check_mp_support(self): | ||||||
|         vmin = sys.version_info[1] |         vmin = sys.version_info[1] | ||||||
|   | |||||||
							
								
								
									
										271
									
								
								copyparty/szip.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										271
									
								
								copyparty/szip.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,271 @@ | |||||||
|  | import os | ||||||
|  | import time | ||||||
|  | import zlib | ||||||
|  | import struct | ||||||
|  | from datetime import datetime | ||||||
|  |  | ||||||
|  | from .sutil import errdesc | ||||||
|  | from .util import yieldfile, sanitize_fn | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def dostime2unix(buf): | ||||||
|  |     t, d = struct.unpack("<HH", buf) | ||||||
|  |  | ||||||
|  |     ts = (t & 0x1F) * 2 | ||||||
|  |     tm = (t >> 5) & 0x3F | ||||||
|  |     th = t >> 11 | ||||||
|  |  | ||||||
|  |     dd = d & 0x1F | ||||||
|  |     dm = (d >> 5) & 0xF | ||||||
|  |     dy = (d >> 9) + 1980 | ||||||
|  |  | ||||||
|  |     tt = (dy, dm, dd, th, tm, ts) | ||||||
|  |     tf = "{:04d}-{:02d}-{:02d} {:02d}:{:02d}:{:02d}" | ||||||
|  |     iso = tf.format(*tt) | ||||||
|  |  | ||||||
|  |     dt = datetime.strptime(iso, "%Y-%m-%d %H:%M:%S") | ||||||
|  |     return int(dt.timestamp()) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def unixtime2dos(ts): | ||||||
|  |     tt = time.gmtime(ts) | ||||||
|  |     dy, dm, dd, th, tm, ts = list(tt)[:6] | ||||||
|  |  | ||||||
|  |     bd = ((dy - 1980) << 9) + (dm << 5) + dd | ||||||
|  |     bt = (th << 11) + (tm << 5) + ts // 2 | ||||||
|  |     return struct.pack("<HH", bt, bd) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def gen_fdesc(sz, crc32, z64): | ||||||
|  |     ret = b"\x50\x4b\x07\x08" | ||||||
|  |     fmt = "<LQQ" if z64 else "<LLL" | ||||||
|  |     ret += struct.pack(fmt, crc32, sz, sz) | ||||||
|  |     return ret | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def gen_hdr(h_pos, fn, sz, lastmod, utf8, crc32, pre_crc): | ||||||
|  |     """ | ||||||
|  |     does regular file headers | ||||||
|  |     and the central directory meme if h_pos is set | ||||||
|  |     (h_pos = absolute position of the regular header) | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # appnote 4.5 / zip 3.0 (2008) / unzip 6.0 (2009) says to add z64 | ||||||
|  |     # extinfo for values which exceed H, but that becomes an off-by-one | ||||||
|  |     # (can't tell if it was clamped or exactly maxval), make it obvious | ||||||
|  |     z64 = sz >= 0xFFFFFFFF | ||||||
|  |     z64v = [sz, sz] if z64 else [] | ||||||
|  |     if h_pos and h_pos >= 0xFFFFFFFF: | ||||||
|  |         # central, also consider ptr to original header | ||||||
|  |         z64v.append(h_pos) | ||||||
|  |  | ||||||
|  |     # confusingly this doesn't bump if h_pos | ||||||
|  |     req_ver = b"\x2d\x00" if z64 else b"\x0a\x00" | ||||||
|  |  | ||||||
|  |     if crc32: | ||||||
|  |         crc32 = struct.pack("<L", crc32) | ||||||
|  |     else: | ||||||
|  |         crc32 = b"\x00" * 4 | ||||||
|  |  | ||||||
|  |     if h_pos is None: | ||||||
|  |         # 4b magic, 2b min-ver | ||||||
|  |         ret = b"\x50\x4b\x03\x04" + req_ver | ||||||
|  |     else: | ||||||
|  |         # 4b magic, 2b spec-ver, 2b min-ver | ||||||
|  |         ret = b"\x50\x4b\x01\x02\x1e\x03" + req_ver | ||||||
|  |  | ||||||
|  |     ret += b"\x00" if pre_crc else b"\x08"  # streaming | ||||||
|  |     ret += b"\x08" if utf8 else b"\x00"  # appnote 6.3.2 (2007) | ||||||
|  |  | ||||||
|  |     # 2b compression, 4b time, 4b crc | ||||||
|  |     ret += b"\x00\x00" + unixtime2dos(lastmod) + crc32 | ||||||
|  |  | ||||||
|  |     # spec says to put zeros when !crc if bit3 (streaming) | ||||||
|  |     # however infozip does actual sz and it even works on winxp | ||||||
|  |     # (same reasning for z64 extradata later) | ||||||
|  |     vsz = 0xFFFFFFFF if z64 else sz | ||||||
|  |     ret += struct.pack("<LL", vsz, vsz) | ||||||
|  |  | ||||||
|  |     # windows support (the "?" replace below too) | ||||||
|  |     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 | ||||||
|  |     ret += struct.pack("<HH", len(bfn), z64_len) | ||||||
|  |  | ||||||
|  |     if h_pos is not None: | ||||||
|  |         # 2b comment, 2b diskno | ||||||
|  |         ret += b"\x00" * 4 | ||||||
|  |  | ||||||
|  |         # 2b internal.attr, 4b external.attr | ||||||
|  |         # infozip-macos: 0100 0000 a481 file:644 | ||||||
|  |         # infozip-macos: 0100 0100 0080 file:000 | ||||||
|  |         ret += b"\x01\x00\x00\x00\xa4\x81" | ||||||
|  |  | ||||||
|  |         # 4b local-header-ofs | ||||||
|  |         ret += struct.pack("<L", min(h_pos, 0xFFFFFFFF)) | ||||||
|  |  | ||||||
|  |     ret += bfn | ||||||
|  |  | ||||||
|  |     if z64v: | ||||||
|  |         ret += struct.pack("<HH" + "Q" * len(z64v), 1, len(z64v) * 8, *z64v) | ||||||
|  |  | ||||||
|  |     return ret | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def gen_ecdr(items, cdir_pos, cdir_end): | ||||||
|  |     """ | ||||||
|  |     summary of all file headers, | ||||||
|  |     usually the zipfile footer unless something clamps | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     ret = b"\x50\x4b\x05\x06" | ||||||
|  |  | ||||||
|  |     # 2b ndisk, 2b disk0 | ||||||
|  |     ret += b"\x00" * 4 | ||||||
|  |  | ||||||
|  |     cdir_sz = cdir_end - cdir_pos | ||||||
|  |  | ||||||
|  |     nitems = min(0xFFFF, len(items)) | ||||||
|  |     csz = min(0xFFFFFFFF, cdir_sz) | ||||||
|  |     cpos = min(0xFFFFFFFF, cdir_pos) | ||||||
|  |  | ||||||
|  |     need_64 = nitems == 0xFFFF or 0xFFFFFFFF in [csz, cpos] | ||||||
|  |  | ||||||
|  |     # 2b tnfiles, 2b dnfiles, 4b dir sz, 4b dir pos | ||||||
|  |     ret += struct.pack("<HHLL", nitems, nitems, csz, cpos) | ||||||
|  |  | ||||||
|  |     # 2b comment length | ||||||
|  |     ret += b"\x00\x00" | ||||||
|  |  | ||||||
|  |     return [ret, need_64] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def gen_ecdr64(items, cdir_pos, cdir_end): | ||||||
|  |     """ | ||||||
|  |     z64 end of central directory | ||||||
|  |     added when numfiles or a headerptr clamps | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     ret = b"\x50\x4b\x06\x06" | ||||||
|  |  | ||||||
|  |     # 8b own length from hereon | ||||||
|  |     ret += b"\x2c" + b"\x00" * 7 | ||||||
|  |  | ||||||
|  |     # 2b spec-ver, 2b min-ver | ||||||
|  |     ret += b"\x1e\x03\x2d\x00" | ||||||
|  |  | ||||||
|  |     # 4b ndisk, 4b disk0 | ||||||
|  |     ret += b"\x00" * 8 | ||||||
|  |  | ||||||
|  |     # 8b tnfiles, 8b dnfiles, 8b dir sz, 8b dir pos | ||||||
|  |     cdir_sz = cdir_end - cdir_pos | ||||||
|  |     ret += struct.pack("<QQQQ", len(items), len(items), cdir_sz, cdir_pos) | ||||||
|  |  | ||||||
|  |     return ret | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def gen_ecdr64_loc(ecdr64_pos): | ||||||
|  |     """ | ||||||
|  |     z64 end of central directory locator | ||||||
|  |     points to ecdr64 | ||||||
|  |     why | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     ret = b"\x50\x4b\x06\x07" | ||||||
|  |  | ||||||
|  |     # 4b cdisk, 8b start of ecdr64, 4b ndisks | ||||||
|  |     ret += struct.pack("<LQL", 0, ecdr64_pos, 1) | ||||||
|  |  | ||||||
|  |     return ret | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class StreamZip(object): | ||||||
|  |     def __init__(self, fgen, utf8=False, pre_crc=False): | ||||||
|  |         self.fgen = fgen | ||||||
|  |         self.utf8 = utf8 | ||||||
|  |         self.pre_crc = pre_crc | ||||||
|  |  | ||||||
|  |         self.pos = 0 | ||||||
|  |         self.items = [] | ||||||
|  |  | ||||||
|  |     def _ct(self, buf): | ||||||
|  |         self.pos += len(buf) | ||||||
|  |         return buf | ||||||
|  |  | ||||||
|  |     def ser(self, f): | ||||||
|  |         name = f["vp"] | ||||||
|  |         src = f["ap"] | ||||||
|  |         st = f["st"] | ||||||
|  |  | ||||||
|  |         sz = st.st_size | ||||||
|  |         ts = st.st_mtime + 1 | ||||||
|  |  | ||||||
|  |         crc = None | ||||||
|  |         if self.pre_crc: | ||||||
|  |             crc = 0 | ||||||
|  |             for buf in yieldfile(src): | ||||||
|  |                 crc = zlib.crc32(buf, crc) | ||||||
|  |  | ||||||
|  |             crc &= 0xFFFFFFFF | ||||||
|  |  | ||||||
|  |         h_pos = self.pos | ||||||
|  |         buf = gen_hdr(None, name, sz, ts, self.utf8, crc, self.pre_crc) | ||||||
|  |         yield self._ct(buf) | ||||||
|  |  | ||||||
|  |         crc = crc or 0 | ||||||
|  |         for buf in yieldfile(src): | ||||||
|  |             if not self.pre_crc: | ||||||
|  |                 crc = zlib.crc32(buf, crc) | ||||||
|  |  | ||||||
|  |             yield self._ct(buf) | ||||||
|  |  | ||||||
|  |         crc &= 0xFFFFFFFF | ||||||
|  |  | ||||||
|  |         self.items.append([name, sz, ts, crc, h_pos]) | ||||||
|  |  | ||||||
|  |         z64 = sz >= 4 * 1024 * 1024 * 1024 | ||||||
|  |  | ||||||
|  |         if z64 or not self.pre_crc: | ||||||
|  |             buf = gen_fdesc(sz, crc, z64) | ||||||
|  |             yield self._ct(buf) | ||||||
|  |  | ||||||
|  |     def gen(self): | ||||||
|  |         errors = [] | ||||||
|  |         for f in self.fgen: | ||||||
|  |             if "err" in f: | ||||||
|  |                 errors.append([f["vp"], f["err"]]) | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             try: | ||||||
|  |                 for x in self.ser(f): | ||||||
|  |                     yield x | ||||||
|  |             except Exception as ex: | ||||||
|  |                 errors.append([f["vp"], repr(ex)]) | ||||||
|  |  | ||||||
|  |         if errors: | ||||||
|  |             errf = errdesc(errors) | ||||||
|  |             print(repr(errf)) | ||||||
|  |             for x in self.ser(errf): | ||||||
|  |                 yield x | ||||||
|  |  | ||||||
|  |         cdir_pos = self.pos | ||||||
|  |         for name, sz, ts, crc, h_pos in self.items: | ||||||
|  |             buf = gen_hdr(h_pos, name, sz, ts, self.utf8, crc, self.pre_crc) | ||||||
|  |             yield self._ct(buf) | ||||||
|  |         cdir_end = self.pos | ||||||
|  |  | ||||||
|  |         _, need_64 = gen_ecdr(self.items, cdir_pos, cdir_end) | ||||||
|  |         if need_64: | ||||||
|  |             ecdir64_pos = self.pos | ||||||
|  |             buf = gen_ecdr64(self.items, cdir_pos, cdir_end) | ||||||
|  |             yield self._ct(buf) | ||||||
|  |  | ||||||
|  |             buf = gen_ecdr64_loc(ecdir64_pos) | ||||||
|  |             yield self._ct(buf) | ||||||
|  |  | ||||||
|  |         ecdr, _ = gen_ecdr(self.items, cdir_pos, cdir_end) | ||||||
|  |         yield self._ct(ecdr) | ||||||
|  |  | ||||||
|  |         if errors: | ||||||
|  |             os.unlink(errf["ap"]) | ||||||
| @@ -68,22 +68,29 @@ class TcpSrv(object): | |||||||
|             self.log("tcpsrv", "listening @ {0}:{1}".format(ip, port)) |             self.log("tcpsrv", "listening @ {0}:{1}".format(ip, port)) | ||||||
|  |  | ||||||
|         while True: |         while True: | ||||||
|             self.log("tcpsrv", "\033[1;30m|%sC-ncli\033[0m" % ("-" * 1,)) |             if self.args.log_conn: | ||||||
|  |                 self.log("tcpsrv", "|%sC-ncli" % ("-" * 1,), c="1;30") | ||||||
|  |  | ||||||
|             if self.num_clients.v >= self.args.nc: |             if self.num_clients.v >= self.args.nc: | ||||||
|                 time.sleep(0.1) |                 time.sleep(0.1) | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             self.log("tcpsrv", "\033[1;30m|%sC-acc1\033[0m" % ("-" * 2,)) |             if self.args.log_conn: | ||||||
|  |                 self.log("tcpsrv", "|%sC-acc1" % ("-" * 2,), c="1;30") | ||||||
|  |  | ||||||
|             ready, _, _ = select.select(self.srv, [], []) |             ready, _, _ = select.select(self.srv, [], []) | ||||||
|             for srv in ready: |             for srv in ready: | ||||||
|                 sck, addr = srv.accept() |                 sck, addr = srv.accept() | ||||||
|                 sip, sport = srv.getsockname() |                 sip, sport = srv.getsockname() | ||||||
|                 self.log( |                 if self.args.log_conn: | ||||||
|                     "%s %s" % addr, |                     self.log( | ||||||
|                     "\033[1;30m|{}C-acc2 \033[0;36m{} \033[3{}m{}".format( |                         "%s %s" % addr, | ||||||
|                         "-" * 3, sip, sport % 8, sport |                         "|{}C-acc2 \033[0;36m{} \033[3{}m{}".format( | ||||||
|                     ), |                             "-" * 3, sip, sport % 8, sport | ||||||
|                 ) |                         ), | ||||||
|  |                         c="1;30", | ||||||
|  |                     ) | ||||||
|  |  | ||||||
|                 self.num_clients.add() |                 self.num_clients.add() | ||||||
|                 self.hub.broker.put(False, "httpconn", sck, addr) |                 self.hub.broker.put(False, "httpconn", sck, addr) | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										281
									
								
								copyparty/u2idx.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										281
									
								
								copyparty/u2idx.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,281 @@ | |||||||
|  | # coding: utf-8 | ||||||
|  | from __future__ import print_function, unicode_literals | ||||||
|  |  | ||||||
|  | import re | ||||||
|  | import os | ||||||
|  | import time | ||||||
|  | import threading | ||||||
|  | from datetime import datetime | ||||||
|  |  | ||||||
|  | from .util import u8safe, s3dec, html_escape, Pebkac | ||||||
|  | from .up2k import up2k_wark_from_hashlist | ||||||
|  |  | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     HAVE_SQLITE3 = True | ||||||
|  |     import sqlite3 | ||||||
|  | except: | ||||||
|  |     HAVE_SQLITE3 = False | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class U2idx(object): | ||||||
|  |     def __init__(self, args, log_func): | ||||||
|  |         self.args = args | ||||||
|  |         self.log_func = log_func | ||||||
|  |         self.timeout = args.srch_time | ||||||
|  |  | ||||||
|  |         if not HAVE_SQLITE3: | ||||||
|  |             self.log("could not load sqlite3; searchign wqill be disabled") | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         self.cur = {} | ||||||
|  |         self.mem_cur = sqlite3.connect(":memory:") | ||||||
|  |         self.mem_cur.execute(r"create table a (b text)") | ||||||
|  |  | ||||||
|  |         self.p_end = None | ||||||
|  |         self.p_dur = 0 | ||||||
|  |  | ||||||
|  |     def log(self, msg, c=0): | ||||||
|  |         self.log_func("u2idx", msg, c) | ||||||
|  |  | ||||||
|  |     def fsearch(self, vols, body): | ||||||
|  |         """search by up2k hashlist""" | ||||||
|  |         if not HAVE_SQLITE3: | ||||||
|  |             return [] | ||||||
|  |  | ||||||
|  |         fsize = body["size"] | ||||||
|  |         fhash = body["hash"] | ||||||
|  |         wark = up2k_wark_from_hashlist(self.args.salt, fsize, fhash) | ||||||
|  |  | ||||||
|  |         uq = "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)) | ||||||
|  |  | ||||||
|  |     def get_cur(self, ptop): | ||||||
|  |         cur = self.cur.get(ptop) | ||||||
|  |         if cur: | ||||||
|  |             return cur | ||||||
|  |  | ||||||
|  |         cur = _open(ptop) | ||||||
|  |         if not cur: | ||||||
|  |             return None | ||||||
|  |  | ||||||
|  |         self.cur[ptop] = cur | ||||||
|  |         return cur | ||||||
|  |  | ||||||
|  |     def search(self, vols, body): | ||||||
|  |         """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) | ||||||
|  |  | ||||||
|  |         uq, uv = _sqlize(qobj) | ||||||
|  |  | ||||||
|  |         qobj = {} | ||||||
|  |         if "tags" in body: | ||||||
|  |             _conv_txt(qobj, body, "tags", "mt.v") | ||||||
|  |  | ||||||
|  |         if "adv" in body: | ||||||
|  |             _conv_adv(qobj, body, "adv") | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             return self.run_query(vols, uq, uv, qobj) | ||||||
|  |         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))) | ||||||
|  |  | ||||||
|  |         done_flag = [] | ||||||
|  |         self.active_id = "{:.6f}_{}".format( | ||||||
|  |             time.time(), threading.current_thread().ident | ||||||
|  |         ) | ||||||
|  |         thr = threading.Thread( | ||||||
|  |             target=self.terminator, | ||||||
|  |             args=( | ||||||
|  |                 self.active_id, | ||||||
|  |                 done_flag, | ||||||
|  |             ), | ||||||
|  |         ) | ||||||
|  |         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) | ||||||
|  |         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) | ||||||
|  |  | ||||||
|  |             if uq: | ||||||
|  |                 where.append(uq) | ||||||
|  |                 v.extend(uv) | ||||||
|  |  | ||||||
|  |             q += " where " + (" and ".join(where)) | ||||||
|  |  | ||||||
|  |         # self.log("q2: {} {}".format(q, repr(v))) | ||||||
|  |  | ||||||
|  |         ret = [] | ||||||
|  |         lim = 1000 | ||||||
|  |         taglist = {} | ||||||
|  |         for (vtop, ptop, flags) in vols: | ||||||
|  |             cur = self.get_cur(ptop) | ||||||
|  |             if not cur: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             self.active_cur = cur | ||||||
|  |  | ||||||
|  |             sret = [] | ||||||
|  |             c = cur.execute(q, v) | ||||||
|  |             for hit in c: | ||||||
|  |                 w, ts, sz, rd, fn = hit | ||||||
|  |                 lim -= 1 | ||||||
|  |                 if lim <= 0: | ||||||
|  |                     break | ||||||
|  |  | ||||||
|  |                 if rd.startswith("//") or fn.startswith("//"): | ||||||
|  |                     rd, fn = s3dec(rd, fn) | ||||||
|  |  | ||||||
|  |                 rp = os.path.join(vtop, rd, fn).replace("\\", "/") | ||||||
|  |                 sret.append({"ts": int(ts), "sz": sz, "rp": rp, "w": w[:16]}) | ||||||
|  |  | ||||||
|  |             for hit in sret: | ||||||
|  |                 w = hit["w"] | ||||||
|  |                 del hit["w"] | ||||||
|  |                 tags = {} | ||||||
|  |                 q2 = "select k, v from mt where w = ? and k != 'x'" | ||||||
|  |                 for k, v2 in cur.execute(q2, (w,)): | ||||||
|  |                     taglist[k] = True | ||||||
|  |                     tags[k] = v2 | ||||||
|  |  | ||||||
|  |                 hit["tags"] = tags | ||||||
|  |  | ||||||
|  |             ret.extend(sret) | ||||||
|  |  | ||||||
|  |         done_flag.append(True) | ||||||
|  |         self.active_id = None | ||||||
|  |  | ||||||
|  |         # undupe hits from multiple metadata keys | ||||||
|  |         if len(ret) > 1: | ||||||
|  |             ret = [ret[0]] + [ | ||||||
|  |                 y for x, y in zip(ret[:-1], ret[1:]) if x["rp"] != y["rp"] | ||||||
|  |             ] | ||||||
|  |  | ||||||
|  |         return ret, list(taglist.keys()) | ||||||
|  |  | ||||||
|  |     def terminator(self, identifier, done_flag): | ||||||
|  |         for _ in range(self.timeout): | ||||||
|  |             time.sleep(1) | ||||||
|  |             if done_flag: | ||||||
|  |                 return | ||||||
|  |  | ||||||
|  |         if identifier == self.active_id: | ||||||
|  |             self.active_cur.connection.interrupt() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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 | ||||||
							
								
								
									
										1183
									
								
								copyparty/up2k.py
									
									
									
									
									
								
							
							
						
						
									
										1183
									
								
								copyparty/up2k.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -10,12 +10,13 @@ import select | |||||||
| import struct | import struct | ||||||
| import hashlib | import hashlib | ||||||
| import platform | import platform | ||||||
|  | import traceback | ||||||
| import threading | import threading | ||||||
| import mimetypes | import mimetypes | ||||||
| import contextlib | import contextlib | ||||||
| import subprocess as sp  # nosec | import subprocess as sp  # nosec | ||||||
|  |  | ||||||
| from .__init__ import PY2, WINDOWS | from .__init__ import PY2, WINDOWS, ANYWIN | ||||||
| from .stolen import surrogateescape | from .stolen import surrogateescape | ||||||
|  |  | ||||||
| FAKE_MP = False | FAKE_MP = False | ||||||
| @@ -48,6 +49,7 @@ HTTPCODE = { | |||||||
|     200: "OK", |     200: "OK", | ||||||
|     204: "No Content", |     204: "No Content", | ||||||
|     206: "Partial Content", |     206: "Partial Content", | ||||||
|  |     302: "Found", | ||||||
|     304: "Not Modified", |     304: "Not Modified", | ||||||
|     400: "Bad Request", |     400: "Bad Request", | ||||||
|     403: "Forbidden", |     403: "Forbidden", | ||||||
| @@ -56,11 +58,58 @@ HTTPCODE = { | |||||||
|     413: "Payload Too Large", |     413: "Payload Too Large", | ||||||
|     416: "Requested Range Not Satisfiable", |     416: "Requested Range Not Satisfiable", | ||||||
|     422: "Unprocessable Entity", |     422: "Unprocessable Entity", | ||||||
|  |     429: "Too Many Requests", | ||||||
|     500: "Internal Server Error", |     500: "Internal Server Error", | ||||||
|     501: "Not Implemented", |     501: "Not Implemented", | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | IMPLICATIONS = [ | ||||||
|  |     ["e2dsa", "e2ds"], | ||||||
|  |     ["e2ds", "e2d"], | ||||||
|  |     ["e2tsr", "e2ts"], | ||||||
|  |     ["e2ts", "e2t"], | ||||||
|  |     ["e2t", "e2d"], | ||||||
|  | ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | REKOBO_KEY = { | ||||||
|  |     v: ln.split(" ", 1)[0] | ||||||
|  |     for ln in """ | ||||||
|  | 1B 6d B | ||||||
|  | 2B 7d Gb F# | ||||||
|  | 3B 8d Db C# | ||||||
|  | 4B 9d Ab G# | ||||||
|  | 5B 10d Eb D# | ||||||
|  | 6B 11d Bb A# | ||||||
|  | 7B 12d F | ||||||
|  | 8B 1d C | ||||||
|  | 9B 2d G | ||||||
|  | 10B 3d D | ||||||
|  | 11B 4d A | ||||||
|  | 12B 5d E | ||||||
|  | 1A 6m Abm G#m | ||||||
|  | 2A 7m Ebm D#m | ||||||
|  | 3A 8m Bbm A#m | ||||||
|  | 4A 9m Fm | ||||||
|  | 5A 10m Cm | ||||||
|  | 6A 11m Gm | ||||||
|  | 7A 12m Dm | ||||||
|  | 8A 1m Am | ||||||
|  | 9A 2m Em | ||||||
|  | 10A 3m Bm | ||||||
|  | 11A 4m Gbm F#m | ||||||
|  | 12A 5m Dbm C#m | ||||||
|  | """.strip().split( | ||||||
|  |         "\n" | ||||||
|  |     ) | ||||||
|  |     for v in ln.strip().split(" ")[1:] | ||||||
|  |     if v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | REKOBO_LKEY = {k.lower(): v for k, v in REKOBO_KEY.items()} | ||||||
|  |  | ||||||
|  |  | ||||||
| class Counter(object): | class Counter(object): | ||||||
|     def __init__(self, v=0): |     def __init__(self, v=0): | ||||||
|         self.v = v |         self.v = v | ||||||
| @@ -99,6 +148,71 @@ class Unrecv(object): | |||||||
|         self.buf = buf + self.buf |         self.buf = buf + self.buf | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ProgressPrinter(threading.Thread): | ||||||
|  |     """ | ||||||
|  |     periodically print progress info without linefeeds | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self): | ||||||
|  |         threading.Thread.__init__(self) | ||||||
|  |         self.daemon = True | ||||||
|  |         self.msg = None | ||||||
|  |         self.end = False | ||||||
|  |         self.start() | ||||||
|  |  | ||||||
|  |     def run(self): | ||||||
|  |         msg = None | ||||||
|  |         while not self.end: | ||||||
|  |             time.sleep(0.1) | ||||||
|  |             if msg == self.msg or self.end: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             msg = self.msg | ||||||
|  |             uprint(" {}\033[K\r".format(msg)) | ||||||
|  |  | ||||||
|  |         print("\033[K", end="") | ||||||
|  |         sys.stdout.flush()  # necessary on win10 even w/ stderr btw | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def uprint(msg): | ||||||
|  |     try: | ||||||
|  |         print(msg, end="") | ||||||
|  |     except UnicodeEncodeError: | ||||||
|  |         try: | ||||||
|  |             print(msg.encode("utf-8", "replace").decode(), end="") | ||||||
|  |         except: | ||||||
|  |             print(msg.encode("ascii", "replace").decode(), end="") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def nuprint(msg): | ||||||
|  |     uprint("{}\n".format(msg)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def rice_tid(): | ||||||
|  |     tid = threading.current_thread().ident | ||||||
|  |     c = struct.unpack(b"B" * 5, struct.pack(b">Q", tid)[-5:]) | ||||||
|  |     return "".join("\033[1;37;48;5;{}m{:02x}".format(x, x) for x in c) + "\033[0m" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def trace(*args, **kwargs): | ||||||
|  |     t = time.time() | ||||||
|  |     stack = "".join( | ||||||
|  |         "\033[36m{}\033[33m{}".format(x[0].split(os.sep)[-1][:-3], x[1]) | ||||||
|  |         for x in traceback.extract_stack()[3:-1] | ||||||
|  |     ) | ||||||
|  |     parts = ["{:.6f}".format(t), rice_tid(), stack] | ||||||
|  |  | ||||||
|  |     if args: | ||||||
|  |         parts.append(repr(args)) | ||||||
|  |  | ||||||
|  |     if kwargs: | ||||||
|  |         parts.append(repr(kwargs)) | ||||||
|  |  | ||||||
|  |     msg = "\033[0m ".join(parts) | ||||||
|  |     # _tracebuf.append(msg) | ||||||
|  |     nuprint(msg) | ||||||
|  |  | ||||||
|  |  | ||||||
| @contextlib.contextmanager | @contextlib.contextmanager | ||||||
| def ren_open(fname, *args, **kwargs): | def ren_open(fname, *args, **kwargs): | ||||||
|     fdir = kwargs.pop("fdir", None) |     fdir = kwargs.pop("fdir", None) | ||||||
| @@ -146,7 +260,7 @@ def ren_open(fname, *args, **kwargs): | |||||||
|  |  | ||||||
|         except OSError as ex_: |         except OSError as ex_: | ||||||
|             ex = ex_ |             ex = ex_ | ||||||
|             if ex.errno != 36: |             if ex.errno not in [36, 63] and (not WINDOWS or ex.errno != 22): | ||||||
|                 raise |                 raise | ||||||
|  |  | ||||||
|         if not b64: |         if not b64: | ||||||
| @@ -437,6 +551,16 @@ def get_spd(nbyte, t0, t=None): | |||||||
|     return "{} \033[0m{}/s\033[0m".format(s1, s2) |     return "{} \033[0m{}/s\033[0m".format(s1, s2) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def s2hms(s, optional_h=False): | ||||||
|  |     s = int(s) | ||||||
|  |     h, s = divmod(s, 3600) | ||||||
|  |     m, s = divmod(s, 60) | ||||||
|  |     if not h and optional_h: | ||||||
|  |         return "{}:{:02}".format(m, s) | ||||||
|  |  | ||||||
|  |     return "{}:{:02}:{:02}".format(h, m, s) | ||||||
|  |  | ||||||
|  |  | ||||||
| def undot(path): | def undot(path): | ||||||
|     ret = [] |     ret = [] | ||||||
|     for node in path.split("/"): |     for node in path.split("/"): | ||||||
| @@ -453,11 +577,12 @@ def undot(path): | |||||||
|     return "/".join(ret) |     return "/".join(ret) | ||||||
|  |  | ||||||
|  |  | ||||||
| def sanitize_fn(fn): | def sanitize_fn(fn, ok="", bad=[]): | ||||||
|     fn = fn.replace("\\", "/").split("/")[-1] |     if "/" not in ok: | ||||||
|  |         fn = fn.replace("\\", "/").split("/")[-1] | ||||||
|  |  | ||||||
|     if WINDOWS: |     if ANYWIN: | ||||||
|         for bad, good in [ |         remap = [ | ||||||
|             ["<", "<"], |             ["<", "<"], | ||||||
|             [">", ">"], |             [">", ">"], | ||||||
|             [":", ":"], |             [":", ":"], | ||||||
| @@ -467,36 +592,49 @@ def sanitize_fn(fn): | |||||||
|             ["|", "|"], |             ["|", "|"], | ||||||
|             ["?", "?"], |             ["?", "?"], | ||||||
|             ["*", "*"], |             ["*", "*"], | ||||||
|         ]: |         ] | ||||||
|             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): |         for n in range(1, 10): | ||||||
|             bad += "com{0} lpt{0}".format(n).split(" ") |             bad += "com{0} lpt{0}".format(n).split(" ") | ||||||
|  |  | ||||||
|         if fn.lower() in bad: |     if fn.lower() in bad: | ||||||
|             fn = "_" + fn |         fn = "_" + fn | ||||||
|  |  | ||||||
|     return fn.strip() |     return fn.strip() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def u8safe(txt): | ||||||
|  |     try: | ||||||
|  |         return txt.encode("utf-8", "xmlcharrefreplace").decode("utf-8", "replace") | ||||||
|  |     except: | ||||||
|  |         return txt.encode("utf-8", "replace").decode("utf-8", "replace") | ||||||
|  |  | ||||||
|  |  | ||||||
| def exclude_dotfiles(filepaths): | def exclude_dotfiles(filepaths): | ||||||
|     for fpath in filepaths: |     return [x for x in filepaths if not x.split("/")[-1].startswith(".")] | ||||||
|         if not fpath.split("/")[-1].startswith("."): |  | ||||||
|             yield fpath |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def html_escape(s, quote=False): | def html_escape(s, quote=False, crlf=False): | ||||||
|     """html.escape but also newlines""" |     """html.escape but also newlines""" | ||||||
|     s = ( |     s = s.replace("&", "&").replace("<", "<").replace(">", ">") | ||||||
|         s.replace("&", "&") |  | ||||||
|         .replace("<", "<") |  | ||||||
|         .replace(">", ">") |  | ||||||
|         .replace("\r", "
") |  | ||||||
|         .replace("\n", "
") |  | ||||||
|     ) |  | ||||||
|     if quote: |     if quote: | ||||||
|         s = s.replace('"', """).replace("'", "'") |         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 |     return s | ||||||
|  |  | ||||||
| @@ -536,6 +674,16 @@ def w8enc(txt): | |||||||
|     return txt.encode(FS_ENCODING, "surrogateescape") |     return txt.encode(FS_ENCODING, "surrogateescape") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def w8b64dec(txt): | ||||||
|  |     """decodes base64(filesystem-bytes) to wtf8""" | ||||||
|  |     return w8dec(base64.urlsafe_b64decode(txt.encode("ascii"))) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def w8b64enc(txt): | ||||||
|  |     """encodes wtf8 to base64(filesystem-bytes)""" | ||||||
|  |     return base64.urlsafe_b64encode(w8enc(txt)).decode("ascii") | ||||||
|  |  | ||||||
|  |  | ||||||
| if PY2 and WINDOWS: | if PY2 and WINDOWS: | ||||||
|     # moonrunes become \x3f with bytestrings, |     # moonrunes become \x3f with bytestrings, | ||||||
|     # losing mojibake support is worth |     # losing mojibake support is worth | ||||||
| @@ -549,6 +697,31 @@ else: | |||||||
|     fsdec = w8dec |     fsdec = w8dec | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def s3enc(mem_cur, rd, fn): | ||||||
|  |     ret = [] | ||||||
|  |     for v in [rd, fn]: | ||||||
|  |         try: | ||||||
|  |             mem_cur.execute("select * from a where b = ?", (v,)) | ||||||
|  |             ret.append(v) | ||||||
|  |         except: | ||||||
|  |             ret.append("//" + w8b64enc(v)) | ||||||
|  |             # self.log("mojien/{} [{}] {}".format(k, v, ret[-1][2:])) | ||||||
|  |  | ||||||
|  |     return tuple(ret) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def s3dec(rd, fn): | ||||||
|  |     ret = [] | ||||||
|  |     for k, v in [["d", rd], ["f", fn]]: | ||||||
|  |         if v.startswith("//"): | ||||||
|  |             ret.append(w8b64dec(v[2:])) | ||||||
|  |             # self.log("mojide/{} [{}] {}".format(k, ret[-1], v[2:])) | ||||||
|  |         else: | ||||||
|  |             ret.append(v) | ||||||
|  |  | ||||||
|  |     return tuple(ret) | ||||||
|  |  | ||||||
|  |  | ||||||
| def atomic_move(src, dst): | def atomic_move(src, dst): | ||||||
|     if not PY2: |     if not PY2: | ||||||
|         os.replace(src, dst) |         os.replace(src, dst) | ||||||
| @@ -583,6 +756,50 @@ def read_socket_unbounded(sr): | |||||||
|         yield buf |         yield buf | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def read_socket_chunked(sr, log=None): | ||||||
|  |     err = "expected chunk length, got [{}] |{}| instead" | ||||||
|  |     while True: | ||||||
|  |         buf = b"" | ||||||
|  |         while b"\r" not in buf: | ||||||
|  |             rbuf = sr.recv(2) | ||||||
|  |             if not rbuf or len(buf) > 16: | ||||||
|  |                 err = err.format(buf.decode("utf-8", "replace"), len(buf)) | ||||||
|  |                 raise Pebkac(400, err) | ||||||
|  |  | ||||||
|  |             buf += rbuf | ||||||
|  |  | ||||||
|  |         if not buf.endswith(b"\n"): | ||||||
|  |             sr.recv(1) | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             chunklen = int(buf.rstrip(b"\r\n"), 16) | ||||||
|  |         except: | ||||||
|  |             err = err.format(buf.decode("utf-8", "replace"), len(buf)) | ||||||
|  |             raise Pebkac(400, err) | ||||||
|  |  | ||||||
|  |         if chunklen == 0: | ||||||
|  |             sr.recv(2)  # \r\n after final chunk | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         if log: | ||||||
|  |             log("receiving {} byte chunk".format(chunklen)) | ||||||
|  |  | ||||||
|  |         for chunk in read_socket(sr, chunklen): | ||||||
|  |             yield chunk | ||||||
|  |  | ||||||
|  |         sr.recv(2)  # \r\n after each chunk too | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def yieldfile(fn): | ||||||
|  |     with open(fsenc(fn), "rb", 512 * 1024) as f: | ||||||
|  |         while True: | ||||||
|  |             buf = f.read(64 * 1024) | ||||||
|  |             if not buf: | ||||||
|  |                 break | ||||||
|  |  | ||||||
|  |             yield buf | ||||||
|  |  | ||||||
|  |  | ||||||
| def hashcopy(actor, fin, fout): | def hashcopy(actor, fin, fout): | ||||||
|     u32_lim = int((2 ** 31) * 0.9) |     u32_lim = int((2 ** 31) * 0.9) | ||||||
|     hashobj = hashlib.sha512() |     hashobj = hashlib.sha512() | ||||||
| @@ -642,6 +859,33 @@ def sendfile_kern(lower, upper, f, s): | |||||||
|     return 0 |     return 0 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def statdir(logger, scandir, lstat, top): | ||||||
|  |     try: | ||||||
|  |         btop = fsenc(top) | ||||||
|  |         if scandir and hasattr(os, "scandir"): | ||||||
|  |             src = "scandir" | ||||||
|  |             with os.scandir(btop) as dh: | ||||||
|  |                 for fh in dh: | ||||||
|  |                     try: | ||||||
|  |                         yield [fsdec(fh.name), fh.stat(follow_symlinks=not lstat)] | ||||||
|  |                     except Exception as ex: | ||||||
|  |                         msg = "scan-stat: \033[36m{} @ {}" | ||||||
|  |                         logger(msg.format(repr(ex), fsdec(fh.path))) | ||||||
|  |         else: | ||||||
|  |             src = "listdir" | ||||||
|  |             fun = os.lstat if lstat else os.stat | ||||||
|  |             for name in os.listdir(btop): | ||||||
|  |                 abspath = os.path.join(btop, name) | ||||||
|  |                 try: | ||||||
|  |                     yield [fsdec(name), fun(abspath)] | ||||||
|  |                 except Exception as ex: | ||||||
|  |                     msg = "list-stat: \033[36m{} @ {}" | ||||||
|  |                     logger(msg.format(repr(ex), fsdec(abspath))) | ||||||
|  |  | ||||||
|  |     except Exception as ex: | ||||||
|  |         logger("{}: \033[31m{} @ {}".format(src, repr(ex), top)) | ||||||
|  |  | ||||||
|  |  | ||||||
| def unescape_cookie(orig): | def unescape_cookie(orig): | ||||||
|     # mw=idk; doot=qwe%2Crty%3Basd+fgh%2Bjkl%25zxc%26vbn  # qwe,rty;asd fgh+jkl%zxc&vbn |     # mw=idk; doot=qwe%2Crty%3Basd+fgh%2Bjkl%25zxc%26vbn  # qwe,rty;asd fgh+jkl%zxc&vbn | ||||||
|     ret = "" |     ret = "" | ||||||
| @@ -696,7 +940,11 @@ def chkcmd(*argv): | |||||||
| def gzip_orig_sz(fn): | def gzip_orig_sz(fn): | ||||||
|     with open(fsenc(fn), "rb") as f: |     with open(fsenc(fn), "rb") as f: | ||||||
|         f.seek(-4, 2) |         f.seek(-4, 2) | ||||||
|         return struct.unpack(b"I", f.read(4))[0] |         rv = f.read(4) | ||||||
|  |         try: | ||||||
|  |             return struct.unpack(b"I", rv)[0] | ||||||
|  |         except: | ||||||
|  |             return struct.unpack("I", rv)[0] | ||||||
|  |  | ||||||
|  |  | ||||||
| def py_desc(): | def py_desc(): | ||||||
| @@ -706,7 +954,11 @@ def py_desc(): | |||||||
|     if ofs > 0: |     if ofs > 0: | ||||||
|         py_ver = py_ver[:ofs] |         py_ver = py_ver[:ofs] | ||||||
|  |  | ||||||
|     bitness = struct.calcsize(b"P") * 8 |     try: | ||||||
|  |         bitness = struct.calcsize(b"P") * 8 | ||||||
|  |     except: | ||||||
|  |         bitness = struct.calcsize("P") * 8 | ||||||
|  |  | ||||||
|     host_os = platform.system() |     host_os = platform.system() | ||||||
|     compiler = platform.python_compiler() |     compiler = platform.python_compiler() | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ html,body,tr,th,td,#files,a { | |||||||
| 	background: none; | 	background: none; | ||||||
| 	font-weight: inherit; | 	font-weight: inherit; | ||||||
| 	font-size: inherit; | 	font-size: inherit; | ||||||
| 	padding: none; | 	padding: 0; | ||||||
| 	border: none; | 	border: none; | ||||||
| } | } | ||||||
| html { | html { | ||||||
| @@ -39,15 +39,22 @@ body { | |||||||
| 	margin: 1.3em 0 0 0; | 	margin: 1.3em 0 0 0; | ||||||
| 	font-size: 1.4em; | 	font-size: 1.4em; | ||||||
| } | } | ||||||
|  | #path #entree { | ||||||
|  | 	margin-left: -.7em; | ||||||
|  | } | ||||||
| #files { | #files { | ||||||
| 	border-collapse: collapse; | 	border-spacing: 0; | ||||||
| 	margin-top: 2em; | 	z-index: 1; | ||||||
|  | 	position: relative; | ||||||
| } | } | ||||||
| #files tbody a { | #files tbody a { | ||||||
| 	display: block; | 	display: block; | ||||||
| 	padding: .3em 0; | 	padding: .3em 0; | ||||||
| } | } | ||||||
| a { | #files tbody div a { | ||||||
|  | 	color: #f5a; | ||||||
|  | } | ||||||
|  | a, #files tbody div a:last-child { | ||||||
| 	color: #fc5; | 	color: #fc5; | ||||||
| 	padding: .2em; | 	padding: .2em; | ||||||
| 	text-decoration: none; | 	text-decoration: none; | ||||||
| @@ -55,6 +62,7 @@ a { | |||||||
| #files a:hover { | #files a:hover { | ||||||
| 	color: #fff; | 	color: #fff; | ||||||
| 	background: #161616; | 	background: #161616; | ||||||
|  | 	text-decoration: underline; | ||||||
| } | } | ||||||
| #files thead a { | #files thead a { | ||||||
| 	color: #999; | 	color: #999; | ||||||
| @@ -65,6 +73,7 @@ a { | |||||||
| } | } | ||||||
| #files thead th { | #files thead th { | ||||||
| 	padding: .5em 1.3em .3em 1.3em; | 	padding: .5em 1.3em .3em 1.3em; | ||||||
|  | 	cursor: pointer; | ||||||
| } | } | ||||||
| #files thead th:last-child { | #files thead th:last-child { | ||||||
| 	background: #444; | 	background: #444; | ||||||
| @@ -81,6 +90,14 @@ a { | |||||||
| #files td { | #files td { | ||||||
| 	margin: 0; | 	margin: 0; | ||||||
| 	padding: 0 .5em; | 	padding: 0 .5em; | ||||||
|  | 	border-bottom: 1px solid #111; | ||||||
|  | } | ||||||
|  | #files td+td+td { | ||||||
|  | 	max-width: 30em; | ||||||
|  | 	overflow: hidden; | ||||||
|  | } | ||||||
|  | #files tr+tr td { | ||||||
|  | 	border-top: 1px solid #383838; | ||||||
| } | } | ||||||
| #files tbody td:nth-child(3) { | #files tbody td:nth-child(3) { | ||||||
| 	font-family: monospace; | 	font-family: monospace; | ||||||
| @@ -100,6 +117,9 @@ a { | |||||||
| 	padding-bottom: 1.3em; | 	padding-bottom: 1.3em; | ||||||
| 	border-bottom: .5em solid #444; | 	border-bottom: .5em solid #444; | ||||||
| } | } | ||||||
|  | #files tbody tr td:last-child { | ||||||
|  | 	white-space: nowrap; | ||||||
|  | } | ||||||
| #files thead th[style] { | #files thead th[style] { | ||||||
| 	width: auto !important; | 	width: auto !important; | ||||||
| } | } | ||||||
| @@ -131,6 +151,15 @@ a { | |||||||
| .logue { | .logue { | ||||||
| 	padding: .2em 1.5em; | 	padding: .2em 1.5em; | ||||||
| } | } | ||||||
|  | .logue:empty { | ||||||
|  | 	display: none; | ||||||
|  | } | ||||||
|  | #pro.logue { | ||||||
|  | 	margin-bottom: .8em; | ||||||
|  | } | ||||||
|  | #epi.logue { | ||||||
|  | 	margin: .8em 0; | ||||||
|  | } | ||||||
| #srv_info { | #srv_info { | ||||||
| 	opacity: .5; | 	opacity: .5; | ||||||
| 	font-size: .8em; | 	font-size: .8em; | ||||||
| @@ -142,11 +171,29 @@ a { | |||||||
| #srv_info span { | #srv_info span { | ||||||
| 	color: #fff; | 	color: #fff; | ||||||
| } | } | ||||||
| a.play { | #files tbody a.play { | ||||||
| 	color: #e70; | 	color: #e70; | ||||||
|  | 	padding: .2em; | ||||||
|  | 	margin: -.2em; | ||||||
| } | } | ||||||
| a.play.act { | #files tbody a.play.act { | ||||||
| 	color: #af0; | 	color: #840; | ||||||
|  | 	text-shadow: 0 0 .3em #b80; | ||||||
|  | } | ||||||
|  | #files tbody tr.sel td { | ||||||
|  | 	color: #fff; | ||||||
|  | 	background: #925; | ||||||
|  | 	border-color: #c37; | ||||||
|  | } | ||||||
|  | #files tr.sel a { | ||||||
|  | 	color: #fff; | ||||||
|  | } | ||||||
|  | #files tr.sel a.play { | ||||||
|  | 	color: #fc5; | ||||||
|  | } | ||||||
|  | #files tr.sel a.play.act { | ||||||
|  | 	color: #fff; | ||||||
|  | 	text-shadow: 0 0 1px #fff; | ||||||
| } | } | ||||||
| #blocked { | #blocked { | ||||||
| 	position: fixed; | 	position: fixed; | ||||||
| @@ -156,7 +203,7 @@ a.play.act { | |||||||
| 	height: 100%; | 	height: 100%; | ||||||
| 	background: #333; | 	background: #333; | ||||||
| 	font-size: 2.5em; | 	font-size: 2.5em; | ||||||
| 	z-index:99; | 	z-index: 99; | ||||||
| } | } | ||||||
| #blk_play, | #blk_play, | ||||||
| #blk_abrt { | #blk_abrt { | ||||||
| @@ -190,6 +237,7 @@ a.play.act { | |||||||
| 	bottom: -6em; | 	bottom: -6em; | ||||||
| 	height: 6em; | 	height: 6em; | ||||||
| 	width: 100%; | 	width: 100%; | ||||||
|  | 	z-index: 3; | ||||||
| 	transition: bottom 0.15s; | 	transition: bottom 0.15s; | ||||||
| } | } | ||||||
| #widget.open { | #widget.open { | ||||||
| @@ -203,7 +251,7 @@ a.play.act { | |||||||
| 	height: 100%; | 	height: 100%; | ||||||
| 	background: #3c3c3c; | 	background: #3c3c3c; | ||||||
| } | } | ||||||
| #wtoggle { | #wtico { | ||||||
| 	cursor: url(/.cpr/dd/1.png), pointer; | 	cursor: url(/.cpr/dd/1.png), pointer; | ||||||
| 	animation: cursor 500ms infinite; | 	animation: cursor 500ms infinite; | ||||||
| } | } | ||||||
| @@ -214,6 +262,9 @@ a.play.act { | |||||||
| 	75% {cursor: url(/.cpr/dd/5.png), pointer} | 	75% {cursor: url(/.cpr/dd/5.png), pointer} | ||||||
| 	85% {cursor: url(/.cpr/dd/1.png), pointer} | 	85% {cursor: url(/.cpr/dd/1.png), pointer} | ||||||
| } | } | ||||||
|  | @keyframes spin { | ||||||
|  | 	100% {transform: rotate(360deg)} | ||||||
|  | } | ||||||
| #wtoggle { | #wtoggle { | ||||||
| 	position: absolute; | 	position: absolute; | ||||||
| 	top: -1.2em; | 	top: -1.2em; | ||||||
| @@ -230,6 +281,33 @@ a.play.act { | |||||||
| 	padding: .2em 0 0 .07em; | 	padding: .2em 0 0 .07em; | ||||||
| 	color: #fff; | 	color: #fff; | ||||||
| } | } | ||||||
|  | #wzip { | ||||||
|  | 	display: none; | ||||||
|  | 	margin-right: .3em; | ||||||
|  | 	padding-right: .3em; | ||||||
|  | 	border-right: .1em solid #555; | ||||||
|  | } | ||||||
|  | #wtoggle, | ||||||
|  | #wtoggle * { | ||||||
|  | 	line-height: 1em; | ||||||
|  | } | ||||||
|  | #wtoggle.sel { | ||||||
|  | 	width: 6.4em; | ||||||
|  | } | ||||||
|  | #wtoggle.sel #wzip { | ||||||
|  | 	display: inline-block; | ||||||
|  | } | ||||||
|  | #wtoggle.sel #wzip a { | ||||||
|  | 	font-size: .4em; | ||||||
|  | 	padding: 0 .3em; | ||||||
|  | 	margin: -.3em .2em; | ||||||
|  | 	position: relative; | ||||||
|  | 	display: inline-block; | ||||||
|  | } | ||||||
|  | #wtoggle.sel #wzip #selzip { | ||||||
|  | 	top: -.6em; | ||||||
|  | 	padding: .4em .3em; | ||||||
|  | } | ||||||
| #barpos, | #barpos, | ||||||
| #barbuf { | #barbuf { | ||||||
| 	position: absolute; | 	position: absolute; | ||||||
| @@ -273,3 +351,540 @@ a.play.act { | |||||||
| 	width: calc(100% - 10.5em); | 	width: calc(100% - 10.5em); | ||||||
| 	background: rgba(0,0,0,0.2); | 	background: rgba(0,0,0,0.2); | ||||||
| } | } | ||||||
|  | @media (min-width: 80em) { | ||||||
|  | 	#barpos, | ||||||
|  | 	#barbuf { | ||||||
|  | 		width: calc(100% - 21em); | ||||||
|  | 		left: 9.8em; | ||||||
|  | 		top: .7em; | ||||||
|  | 		height: 1.6em; | ||||||
|  | 		bottom: auto; | ||||||
|  | 	} | ||||||
|  | 	#widget { | ||||||
|  | 		bottom: -3.2em; | ||||||
|  | 		height: 3.2em; | ||||||
|  | 	} | ||||||
|  | 	#pvol { | ||||||
|  | 		max-width: 9em; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .opview { | ||||||
|  | 	display: none; | ||||||
|  | } | ||||||
|  | .opview.act { | ||||||
|  | 	display: block; | ||||||
|  | } | ||||||
|  | #ops a { | ||||||
|  | 	color: #fc5; | ||||||
|  | 	font-size: 1.5em; | ||||||
|  | 	padding: .25em .3em; | ||||||
|  | 	margin: 0; | ||||||
|  | 	outline: none; | ||||||
|  | } | ||||||
|  | #ops a.act { | ||||||
|  | 	background: #281838; | ||||||
|  | 	border-radius: 0 0 .2em .2em; | ||||||
|  | 	border-bottom: .3em solid #d90; | ||||||
|  | 	box-shadow: 0 -.15em .2em #000 inset; | ||||||
|  | 	padding-bottom: .3em; | ||||||
|  | } | ||||||
|  | #ops i { | ||||||
|  | 	font-size: 1.5em; | ||||||
|  | } | ||||||
|  | #ops i:before { | ||||||
|  | 	content: 'x'; | ||||||
|  | 	color: #282828; | ||||||
|  | 	text-shadow: 0 0 .08em #01a7e1; | ||||||
|  | 	position: relative; | ||||||
|  | } | ||||||
|  | #ops i:after { | ||||||
|  | 	content: 'x'; | ||||||
|  | 	color: #282828; | ||||||
|  | 	text-shadow: 0 0 .08em #ff3f1a; | ||||||
|  | 	margin-left: -.35em; | ||||||
|  | 	font-size: 1.05em; | ||||||
|  | } | ||||||
|  | #ops, | ||||||
|  | .opbox { | ||||||
|  | 	border: 1px solid #3a3a3a; | ||||||
|  | 	box-shadow: 0 0 1em #222 inset; | ||||||
|  | } | ||||||
|  | #ops { | ||||||
|  | 	background: #333; | ||||||
|  | 	margin: 1.7em 1.5em 0 1.5em; | ||||||
|  | 	padding: .3em .6em; | ||||||
|  | 	border-radius: .3em; | ||||||
|  | 	border-width: .15em 0; | ||||||
|  | 	white-space: nowrap; | ||||||
|  | } | ||||||
|  | .opbox { | ||||||
|  | 	background: #2d2d2d; | ||||||
|  | 	margin: 1.5em 0 0 0; | ||||||
|  | 	padding: .5em; | ||||||
|  | 	border-radius: 0 1em 1em 0; | ||||||
|  | 	border-width: .15em .3em .3em 0; | ||||||
|  | 	max-width: 40em; | ||||||
|  | } | ||||||
|  | .opbox input { | ||||||
|  | 	margin: .5em; | ||||||
|  | } | ||||||
|  | .opview input[type=text] { | ||||||
|  | 	color: #fff; | ||||||
|  | 	background: #383838; | ||||||
|  | 	border: none; | ||||||
|  | 	box-shadow: 0 0 .3em #222; | ||||||
|  | 	border-bottom: 1px solid #fc5; | ||||||
|  | 	border-radius: .2em; | ||||||
|  | 	padding: .2em .3em; | ||||||
|  | } | ||||||
|  | input[type="checkbox"]+label { | ||||||
|  | 	color: #f5a; | ||||||
|  | } | ||||||
|  | input[type="checkbox"]:checked+label { | ||||||
|  | 	color: #fc5; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #srch_form { | ||||||
|  | 	border: 1px solid #3a3a3a; | ||||||
|  | 	box-shadow: 0 0 1em #222 inset; | ||||||
|  | 	background: #2d2d2d; | ||||||
|  | 	border-radius: .4em; | ||||||
|  | 	margin: 1.4em; | ||||||
|  | 	margin-bottom: 0; | ||||||
|  | 	padding: 0 .5em .5em 0; | ||||||
|  | } | ||||||
|  | #srch_form table { | ||||||
|  | 	display: inline-block; | ||||||
|  | } | ||||||
|  | #srch_form td { | ||||||
|  | 	padding: .6em .6em; | ||||||
|  | } | ||||||
|  | #srch_form td:first-child { | ||||||
|  | 	width: 3em; | ||||||
|  | 	padding-right: .2em; | ||||||
|  | 	text-align: right; | ||||||
|  | } | ||||||
|  | #op_search input { | ||||||
|  | 	margin: 0; | ||||||
|  | } | ||||||
|  | #srch_q { | ||||||
|  | 	white-space: pre; | ||||||
|  | 	color: #f80; | ||||||
|  | 	height: 1em; | ||||||
|  | 	margin: .2em 0 -1em 1.6em; | ||||||
|  | } | ||||||
|  | #files td div span { | ||||||
|  | 	color: #fff; | ||||||
|  | 	padding: 0 .4em; | ||||||
|  | 	font-weight: bold; | ||||||
|  | 	font-style: italic; | ||||||
|  | } | ||||||
|  | #files td div a:hover { | ||||||
|  | 	background: #444; | ||||||
|  | 	color: #fff; | ||||||
|  | } | ||||||
|  | #files td div a { | ||||||
|  | 	display: inline-block; | ||||||
|  | 	white-space: nowrap; | ||||||
|  | } | ||||||
|  | #files td div a:last-child { | ||||||
|  | 	width: 100%; | ||||||
|  | } | ||||||
|  | #files td div { | ||||||
|  | 	border-collapse: collapse; | ||||||
|  | 	width: 100%; | ||||||
|  | } | ||||||
|  | #wrap { | ||||||
|  | 	margin-top: 2em; | ||||||
|  | } | ||||||
|  | #tree { | ||||||
|  | 	display: none; | ||||||
|  | 	position: absolute; | ||||||
|  | 	left: 0; | ||||||
|  | 	bottom: 0; | ||||||
|  | 	top: 7em; | ||||||
|  | 	padding-top: .2em; | ||||||
|  | 	overflow-y: auto; | ||||||
|  | 	-ms-scroll-chaining: none; | ||||||
|  | 	overscroll-behavior-y: none; | ||||||
|  | 	scrollbar-color: #eb0 #333; | ||||||
|  | } | ||||||
|  | #thx_ff { | ||||||
|  | 	padding: 5em 0; | ||||||
|  | } | ||||||
|  | #tree::-webkit-scrollbar-track, | ||||||
|  | #tree::-webkit-scrollbar { | ||||||
|  | 	background: #333; | ||||||
|  | } | ||||||
|  | #tree::-webkit-scrollbar-thumb { | ||||||
|  | 	background: #eb0; | ||||||
|  | } | ||||||
|  | #tree:hover { | ||||||
|  | 	z-index: 2; | ||||||
|  | } | ||||||
|  | #treeul { | ||||||
|  | 	position: relative; | ||||||
|  | 	left: -1.7em; | ||||||
|  | 	width: calc(100% + 1.3em); | ||||||
|  | } | ||||||
|  | .tglbtn, | ||||||
|  | #tree>a+a { | ||||||
|  | 	padding: .2em .4em; | ||||||
|  | 	font-size: 1.2em; | ||||||
|  | 	background: #2a2a2a; | ||||||
|  | 	box-shadow: 0 .1em .2em #222 inset; | ||||||
|  | 	border-radius: .3em; | ||||||
|  | 	margin: .2em; | ||||||
|  | 	position: relative; | ||||||
|  | 	top: -.2em; | ||||||
|  | } | ||||||
|  | .tglbtn:hover, | ||||||
|  | #tree>a+a:hover { | ||||||
|  | 	background: #805; | ||||||
|  | } | ||||||
|  | .tglbtn.on, | ||||||
|  | #tree>a+a.on { | ||||||
|  | 	background: #fc4; | ||||||
|  | 	color: #400; | ||||||
|  | 	text-shadow: none; | ||||||
|  | } | ||||||
|  | #detree { | ||||||
|  | 	padding: .3em .5em; | ||||||
|  | 	font-size: 1.5em; | ||||||
|  | } | ||||||
|  | #tree ul, | ||||||
|  | #tree li { | ||||||
|  | 	padding: 0; | ||||||
|  | 	margin: 0; | ||||||
|  | } | ||||||
|  | #tree ul { | ||||||
|  | 	border-left: .2em solid #555; | ||||||
|  | } | ||||||
|  | #tree li { | ||||||
|  | 	margin-left: 1em; | ||||||
|  | 	list-style: none; | ||||||
|  | 	border-top: 1px solid #4c4c4c; | ||||||
|  | 	border-bottom: 1px solid #222; | ||||||
|  | } | ||||||
|  | #tree li:last-child { | ||||||
|  | 	border-bottom: none; | ||||||
|  | } | ||||||
|  | #treeul a.hl { | ||||||
|  | 	color: #400; | ||||||
|  | 	background: #fc4; | ||||||
|  | 	border-radius: .3em; | ||||||
|  | 	text-shadow: none; | ||||||
|  | } | ||||||
|  | #treeul a { | ||||||
|  | 	display: inline-block; | ||||||
|  | } | ||||||
|  | #treeul a+a { | ||||||
|  | 	width: calc(100% - 2em); | ||||||
|  | 	background: #333; | ||||||
|  | 	line-height: 1em; | ||||||
|  | } | ||||||
|  | #treeul a+a:hover { | ||||||
|  | 	background: #222; | ||||||
|  | 	color: #fff; | ||||||
|  | } | ||||||
|  | #treeul a:first-child { | ||||||
|  | 	font-family: monospace, monospace; | ||||||
|  | } | ||||||
|  | .dumb_loader_thing { | ||||||
|  | 	display: inline-block; | ||||||
|  | 	margin: 1em .3em 1em 1em; | ||||||
|  | 	padding: 0 1.2em 0 0; | ||||||
|  | 	font-size: 4em; | ||||||
|  | 	animation: spin 1s linear infinite; | ||||||
|  | 	position: absolute; | ||||||
|  | 	z-index: 9; | ||||||
|  | } | ||||||
|  | #files .cfg { | ||||||
|  | 	display: none; | ||||||
|  | 	font-size: 2em; | ||||||
|  | 	white-space: nowrap; | ||||||
|  | } | ||||||
|  | #files th:hover .cfg, | ||||||
|  | #files th.min .cfg { | ||||||
|  | 	display: block; | ||||||
|  | 	width: 1em; | ||||||
|  | 	border-radius: .2em; | ||||||
|  | 	margin: -1.3em auto 0 auto; | ||||||
|  | 	background: #444; | ||||||
|  | } | ||||||
|  | #files th.min .cfg { | ||||||
|  | 	margin: -.6em; | ||||||
|  | } | ||||||
|  | #files>thead>tr>th.min span { | ||||||
|  | 	position: absolute; | ||||||
|  | 	transform: rotate(270deg); | ||||||
|  | 	background: linear-gradient(90deg, rgba(68,68,68,0), rgba(68,68,68,0.5) 70%, #444); | ||||||
|  | 	margin-left: -4.6em; | ||||||
|  | 	padding: .4em; | ||||||
|  | 	top: 5.4em; | ||||||
|  | 	width: 8em; | ||||||
|  | 	text-align: right; | ||||||
|  | 	letter-spacing: .04em; | ||||||
|  | } | ||||||
|  | #files td:nth-child(2n) { | ||||||
|  | 	color: #f5a; | ||||||
|  | } | ||||||
|  | #files td.min a { | ||||||
|  | 	display: none; | ||||||
|  | } | ||||||
|  | #files tr.play td, | ||||||
|  | #files tr.play div a { | ||||||
|  | 	background: #fc4; | ||||||
|  | 	border-color: transparent; | ||||||
|  | 	color: #400; | ||||||
|  | 	text-shadow: none; | ||||||
|  | } | ||||||
|  | #files tr.play a { | ||||||
|  | 	color: inherit; | ||||||
|  | } | ||||||
|  | #files tr.play a:hover { | ||||||
|  | 	color: #300; | ||||||
|  | 	background: #fea; | ||||||
|  | } | ||||||
|  | #op_cfg { | ||||||
|  | 	max-width: none; | ||||||
|  | 	margin-right: 1.5em; | ||||||
|  | } | ||||||
|  | #op_cfg>div>a { | ||||||
|  | 	line-height: 2em; | ||||||
|  | } | ||||||
|  | #op_cfg>div>span { | ||||||
|  | 	display: inline-block; | ||||||
|  | 	padding: .2em .4em; | ||||||
|  | } | ||||||
|  | #op_cfg h3 { | ||||||
|  | 	margin: .8em 0 0 .6em; | ||||||
|  | 	padding: 0; | ||||||
|  | 	border-bottom: 1px solid #555; | ||||||
|  | } | ||||||
|  | #opdesc { | ||||||
|  | 	display: none; | ||||||
|  | } | ||||||
|  | #ops:hover #opdesc { | ||||||
|  | 	display: block; | ||||||
|  | 	background: linear-gradient(0deg,#555, #4c4c4c 80%, #444); | ||||||
|  | 	box-shadow: 0 .3em 1em #222; | ||||||
|  | 	padding: 1em; | ||||||
|  | 	border-radius: .3em; | ||||||
|  | 	position: absolute; | ||||||
|  | 	z-index: 3; | ||||||
|  | 	top: 6em; | ||||||
|  | 	right: 1.5em; | ||||||
|  | } | ||||||
|  | #ops:hover #opdesc.off { | ||||||
|  | 	display: none; | ||||||
|  | } | ||||||
|  | #opdesc code { | ||||||
|  | 	background: #3c3c3c; | ||||||
|  | 	padding: .2em .3em; | ||||||
|  | 	border-top: 1px solid #777; | ||||||
|  | 	border-radius: .3em; | ||||||
|  | 	font-family: monospace, monospace; | ||||||
|  | 	line-height: 2em; | ||||||
|  | } | ||||||
|  | #pvol, | ||||||
|  | #barbuf, | ||||||
|  | #barpos, | ||||||
|  | #u2conf label { | ||||||
|  | 	-webkit-user-select: none; | ||||||
|  | 	-moz-user-select: none; | ||||||
|  | 	-ms-user-select: none; | ||||||
|  | 	user-select: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | html.light { | ||||||
|  | 	color: #333; | ||||||
|  | 	background: #eee; | ||||||
|  | 	text-shadow: none; | ||||||
|  | } | ||||||
|  | html.light #ops, | ||||||
|  | html.light .opbox, | ||||||
|  | html.light #srch_form { | ||||||
|  | 	background: #f7f7f7; | ||||||
|  | 	box-shadow: 0 0 .3em #ddd; | ||||||
|  | 	border-color: #f7f7f7; | ||||||
|  | } | ||||||
|  | html.light #ops a.act { | ||||||
|  | 	box-shadow: 0 .2em .2em #ccc; | ||||||
|  | 	background: #fff; | ||||||
|  | 	border-color: #07a; | ||||||
|  | 	padding-top: .4em; | ||||||
|  | } | ||||||
|  | html.light #op_cfg h3 { | ||||||
|  | 	border-color: #ccc; | ||||||
|  | } | ||||||
|  | html.light .tglbtn, | ||||||
|  | html.light #tree > a + a { | ||||||
|  | 	color: #666; | ||||||
|  | 	background: #ddd; | ||||||
|  | 	box-shadow: none; | ||||||
|  | } | ||||||
|  | html.light .tglbtn:hover, | ||||||
|  | html.light #tree > a + a:hover { | ||||||
|  | 	background: #caf; | ||||||
|  | } | ||||||
|  | html.light .tglbtn.on, | ||||||
|  | html.light #tree > a + a.on { | ||||||
|  | 	background: #4a0; | ||||||
|  | 	color: #fff; | ||||||
|  | } | ||||||
|  | html.light #srv_info { | ||||||
|  | 	color: #c83; | ||||||
|  | 	text-shadow: 1px 1px 0 #fff; | ||||||
|  | } | ||||||
|  | html.light #srv_info span { | ||||||
|  | 	color: #000; | ||||||
|  | } | ||||||
|  | html.light #treeul a+a { | ||||||
|  | 	background: inherit; | ||||||
|  | 	color: #06a; | ||||||
|  | } | ||||||
|  | html.light #treeul a.hl { | ||||||
|  | 	background: #07a; | ||||||
|  | 	color: #fff; | ||||||
|  | } | ||||||
|  | html.light #tree li { | ||||||
|  | 	border-color: #ddd #fff #f7f7f7 #fff; | ||||||
|  | } | ||||||
|  | html.light #tree ul { | ||||||
|  | 	border-color: #ccc; | ||||||
|  | } | ||||||
|  | html.light a, | ||||||
|  | html.light #ops a, | ||||||
|  | html.light #files tbody div a:last-child { | ||||||
|  | 	color: #06a; | ||||||
|  | } | ||||||
|  | html.light #files tbody { | ||||||
|  | 	background: #f7f7f7; | ||||||
|  | } | ||||||
|  | html.light #files { | ||||||
|  | 	box-shadow: 0 0 .3em #ccc; | ||||||
|  | } | ||||||
|  | html.light #files thead th { | ||||||
|  | 	background: #eee; | ||||||
|  | } | ||||||
|  | html.light #files tr td { | ||||||
|  | 	border-top: 1px solid #ddd; | ||||||
|  | } | ||||||
|  | html.light #files td { | ||||||
|  | 	border-bottom: 1px solid #f7f7f7; | ||||||
|  | } | ||||||
|  | html.light #files tbody tr:last-child td { | ||||||
|  | 	border-bottom: .2em solid #ccc; | ||||||
|  | } | ||||||
|  | html.light #files td:nth-child(2n) { | ||||||
|  | 	color: #d38; | ||||||
|  | } | ||||||
|  | html.light #files tr:hover td { | ||||||
|  | 	background: #fff; | ||||||
|  | } | ||||||
|  | html.light #files tbody a.play { | ||||||
|  | 	color: #c0f; | ||||||
|  | } | ||||||
|  | html.light tr.play td { | ||||||
|  | 	background: #fc5; | ||||||
|  | } | ||||||
|  | html.light tr.play a { | ||||||
|  | 	color: #406; | ||||||
|  | } | ||||||
|  | html.light #files th:hover .cfg, | ||||||
|  | html.light #files th.min .cfg { | ||||||
|  | 	background: #ccc; | ||||||
|  | } | ||||||
|  | html.light #files > thead > tr > th.min span { | ||||||
|  | 	background: linear-gradient(90deg, rgba(204,204,204,0), rgba(204,204,204,0.5) 70%, #ccc); | ||||||
|  | } | ||||||
|  | html.light #blocked { | ||||||
|  | 	background: #eee; | ||||||
|  | } | ||||||
|  | html.light #blk_play a, | ||||||
|  | html.light #blk_abrt a { | ||||||
|  | 	background: #fff; | ||||||
|  | 	box-shadow: 0 .2em .4em #ddd; | ||||||
|  | } | ||||||
|  | html.light #widget a { | ||||||
|  | 	color: #fc5; | ||||||
|  | } | ||||||
|  | html.light #files tr.sel:hover td { | ||||||
|  | 	background: #c37; | ||||||
|  | } | ||||||
|  | html.light #files tr.sel td { | ||||||
|  | 	color: #fff; | ||||||
|  | } | ||||||
|  | html.light #files tr.sel a { | ||||||
|  | 	color: #fff; | ||||||
|  | } | ||||||
|  | html.light #files tr.sel a.play.act { | ||||||
|  | 	color: #fb0; | ||||||
|  | } | ||||||
|  | html.light input[type="checkbox"] + label { | ||||||
|  | 	color: #333; | ||||||
|  | } | ||||||
|  | html.light .opview input[type="text"] { | ||||||
|  | 	background: #fff; | ||||||
|  | 	color: #333; | ||||||
|  | 	box-shadow: 0 0 2px #888; | ||||||
|  | 	border-color: #38d; | ||||||
|  | } | ||||||
|  | html.light #ops:hover #opdesc { | ||||||
|  | 	background: #fff; | ||||||
|  | 	box-shadow: 0 .3em 1em #ccc; | ||||||
|  | } | ||||||
|  | html.light #opdesc code { | ||||||
|  | 	background: #060; | ||||||
|  | 	color: #fff; | ||||||
|  | } | ||||||
|  | html.light #u2tab a>span, | ||||||
|  | html.light #files td div span { | ||||||
|  | 	color: #000; | ||||||
|  | } | ||||||
|  | html.light #path { | ||||||
|  | 	background: #f7f7f7; | ||||||
|  | 	text-shadow: none; | ||||||
|  | 	box-shadow: 0 0 .3em #bbb; | ||||||
|  | } | ||||||
|  | html.light #path a { | ||||||
|  | 	color: #333; | ||||||
|  | } | ||||||
|  | html.light #path a:not(:last-child)::after { | ||||||
|  | 	border-color: #ccc; | ||||||
|  | 	background: none; | ||||||
|  | 	border-width: .1em .1em 0 0; | ||||||
|  | 	margin: -.2em .3em -.2em -.3em; | ||||||
|  | } | ||||||
|  | html.light #path a:hover { | ||||||
|  | 	background: none; | ||||||
|  | 	color: #60a; | ||||||
|  | } | ||||||
|  | html.light #files tbody div a { | ||||||
|  | 	color: #d38; | ||||||
|  | } | ||||||
|  | html.light #files a:hover, | ||||||
|  | html.light #files tr.sel a:hover { | ||||||
|  | 	color: #000; | ||||||
|  | 	background: #fff; | ||||||
|  | } | ||||||
|  | html.light #tree { | ||||||
|  | 	scrollbar-color: #a70 #ddd; | ||||||
|  | } | ||||||
|  | html.light #tree::-webkit-scrollbar-track, | ||||||
|  | html.light #tree::-webkit-scrollbar { | ||||||
|  | 	background: #ddd; | ||||||
|  | } | ||||||
|  | #tree::-webkit-scrollbar-thumb { | ||||||
|  | 	background: #da0; | ||||||
|  | } | ||||||
| @@ -7,59 +7,122 @@ | |||||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge"> |     <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||||
|     <meta name="viewport" content="width=device-width, initial-scale=0.8"> |     <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/browser.css{{ ts }}"> | ||||||
|     {%- if can_upload %} |  | ||||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/upload.css{{ ts }}"> |     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/upload.css{{ ts }}"> | ||||||
|     {%- endif %} |  | ||||||
| </head> | </head> | ||||||
|  |  | ||||||
| <body> | <body> | ||||||
|     {%- if can_upload %} |     <div id="ops"> | ||||||
|  |         <a href="#" data-dest="" data-desc="close submenu">---</a> | ||||||
|  |         {%- if have_up2k_idx %} | ||||||
|  |         <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> | ||||||
|  |         <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="read 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="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' %} |     {%- include 'upload.html' %} | ||||||
|     {%- endif %} |  | ||||||
|  |     <div id="op_cfg" class="opview opbox"> | ||||||
|  |         <h3>switches</h3> | ||||||
|  |         <div> | ||||||
|  |             <a id="tooltips" class="tglbtn" href="#">tooltips</a> | ||||||
|  |             <a id="lightmode" class="tglbtn" href="#">lightmode</a> | ||||||
|  |         </div> | ||||||
|  |         {%- if have_zip %} | ||||||
|  |         <h3>folder download</h3> | ||||||
|  |         <div id="arc_fmt"></div> | ||||||
|  |         {%- endif %} | ||||||
|  |         <h3>key notation</h3> | ||||||
|  |         <div id="key_notation"></div> | ||||||
|  |     </div> | ||||||
|      |      | ||||||
|     <h1 id="path"> |     <h1 id="path"> | ||||||
|  |         <a href="#" id="entree">🌲</a> | ||||||
|         {%- for n in vpnodes %} |         {%- for n in vpnodes %} | ||||||
|         <a href="/{{ n[0] }}">{{ n[1] }}</a> |         <a href="/{{ n[0] }}">{{ n[1] }}</a> | ||||||
|         {%- endfor %} |         {%- endfor %} | ||||||
|     </h1> |     </h1> | ||||||
|      |      | ||||||
|     {%- if can_read %} |     <div id="tree"> | ||||||
|     {%- if prologue %} |         <a href="#" id="detree">🍞...</a> | ||||||
|     <div id="pro" class="logue">{{ prologue }}</div> |         <a href="#" step="2" id="twobytwo">+</a> | ||||||
|     {%- endif %} |         <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="wrap"> | ||||||
|  |  | ||||||
|  |     <div id="pro" class="logue">{{ logues[0] }}</div> | ||||||
|  |  | ||||||
|     <table id="files"> |     <table id="files"> | ||||||
|         <thead> |         <thead> | ||||||
|             <tr> |             <tr> | ||||||
|                 <th></th> |                 <th name="lead"><span>c</span></th> | ||||||
|                 <th>File Name</th> |                 <th name="href"><span>File Name</span></th> | ||||||
|                 <th sort="int">File Size</th> |                 <th name="sz" sort="int"><span>Size</span></th> | ||||||
|                 <th>T</th> |                 {%- for k in taglist %} | ||||||
|                 <th>Date</th> |                     {%- 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> |             </tr> | ||||||
|         </thead> |         </thead> | ||||||
|         <tbody> |         <tbody> | ||||||
|  |  | ||||||
| {%- for f in files %} | {%- for f in files %} | ||||||
| <tr><td>{{ f[0] }}</td><td><a href="{{ f[1] }}">{{ f[2] }}</a></td><td>{{ f[3] }}</td><td>{{ f[4] }}</td><td>{{ f[5] }}</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 %} | {%- endfor %} | ||||||
|  |  | ||||||
|         </tbody> |         </tbody> | ||||||
|     </table> |     </table> | ||||||
|      |      | ||||||
|     {%- if epilogue %} |     <div id="epi" class="logue">{{ logues[1] }}</div> | ||||||
|     <div id="epi" class="logue">{{ epilogue }}</div> |  | ||||||
|     {%- endif %} |  | ||||||
|     {%- endif %} |  | ||||||
|  |  | ||||||
|     <h2><a href="?h">control-panel</a></h2> |     <h2><a href="?h">control-panel</a></h2> | ||||||
|  |  | ||||||
|  | </div> | ||||||
|  |  | ||||||
|     {%- if srv_info %} |     {%- if srv_info %} | ||||||
|     <div id="srv_info"><span>{{ srv_info }}</span></div> |     <div id="srv_info"><span>{{ srv_info }}</span></div> | ||||||
|     {%- endif %} |     {%- endif %} | ||||||
|  |  | ||||||
|     <div id="widget"> |     <div id="widget"> | ||||||
|         <div id="wtoggle">♫</div> |         <div id="wtoggle"> | ||||||
|  |             <span id="wzip"> | ||||||
|  |                 <a href="#" id="selall">sel.<br />all</a> | ||||||
|  |                 <a href="#" id="selinv">sel.<br />inv.</a> | ||||||
|  |                 <a href="#" id="selzip">zip</a> | ||||||
|  |             </span><a | ||||||
|  |                 href="#" id="wtico">♫</a> | ||||||
|  |         </div> | ||||||
|         <div id="widgeti"> |         <div id="widgeti"> | ||||||
|             <div id="pctl"><a href="#" id="bprev">⏮</a><a href="#" id="bplay">▶</a><a href="#" id="bnext">⏭</a></div> |             <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="pvol" width="288" height="38"></canvas> | ||||||
| @@ -67,16 +130,16 @@ | |||||||
|             <canvas id="barbuf"></canvas> |             <canvas id="barbuf"></canvas> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
|      |  | ||||||
|     <script src="/.cpr/util.js{{ ts }}"></script> |  | ||||||
|  |  | ||||||
|     {%- if can_read %} |     <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/browser.js{{ ts }}"></script> | ||||||
|     {%- endif %} |  | ||||||
|      |  | ||||||
|     {%- if can_upload %} |  | ||||||
|     <script src="/.cpr/up2k.js{{ ts }}"></script> |     <script src="/.cpr/up2k.js{{ ts }}"></script> | ||||||
|     {%- endif %} |     <script> | ||||||
|  |         apply_perms({{ perms }}); | ||||||
|  |     </script> | ||||||
| </body> | </body> | ||||||
|  |  | ||||||
| </html> | </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> | ||||||
| @@ -50,6 +50,9 @@ pre code:last-child { | |||||||
| pre code::before { | pre code::before { | ||||||
| 	content: counter(precode); | 	content: counter(precode); | ||||||
| 	-webkit-user-select: none; | 	-webkit-user-select: none; | ||||||
|  | 	-moz-user-select: none; | ||||||
|  | 	-ms-user-select: none; | ||||||
|  | 	user-select: none; | ||||||
| 	display: inline-block; | 	display: inline-block; | ||||||
| 	text-align: right; | 	text-align: right; | ||||||
| 	font-size: .75em; | 	font-size: .75em; | ||||||
| @@ -591,12 +594,3 @@ blink { | |||||||
| 		color: #940; | 		color: #940; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| /* |  | ||||||
| *[data-ln]:before { |  | ||||||
| 	content: attr(data-ln); |  | ||||||
| 	font-size: .8em; |  | ||||||
| 	margin: 0 .4em; |  | ||||||
| 	color: #f0c; |  | ||||||
| } |  | ||||||
| */ |  | ||||||
| @@ -138,16 +138,16 @@ var md_opt = { | |||||||
|         document.documentElement.setAttribute("class", dark ? "dark" : ""); |         document.documentElement.setAttribute("class", dark ? "dark" : ""); | ||||||
|         btn.innerHTML = "go " + (dark ? "light" : "dark"); |         btn.innerHTML = "go " + (dark ? "light" : "dark"); | ||||||
|         if (window.localStorage) |         if (window.localStorage) | ||||||
|             localStorage.setItem('darkmode', dark ? 1 : 0); |             localStorage.setItem('lightmode', dark ? 0 : 1); | ||||||
|     }; |     }; | ||||||
|     btn.onclick = toggle; |     btn.onclick = toggle; | ||||||
|     if (window.localStorage && localStorage.getItem('darkmode') == 1) |     if (window.localStorage && localStorage.getItem('lightmode') != 1) | ||||||
| 		toggle(); | 		toggle(); | ||||||
| })(); | })(); | ||||||
|  |  | ||||||
| 	</script> | 	</script> | ||||||
|     <script src="/.cpr/util.js"></script> |     <script src="/.cpr/util.js"></script> | ||||||
| 	<script src="/.cpr/deps/marked.full.js"></script> | 	<script src="/.cpr/deps/marked.js"></script> | ||||||
| 	<script src="/.cpr/md.js"></script> | 	<script src="/.cpr/md.js"></script> | ||||||
| 	{%- if edit %} | 	{%- if edit %} | ||||||
| 	<script src="/.cpr/md2.js"></script> | 	<script src="/.cpr/md2.js"></script> | ||||||
|   | |||||||
| @@ -46,7 +46,7 @@ function statify(obj) { | |||||||
|     var ua = navigator.userAgent; |     var ua = navigator.userAgent; | ||||||
|     if (ua.indexOf(') Gecko/') !== -1 && /Linux| Mac /.exec(ua)) { |     if (ua.indexOf(') Gecko/') !== -1 && /Linux| Mac /.exec(ua)) { | ||||||
|         // necessary on ff-68.7 at least |         // necessary on ff-68.7 at least | ||||||
|         var s = document.createElement('style'); |         var s = mknod('style'); | ||||||
|         s.innerHTML = '@page { margin: .5in .6in .8in .6in; }'; |         s.innerHTML = '@page { margin: .5in .6in .8in .6in; }'; | ||||||
|         console.log(s.innerHTML); |         console.log(s.innerHTML); | ||||||
|         document.head.appendChild(s); |         document.head.appendChild(s); | ||||||
| @@ -65,7 +65,7 @@ function statify(obj) { | |||||||
|         if (a > 0) |         if (a > 0) | ||||||
|             loc.push(n[a]); |             loc.push(n[a]); | ||||||
|  |  | ||||||
|         var dec = hesc(decodeURIComponent(n[a])); |         var dec = hesc(uricom_dec(n[a])[0]); | ||||||
|  |  | ||||||
|         nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>'); |         nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>'); | ||||||
|     } |     } | ||||||
| @@ -175,12 +175,12 @@ function md_plug_err(ex, js) { | |||||||
|         msg = "Line " + ln + ", " + msg; |         msg = "Line " + ln + ", " + msg; | ||||||
|         var lns = js.split('\n'); |         var lns = js.split('\n'); | ||||||
|         if (ln < lns.length) { |         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.style.cssText = 'color:#ac2;font-size:.9em;font-family:scp;display:block'; | ||||||
|             o.textContent = lns[ln - 1]; |             o.textContent = lns[ln - 1]; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     errbox = document.createElement('div'); |     errbox = mknod('div'); | ||||||
|     errbox.setAttribute('id', 'md_errbox'); |     errbox.setAttribute('id', 'md_errbox'); | ||||||
|     errbox.style.cssText = 'position:absolute;top:0;left:0;padding:1em .5em;background:#2b2b2b;color:#fc5' |     errbox.style.cssText = 'position:absolute;top:0;left:0;padding:1em .5em;background:#2b2b2b;color:#fc5' | ||||||
|     errbox.textContent = msg; |     errbox.textContent = msg; | ||||||
| @@ -524,11 +524,9 @@ dom_navtgl.onclick = function () { | |||||||
|     dom_navtgl.innerHTML = hidden ? 'show nav' : 'hide nav'; |     dom_navtgl.innerHTML = hidden ? 'show nav' : 'hide nav'; | ||||||
|     dom_nav.style.display = hidden ? 'none' : 'block'; |     dom_nav.style.display = hidden ? 'none' : 'block'; | ||||||
|  |  | ||||||
|     if (window.localStorage) |     swrite('hidenav', hidden ? 1 : 0); | ||||||
|         localStorage.setItem('hidenav', hidden ? 1 : 0); |  | ||||||
|  |  | ||||||
|     redraw(); |     redraw(); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| if (window.localStorage && localStorage.getItem('hidenav') == 1) | if (sread('hidenav') == 1) | ||||||
|     dom_navtgl.onclick(); |     dom_navtgl.onclick(); | ||||||
|   | |||||||
| @@ -1,128 +1,125 @@ | |||||||
| #toc { | #toc { | ||||||
|     display: none; | 	display: none; | ||||||
| } | } | ||||||
| #mtw { | #mtw { | ||||||
|     display: block; | 	display: block; | ||||||
|     position: fixed; | 	position: fixed; | ||||||
|     left: .5em; | 	left: .5em; | ||||||
|     bottom: 0; | 	bottom: 0; | ||||||
|     width: calc(100% - 56em); | 	width: calc(100% - 56em); | ||||||
| } | } | ||||||
| #mw { | #mw { | ||||||
|     left: calc(100% - 55em); | 	left: calc(100% - 55em); | ||||||
|     overflow-y: auto; | 	overflow-y: auto; | ||||||
|     position: fixed; | 	position: fixed; | ||||||
|     bottom: 0; | 	bottom: 0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| /* single-screen */ | /* single-screen */ | ||||||
| #mtw.preview, | #mtw.preview, | ||||||
| #mw.editor { | #mw.editor { | ||||||
|     opacity: 0; | 	opacity: 0; | ||||||
|     z-index: 1; | 	z-index: 1; | ||||||
| } | } | ||||||
| #mw.preview, | #mw.preview, | ||||||
| #mtw.editor { | #mtw.editor { | ||||||
|     z-index: 5; | 	z-index: 5; | ||||||
| } | } | ||||||
| #mtw.single, | #mtw.single, | ||||||
| #mw.single { | #mw.single { | ||||||
|     margin: 0; | 	margin: 0; | ||||||
|     left: 1em; | 	left: 1em; | ||||||
|     left: max(1em, calc((100% - 56em) / 2)); | 	left: max(1em, calc((100% - 56em) / 2)); | ||||||
| } | } | ||||||
| #mtw.single { | #mtw.single { | ||||||
|     width: 55em; | 	width: 55em; | ||||||
|     width: min(55em, calc(100% - 2em)); | 	width: min(55em, calc(100% - 2em)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| #mp { | #mp { | ||||||
|     position: relative; | 	position: relative; | ||||||
| } | } | ||||||
| #mt, #mtr { | #mt, #mtr { | ||||||
|     width: 100%; | 	width: 100%; | ||||||
|     height: calc(100% - 1px); | 	height: calc(100% - 1px); | ||||||
|     color: #444; | 	color: #444; | ||||||
|     background: #f7f7f7; | 	background: #f7f7f7; | ||||||
|     border: 1px solid #999; | 	border: 1px solid #999; | ||||||
|     outline: none; | 	outline: none; | ||||||
|     padding: 0; | 	padding: 0; | ||||||
|     margin: 0; | 	margin: 0; | ||||||
|     font-family: 'consolas', monospace, monospace; | 	font-family: 'consolas', monospace, monospace; | ||||||
|     white-space: pre-wrap; | 	white-space: pre-wrap; | ||||||
|     word-break: break-word; | 	word-break: break-word; | ||||||
|     overflow-wrap: break-word; | 	overflow-wrap: break-word; | ||||||
|     word-wrap: break-word; /*ie*/ | 	word-wrap: break-word; /*ie*/ | ||||||
|     overflow-y: scroll; | 	overflow-y: scroll; | ||||||
|     line-height: 1.3em; | 	line-height: 1.3em; | ||||||
|     font-size: .9em; | 	font-size: .9em; | ||||||
|     position: relative; | 	position: relative; | ||||||
|     scrollbar-color: #eb0 #f7f7f7; | 	scrollbar-color: #eb0 #f7f7f7; | ||||||
| } | } | ||||||
| html.dark #mt { | html.dark #mt { | ||||||
|     color: #eee; | 	color: #eee; | ||||||
|     background: #222; | 	background: #222; | ||||||
|     border: 1px solid #777; | 	border: 1px solid #777; | ||||||
|     scrollbar-color: #b80 #282828; | 	scrollbar-color: #b80 #282828; | ||||||
| } | } | ||||||
| #mtr { | #mtr { | ||||||
|     position: absolute; | 	position: absolute; | ||||||
|     top: 0; | 	top: 0; | ||||||
|     left: 0; | 	left: 0; | ||||||
| } | } | ||||||
| #save.force-save { | #save.force-save { | ||||||
|     color: #400; | 	color: #400; | ||||||
|     background: #f97; | 	background: #f97; | ||||||
|     border-radius: .15em; | 	border-radius: .15em; | ||||||
| } | } | ||||||
| html.dark #save.force-save { | html.dark #save.force-save { | ||||||
|     color: #fca; | 	color: #fca; | ||||||
|     background: #720; | 	background: #720; | ||||||
| } | } | ||||||
| #save.disabled { | #save.disabled { | ||||||
|     opacity: .4; | 	opacity: .4; | ||||||
| } | } | ||||||
| #helpbox, | #helpbox, | ||||||
| #toast { | #toast { | ||||||
|     background: #f7f7f7; | 	background: #f7f7f7; | ||||||
|     border-radius: .4em; | 	border-radius: .4em; | ||||||
|     z-index: 9001; | 	z-index: 9001; | ||||||
| } | } | ||||||
| #helpbox { | #helpbox { | ||||||
|     display: none; | 	display: none; | ||||||
|     position: fixed; | 	position: fixed; | ||||||
|     padding: 2em; | 	padding: 2em; | ||||||
|     top: 4em; | 	top: 4em; | ||||||
|     overflow-y: auto; | 	overflow-y: auto; | ||||||
|     box-shadow: 0 .5em 2em #777; | 	box-shadow: 0 .5em 2em #777; | ||||||
|     height: calc(100% - 12em); | 	height: calc(100% - 12em); | ||||||
|     left: calc(50% - 15em); | 	left: calc(50% - 15em); | ||||||
|     right: 0; | 	right: 0; | ||||||
|     width: 30em; | 	width: 30em; | ||||||
| } | } | ||||||
| #helpclose { | #helpclose { | ||||||
|     display: block; | 	display: block; | ||||||
| } | } | ||||||
| html.dark #helpbox { | html.dark #helpbox { | ||||||
|     box-shadow: 0 .5em 2em #444; | 	box-shadow: 0 .5em 2em #444; | ||||||
| } | } | ||||||
| html.dark #helpbox, | html.dark #helpbox, | ||||||
| html.dark #toast { | html.dark #toast { | ||||||
|     background: #222; | 	background: #222; | ||||||
|     border: 1px solid #079; | 	border: 1px solid #079; | ||||||
|     border-width: 1px 0; | 	border-width: 1px 0; | ||||||
| } | } | ||||||
| #toast { | #toast { | ||||||
|     font-weight: bold; | 	font-weight: bold; | ||||||
|     text-align: center; | 	text-align: center; | ||||||
|     padding: .6em 0; | 	padding: .6em 0; | ||||||
|     position: fixed; | 	position: fixed; | ||||||
|     z-index: 9001; | 	top: 30%; | ||||||
|     top: 30%; | 	transition: opacity 0.2s ease-in-out; | ||||||
|     transition: opacity 0.2s ease-in-out; | 	opacity: 1; | ||||||
|     opacity: 1; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| # mt {opacity: .5;top:1px} |  | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ var dom_sbs = ebi('sbs'); | |||||||
| var dom_nsbs = ebi('nsbs'); | var dom_nsbs = ebi('nsbs'); | ||||||
| var dom_tbox = ebi('toolsbox'); | var dom_tbox = ebi('toolsbox'); | ||||||
| var dom_ref = (function () { | var dom_ref = (function () { | ||||||
|     var d = document.createElement('div'); |     var d = mknod('div'); | ||||||
|     d.setAttribute('id', 'mtr'); |     d.setAttribute('id', 'mtr'); | ||||||
|     dom_swrap.appendChild(d); |     dom_swrap.appendChild(d); | ||||||
|     d = ebi('mtr'); |     d = ebi('mtr'); | ||||||
| @@ -71,7 +71,7 @@ var map_src = []; | |||||||
| var map_pre = []; | var map_pre = []; | ||||||
| function genmap(dom, oldmap) { | function genmap(dom, oldmap) { | ||||||
|     var find = nlines; |     var find = nlines; | ||||||
|     while (oldmap && find --> 0) { |     while (oldmap && find-- > 0) { | ||||||
|         var tmap = genmapq(dom, '*[data-ln="' + find + '"]'); |         var tmap = genmapq(dom, '*[data-ln="' + find + '"]'); | ||||||
|         if (!tmap || !tmap.length) |         if (!tmap || !tmap.length) | ||||||
|             continue; |             continue; | ||||||
| @@ -94,7 +94,7 @@ var nlines = 0; | |||||||
| var draw_md = (function () { | var draw_md = (function () { | ||||||
|     var delay = 1; |     var delay = 1; | ||||||
|     function draw_md() { |     function draw_md() { | ||||||
|         var t0 = new Date().getTime(); |         var t0 = Date.now(); | ||||||
|         var src = dom_src.value; |         var src = dom_src.value; | ||||||
|         convert_markdown(src, dom_pre); |         convert_markdown(src, dom_pre); | ||||||
|  |  | ||||||
| @@ -110,7 +110,7 @@ var draw_md = (function () { | |||||||
|  |  | ||||||
|         cls(ebi('save'), 'disabled', src == server_md); |         cls(ebi('save'), 'disabled', src == server_md); | ||||||
|  |  | ||||||
|         var t1 = new Date().getTime(); |         var t1 = Date.now(); | ||||||
|         delay = t1 - t0 > 100 ? 25 : 1; |         delay = t1 - t0 > 100 ? 25 : 1; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -252,7 +252,7 @@ function Modpoll() { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         console.log('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(); |         var xhr = new XMLHttpRequest(); | ||||||
|         xhr.modpoll = this; |         xhr.modpoll = this; | ||||||
|         xhr.open('GET', url, true); |         xhr.open('GET', url, true); | ||||||
| @@ -399,7 +399,7 @@ function save_cb() { | |||||||
|  |  | ||||||
| function run_savechk(lastmod, txt, btn, ntry) { | function run_savechk(lastmod, txt, btn, ntry) { | ||||||
|     // download the saved doc from the server and compare |     // 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(); |     var xhr = new XMLHttpRequest(); | ||||||
|     xhr.open('GET', url, true); |     xhr.open('GET', url, true); | ||||||
|     xhr.responseType = 'text'; |     xhr.responseType = 'text'; | ||||||
| @@ -455,7 +455,7 @@ function toast(autoclose, style, width, msg) { | |||||||
|         ok.parentNode.removeChild(ok); |         ok.parentNode.removeChild(ok); | ||||||
|  |  | ||||||
|     style = "width:" + width + "em;left:calc(50% - " + (width / 2) + "em);" + style; |     style = "width:" + width + "em;left:calc(50% - " + (width / 2) + "em);" + style; | ||||||
|     ok = document.createElement('div'); |     ok = mknod('div'); | ||||||
|     ok.setAttribute('id', 'toast'); |     ok.setAttribute('id', 'toast'); | ||||||
|     ok.setAttribute('style', style); |     ok.setAttribute('style', style); | ||||||
|     ok.innerHTML = msg; |     ok.innerHTML = msg; | ||||||
| @@ -1049,7 +1049,7 @@ action_stack = (function () { | |||||||
|         var p1 = from.length, |         var p1 = from.length, | ||||||
|             p2 = to.length; |             p2 = to.length; | ||||||
|  |  | ||||||
|         while (p1 --> 0 && p2 --> 0) |         while (p1-- > 0 && p2-- > 0) | ||||||
|             if (from[p1] != to[p2]) |             if (from[p1] != to[p2]) | ||||||
|                 break; |                 break; | ||||||
|  |  | ||||||
| @@ -1142,14 +1142,3 @@ action_stack = (function () { | |||||||
|         _ref: ref |         _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 .editor-toolbar.disabled-for-preview>button:not(.no-disable) { opacity: .35 } | ||||||
|  |  | ||||||
| html { | html { | ||||||
|     line-height: 1.5em; | 	line-height: 1.5em; | ||||||
| } | } | ||||||
| html, body { | html, body { | ||||||
|     margin: 0; | 	margin: 0; | ||||||
|     padding: 0; | 	padding: 0; | ||||||
|     min-height: 100%; | 	min-height: 100%; | ||||||
|     font-family: sans-serif; | 	font-family: sans-serif; | ||||||
|     background: #f7f7f7; | 	background: #f7f7f7; | ||||||
|     color: #333; | 	color: #333; | ||||||
| } | } | ||||||
| #mn { | #mn { | ||||||
|     font-weight: normal; | 	font-weight: normal; | ||||||
|     margin: 1.3em 0 .7em 1em; | 	margin: 1.3em 0 .7em 1em; | ||||||
| } | } | ||||||
| #mn a { | #mn a { | ||||||
|     color: #444; | 	color: #444; | ||||||
|     margin: 0 0 0 -.2em; | 	margin: 0 0 0 -.2em; | ||||||
|     padding: 0 0 0 .4em; | 	padding: 0 0 0 .4em; | ||||||
|     text-decoration: none; | 	text-decoration: none; | ||||||
|     /* ie: */ | 	/* ie: */ | ||||||
|     border-bottom: .1em solid #777\9; | 	border-bottom: .1em solid #777\9; | ||||||
|     margin-right: 1em\9; | 	margin-right: 1em\9; | ||||||
| } | } | ||||||
| #mn a:first-child { | #mn a:first-child { | ||||||
|     padding-left: .5em; | 	padding-left: .5em; | ||||||
| } | } | ||||||
| #mn a:last-child { | #mn a:last-child { | ||||||
|     padding-right: .5em; | 	padding-right: .5em; | ||||||
| } | } | ||||||
| #mn a:not(:last-child):after { | #mn a:not(:last-child):after { | ||||||
|     content: ''; | 	content: ''; | ||||||
|     width: 1.05em; | 	width: 1.05em; | ||||||
|     height: 1.05em; | 	height: 1.05em; | ||||||
|     margin: -.2em .3em -.2em -.4em; | 	margin: -.2em .3em -.2em -.4em; | ||||||
|     display: inline-block; | 	display: inline-block; | ||||||
|     border: 1px solid rgba(0,0,0,0.2); | 	border: 1px solid rgba(0,0,0,0.2); | ||||||
|     border-width: .2em .2em 0 0; | 	border-width: .2em .2em 0 0; | ||||||
|     transform: rotate(45deg); | 	transform: rotate(45deg); | ||||||
| } | } | ||||||
| #mn a:hover { | #mn a:hover { | ||||||
|     color: #000; | 	color: #000; | ||||||
|     text-decoration: underline; | 	text-decoration: underline; | ||||||
| } | } | ||||||
|  |  | ||||||
| html .editor-toolbar>button.disabled { | html .editor-toolbar>button.disabled { | ||||||
|     opacity: .35; | 	opacity: .35; | ||||||
|     pointer-events: none; | 	pointer-events: none; | ||||||
| } | } | ||||||
| html .editor-toolbar>button.save.force-save { | 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; | 	line-height: 1.1em; | ||||||
| } | } | ||||||
| .mdo a { | .mdo a { | ||||||
|     color: #fff; | 	color: #fff; | ||||||
|     background: #39b; | 	background: #39b; | ||||||
|     text-decoration: none; | 	text-decoration: none; | ||||||
|     padding: 0 .3em; | 	padding: 0 .3em; | ||||||
|     border: none; | 	border: none; | ||||||
|     border-bottom: .07em solid #079; | 	border-bottom: .07em solid #079; | ||||||
| } | } | ||||||
| .mdo h2 { | .mdo h2 { | ||||||
|     color: #fff; | 	color: #fff; | ||||||
|     background: #555; | 	background: #555; | ||||||
|     margin-top: 2em; | 	margin-top: 2em; | ||||||
|     border-bottom: .22em solid #999; | 	border-bottom: .22em solid #999; | ||||||
|     border-top: none; | 	border-top: none; | ||||||
| } | } | ||||||
| .mdo h1 { | .mdo h1 { | ||||||
|     color: #fff; | 	color: #fff; | ||||||
|     background: #444; | 	background: #444; | ||||||
|     font-weight: normal; | 	font-weight: normal; | ||||||
|     border-top: .4em solid #fb0; | 	border-top: .4em solid #fb0; | ||||||
|     border-bottom: .4em solid #777; | 	border-bottom: .4em solid #777; | ||||||
|     border-radius: 0 1em 0 1em; | 	border-radius: 0 1em 0 1em; | ||||||
|     margin: 3em 0 1em 0; | 	margin: 3em 0 1em 0; | ||||||
|     padding: .5em 0; | 	padding: .5em 0; | ||||||
| } | } | ||||||
| h1, h2 { | h1, h2 { | ||||||
| 	line-height: 1.5em; | 	line-height: 1.5em; | ||||||
| @@ -197,14 +187,14 @@ th { | |||||||
|  |  | ||||||
| /* mde support */ | /* mde support */ | ||||||
| .mdo { | .mdo { | ||||||
|     padding: 1em; | 	padding: 1em; | ||||||
|     background: #f7f7f7; | 	background: #f7f7f7; | ||||||
| } | } | ||||||
| html.dark .mdo { | html.dark .mdo { | ||||||
|     background: #1c1c1c; | 	background: #1c1c1c; | ||||||
| } | } | ||||||
| .CodeMirror { | .CodeMirror { | ||||||
|     background: #f7f7f7; | 	background: #f7f7f7; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -214,108 +204,108 @@ html.dark .mdo { | |||||||
| /* darkmode */ | /* darkmode */ | ||||||
| html.dark .mdo, | html.dark .mdo, | ||||||
| html.dark .CodeMirror { | html.dark .CodeMirror { | ||||||
|     border-color: #222; | 	border-color: #222; | ||||||
| } | } | ||||||
| html.dark, | html.dark, | ||||||
| html.dark body, | html.dark body, | ||||||
| html.dark .CodeMirror { | html.dark .CodeMirror { | ||||||
|     background: #222; | 	background: #222; | ||||||
|     color: #ccc; | 	color: #ccc; | ||||||
| } | } | ||||||
| html.dark .CodeMirror-cursor { | html.dark .CodeMirror-cursor { | ||||||
|     border-color: #fff; | 	border-color: #fff; | ||||||
| } | } | ||||||
| html.dark .CodeMirror-selected { | 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-selected, | ||||||
| html.dark .CodeMirror-selectedtext { | html.dark .CodeMirror-selectedtext { | ||||||
|     border-radius: .1em; | 	border-radius: .1em; | ||||||
|     background: #246; | 	background: #246; | ||||||
|     color: #fff; | 	color: #fff; | ||||||
| } | } | ||||||
| html.dark .mdo a { | html.dark .mdo a { | ||||||
|     background: #057; | 	background: #057; | ||||||
| } | } | ||||||
| html.dark .mdo h1 a, html.dark .mdo h4 a, | html.dark .mdo h1 a, html.dark .mdo h4 a, | ||||||
| html.dark .mdo h2 a, html.dark .mdo h5 a, | html.dark .mdo h2 a, html.dark .mdo h5 a, | ||||||
| html.dark .mdo h3 a, html.dark .mdo h6 a { | html.dark .mdo h3 a, html.dark .mdo h6 a { | ||||||
|     color: inherit; | 	color: inherit; | ||||||
|     background: none; | 	background: none; | ||||||
| } | } | ||||||
| html.dark pre, | html.dark pre, | ||||||
| html.dark code { | html.dark code { | ||||||
|     color: #8c0; | 	color: #8c0; | ||||||
|     background: #1a1a1a; | 	background: #1a1a1a; | ||||||
|     border: .07em solid #333; | 	border: .07em solid #333; | ||||||
| } | } | ||||||
| html.dark .mdo ul, | html.dark .mdo ul, | ||||||
| html.dark .mdo ol { | html.dark .mdo ol { | ||||||
|     border-color: #444; | 	border-color: #444; | ||||||
| } | } | ||||||
| html.dark .mdo>ul, | html.dark .mdo>ul, | ||||||
| html.dark .mdo>ol { | html.dark .mdo>ol { | ||||||
|     border-color: #555; | 	border-color: #555; | ||||||
| } | } | ||||||
| html.dark strong { | html.dark strong { | ||||||
|     color: #fff; | 	color: #fff; | ||||||
| } | } | ||||||
| html.dark p>em, | html.dark p>em, | ||||||
| html.dark li>em, | html.dark li>em, | ||||||
| html.dark td>em { | html.dark td>em { | ||||||
|     color: #f94; | 	color: #f94; | ||||||
|     border-color: #666; | 	border-color: #666; | ||||||
| } | } | ||||||
| html.dark h1 { | html.dark h1 { | ||||||
|     background: #383838; | 	background: #383838; | ||||||
|     border-top: .4em solid #b80; | 	border-top: .4em solid #b80; | ||||||
|     border-bottom: .4em solid #4c4c4c; | 	border-bottom: .4em solid #4c4c4c; | ||||||
| } | } | ||||||
| html.dark h2 { | html.dark h2 { | ||||||
|     background: #444; | 	background: #444; | ||||||
|     border-bottom: .22em solid #555; | 	border-bottom: .22em solid #555; | ||||||
| } | } | ||||||
| html.dark td, | html.dark td, | ||||||
| html.dark th { | html.dark th { | ||||||
|     border-color: #444; | 	border-color: #444; | ||||||
| } | } | ||||||
| html.dark blockquote { | html.dark blockquote { | ||||||
|     background: #282828; | 	background: #282828; | ||||||
|     border: .07em dashed #444; | 	border: .07em dashed #444; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| html.dark #mn a { | html.dark #mn a { | ||||||
|     color: #ccc; | 	color: #ccc; | ||||||
| } | } | ||||||
| html.dark #mn a:not(:last-child):after { | 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 { | html.dark .editor-toolbar { | ||||||
|     border-color: #2c2c2c; | 	border-color: #2c2c2c; | ||||||
|     background: #1c1c1c; | 	background: #1c1c1c; | ||||||
| } | } | ||||||
| html.dark .editor-toolbar>i.separator { | html.dark .editor-toolbar>i.separator { | ||||||
|     border-left: 1px solid #444; | 	border-left: 1px solid #444; | ||||||
|     border-right: 1px solid #111; | 	border-right: 1px solid #111; | ||||||
| } | } | ||||||
| html.dark .editor-toolbar>button { | html.dark .editor-toolbar>button { | ||||||
|     margin-left: -1px; border: 1px solid rgba(255,255,255,0.1); | 	margin-left: -1px; border: 1px solid rgba(255,255,255,0.1); | ||||||
|     color: #aaa; | 	color: #aaa; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| html.dark .editor-toolbar>button:hover { | html.dark .editor-toolbar>button:hover { | ||||||
|     color: #333; | 	color: #333; | ||||||
| } | } | ||||||
| html.dark .editor-toolbar>button.active { | html.dark .editor-toolbar>button.active { | ||||||
|     color: #333; | 	color: #333; | ||||||
|     border-color: #ec1; | 	border-color: #ec1; | ||||||
|     background: #c90; | 	background: #c90; | ||||||
| } | } | ||||||
| html.dark .editor-toolbar::after, | html.dark .editor-toolbar::after, | ||||||
| html.dark .editor-toolbar::before { | html.dark .editor-toolbar::before { | ||||||
|     background: none; | 	background: none; | ||||||
| } | } | ||||||
| @@ -31,12 +31,12 @@ var md_opt = { | |||||||
|  |  | ||||||
| var lightswitch = (function () { | var lightswitch = (function () { | ||||||
| 	var fun = function () { | 	var fun = function () { | ||||||
| 		var dark = !!!document.documentElement.getAttribute("class"); | 		var dark = !document.documentElement.getAttribute("class"); | ||||||
| 		document.documentElement.setAttribute("class", dark ? "dark" : ""); | 		document.documentElement.setAttribute("class", dark ? "dark" : ""); | ||||||
| 		if (window.localStorage) | 		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(); | 		fun(); | ||||||
| 	 | 	 | ||||||
| 	return fun; | 	return fun; | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ var dom_md = ebi('mt'); | |||||||
|         if (a > 0) |         if (a > 0) | ||||||
|             loc.push(n[a]); |             loc.push(n[a]); | ||||||
|  |  | ||||||
|         var dec = decodeURIComponent(n[a]).replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">"); |         var dec = uricom_dec(n[a])[0].replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">"); | ||||||
|  |  | ||||||
|         nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>'); |         nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>'); | ||||||
|     } |     } | ||||||
| @@ -71,7 +71,7 @@ var mde = (function () { | |||||||
| })(); | })(); | ||||||
|  |  | ||||||
| function set_jumpto() { | function set_jumpto() { | ||||||
|     document.querySelector('.editor-preview-side').onclick = jumpto; |     QS('.editor-preview-side').onclick = jumpto; | ||||||
| } | } | ||||||
|  |  | ||||||
| function jumpto(ev) { | function jumpto(ev) { | ||||||
| @@ -94,7 +94,7 @@ function md_changed(mde, on_srv) { | |||||||
|         window.md_saved = mde.value(); |         window.md_saved = mde.value(); | ||||||
|  |  | ||||||
|     var md_now = 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) |     if (md_now == window.md_saved) | ||||||
|         save_btn.classList.add('disabled'); |         save_btn.classList.add('disabled'); | ||||||
| @@ -105,7 +105,7 @@ function md_changed(mde, on_srv) { | |||||||
| } | } | ||||||
|  |  | ||||||
| function save(mde) { | 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')) { |     if (save_btn.classList.contains('disabled')) { | ||||||
|         alert('there is nothing to save'); |         alert('there is nothing to save'); | ||||||
|         return; |         return; | ||||||
| @@ -212,7 +212,7 @@ function save_chk() { | |||||||
|     last_modified = this.lastmod; |     last_modified = this.lastmod; | ||||||
|     md_changed(this.mde, true); |     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.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✔️'; |     ok.innerHTML = 'OK✔️'; | ||||||
|     var parent = ebi('m'); |     var parent = ebi('m'); | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ html,body,tr,th,td,#files,a { | |||||||
| 	background: none; | 	background: none; | ||||||
| 	font-weight: inherit; | 	font-weight: inherit; | ||||||
| 	font-size: inherit; | 	font-size: inherit; | ||||||
| 	padding: none; | 	padding: 0; | ||||||
| 	border: none; | 	border: none; | ||||||
| } | } | ||||||
| html { | html { | ||||||
| @@ -20,8 +20,8 @@ body { | |||||||
| 	padding-bottom: 5em; | 	padding-bottom: 5em; | ||||||
| } | } | ||||||
| #box { | #box { | ||||||
|     padding: .5em 1em; | 	padding: .5em 1em; | ||||||
|     background: #2c2c2c; | 	background: #2c2c2c; | ||||||
| } | } | ||||||
| pre { | pre { | ||||||
| 	font-family: monospace, monospace; | 	font-family: monospace, monospace; | ||||||
|   | |||||||
| @@ -13,23 +13,27 @@ | |||||||
|     <div id="wrap"> |     <div id="wrap"> | ||||||
|         <p>hello {{ this.uname }}</p> |         <p>hello {{ this.uname }}</p> | ||||||
|  |  | ||||||
|  |         {%- if rvol %} | ||||||
|         <h1>you can browse these:</h1> |         <h1>you can browse these:</h1> | ||||||
|         <ul> |         <ul> | ||||||
|             {% for mp in rvol %} |             {% for mp in rvol %} | ||||||
|             <li><a href="/{{ mp }}">/{{ mp }}</a></li> |             <li><a href="/{{ mp }}{{ url_suf }}">/{{ mp }}</a></li> | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
|         </ul> |         </ul> | ||||||
|  |         {%- endif %} | ||||||
|  |  | ||||||
|  |         {%- if wvol %} | ||||||
|         <h1>you can upload to:</h1> |         <h1>you can upload to:</h1> | ||||||
|         <ul> |         <ul> | ||||||
|             {% for mp in wvol %} |             {% for mp in wvol %} | ||||||
|             <li><a href="/{{ mp }}">/{{ mp }}</a></li> |             <li><a href="/{{ mp }}{{ url_suf }}">/{{ mp }}</a></li> | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
|         </ul> |         </ul> | ||||||
|  |         {%- endif %} | ||||||
|  |  | ||||||
|         <h1>login for more:</h1> |         <h1>login for more:</h1> | ||||||
|         <ul> |         <ul> | ||||||
|             <form method="post" enctype="multipart/form-data" action="/"> |             <form method="post" enctype="multipart/form-data" action="/{{ url_suf }}"> | ||||||
|                 <input type="hidden" name="act" value="login" /> |                 <input type="hidden" name="act" value="login" /> | ||||||
|                 <input type="password" name="cppwd" /> |                 <input type="password" name="cppwd" /> | ||||||
|                 <input type="submit" value="Login" /> |                 <input type="submit" value="Login" /> | ||||||
| @@ -38,7 +42,7 @@ | |||||||
|     </div> |     </div> | ||||||
|     <script> |     <script> | ||||||
|  |  | ||||||
| if (window.localStorage && localStorage.getItem('darkmode') == 1) | if (window.localStorage && localStorage.getItem('lightmode') != 1) | ||||||
|     document.documentElement.setAttribute("class", "dark"); |     document.documentElement.setAttribute("class", "dark"); | ||||||
|  |  | ||||||
| </script> | </script> | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,92 +1,4 @@ | |||||||
| .opview { |  | ||||||
| 	display: none; |  | ||||||
| } |  | ||||||
| .opview.act { |  | ||||||
| 	display: block; |  | ||||||
| } |  | ||||||
| #ops a { |  | ||||||
| 	color: #fc5; |  | ||||||
| 	font-size: 1.5em; |  | ||||||
| 	padding: 0 .3em; |  | ||||||
| 	margin: 0; |  | ||||||
| 	outline: none; |  | ||||||
| } |  | ||||||
| #ops a.act { |  | ||||||
| 	text-decoration: underline; |  | ||||||
| } |  | ||||||
| /* |  | ||||||
| #ops a+a:after, |  | ||||||
| #ops a:first-child:after { |  | ||||||
| 	content: 'x'; |  | ||||||
| 	color: #282828; |  | ||||||
| 	text-shadow: 0 0 .08em #01a7e1; |  | ||||||
| 	margin-left: .3em; |  | ||||||
| 	position: relative; |  | ||||||
| } |  | ||||||
| #ops a+a:before { |  | ||||||
| 	content: 'x'; |  | ||||||
| 	color: #282828; |  | ||||||
| 	text-shadow: 0 0 .08em #ff3f1a; |  | ||||||
| 	margin-right: .3em; |  | ||||||
| 	margin-left: -.3em; |  | ||||||
| } |  | ||||||
| #ops a:last-child:after { |  | ||||||
| 	content: ''; |  | ||||||
| } |  | ||||||
| #ops a.act:before, |  | ||||||
| #ops a.act:after { |  | ||||||
| 	text-decoration: none !important; |  | ||||||
| } |  | ||||||
| */ |  | ||||||
| #ops i { |  | ||||||
| 	font-size: 1.5em; |  | ||||||
| } |  | ||||||
| #ops i:before { |  | ||||||
| 	content: 'x'; |  | ||||||
| 	color: #282828; |  | ||||||
| 	text-shadow: 0 0 .08em #01a7e1; |  | ||||||
| 	position: relative; |  | ||||||
| } |  | ||||||
| #ops i:after { |  | ||||||
| 	content: 'x'; |  | ||||||
| 	color: #282828; |  | ||||||
| 	text-shadow: 0 0 .08em #ff3f1a; |  | ||||||
| 	margin-left: -.35em; |  | ||||||
| 	font-size: 1.05em; |  | ||||||
| } |  | ||||||
| #ops, |  | ||||||
| .opbox { |  | ||||||
| 	border: 1px solid #3a3a3a; |  | ||||||
| 	box-shadow: 0 0 1em #222 inset; |  | ||||||
| } |  | ||||||
| #ops { |  | ||||||
| 	display: none; |  | ||||||
| 	background: #333; |  | ||||||
| 	margin: 1.7em 1.5em 0 1.5em; |  | ||||||
| 	padding: .3em .6em; |  | ||||||
| 	border-radius: .3em; |  | ||||||
| 	border-width: .15em 0; |  | ||||||
| } |  | ||||||
| .opbox { |  | ||||||
| 	background: #2d2d2d; |  | ||||||
| 	margin: 1.5em 0 0 0; |  | ||||||
| 	padding: .5em; |  | ||||||
| 	border-radius: 0 1em 1em 0; |  | ||||||
| 	border-width: .15em .3em .3em 0; |  | ||||||
| 	max-width: 40em; |  | ||||||
| } |  | ||||||
| .opbox input { |  | ||||||
| 	margin: .5em; |  | ||||||
| } |  | ||||||
| .opbox input[type=text] { |  | ||||||
| 	color: #fff; |  | ||||||
| 	background: #383838; |  | ||||||
| 	border: none; |  | ||||||
| 	box-shadow: 0 0 .3em #222; |  | ||||||
| 	border-bottom: 1px solid #fc5; |  | ||||||
| 	border-radius: .2em; |  | ||||||
| 	padding: .2em .3em; |  | ||||||
| } |  | ||||||
| #op_up2k { | #op_up2k { | ||||||
| 	padding: 0 1em 1em 1em; | 	padding: 0 1em 1em 1em; | ||||||
| } | } | ||||||
| @@ -94,6 +6,9 @@ | |||||||
| 	position: absolute; | 	position: absolute; | ||||||
| 	top: 0; | 	top: 0; | ||||||
| 	left: 0; | 	left: 0; | ||||||
|  | 	width: 2px; | ||||||
|  | 	height: 2px; | ||||||
|  | 	overflow: hidden; | ||||||
| } | } | ||||||
| #u2form input { | #u2form input { | ||||||
| 	background: #444; | 	background: #444; | ||||||
| @@ -104,10 +19,10 @@ | |||||||
| 	color: #f87; | 	color: #f87; | ||||||
| 	padding: .5em; | 	padding: .5em; | ||||||
| } | } | ||||||
| #u2form { | #u2err.msg { | ||||||
| 	width: 2px; | 	color: #999; | ||||||
| 	height: 2px; | 	padding: .5em; | ||||||
| 	overflow: hidden; | 	font-size: .9em; | ||||||
| } | } | ||||||
| #u2btn { | #u2btn { | ||||||
| 	color: #eee; | 	color: #eee; | ||||||
| @@ -117,17 +32,32 @@ | |||||||
| 	background: linear-gradient(to bottom, #367 0%, #489 50%, #38788a 51%, #367 100%); | 	background: linear-gradient(to bottom, #367 0%, #489 50%, #38788a 51%, #367 100%); | ||||||
| 	filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#489', endColorstr='#38788a', GradientType=0); | 	filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#489', endColorstr='#38788a', GradientType=0); | ||||||
| 	text-decoration: none; | 	text-decoration: none; | ||||||
| 	line-height: 1.5em; | 	line-height: 1.3em; | ||||||
| 	border: 1px solid #222; | 	border: 1px solid #222; | ||||||
| 	border-radius: .4em; | 	border-radius: .4em; | ||||||
| 	text-align: center; | 	text-align: center; | ||||||
| 	font-size: 2em; | 	font-size: 1.5em; | ||||||
| 	margin: 1em auto; | 	margin: .5em auto; | ||||||
| 	padding: 1em 0; | 	padding: .8em 0; | ||||||
| 	width: 12em; | 	width: 16em; | ||||||
| 	cursor: pointer; | 	cursor: pointer; | ||||||
| 	box-shadow: .4em .4em 0 #111; | 	box-shadow: .4em .4em 0 #111; | ||||||
| } | } | ||||||
|  | #op_up2k.srch #u2btn { | ||||||
|  | 	background: linear-gradient(to bottom, #ca3 0%, #fd8 50%, #fc6 51%, #b92 100%); | ||||||
|  | 	text-shadow: 1px 1px 1px #fc6; | ||||||
|  | 	color: #333; | ||||||
|  | } | ||||||
|  | #u2conf #u2btn { | ||||||
|  | 	margin: -1.5em 0; | ||||||
|  | 	padding: .8em 0; | ||||||
|  | 	width: 100%; | ||||||
|  | 	max-width: 12em; | ||||||
|  | 	display: inline-block; | ||||||
|  | } | ||||||
|  | #u2conf #u2btn_cw { | ||||||
|  | 	text-align: right; | ||||||
|  | } | ||||||
| #u2notbtn { | #u2notbtn { | ||||||
| 	display: none; | 	display: none; | ||||||
| 	text-align: center; | 	text-align: center; | ||||||
| @@ -142,6 +72,9 @@ | |||||||
| 	width: calc(100% - 2em); | 	width: calc(100% - 2em); | ||||||
| 	max-width: 100em; | 	max-width: 100em; | ||||||
| } | } | ||||||
|  | #op_up2k.srch #u2tab { | ||||||
|  | 	max-width: none; | ||||||
|  | } | ||||||
| #u2tab td { | #u2tab td { | ||||||
| 	border: 1px solid #ccc; | 	border: 1px solid #ccc; | ||||||
| 	border-width: 0 0px 1px 0; | 	border-width: 0 0px 1px 0; | ||||||
| @@ -149,16 +82,61 @@ | |||||||
| } | } | ||||||
| #u2tab td:nth-child(2) { | #u2tab td:nth-child(2) { | ||||||
| 	width: 5em; | 	width: 5em; | ||||||
|  | 	white-space: nowrap; | ||||||
| } | } | ||||||
| #u2tab td:nth-child(3) { | #u2tab td:nth-child(3) { | ||||||
| 	width: 40%; | 	width: 40%; | ||||||
| } | } | ||||||
| #u2tab tr+tr:hover td { | #op_up2k.srch #u2tab td:nth-child(3) { | ||||||
|  | 	font-family: sans-serif; | ||||||
|  | 	width: auto; | ||||||
|  | } | ||||||
|  | #u2tab tbody tr:hover td { | ||||||
| 	background: #222; | 	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 { | #u2conf { | ||||||
| 	margin: 1em auto; | 	margin: 1em auto; | ||||||
| 	width: 26em; | 	width: 30em; | ||||||
|  | } | ||||||
|  | #u2conf.has_btn { | ||||||
|  | 	width: 48em; | ||||||
| } | } | ||||||
| #u2conf * { | #u2conf * { | ||||||
| 	text-align: center; | 	text-align: center; | ||||||
| @@ -169,12 +147,16 @@ | |||||||
| 	outline: none; | 	outline: none; | ||||||
| } | } | ||||||
| #u2conf .txtbox { | #u2conf .txtbox { | ||||||
| 	width: 4em; | 	width: 3em; | ||||||
| 	color: #fff; | 	color: #fff; | ||||||
| 	background: #444; | 	background: #444; | ||||||
| 	border: 1px solid #777; | 	border: 1px solid #777; | ||||||
| 	font-size: 1.2em; | 	font-size: 1.2em; | ||||||
| 	padding: .15em 0; | 	padding: .15em 0; | ||||||
|  | 	height: 1.05em; | ||||||
|  | } | ||||||
|  | #u2conf .txtbox.err { | ||||||
|  | 	background: #922; | ||||||
| } | } | ||||||
| #u2conf a { | #u2conf a { | ||||||
| 	color: #fff; | 	color: #fff; | ||||||
| @@ -183,45 +165,133 @@ | |||||||
| 	border-radius: .1em; | 	border-radius: .1em; | ||||||
| 	font-size: 1.5em; | 	font-size: 1.5em; | ||||||
| 	padding: .1em 0; | 	padding: .1em 0; | ||||||
| 	margin: 0 -.25em; | 	margin: 0 -1px; | ||||||
| 	width: 1.5em; | 	width: 1.5em; | ||||||
| 	height: 1em; | 	height: 1em; | ||||||
| 	display: inline-block; | 	display: inline-block; | ||||||
| 	position: relative; | 	position: relative; | ||||||
| 	line-height: 1em; | 	bottom: -0.08em; | ||||||
| 	bottom: -.08em; |  | ||||||
| } | } | ||||||
| #u2conf input+a { | #u2conf input+a { | ||||||
| 	background: #d80; | 	background: #d80; | ||||||
| } | } | ||||||
|  | #u2conf label { | ||||||
|  | 	font-size: 1.6em; | ||||||
|  | 	width: 2em; | ||||||
|  | 	height: 1em; | ||||||
|  | 	padding: .4em 0; | ||||||
|  | 	display: block; | ||||||
|  | 	border-radius: .25em; | ||||||
|  | } | ||||||
|  | #u2conf input[type="checkbox"] { | ||||||
|  | 	position: relative; | ||||||
|  | 	opacity: .02; | ||||||
|  | 	top: 2em; | ||||||
|  | } | ||||||
| #u2conf input[type="checkbox"]+label { | #u2conf input[type="checkbox"]+label { | ||||||
| 	color: #f5a; | 	position: relative; | ||||||
|  | 	background: #603; | ||||||
|  | 	border-bottom: .2em solid #a16; | ||||||
|  | 	box-shadow: 0 .1em .3em #a00 inset; | ||||||
| } | } | ||||||
| #u2conf input[type="checkbox"]:checked+label { | #u2conf input[type="checkbox"]:checked+label { | ||||||
| 	color: #fc5; | 	background: #6a1; | ||||||
|  | 	border-bottom: .2em solid #efa; | ||||||
|  | 	box-shadow: 0 .1em .5em #0c0; | ||||||
|  | } | ||||||
|  | #u2conf input[type="checkbox"]+label:hover { | ||||||
|  | 	box-shadow: 0 .1em .3em #fb0; | ||||||
|  | 	border-color: #fb0; | ||||||
|  | } | ||||||
|  | #op_up2k.srch #u2conf td:nth-child(1)>*, | ||||||
|  | #op_up2k.srch #u2conf td:nth-child(2)>*, | ||||||
|  | #op_up2k.srch #u2conf td:nth-child(3)>* { | ||||||
|  | 	background: #777; | ||||||
|  | 	border-color: #ccc; | ||||||
|  | 	box-shadow: none; | ||||||
|  | 	opacity: .2; | ||||||
|  | } | ||||||
|  | #u2cdesc { | ||||||
|  | 	position: absolute; | ||||||
|  | 	width: 34em; | ||||||
|  | 	left: calc(50% - 15em); | ||||||
|  | 	background: #222; | ||||||
|  | 	border: 0 solid #555; | ||||||
|  | 	text-align: center; | ||||||
|  | 	overflow: hidden; | ||||||
|  | 	margin: 0 -2em; | ||||||
|  | 	padding: 0 1em; | ||||||
|  | 	height: 0; | ||||||
|  | 	opacity: .1; | ||||||
|  | 	transition: all 0.14s ease-in-out; | ||||||
|  | 	box-shadow: 0 .2em .5em #222; | ||||||
|  | 	border-radius: .4em; | ||||||
|  | 	z-index: 1; | ||||||
|  | } | ||||||
|  | #u2cdesc.show { | ||||||
|  | 	padding: 1em; | ||||||
|  | 	height: auto; | ||||||
|  | 	border-width: .2em 0; | ||||||
|  | 	opacity: 1; | ||||||
| } | } | ||||||
| #u2foot { | #u2foot { | ||||||
| 	color: #fff; | 	color: #fff; | ||||||
| 	font-style: italic; | 	font-style: italic; | ||||||
| } | } | ||||||
|  | #u2footfoot { | ||||||
|  | 	margin-bottom: -1em; | ||||||
|  | } | ||||||
| .prog { | .prog { | ||||||
| 	font-family: monospace; | 	font-family: monospace; | ||||||
| } | } | ||||||
| .prog>div { | #u2tab a>span { | ||||||
| 	display: inline-block; | 	font-weight: bold; | ||||||
| 	position: relative; | 	font-style: italic; | ||||||
| 	overflow: hidden; | 	color: #fff; | ||||||
| 	margin: 0; | 	padding-left: .2em; | ||||||
| 	padding: 0; |  | ||||||
| 	height: 1.1em; |  | ||||||
| 	margin-bottom: -.15em; |  | ||||||
| 	box-shadow: -1px -1px 0 inset rgba(255,255,255,0.1); |  | ||||||
| } | } | ||||||
| .prog>div>div { | #u2cleanup { | ||||||
| 	width: 0%; | 	float: right; | ||||||
| 	position: absolute; | 	margin-bottom: -.3em; | ||||||
| 	left: 0; | } | ||||||
| 	top: 0; |  | ||||||
| 	bottom: 0; |  | ||||||
| 	background: #0a0; |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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 #u2cdesc { | ||||||
|  | 	background: #fff; | ||||||
|  | 	border: none; | ||||||
|  | } | ||||||
|  | html.light #op_up2k.srch #u2btn { | ||||||
|  | 	border-color: #a80; | ||||||
|  | } | ||||||
|  | html.light #u2foot { | ||||||
|  | 	color: #000; | ||||||
|  | } | ||||||
|  | html.light #u2tab tbody tr:hover td { | ||||||
|  | 	background: #fff; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,14 +1,7 @@ | |||||||
|     <div id="ops"><a |  | ||||||
|         href="#" data-dest="">---</a><i></i><a |  | ||||||
|         href="#" data-dest="up2k">up2k</a><i></i><a |  | ||||||
|         href="#" data-dest="bup">bup</a><i></i><a |  | ||||||
|         href="#" data-dest="mkdir">mkdir</a><i></i><a |  | ||||||
|         href="#" data-dest="new_md">new.md</a><i></i><a |  | ||||||
|         href="#" data-dest="msg">msg</a></div> |  | ||||||
|  |  | ||||||
|     <div id="op_bup" class="opview opbox act"> |     <div id="op_bup" class="opview opbox act"> | ||||||
|         <div id="u2err"></div> |         <div id="u2err"></div> | ||||||
|         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}"> |         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||||
|             <input type="hidden" name="act" value="bput" /> |             <input type="hidden" name="act" value="bput" /> | ||||||
|             <input type="file" name="f" multiple><br /> |             <input type="file" name="f" multiple><br /> | ||||||
|             <input type="submit" value="start upload"> |             <input type="submit" value="start upload"> | ||||||
| @@ -16,7 +9,7 @@ | |||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|     <div id="op_mkdir" class="opview opbox act"> |     <div id="op_mkdir" class="opview opbox act"> | ||||||
|         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}"> |         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||||
|             <input type="hidden" name="act" value="mkdir" /> |             <input type="hidden" name="act" value="mkdir" /> | ||||||
|             <input type="text" name="name" size="30"> |             <input type="text" name="name" size="30"> | ||||||
|             <input type="submit" value="mkdir"> |             <input type="submit" value="mkdir"> | ||||||
| @@ -24,17 +17,17 @@ | |||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|     <div id="op_new_md" class="opview opbox"> |     <div id="op_new_md" class="opview opbox"> | ||||||
|         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}"> |         <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}"> | ||||||
|             <input type="hidden" name="act" value="new_md" /> |             <input type="hidden" name="act" value="new_md" /> | ||||||
|             <input type="text" name="name" size="30"> |             <input type="text" name="name" size="30"> | ||||||
|             <input type="submit" value="create doc"> |             <input type="submit" value="create doc"> | ||||||
|         </form> |         </form> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|     <div id="op_msg" class="opview opbox"> |     <div id="op_msg" class="opview opbox act"> | ||||||
|         <form method="post" enctype="application/x-www-form-urlencoded" accept-charset="utf-8" action="/{{ vdir }}"> |         <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="text" name="msg" size="30"> | ||||||
|             <input type="submit" value="send"> |             <input type="submit" value="send msg"> | ||||||
|         </form> |         </form> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
| @@ -43,40 +36,68 @@ | |||||||
|  |  | ||||||
|             <table id="u2conf"> |             <table id="u2conf"> | ||||||
|                 <tr> |                 <tr> | ||||||
|                     <td>parallel uploads</td> |                     <td><br />parallel uploads:</td> | ||||||
|                 </tr> |                     <td rowspan="2"> | ||||||
|                 <tr> |  | ||||||
|                     <td> |  | ||||||
|                         <a href="#" id="nthread_sub">–</a> |  | ||||||
|                         <input class="txtbox" id="nthread" value="2" /> |  | ||||||
|                         <a href="#" id="nthread_add">+</a> |  | ||||||
|                     </td> |  | ||||||
|                     <td rowspan="2" style="padding-left:1.5em"> |  | ||||||
|                         <input type="checkbox" id="multitask" /> |                         <input type="checkbox" id="multitask" /> | ||||||
|                         <label for="multitask">hash while<br />uploading</label> |                         <label for="multitask" alt="continue hashing other files while uploading">🏃</label> | ||||||
|                     </td> |                     </td> | ||||||
|                     <td rowspan="2"> |                     <td rowspan="2"> | ||||||
|                         <input type="checkbox" id="ask_up" /> |                         <input type="checkbox" id="ask_up" /> | ||||||
|                         <label for="ask_up">ask for<br />confirmation</label> |                         <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><br />  | ||||||
|                     </td> |                     </td> | ||||||
|                 </tr> |                 </tr> | ||||||
|             </table> |             </table> | ||||||
|  |  | ||||||
|  |             <div id="u2cdesc"></div> | ||||||
|  |  | ||||||
|             <div id="u2notbtn"></div> |             <div id="u2notbtn"></div> | ||||||
|  |  | ||||||
|             <div id="u2btn"> |             <div id="u2btn_ct"> | ||||||
|                 drop files here<br /> |                 <div id="u2btn"> | ||||||
|                 (or click me) |                     <span id="u2bm"></span><br /> | ||||||
|  |                     drag/drop files<br /> | ||||||
|  |                     and folders here<br /> | ||||||
|  |                     (or click me) | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |  | ||||||
|  |             <div id="u2cards"> | ||||||
|  |                 <a href="#" act="ok">ok <span>0</span></a><a | ||||||
|  |                 href="#" act="ng">ng <span>0</span></a><a | ||||||
|  |                 href="#" act="done">done <span>0</span></a><a | ||||||
|  |                 href="#" act="bz" class="act">busy <span>0</span></a><a | ||||||
|  |                 href="#" act="q">que <span>0</span></a> | ||||||
|             </div> |             </div> | ||||||
|  |  | ||||||
|             <table id="u2tab"> |             <table id="u2tab"> | ||||||
|                 <tr> |                 <thead> | ||||||
|                     <td>filename</td> |                     <tr> | ||||||
|                     <td>status</td> |                         <td>filename</td> | ||||||
|                     <td>progress</td> |                         <td>status</td> | ||||||
|                 </tr> |                         <td>progress<a href="#" id="u2cleanup">cleanup</a></td> | ||||||
|  |                     </tr> | ||||||
|  |                 </thead> | ||||||
|  |                 <tbody></tbody> | ||||||
|             </table> |             </table> | ||||||
|  |  | ||||||
|             <p id="u2foot"></p> |             <p id="u2foot"></p> | ||||||
|             <p>( if you don't need lastmod timestamps, resumable uploads or progress bars just use the <a href="#" id="u2nope">basic uploader</a>)</p> |             <p id="u2footfoot" data-perm="write">( you can use the <a href="#" id="u2nope">basic uploader</a> if you don't need lastmod timestamps, resumable uploads, or progress bars )</p> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
| @@ -1,5 +1,15 @@ | |||||||
| "use strict"; | "use strict"; | ||||||
|  |  | ||||||
|  | if (!window['console']) | ||||||
|  |     window['console'] = { | ||||||
|  |         "log": function (msg) { } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | var clickev = window.Touch ? 'touchstart' : 'click', | ||||||
|  |     ANDROID = /(android)/i.test(navigator.userAgent); | ||||||
|  |  | ||||||
|  |  | ||||||
| // error handler for mobile devices | // error handler for mobile devices | ||||||
| function hcroak(msg) { | function hcroak(msg) { | ||||||
|     document.body.innerHTML = msg; |     document.body.innerHTML = msg; | ||||||
| @@ -23,6 +33,7 @@ function esc(txt) { | |||||||
| } | } | ||||||
| function vis_exh(msg, url, lineNo, columnNo, error) { | function vis_exh(msg, url, lineNo, columnNo, error) { | ||||||
|     window.onerror = undefined; |     window.onerror = undefined; | ||||||
|  |     window['vis_exh'] = null; | ||||||
|     var html = ['<h1>you hit a bug!</h1><p>please screenshot this error and send me a copy arigathanks gozaimuch (ed/irc.rizon.net or ed#2644)</p><p>', |     var html = ['<h1>you hit a bug!</h1><p>please screenshot this error and send me a copy arigathanks gozaimuch (ed/irc.rizon.net or ed#2644)</p><p>', | ||||||
|         esc(String(msg)), '</p><p>', esc(url + ' @' + lineNo + ':' + columnNo), '</p>']; |         esc(String(msg)), '</p><p>', esc(url + ' @' + lineNo + ':' + columnNo), '</p>']; | ||||||
|  |  | ||||||
| @@ -39,8 +50,25 @@ function vis_exh(msg, url, lineNo, columnNo, error) { | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| function ebi(id) { | var ebi = document.getElementById.bind(document), | ||||||
|     return document.getElementById(id); |     QS = document.querySelector.bind(document), | ||||||
|  |     QSA = document.querySelectorAll.bind(document), | ||||||
|  |     mknod = document.createElement.bind(document); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function ev(e) { | ||||||
|  |     e = e || window.event; | ||||||
|  |     if (!e) | ||||||
|  |         return; | ||||||
|  |  | ||||||
|  |     if (e.preventDefault) | ||||||
|  |         e.preventDefault() | ||||||
|  |  | ||||||
|  |     if (e.stopPropagation) | ||||||
|  |         e.stopPropagation(); | ||||||
|  |  | ||||||
|  |     e.returnValue = false; | ||||||
|  |     return e; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -64,7 +92,7 @@ if (!String.startsWith) { | |||||||
| // https://stackoverflow.com/a/950146 | // https://stackoverflow.com/a/950146 | ||||||
| function import_js(url, cb) { | function import_js(url, cb) { | ||||||
|     var head = document.head || document.getElementsByTagName('head')[0]; |     var head = document.head || document.getElementsByTagName('head')[0]; | ||||||
|     var script = document.createElement('script'); |     var script = mknod('script'); | ||||||
|     script.type = 'text/javascript'; |     script.type = 'text/javascript'; | ||||||
|     script.src = url; |     script.src = url; | ||||||
|  |  | ||||||
| @@ -75,35 +103,411 @@ function import_js(url, cb) { | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| function sortTable(table, col) { | var crctab = (function () { | ||||||
|     var tb = table.tBodies[0], // use `<tbody>` to ignore `<thead>` and `<tfoot>` rows |     var c, tab = []; | ||||||
|  |     for (var n = 0; n < 256; n++) { | ||||||
|  |         c = n; | ||||||
|  |         for (var k = 0; k < 8; k++) { | ||||||
|  |             c = ((c & 1) ? (0xEDB88320 ^ (c >>> 1)) : (c >>> 1)); | ||||||
|  |         } | ||||||
|  |         tab[n] = c; | ||||||
|  |     } | ||||||
|  |     return tab; | ||||||
|  | })(); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function crc32(str) { | ||||||
|  |     var crc = 0 ^ (-1); | ||||||
|  |     for (var i = 0; i < str.length; i++) { | ||||||
|  |         crc = (crc >>> 8) ^ crctab[(crc ^ str.charCodeAt(i)) & 0xFF]; | ||||||
|  |     } | ||||||
|  |     return ((crc ^ (-1)) >>> 0).toString(16); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function clmod(obj, cls, add) { | ||||||
|  |     var re = new RegExp('\\s*\\b' + cls + '\\s*\\b', 'g'); | ||||||
|  |     if (add == 't') | ||||||
|  |         add = !re.test(obj.className); | ||||||
|  |  | ||||||
|  |     obj.className = obj.className.replace(re, ' ') + (add ? ' ' + cls : ''); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function sortfiles(nodes) { | ||||||
|  |     var sopts = jread('fsort', [["lead", -1, ""], ["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.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) | ||||||
|  |                         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); | ||||||
|  |     } | ||||||
|  |     return nodes; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function sortTable(table, col, cb) { | ||||||
|  |     var tb = table.tBodies[0], | ||||||
|         th = table.tHead.rows[0].cells, |         th = table.tHead.rows[0].cells, | ||||||
|         tr = Array.prototype.slice.call(tb.rows, 0), |         tr = Array.prototype.slice.call(tb.rows, 0), | ||||||
|         i, reverse = th[col].className == 'sort1' ? -1 : 1; |         i, reverse = th[col].className.indexOf('sort1') !== -1 ? -1 : 1; | ||||||
|     for (var a = 0, thl = th.length; a < thl; a++) |     for (var a = 0, thl = th.length; a < thl; a++) | ||||||
|         th[a].className = ''; |         th[a].className = th[a].className.replace(/ *sort-?1 */, " "); | ||||||
|     th[col].className = 'sort' + reverse; |     th[col].className += ' sort' + reverse; | ||||||
|     var stype = th[col].getAttribute('sort'); |     var stype = th[col].getAttribute('sort'); | ||||||
|     tr = tr.sort(function (a, b) { |     try { | ||||||
|         var v1 = a.cells[col].textContent.trim(); |         var nrules = [], rules = jread("fsort", []); | ||||||
|         var v2 = b.cells[col].textContent.trim(); |         rules.unshift([th[col].getAttribute('name'), reverse, stype || '']); | ||||||
|         if (stype == 'int') { |         for (var a = 0; a < rules.length; a++) { | ||||||
|             v1 = parseInt(v1.replace(/,/g, '')); |             var add = true; | ||||||
|             v2 = parseInt(v2.replace(/,/g, '')); |             for (var b = 0; b < a; b++) | ||||||
|             return reverse * (v1 - v2); |                 if (rules[a][0] == rules[b][0]) | ||||||
|  |                     add = false; | ||||||
|  |  | ||||||
|  |             if (add) | ||||||
|  |                 nrules.push(rules[a]); | ||||||
|  |  | ||||||
|  |             if (nrules.length >= 10) | ||||||
|  |                 break; | ||||||
|         } |         } | ||||||
|         return reverse * (v1.localeCompare(v2)); |         jwrite("fsort", nrules); | ||||||
|  |     } | ||||||
|  |     catch (ex) { | ||||||
|  |         console.log("failed to persist sort rules, resetting: " + ex); | ||||||
|  |         jwrite("fsort", null); | ||||||
|  |     } | ||||||
|  |     var vl = []; | ||||||
|  |     for (var a = 0; a < tr.length; a++) { | ||||||
|  |         var cell = tr[a].cells[col]; | ||||||
|  |         if (!cell) { | ||||||
|  |             vl.push([null, a]); | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  |         var v = cell.getAttribute('sortv') || cell.textContent.trim(); | ||||||
|  |         if (stype == 'int') { | ||||||
|  |             v = parseInt(v.replace(/[, ]/g, '')) || 0; | ||||||
|  |         } | ||||||
|  |         vl.push([v, a]); | ||||||
|  |     } | ||||||
|  |     vl.sort(function (a, b) { | ||||||
|  |         a = a[0]; | ||||||
|  |         b = b[0]; | ||||||
|  |         if (a === null) | ||||||
|  |             return -1; | ||||||
|  |         if (b === null) | ||||||
|  |             return 1; | ||||||
|  |  | ||||||
|  |         if (stype == 'int') { | ||||||
|  |             return reverse * (a - b); | ||||||
|  |         } | ||||||
|  |         return reverse * (a.localeCompare(b)); | ||||||
|     }); |     }); | ||||||
|     for (i = 0; i < tr.length; ++i) tb.appendChild(tr[i]); |     for (i = 0; i < tr.length; ++i) tb.appendChild(tr[vl[i][1]]); | ||||||
|  |     if (cb) cb(); | ||||||
| } | } | ||||||
| function makeSortable(table) { | function makeSortable(table, cb) { | ||||||
|     var th = table.tHead, i; |     var th = table.tHead, i; | ||||||
|     th && (th = th.rows[0]) && (th = th.cells); |     th && (th = th.rows[0]) && (th = th.cells); | ||||||
|     if (th) i = th.length; |     if (th) i = th.length; | ||||||
|     else return; // if no `<thead>` then do nothing |     else return; // if no `<thead>` then do nothing | ||||||
|     while (--i >= 0) (function (i) { |     while (--i >= 0) (function (i) { | ||||||
|         th[i].onclick = function () { |         th[i].onclick = function (e) { | ||||||
|             sortTable(table, i); |             ev(e); | ||||||
|  |             sortTable(table, i, cb); | ||||||
|         }; |         }; | ||||||
|     }(i)); |     }(i)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | (function () { | ||||||
|  |     var ops = QSA('#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 = QS('.opview.act input:not([type="hidden"])') | ||||||
|  |     if (input) | ||||||
|  |         input.focus(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function goto(dest) { | ||||||
|  |     var obj = QSA('.opview.act'); | ||||||
|  |     for (var a = obj.length - 1; a >= 0; a--) | ||||||
|  |         clmod(obj[a], 'act'); | ||||||
|  |  | ||||||
|  |     obj = QSA('#ops>a'); | ||||||
|  |     for (var a = obj.length - 1; a >= 0; a--) | ||||||
|  |         clmod(obj[a], 'act'); | ||||||
|  |  | ||||||
|  |     if (dest) { | ||||||
|  |         var ui = ebi('op_' + dest); | ||||||
|  |         clmod(ui, 'act', true); | ||||||
|  |         QS('#ops>a[data-dest=' + dest + ']').className += " 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 = '/'; | ||||||
|  |     if (rp && rp.charAt(0) == '/') | ||||||
|  |         rp = rp.slice(1); | ||||||
|  |  | ||||||
|  |     while (rp) { | ||||||
|  |         var link = rp; | ||||||
|  |         var ofs = rp.indexOf('/'); | ||||||
|  |         if (ofs === -1) { | ||||||
|  |             rp = null; | ||||||
|  |         } | ||||||
|  |         else { | ||||||
|  |             link = rp.slice(0, ofs + 1); | ||||||
|  |             rp = rp.slice(ofs + 1); | ||||||
|  |         } | ||||||
|  |         var vlink = link; | ||||||
|  |         if (link.indexOf('/') !== -1) | ||||||
|  |             vlink = link.slice(0, -1) + '<span>/</span>'; | ||||||
|  |  | ||||||
|  |         ret.push('<a href="' + apath + link + '">' + vlink + '</a>'); | ||||||
|  |         apath += link; | ||||||
|  |     } | ||||||
|  |     return ret; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function uricom_enc(txt, do_fb_enc) { | ||||||
|  |     try { | ||||||
|  |         return encodeURIComponent(txt); | ||||||
|  |     } | ||||||
|  |     catch (ex) { | ||||||
|  |         console.log("uce-err [" + txt + "]"); | ||||||
|  |         if (do_fb_enc) | ||||||
|  |             return esc(txt); | ||||||
|  |  | ||||||
|  |         return txt; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function uricom_dec(txt) { | ||||||
|  |     try { | ||||||
|  |         return [decodeURIComponent(txt), true]; | ||||||
|  |     } | ||||||
|  |     catch (ex) { | ||||||
|  |         console.log("ucd-err [" + txt + "]"); | ||||||
|  |         return [txt, false]; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function get_evpath() { | ||||||
|  |     var ret = document.location.pathname; | ||||||
|  |  | ||||||
|  |     if (ret.indexOf('/') !== 0) | ||||||
|  |         ret = '/' + ret; | ||||||
|  |  | ||||||
|  |     if (ret.lastIndexOf('/') !== ret.length - 1) | ||||||
|  |         ret += '/'; | ||||||
|  |  | ||||||
|  |     return ret; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function get_vpath() { | ||||||
|  |     return uricom_dec(get_evpath())[0]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function unix2iso(ts) { | ||||||
|  |     return new Date(ts * 1000).toISOString().replace("T", " ").slice(0, -5); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function s2ms(s) { | ||||||
|  |     s = Math.floor(s); | ||||||
|  |     var m = Math.floor(s / 60); | ||||||
|  |     return m + ":" + ("0" + (s - m * 60)).slice(-2); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function has(haystack, needle) { | ||||||
|  |     for (var a = 0; a < haystack.length; a++) | ||||||
|  |         if (haystack[a] == needle) | ||||||
|  |             return true; | ||||||
|  |  | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function sread(key) { | ||||||
|  |     if (window.localStorage) | ||||||
|  |         return localStorage.getItem(key); | ||||||
|  |  | ||||||
|  |     return null; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function swrite(key, val) { | ||||||
|  |     if (window.localStorage) { | ||||||
|  |         if (val === undefined || val === null) | ||||||
|  |             localStorage.removeItem(key); | ||||||
|  |         else | ||||||
|  |             localStorage.setItem(key, val); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function jread(key, fb) { | ||||||
|  |     var str = sread(key); | ||||||
|  |     if (!str) | ||||||
|  |         return fb; | ||||||
|  |  | ||||||
|  |     return JSON.parse(str); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function jwrite(key, val) { | ||||||
|  |     if (!val) | ||||||
|  |         swrite(key); | ||||||
|  |     else | ||||||
|  |         swrite(key, JSON.stringify(val)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function icfg_get(name, defval) { | ||||||
|  |     var o = ebi(name); | ||||||
|  |  | ||||||
|  |     var val = parseInt(sread(name)); | ||||||
|  |     if (isNaN(val)) | ||||||
|  |         return parseInt(o ? o.value : defval); | ||||||
|  |  | ||||||
|  |     if (o) | ||||||
|  |         o.value = val; | ||||||
|  |  | ||||||
|  |     return val; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function bcfg_get(name, defval) { | ||||||
|  |     var o = ebi(name); | ||||||
|  |     if (!o) | ||||||
|  |         return defval; | ||||||
|  |  | ||||||
|  |     var val = sread(name); | ||||||
|  |     if (val === null) | ||||||
|  |         val = defval; | ||||||
|  |     else | ||||||
|  |         val = (val == '1'); | ||||||
|  |  | ||||||
|  |     bcfg_upd_ui(name, val); | ||||||
|  |     return val; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function bcfg_set(name, val) { | ||||||
|  |     swrite(name, val ? '1' : '0'); | ||||||
|  |     bcfg_upd_ui(name, val); | ||||||
|  |     return val; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function bcfg_upd_ui(name, val) { | ||||||
|  |     var o = ebi(name); | ||||||
|  |     if (!o) | ||||||
|  |         return; | ||||||
|  |  | ||||||
|  |     if (o.getAttribute('type') == 'checkbox') | ||||||
|  |         o.checked = val; | ||||||
|  |     else if (o) { | ||||||
|  |         clmod(o, 'on', val); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | function hist_push(url) { | ||||||
|  |     console.log("h-push " + url); | ||||||
|  |     history.pushState(url, url, url); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function hist_replace(url) { | ||||||
|  |     console.log("h-repl " + url); | ||||||
|  |     history.replaceState(url, url, url); | ||||||
|  | } | ||||||
|   | |||||||
| @@ -32,9 +32,13 @@ r | |||||||
|  |  | ||||||
| # and a folder where anyone can upload | # and a folder where anyone can upload | ||||||
| # but nobody can see the contents | # 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 | /home/ed/inc | ||||||
| /dump | /dump | ||||||
| w | w | ||||||
|  | c e2d | ||||||
|  | c nodupe | ||||||
|  |  | ||||||
| # this entire config file can be replaced with these arguments: | # this entire config file can be replaced with these arguments: | ||||||
| # -u ed:123 -u k:k -v .::r:aed -v priv:priv:rk:aed -v /home/ed/Music:music:r -v /home/ed/inc:dump:w | # -u ed:123 -u k:k -v .::r: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> | ||||||
							
								
								
									
										242
									
								
								docs/music-analysis.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										242
									
								
								docs/music-analysis.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,242 @@ | |||||||
|  | #!/bin/bash | ||||||
|  | echo please dont actually run this as a scriopt | ||||||
|  | exit 1 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # dependency-heavy, not particularly good fit | ||||||
|  | pacman -S llvm10 | ||||||
|  | python3 -m pip install --user librosa | ||||||
|  | git clone https://github.com/librosa/librosa.git | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # correct bpm for tracks with bad tags | ||||||
|  | br=' | ||||||
|  | /Trip Trip Trip\(Hardcore Edit\).mp3/ {v=176} | ||||||
|  | /World!!.BIG_SOS/ {v=175} | ||||||
|  | /\/08\..*\(BIG_SOS Bootleg\)\.mp3/ {v=175} | ||||||
|  | /もってけ!セーラ服.Asterisk DnB/ {v=175} | ||||||
|  | /Rondo\(Asterisk DnB Re.mp3/ {v=175} | ||||||
|  | /Ray Nautica 175 Edit/ {v=175;x="thunk"} | ||||||
|  | /TOKIMEKI Language.Jauz/ {v=174} | ||||||
|  | /YUPPUN Hardcore Remix\).mp3/ {v=174;x="keeps drifting"} | ||||||
|  | /(èâAâï.î╧ûδ|バーチャリアル.狐耶)J-Core Remix\).mp3/ {v=172;x="hard"} | ||||||
|  | /lucky train..Freezer/ {v=170} | ||||||
|  | /Alf zero Bootleg ReMix/ {v=170} | ||||||
|  | /Prisoner of Love.Kacky/ {v=170} | ||||||
|  | /火炎 .Qota/ {v=170} | ||||||
|  | /\(hu-zin Bootleg\)\.mp3/ {v=170} | ||||||
|  | /15. STRAIGHT BET\(Milynn Bootleg\)\.mp3/ {v=170} | ||||||
|  | /\/13.*\(Milynn Bootleg\)\.mp3/ {v=167;x="way hard"} | ||||||
|  | /COLOR PLANET .10SAI . nijikon Remix\)\.mp3/ {v=165} | ||||||
|  | /11\. (朝はご飯派|Æ⌐é═é▓ö╤öh)\.mp3/ {v=162} | ||||||
|  | /09\. Where.s the core/ {v=160} | ||||||
|  | /PLANET\(Koushif Jersey Club Bootleg\)remaster.mp3/ {v=160;x="starts ez turns bs"} | ||||||
|  | /kened Soul - Madeon x Angel Beats!.mp3/ {v=160} | ||||||
|  | /Dear Moments\(Mother Harlot Bootleg\)\.mp3/ {v=150} | ||||||
|  | /POWER.Ringos UKG/ {v=140} | ||||||
|  | /ブルー・フィールド\(Ringos UKG Remix\).mp3/ {v=135} | ||||||
|  | /プラチナジェット.Ringo Remix..mp3/ {v=131.2} | ||||||
|  | /Mirrorball Love \(TKM Bootleg Mix\).mp3/ {v=130} | ||||||
|  | /Photon Melodies \(TKM Bootleg Mix\).mp3/ {v=128} | ||||||
|  | /Trap of Love \(TKM Bootleg Mix\).mp3/ {v=128} | ||||||
|  | /One Step \(TKM Bootleg Mix\)\.mp3/ {v=126} | ||||||
|  | /04 (トリカムイ岩|âgâèâJâÇâCèΓ).mp3/ {v=125} | ||||||
|  | /Get your Wish \(NAWN REMIX\)\.mp3/ {v=95} | ||||||
|  | /Flicker .Nitro Fun/ {v=92} | ||||||
|  | /\/14\..*suicat Remix/ {v=85.5;x="tricky"} | ||||||
|  | /Yanagi Nagi - Harumodoki \(EO Remix\)\.mp3/ {v=150} | ||||||
|  | /Azure - Nicology\.mp3/ {v=128;x="off by 5 how"} | ||||||
|  | ' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # afun host, collects/grades the results | ||||||
|  | runfun() { cores=8; touch run; rm -f /dev/shm/mres.*; t00=$(date +%s); tbc() { bc | sed -r 's/(\.[0-9]{2}).*/\1/'; }; for ((core=0; core<$cores; core++)); do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db 'select dur.w, dur.v, bpm.v from mt bpm join mt dur on bpm.w = dur.w where bpm.k = ".bpm" and dur.k = ".dur" order by dur.w' | uniq -w16 | while IFS=\| read w dur bpm; do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db "select rd, fn from up where substr(w,1,16) = '$w'" | sed -r "s/^/$bpm /"; done | grep mir/cr | tr \| / | awk '{v=$1;sub(/[^ ]+ /,"")} '"$br"' {printf "%s %s\n",v,$0}' | while read bpm fn; do [ -e run ] || break; n=$((n+1)); ncore=$((n%cores)); [ $ncore -eq $core ] || continue; t0=$(date +%s.%N); (afun || exit 1; t=$(date +%s.%N); td=$(echo "scale=3; $t - $t0" | tbc); bd=$(echo "scale=3; $bpm / $py" | tbc); printf '%4s sec, %4s orig, %6s py, %4s div, %s\n' $td $bpm $py $bd "$fn") | tee -a /dev/shm/mres.$ncore; rv=${PIPESTATUS[0]}; [ $rv -eq 0 ] || { echo "FAULT($rv): $fn"; }; done & done; wait 2>/dev/null; cat /dev/shm/mres.* | awk 'function prt(c) {printf "\033[3%sm%s\033[0m\n",c,$0} $8!="div,"{next} $5!~/^[0-9\.]+/{next} {meta=$3;det=$5;div=meta/det} div<0.7{det/=2} div>1.3{det*=2} {idet=sprintf("%.0f",det)} {idiff=idet-meta} meta>idet{idiff=meta-idet} idiff==0{n0++;prt(6);next} idiff==1{n1++;prt(3);next} idiff>10{nx++;prt(1);next} {n10++;prt(5)} END {printf "ok: %d   1off: %2s   (%3s)   10off: %2s   (%3s)   fail: %2s\n",n0,n1,n0+n1,n10,n0+n1+n10,nx}'; te=$(date +%s); echo $((te-t00)) sec spent; } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # ok:   8   1off: 62   ( 70)   10off: 86   (156)   fail: 25   # 105 sec,  librosa @ 8c archvm on 3700x w10 | ||||||
|  | # ok:   4   1off: 59   ( 63)   10off: 65   (128)   fail: 53   # using original tags (bad) | ||||||
|  | afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -t 60 /dev/shm/$core.wav || return 1; py="$(/home/ed/src/librosa/examples/beat_tracker.py /dev/shm/$core.wav x 2>&1 | awk 'BEGIN {v=1} /^Estimated tempo: /{v=$3} END {print v}')"; } runfun | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # ok: 119   1off:  5   (124)   10off:  8   (132)   fail: 49   # 51 sec,  vamp-example-fixedtempo | ||||||
|  | # ok: 109   1off:  4   (113)   10off:  9   (122)   fail: 59   # bad-tags | ||||||
|  | afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 22050 -f f32le /dev/shm/$core.pcm || return 1; py="$(python3 -c 'import vamp; import numpy as np; f = open("/dev/shm/'$core'.pcm", "rb"); d = np.fromfile(f, dtype=np.float32); c = vamp.collect(d, 22050, "vamp-example-plugins:fixedtempo", parameters={"maxdflen":40}); print(c["list"][0]["label"].split(" ")[0])')"; }; runfun | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # ok: 102   1off: 61   (163)   10off: 12   (175)   fail:  6   # 61 sec,  vamp-qm-tempotracker | ||||||
|  | # ok:  80   1off: 48   (128)   10off: 11   (139)   fail: 42   # bad-tags | ||||||
|  | afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 22050 -f f32le /dev/shm/$core.pcm || return 1; py="$(python3 -c 'import vamp; import numpy as np; f = open("/dev/shm/'$core'.pcm", "rb"); d = np.fromfile(f, dtype=np.float32); c = vamp.collect(d, 22050, "qm-vamp-plugins:qm-tempotracker", parameters={"inputtempo":150}); v = [float(x["label"].split(" ")[0]) for x in c["list"] if x["label"]]; v = list(sorted(v))[len(v)//4:-len(v)//4]; print(round(sum(v) / len(v), 1))')"; }; runfun | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # ok: 133   1off: 32   (165)   10off: 12   (177)   fail:  3   # 51 sec,  vamp-beatroot | ||||||
|  | # ok: 101   1off: 22   (123)   10off: 16   (139)   fail: 39   # bad-tags | ||||||
|  | # note: some tracks fully fail to analyze (unlike the others which always provide a guess) | ||||||
|  | afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 22050 -f f32le /dev/shm/$core.pcm || return 1; py="$(python3 -c 'import vamp; import numpy as np; f = open("/dev/shm/'$core'.pcm", "rb"); d = np.fromfile(f, dtype=np.float32); c = vamp.collect(d, 22050, "beatroot-vamp:beatroot"); cl=c["list"]; print(round(60*((len(cl)-1)/(float(cl[-1]["timestamp"]-cl[1]["timestamp"]))), 2))')"; }; runfun | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # ok: 124   1off:  9   (133)   10off: 40   (173)   fail:  8   # 231 sec,  essentia/full | ||||||
|  | # ok: 109   1off:  8   (117)   10off: 22   (139)   fail: 42   # bad-tags | ||||||
|  | afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 /dev/shm/$core.wav || return 1; py="$(python3 -c 'import essentia; import essentia.standard as es; fe, fef = es.MusicExtractor(lowlevelStats=["mean", "stdev"], rhythmStats=["mean", "stdev"], tonalStats=["mean", "stdev"])("/dev/shm/'$core'.wav"); print("{:.2f}".format(fe["rhythm.bpm"]))')"; }; runfun | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # ok: 113   1off: 18   (131)   10off: 46   (177)   fail:  4   # 134 sec,  essentia/re2013 | ||||||
|  | # ok: 101   1off: 15   (116)   10off: 26   (142)   fail: 39   # bad-tags | ||||||
|  | afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 /dev/shm/$core.wav || return 1; py="$(python3 -c 'from essentia.standard import *; a=MonoLoader(filename="/dev/shm/'$core'.wav")(); bpm,beats,confidence,_,intervals=RhythmExtractor2013(method="multifeature")(a); print("{:.2f}".format(bpm))')"; }; runfun | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ######################################################################## | ||||||
|  | ## | ||||||
|  | ##  key detectyion | ||||||
|  | ## | ||||||
|  | ######################################################################## | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # console scriptlet reusing keytabs from browser.js | ||||||
|  | var m=''; for (var a=0; a<24; a++) m += 's/\\|(' + maps["traktor_sharps"][a].trim() + "|" + maps["rekobo_classic"][a].trim() + "|" + maps["traktor_musical"][a].trim() + "|" + maps["traktor_open"][a].trim() + ')$/|' + maps["rekobo_alnum"][a].trim() + '/;'; console.log(m); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # translate to camelot | ||||||
|  | re='s/\|(B|B|B|6d)$/|1B/;s/\|(F#|F#|Gb|7d)$/|2B/;s/\|(C#|Db|Db|8d)$/|3B/;s/\|(G#|Ab|Ab|9d)$/|4B/;s/\|(D#|Eb|Eb|10d)$/|5B/;s/\|(A#|Bb|Bb|11d)$/|6B/;s/\|(F|F|F|12d)$/|7B/;s/\|(C|C|C|1d)$/|8B/;s/\|(G|G|G|2d)$/|9B/;s/\|(D|D|D|3d)$/|10B/;s/\|(A|A|A|4d)$/|11B/;s/\|(E|E|E|5d)$/|12B/;s/\|(G#m|Abm|Abm|6m)$/|1A/;s/\|(D#m|Ebm|Ebm|7m)$/|2A/;s/\|(A#m|Bbm|Bbm|8m)$/|3A/;s/\|(Fm|Fm|Fm|9m)$/|4A/;s/\|(Cm|Cm|Cm|10m)$/|5A/;s/\|(Gm|Gm|Gm|11m)$/|6A/;s/\|(Dm|Dm|Dm|12m)$/|7A/;s/\|(Am|Am|Am|1m)$/|8A/;s/\|(Em|Em|Em|2m)$/|9A/;s/\|(Bm|Bm|Bm|3m)$/|10A/;s/\|(F#m|F#m|Gbm|4m)$/|11A/;s/\|(C#m|Dbm|Dbm|5m)$/|12A/;' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # runner/wrapper | ||||||
|  | runfun() { cores=8; touch run; tbc() { bc | sed -r 's/(\.[0-9]{2}).*/\1/'; }; for ((core=0; core<$cores; core++)); do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db 'select dur.w, dur.v, key.v from mt key join mt dur on key.w = dur.w where key.k = "key" and dur.k = ".dur" order by dur.w' | uniq -w16 | grep -vE '(Off-Key|None)$' | sed -r "s/ //g;$re" | uniq -w16 | while IFS=\| read w dur bpm; do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db "select rd, fn from up where substr(w,1,16) = '$w'" | sed -r "s/^/$bpm /"; done| grep mir/cr | tr \| / | while read key fn; do [ -e run ] || break; n=$((n+1)); ncore=$((n%cores)); [ $ncore -eq $core ] || continue; t0=$(date +%s.%N); (afun || exit 1; t=$(date +%s.%N); td=$(echo "scale=3; $t - $t0" | tbc); [ "$key" = "$py" ] && c=2 || c=5; printf '%4s sec, %4s orig, \033[3%dm%4s py,\033[0m %s\n' $td "$key" $c "$py" "$fn") || break; done & done; time wait 2>/dev/null; } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # ok: 26   1off: 10   2off: 1   fail: 3   #  15 sec, keyfinder | ||||||
|  | afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 -t 60 /dev/shm/$core.wav || break; py="$(python3 -c 'import sys; import keyfinder; print(keyfinder.key(sys.argv[1]).camelot())' "/dev/shm/$core.wav")"; }; runfun | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # https://github.com/MTG/essentia/raw/master/src/examples/tutorial/example_key_by_steps_streaming.py | ||||||
|  | # https://essentia.upf.edu/reference/std_Key.html  # edma edmm braw bgate | ||||||
|  | sed -ri 's/^(key = Key\().*/\1profileType="bgate")/' example_key_by_steps_streaming.py | ||||||
|  | afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 -t 60 /dev/shm/$core.wav || break; py="$(python3 example_key_by_steps_streaming.py /dev/shm/$core.{wav,yml} 2>/dev/null | sed -r "s/ major//;s/ minor/m/;s/^/|/;$re;s/.//")"; }; runfun | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ######################################################################## | ||||||
|  | ## | ||||||
|  | ##  misc | ||||||
|  | ## | ||||||
|  | ######################################################################## | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | python3 -m pip install --user vamp | ||||||
|  |  | ||||||
|  | import librosa | ||||||
|  | d, r = librosa.load('/dev/shm/0.wav') | ||||||
|  | d.dtype | ||||||
|  | # dtype('float32') | ||||||
|  | d.shape | ||||||
|  | # (1323000,) | ||||||
|  | d | ||||||
|  | # array([-1.9614939e-08,  1.8037968e-08, -1.4106059e-08, ..., | ||||||
|  | #         1.2024145e-01,  2.7462116e-01,  1.6202132e-01], dtype=float32) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | import vamp | ||||||
|  | c = vamp.collect(d, r, "vamp-example-plugins:fixedtempo") | ||||||
|  | c | ||||||
|  | # {'list': [{'timestamp':  0.005804988, 'duration':  9.999092971, 'label': '110.0 bpm', 'values': array([109.98116], dtype=float32)}]} | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ffmpeg -ss 48 -i /mnt/Users/ed/Music/mir/cr-a/'I Beg You(ths Bootleg).wav' -ac 1 -ar 22050 -f f32le -t 60 /dev/shm/f32.pcm | ||||||
|  |  | ||||||
|  | import numpy as np | ||||||
|  | f = open('/dev/shm/f32.pcm', 'rb') | ||||||
|  | d = np.fromfile(f, dtype=np.float32) | ||||||
|  | d | ||||||
|  | array([-0.17803933, -0.27206388, -0.41586545, ..., -0.04940119, | ||||||
|  |        -0.0267825 , -0.03564296], dtype=float32) | ||||||
|  |  | ||||||
|  | d = np.reshape(d, [1, -1]) | ||||||
|  | d | ||||||
|  | array([[-0.17803933, -0.27206388, -0.41586545, ..., -0.04940119, | ||||||
|  |         -0.0267825 , -0.03564296]], dtype=float32) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | import vampyhost | ||||||
|  | print("\n".join(vampyhost.list_plugins())) | ||||||
|  |  | ||||||
|  | mvamp:marsyas_bextract_centroid | ||||||
|  | mvamp:marsyas_bextract_lpcc | ||||||
|  | mvamp:marsyas_bextract_lsp | ||||||
|  | mvamp:marsyas_bextract_mfcc | ||||||
|  | mvamp:marsyas_bextract_rolloff | ||||||
|  | mvamp:marsyas_bextract_scf | ||||||
|  | mvamp:marsyas_bextract_sfm | ||||||
|  | mvamp:marsyas_bextract_zero_crossings | ||||||
|  | mvamp:marsyas_ibt | ||||||
|  | mvamp:zerocrossing | ||||||
|  | qm-vamp-plugins:qm-adaptivespectrogram | ||||||
|  | qm-vamp-plugins:qm-barbeattracker | ||||||
|  | qm-vamp-plugins:qm-chromagram | ||||||
|  | qm-vamp-plugins:qm-constantq | ||||||
|  | qm-vamp-plugins:qm-dwt | ||||||
|  | qm-vamp-plugins:qm-keydetector | ||||||
|  | qm-vamp-plugins:qm-mfcc | ||||||
|  | qm-vamp-plugins:qm-onsetdetector | ||||||
|  | qm-vamp-plugins:qm-segmenter | ||||||
|  | qm-vamp-plugins:qm-similarity | ||||||
|  | qm-vamp-plugins:qm-tempotracker | ||||||
|  | qm-vamp-plugins:qm-tonalchange | ||||||
|  | qm-vamp-plugins:qm-transcription | ||||||
|  | vamp-aubio:aubiomelenergy | ||||||
|  | vamp-aubio:aubiomfcc | ||||||
|  | vamp-aubio:aubionotes | ||||||
|  | vamp-aubio:aubioonset | ||||||
|  | vamp-aubio:aubiopitch | ||||||
|  | vamp-aubio:aubiosilence | ||||||
|  | vamp-aubio:aubiospecdesc | ||||||
|  | vamp-aubio:aubiotempo | ||||||
|  | vamp-example-plugins:amplitudefollower | ||||||
|  | vamp-example-plugins:fixedtempo | ||||||
|  | vamp-example-plugins:percussiononsets | ||||||
|  | vamp-example-plugins:powerspectrum | ||||||
|  | vamp-example-plugins:spectralcentroid | ||||||
|  | vamp-example-plugins:zerocrossing | ||||||
|  | vamp-rubberband:rubberband | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | plug = vampyhost.load_plugin("vamp-example-plugins:fixedtempo", 22050, 0) | ||||||
|  | plug.info | ||||||
|  | {'apiVersion': 2, 'pluginVersion': 1, 'identifier': 'fixedtempo', 'name': 'Simple Fixed Tempo Estimator', 'description': 'Study a short section of audio and estimate its tempo, assuming the tempo is constant', 'maker': 'Vamp SDK Example Plugins', 'copyright': 'Code copyright 2008 Queen Mary, University of London.  Freely redistributable (BSD license)'} | ||||||
|  | plug = vampyhost.load_plugin("qm-vamp-plugins:qm-tempotracker", 22050, 0) | ||||||
|  | from pprint import pprint; pprint(plug.parameters) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | for c in plug.parameters: print("{} \033[36m{}  [\033[33m{}\033[36m] = {}\033[0m".format(c["identifier"], c["name"], "\033[36m, \033[33m".join(c["valueNames"]), c["valueNames"][int(c["defaultValue"])])) if "valueNames" in c else print("{} \033[36m{}  [\033[33m{}..{}\033[36m] = {}\033[0m".format(c["identifier"], c["name"], c["minValue"], c["maxValue"], c["defaultValue"])) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | beatroot-vamp:beatroot | ||||||
|  | cl=c["list"]; 60*((len(cl)-1)/(float(cl[-1]["timestamp"]-cl[1]["timestamp"]))) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ffmpeg -ss 48 -i /mnt/Users/ed/Music/mir/cr-a/'I Beg You(ths Bootleg).wav' -ac 1 -ar 22050 -f f32le -t 60 /dev/shm/f32.pcm | ||||||
|  | # 128 bpm, key 5A Cm | ||||||
|  |  | ||||||
|  | import vamp | ||||||
|  | import numpy as np | ||||||
|  | f = open('/dev/shm/f32.pcm', 'rb') | ||||||
|  | d = np.fromfile(f, dtype=np.float32) | ||||||
|  | c = vamp.collect(d, 22050, "vamp-example-plugins:fixedtempo", parameters={"maxdflen":40}) | ||||||
|  | c["list"][0]["label"] | ||||||
|  | # 127.6 bpm | ||||||
|  |  | ||||||
|  | c = vamp.collect(d, 22050, "qm-vamp-plugins:qm-tempotracker", parameters={"inputtempo":150}) | ||||||
|  | print("\n".join([v["label"] for v in c["list"] if v["label"]])) | ||||||
|  | v = [float(x["label"].split(' ')[0]) for x in c["list"] if x["label"]] | ||||||
|  | v = list(sorted(v))[len(v)//4:-len(v)//4] | ||||||
|  | v = sum(v) / len(v) | ||||||
|  | # 128.1 bpm | ||||||
|  |  | ||||||
| @@ -11,6 +11,13 @@ gzip -d < .hist/up2k.snap | jq -r '.[].tnam' | while IFS= read -r f; do rm -f -- | |||||||
| gzip -d < .hist/up2k.snap | jq -r '.[].name' | while IFS= read -r f; do wc -c -- "$f" | grep -qiE '^[^0-9a-z]*0' && rm -f -- "$f"; done | gzip -d < .hist/up2k.snap | jq -r '.[].name' | while IFS= read -r f; do wc -c -- "$f" | grep -qiE '^[^0-9a-z]*0' && rm -f -- "$f"; done | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## | ||||||
|  | ## detect partial uploads based on file contents | ||||||
|  | ##  (in case of context loss or old copyparties) | ||||||
|  |  | ||||||
|  | echo; find -type f | while IFS= read -r x; do printf '\033[A\033[36m%s\033[K\033[0m\n' "$x"; tail -c$((1024*1024)) <"$x" | xxd -a | awk 'NR==1&&/^[0: ]+.{16}$/{next} NR==2&&/^\*$/{next} NR==3&&/^[0f]+: [0 ]+65 +.{16}$/{next} {e=1} END {exit e}' || continue; printf '\033[A\033[31msus:\033[33m %s \033[0m\n\n' "$x"; done | ||||||
|  |  | ||||||
|  |  | ||||||
| ## | ## | ||||||
| ## create a test payload | ## create a test payload | ||||||
|  |  | ||||||
| @@ -60,6 +67,43 @@ wget -S --header='Accept-Encoding: gzip' -U 'MSIE 6.0; SV1' http://127.0.0.1:392 | |||||||
| shab64() { sp=$1; f="$2"; v=0; sz=$(stat -c%s "$f"); while true; do w=$((v+sp*1024*1024)); printf $(tail -c +$((v+1)) "$f" | head -c $((w-v)) | sha512sum | cut -c-64 | sed -r 's/ .*//;s/(..)/\\x\1/g') | base64 -w0 | cut -c-43 | tr '+/' '-_'; v=$w; [ $v -lt $sz ] || break; done; } | shab64() { sp=$1; f="$2"; v=0; sz=$(stat -c%s "$f"); while true; do w=$((v+sp*1024*1024)); printf $(tail -c +$((v+1)) "$f" | head -c $((w-v)) | sha512sum | cut -c-64 | sed -r 's/ .*//;s/(..)/\\x\1/g') | base64 -w0 | cut -c-43 | tr '+/' '-_'; v=$w; [ $v -lt $sz ] || break; done; } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## | ||||||
|  | ## poll url for performance issues | ||||||
|  |  | ||||||
|  | command -v gdate && date() { gdate "$@"; }; while true; do t=$(date +%s.%N); (time wget http://127.0.0.1:3923/?ls -qO- | jq -C '.files[]|{sz:.sz,ta:.tags.artist,tb:.tags.".bpm"}|del(.[]|select(.==null))' | awk -F\" '/"/{t[$2]++} END {for (k in t){v=t[k];p=sprintf("%" (v+1) "s",v);gsub(/ /,"#",p);printf "\033[36m%s\033[33m%s   ",k,p}}') 2>&1 | awk -v ts=$t 'NR==1{t1=$0} NR==2{sub(/.*0m/,"");sub(/s$/,"");t2=$0;c=2; if(t2>0.3){c=3} if(t2>0.8){c=1} } END{sub(/[0-9]{6}$/,"",ts);printf "%s   \033[3%dm%s   %s\033[0m\n",ts,c,t2,t1}'; sleep 0.1 || break; done | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## | ||||||
|  | ## js oneliners | ||||||
|  |  | ||||||
|  | # get all up2k search result URLs | ||||||
|  | var t=[]; var b=document.location.href.split('#')[0].slice(0, -1); document.querySelectorAll('#u2tab .prog a').forEach((x) => {t.push(b+encodeURI(x.getAttribute("href")))}); console.log(t.join("\n")); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## | ||||||
|  | ## sqlite3 stuff | ||||||
|  |  | ||||||
|  | # find dupe metadata keys | ||||||
|  | sqlite3 up2k.db 'select mt1.w, mt1.k, mt1.v, mt2.v from mt mt1 inner join mt mt2 on mt1.w = mt2.w where mt1.k = mt2.k and mt1.rowid != mt2.rowid' | ||||||
|  |  | ||||||
|  | # partial reindex by deleting all tags for a list of files | ||||||
|  | time sqlite3 up2k.db 'select mt1.w from mt mt1 inner join mt mt2 on mt1.w = mt2.w where mt1.k = +mt2.k and mt1.rowid != mt2.rowid'  > warks | ||||||
|  | cat warks | while IFS= read -r x; do sqlite3 up2k.db "delete from mt where w = '$x'"; done | ||||||
|  |  | ||||||
|  | # dump all dbs | ||||||
|  | find -iname up2k.db | while IFS= read -r x; do sqlite3 "$x" 'select substr(w,1,12), rd, fn from up' | sed -r 's/\|/ \| /g' | while IFS= read -r y; do printf '%s | %s\n' "$x" "$y"; done; done | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## | ||||||
|  | ## media | ||||||
|  |  | ||||||
|  | # split track into test files | ||||||
|  | e=6; s=10; d=~/dev/copyparty/srv/aus; n=1; p=0; e=$((e*60)); rm -rf $d; mkdir $d; while true; do ffmpeg -hide_banner -ss $p -i 'nervous_testpilot - office.mp3' -c copy -t $s $d/$(printf %04d $n).mp3; n=$((n+1)); p=$((p+s)); [ $p -gt $e ] && break; done | ||||||
|  |  | ||||||
|  | -v srv/aus:aus:r:ce2dsa:ce2ts:cmtp=fgsfds=bin/mtag/sleep.py | ||||||
|  | sqlite3 .hist/up2k.db 'select * from mt where k="fgsfds" or k="t:mtp"' | tee /dev/stderr | wc -l | ||||||
|  |  | ||||||
|  |  | ||||||
| ## | ## | ||||||
| ## vscode | ## vscode | ||||||
|  |  | ||||||
| @@ -89,6 +133,19 @@ for d in /usr /var; do find $d -type f -size +30M 2>/dev/null; done | while IFS= | |||||||
| brew install python@2 | brew install python@2 | ||||||
| pip install virtualenv | 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 | ||||||
|  |  | ||||||
|  |  | ||||||
| ## | ## | ||||||
| ## http 206 | ## http 206 | ||||||
| @@ -114,7 +171,7 @@ Range: bytes=26-         Content-Range: bytes */26 | |||||||
|  |  | ||||||
| var tsh = []; | var tsh = []; | ||||||
| function convert_markdown(md_text, dest_dom) { | function convert_markdown(md_text, dest_dom) { | ||||||
|     tsh.push(new Date().getTime()); |     tsh.push(Date.now()); | ||||||
|     while (tsh.length > 10) |     while (tsh.length > 10) | ||||||
|         tsh.shift(); |         tsh.shift(); | ||||||
|     if (tsh.length > 1) { |     if (tsh.length > 1) { | ||||||
|   | |||||||
| @@ -20,6 +20,7 @@ set -e | |||||||
| # -rwxr-xr-x  0 ed ed  183808 Nov 19 00:43 copyparty-extras/sfx-lite/copyparty-sfx.py | # -rwxr-xr-x  0 ed ed  183808 Nov 19 00:43 copyparty-extras/sfx-lite/copyparty-sfx.py | ||||||
|  |  | ||||||
|  |  | ||||||
|  | command -v gnutar && tar() { gnutar "$@"; } | ||||||
| command -v gtar && tar() { gtar "$@"; } | command -v gtar && tar() { gtar "$@"; } | ||||||
| command -v gsed && sed() { gsed "$@"; } | command -v gsed && sed() { gsed "$@"; } | ||||||
| td="$(mktemp -d)" | td="$(mktemp -d)" | ||||||
| @@ -29,11 +30,11 @@ pwd | |||||||
|  |  | ||||||
|  |  | ||||||
| dl_text() { | dl_text() { | ||||||
| 	command -v curl && exec curl "$@" | 	command -v curl >/dev/null && exec curl "$@" | ||||||
| 	exec wget -O- "$@" | 	exec wget -O- "$@" | ||||||
| } | } | ||||||
| dl_files() { | dl_files() { | ||||||
| 	command -v curl && exec curl -L --remote-name-all "$@" | 	command -v curl >/dev/null && exec curl -L --remote-name-all "$@" | ||||||
| 	exec wget "$@" | 	exec wget "$@" | ||||||
| } | } | ||||||
| export -f dl_files | export -f dl_files | ||||||
|   | |||||||
| @@ -1,12 +1,10 @@ | |||||||
| FROM    alpine:3.11 | FROM    alpine:3.13 | ||||||
| WORKDIR /z | WORKDIR /z | ||||||
| ENV     ver_asmcrypto=2821dd1dedd1196c378f5854037dda5c869313f3 \ | ENV     ver_asmcrypto=5b994303a9d3e27e0915f72a10b6c2c51535a4dc \ | ||||||
|         ver_markdownit=10.0.0 \ |  | ||||||
|         ver_showdown=1.9.1 \ |  | ||||||
|         ver_marked=1.1.0 \ |         ver_marked=1.1.0 \ | ||||||
|         ver_ogvjs=1.6.1 \ |         ver_ogvjs=1.8.0 \ | ||||||
|         ver_mde=2.10.1 \ |         ver_mde=2.14.0 \ | ||||||
|         ver_codemirror=5.53.2 \ |         ver_codemirror=5.59.3 \ | ||||||
|         ver_fontawesome=5.13.0 \ |         ver_fontawesome=5.13.0 \ | ||||||
|         ver_zopfli=1.0.3 |         ver_zopfli=1.0.3 | ||||||
|  |  | ||||||
| @@ -17,7 +15,7 @@ RUN     mkdir -p /z/dist/no-pk \ | |||||||
|         && wget https://fonts.gstatic.com/s/sourcecodepro/v11/HI_SiYsKILxRpg3hIP6sJ7fM7PqlPevW.woff2 -O scp.woff2 \ |         && wget https://fonts.gstatic.com/s/sourcecodepro/v11/HI_SiYsKILxRpg3hIP6sJ7fM7PqlPevW.woff2 -O scp.woff2 \ | ||||||
|         && apk add cmake make g++ git bash npm patch wget tar pigz brotli gzip unzip python3 python3-dev brotli py3-brotli \ |         && apk add cmake make g++ git bash npm patch wget tar pigz brotli gzip unzip python3 python3-dev brotli py3-brotli \ | ||||||
|         && wget https://github.com/brion/ogv.js/releases/download/$ver_ogvjs/ogvjs-$ver_ogvjs.zip -O ogvjs.zip \ |         && wget https://github.com/brion/ogv.js/releases/download/$ver_ogvjs/ogvjs-$ver_ogvjs.zip -O ogvjs.zip \ | ||||||
|         && wget https://github.com/asmcrypto/asmcrypto.js/archive/$ver_asmcrypto.tar.gz -O asmcrypto.tgz \ |         && wget https://github.com/openpgpjs/asmcrypto.js/archive/$ver_asmcrypto.tar.gz -O asmcrypto.tgz \ | ||||||
|         && wget https://github.com/markedjs/marked/archive/v$ver_marked.tar.gz -O marked.tgz \ |         && wget https://github.com/markedjs/marked/archive/v$ver_marked.tar.gz -O marked.tgz \ | ||||||
|         && wget https://github.com/Ionaru/easy-markdown-editor/archive/$ver_mde.tar.gz -O mde.tgz \ |         && wget https://github.com/Ionaru/easy-markdown-editor/archive/$ver_mde.tar.gz -O mde.tgz \ | ||||||
|         && wget https://github.com/codemirror/CodeMirror/archive/$ver_codemirror.tar.gz -O codemirror.tgz \ |         && wget https://github.com/codemirror/CodeMirror/archive/$ver_codemirror.tar.gz -O codemirror.tgz \ | ||||||
| @@ -52,6 +50,7 @@ RUN     tar -xf zopfli.tgz \ | |||||||
|             -S . \ |             -S . \ | ||||||
|         && make -C build \ |         && make -C build \ | ||||||
|         && make -C build install \ |         && make -C build install \ | ||||||
|  |         && python3 -m ensurepip \ | ||||||
|         && python3 -m pip install fonttools zopfli |         && python3 -m pip install fonttools zopfli | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| diff -NarU2 CodeMirror-orig/mode/gfm/gfm.js CodeMirror-edit/mode/gfm/gfm.js | diff -NarU2 codemirror-5.59.3-orig/mode/gfm/gfm.js codemirror-5.59.3/mode/gfm/gfm.js | ||||||
| --- CodeMirror-orig/mode/gfm/gfm.js	2020-04-21 12:47:20.000000000 +0200 | --- codemirror-5.59.3-orig/mode/gfm/gfm.js	2021-02-20 21:24:57.000000000 +0000 | ||||||
| +++ CodeMirror-edit/mode/gfm/gfm.js	2020-05-02 02:13:32.142131800 +0200 | +++ codemirror-5.59.3/mode/gfm/gfm.js	2021-02-21 20:42:02.166174775 +0000 | ||||||
| @@ -97,5 +97,5 @@ | @@ -97,5 +97,5 @@ | ||||||
|          } |          } | ||||||
|        } |        } | ||||||
| @@ -15,9 +15,9 @@ diff -NarU2 CodeMirror-orig/mode/gfm/gfm.js CodeMirror-edit/mode/gfm/gfm.js | |||||||
| +      }*/ | +      }*/ | ||||||
|        stream.next(); |        stream.next(); | ||||||
|        return null; |        return null; | ||||||
| diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js | diff -NarU2 codemirror-5.59.3-orig/mode/meta.js codemirror-5.59.3/mode/meta.js | ||||||
| --- CodeMirror-orig/mode/meta.js	2020-04-21 12:47:20.000000000 +0200 | --- codemirror-5.59.3-orig/mode/meta.js	2021-02-20 21:24:57.000000000 +0000 | ||||||
| +++ CodeMirror-edit/mode/meta.js	2020-05-02 03:56:58.852408400 +0200 | +++ codemirror-5.59.3/mode/meta.js	2021-02-21 20:42:54.798742821 +0000 | ||||||
| @@ -13,4 +13,5 @@ | @@ -13,4 +13,5 @@ | ||||||
|   |   | ||||||
|    CodeMirror.modeInfo = [ |    CodeMirror.modeInfo = [ | ||||||
| @@ -28,7 +28,7 @@ diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js | |||||||
|      {name: "Gas", mime: "text/x-gas", mode: "gas", ext: ["s"]}, |      {name: "Gas", mime: "text/x-gas", mode: "gas", ext: ["s"]}, | ||||||
|      {name: "Gherkin", mime: "text/x-feature", mode: "gherkin", ext: ["feature"]}, |      {name: "Gherkin", mime: "text/x-feature", mode: "gherkin", ext: ["feature"]}, | ||||||
| +    */ | +    */ | ||||||
|      {name: "GitHub Flavored Markdown", mime: "text/x-gfm", mode: "gfm", file: /^(readme|contributing|history).md$/i}, |      {name: "GitHub Flavored Markdown", mime: "text/x-gfm", mode: "gfm", file: /^(readme|contributing|history)\.md$/i}, | ||||||
| +    /* | +    /* | ||||||
|      {name: "Go", mime: "text/x-go", mode: "go", ext: ["go"]}, |      {name: "Go", mime: "text/x-go", mode: "go", ext: ["go"]}, | ||||||
|      {name: "Groovy", mime: "text/x-groovy", mode: "groovy", ext: ["groovy", "gradle"], file: /^Jenkinsfile$/}, |      {name: "Groovy", mime: "text/x-groovy", mode: "groovy", ext: ["groovy", "gradle"], file: /^Jenkinsfile$/}, | ||||||
| @@ -56,16 +56,16 @@ diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js | |||||||
| +    /* | +    /* | ||||||
|      {name: "XQuery", mime: "application/xquery", mode: "xquery", ext: ["xy", "xquery"]}, |      {name: "XQuery", mime: "application/xquery", mode: "xquery", ext: ["xy", "xquery"]}, | ||||||
|      {name: "Yacas", mime: "text/x-yacas", mode: "yacas", ext: ["ys"]}, |      {name: "Yacas", mime: "text/x-yacas", mode: "yacas", ext: ["ys"]}, | ||||||
| @@ -171,4 +180,5 @@ | @@ -172,4 +181,5 @@ | ||||||
|      {name: "xu", mime: "text/x-xu", mode: "mscgen", ext: ["xu"]}, |      {name: "msgenny", mime: "text/x-msgenny", mode: "mscgen", ext: ["msgenny"]}, | ||||||
|      {name: "msgenny", mime: "text/x-msgenny", mode: "mscgen", ext: ["msgenny"]} |      {name: "WebAssembly", mime: "text/webassembly", mode: "wast", ext: ["wat", "wast"]}, | ||||||
| +    */ | +    */ | ||||||
|    ]; |    ]; | ||||||
|    // Ensure all modes have a mime property for backwards compatibility |    // Ensure all modes have a mime property for backwards compatibility | ||||||
| diff -NarU2 CodeMirror-orig/src/display/selection.js CodeMirror-edit/src/display/selection.js | diff -NarU2 codemirror-5.59.3-orig/src/display/selection.js codemirror-5.59.3/src/display/selection.js | ||||||
| --- CodeMirror-orig/src/display/selection.js	2020-04-21 12:47:20.000000000 +0200 | --- codemirror-5.59.3-orig/src/display/selection.js	2021-02-20 21:24:57.000000000 +0000 | ||||||
| +++ CodeMirror-edit/src/display/selection.js	2020-05-02 03:27:30.144662800 +0200 | +++ codemirror-5.59.3/src/display/selection.js	2021-02-21 20:44:14.860894328 +0000 | ||||||
| @@ -83,29 +83,21 @@ | @@ -84,29 +84,21 @@ | ||||||
|      let order = getOrder(lineObj, doc.direction) |      let order = getOrder(lineObj, doc.direction) | ||||||
|      iterateBidiSections(order, fromArg || 0, toArg == null ? lineLen : toArg, (from, to, dir, i) => { |      iterateBidiSections(order, fromArg || 0, toArg == null ? lineLen : toArg, (from, to, dir, i) => { | ||||||
| -      let ltr = dir == "ltr" | -      let ltr = dir == "ltr" | ||||||
| @@ -105,24 +105,24 @@ diff -NarU2 CodeMirror-orig/src/display/selection.js CodeMirror-edit/src/display | |||||||
| +          botRight = openEnd && last ? rightSide : toPos.right | +          botRight = openEnd && last ? rightSide : toPos.right | ||||||
|          add(topLeft, fromPos.top, topRight - topLeft, fromPos.bottom) |          add(topLeft, fromPos.top, topRight - topLeft, fromPos.bottom) | ||||||
|          if (fromPos.bottom < toPos.top) add(leftSide, fromPos.bottom, null, toPos.top) |          if (fromPos.bottom < toPos.top) add(leftSide, fromPos.bottom, null, toPos.top) | ||||||
| diff -NarU2 CodeMirror-orig/src/input/ContentEditableInput.js CodeMirror-edit/src/input/ContentEditableInput.js | diff -NarU2 codemirror-5.59.3-orig/src/input/ContentEditableInput.js codemirror-5.59.3/src/input/ContentEditableInput.js | ||||||
| --- CodeMirror-orig/src/input/ContentEditableInput.js	2020-04-21 12:47:20.000000000 +0200 | --- codemirror-5.59.3-orig/src/input/ContentEditableInput.js	2021-02-20 21:24:57.000000000 +0000 | ||||||
| +++ CodeMirror-edit/src/input/ContentEditableInput.js	2020-05-02 03:33:05.707995500 +0200 | +++ codemirror-5.59.3/src/input/ContentEditableInput.js	2021-02-21 20:44:33.273953867 +0000 | ||||||
| @@ -391,4 +391,5 @@ | @@ -399,4 +399,5 @@ | ||||||
|    let info = mapFromLineView(view, line, pos.line) |    let info = mapFromLineView(view, line, pos.line) | ||||||
|   |   | ||||||
| +  /* | +  /* | ||||||
|    let order = getOrder(line, cm.doc.direction), side = "left" |    let order = getOrder(line, cm.doc.direction), side = "left" | ||||||
|    if (order) { |    if (order) { | ||||||
| @@ -396,4 +397,5 @@ | @@ -404,4 +405,5 @@ | ||||||
|      side = partPos % 2 ? "right" : "left" |      side = partPos % 2 ? "right" : "left" | ||||||
|    } |    } | ||||||
| +  */ | +  */ | ||||||
|    let result = nodeAndOffsetInLineMap(info.map, pos.ch, side) |    let result = nodeAndOffsetInLineMap(info.map, pos.ch, side) | ||||||
|    result.offset = result.collapse == "right" ? result.end : result.start |    result.offset = result.collapse == "right" ? result.end : result.start | ||||||
| diff -NarU2 CodeMirror-orig/src/input/movement.js CodeMirror-edit/src/input/movement.js | diff -NarU2 codemirror-5.59.3-orig/src/input/movement.js codemirror-5.59.3/src/input/movement.js | ||||||
| --- CodeMirror-orig/src/input/movement.js	2020-04-21 12:47:20.000000000 +0200 | --- codemirror-5.59.3-orig/src/input/movement.js	2021-02-20 21:24:57.000000000 +0000 | ||||||
| +++ CodeMirror-edit/src/input/movement.js	2020-05-02 03:31:19.710773500 +0200 | +++ codemirror-5.59.3/src/input/movement.js	2021-02-21 20:45:12.763093671 +0000 | ||||||
| @@ -15,4 +15,5 @@ | @@ -15,4 +15,5 @@ | ||||||
|   |   | ||||||
|  export function endOfLine(visually, cm, lineObj, lineNo, dir) { |  export function endOfLine(visually, cm, lineObj, lineNo, dir) { | ||||||
| @@ -146,9 +146,9 @@ diff -NarU2 CodeMirror-orig/src/input/movement.js CodeMirror-edit/src/input/move | |||||||
|    return null |    return null | ||||||
| +  */ | +  */ | ||||||
|  } |  } | ||||||
| diff -NarU2 CodeMirror-orig/src/line/line_data.js CodeMirror-edit/src/line/line_data.js | diff -NarU2 codemirror-5.59.3-orig/src/line/line_data.js codemirror-5.59.3/src/line/line_data.js | ||||||
| --- CodeMirror-orig/src/line/line_data.js	2020-04-21 12:47:20.000000000 +0200 | --- codemirror-5.59.3-orig/src/line/line_data.js	2021-02-20 21:24:57.000000000 +0000 | ||||||
| +++ CodeMirror-edit/src/line/line_data.js	2020-05-02 03:17:02.785065000 +0200 | +++ codemirror-5.59.3/src/line/line_data.js	2021-02-21 20:45:36.472549599 +0000 | ||||||
| @@ -79,6 +79,6 @@ | @@ -79,6 +79,6 @@ | ||||||
|      // Optionally wire in some hacks into the token-rendering |      // Optionally wire in some hacks into the token-rendering | ||||||
|      // algorithm, to deal with browser quirks. |      // algorithm, to deal with browser quirks. | ||||||
| @@ -158,9 +158,9 @@ diff -NarU2 CodeMirror-orig/src/line/line_data.js CodeMirror-edit/src/line/line_ | |||||||
| +    //  builder.addToken = buildTokenBadBidi(builder.addToken, order) | +    //  builder.addToken = buildTokenBadBidi(builder.addToken, order) | ||||||
|      builder.map = [] |      builder.map = [] | ||||||
|      let allowFrontierUpdate = lineView != cm.display.externalMeasured && lineNo(line) |      let allowFrontierUpdate = lineView != cm.display.externalMeasured && lineNo(line) | ||||||
| diff -NarU2 CodeMirror-orig/src/measurement/position_measurement.js CodeMirror-edit/src/measurement/position_measurement.js | diff -NarU2 codemirror-5.59.3-orig/src/measurement/position_measurement.js codemirror-5.59.3/src/measurement/position_measurement.js | ||||||
| --- CodeMirror-orig/src/measurement/position_measurement.js	2020-04-21 12:47:20.000000000 +0200 | --- codemirror-5.59.3-orig/src/measurement/position_measurement.js	2021-02-20 21:24:57.000000000 +0000 | ||||||
| +++ CodeMirror-edit/src/measurement/position_measurement.js	2020-05-02 03:35:20.674159600 +0200 | +++ codemirror-5.59.3/src/measurement/position_measurement.js	2021-02-21 20:50:52.372945293 +0000 | ||||||
| @@ -380,5 +380,6 @@ | @@ -380,5 +380,6 @@ | ||||||
|      sticky = "after" |      sticky = "after" | ||||||
|    } |    } | ||||||
| @@ -199,9 +199,9 @@ diff -NarU2 CodeMirror-orig/src/measurement/position_measurement.js CodeMirror-e | |||||||
| +*/ | +*/ | ||||||
|   |   | ||||||
|  let measureText |  let measureText | ||||||
| diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js | diff -NarU2 codemirror-5.59.3-orig/src/util/bidi.js codemirror-5.59.3/src/util/bidi.js | ||||||
| --- CodeMirror-orig/src/util/bidi.js	2020-04-21 12:47:20.000000000 +0200 | --- codemirror-5.59.3-orig/src/util/bidi.js	2021-02-20 21:24:57.000000000 +0000 | ||||||
| +++ CodeMirror-edit/src/util/bidi.js	2020-05-02 03:12:44.418649800 +0200 | +++ codemirror-5.59.3/src/util/bidi.js	2021-02-21 20:52:18.168092225 +0000 | ||||||
| @@ -4,5 +4,5 @@ | @@ -4,5 +4,5 @@ | ||||||
|   |   | ||||||
|  export function iterateBidiSections(order, from, to, f) { |  export function iterateBidiSections(order, from, to, f) { | ||||||
| @@ -239,20 +239,19 @@ diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js | |||||||
| +  var fun = function(str, direction) { | +  var fun = function(str, direction) { | ||||||
|      let outerType = direction == "ltr" ? "L" : "R" |      let outerType = direction == "ltr" ? "L" : "R" | ||||||
|   |   | ||||||
| @@ -204,12 +210,16 @@ | @@ -204,5 +210,11 @@ | ||||||
|      return direction == "rtl" ? order.reverse() : order |      return direction == "rtl" ? order.reverse() : order | ||||||
|    } |    } | ||||||
| -})() |  | ||||||
|   |  | ||||||
| +  return function(str, direction) { | +  return function(str, direction) { | ||||||
| +    var ret = fun(str, direction); | +    var ret = fun(str, direction); | ||||||
| +    console.log("bidiOrdering inner ([%s], %s) => [%s]", str, direction, ret); | +    console.log("bidiOrdering inner ([%s], %s) => [%s]", str, direction, ret); | ||||||
| +    return ret; | +    return ret; | ||||||
| +  } | +  } | ||||||
| +})() |  })() | ||||||
| +*/ | +*/ | ||||||
|  |   | ||||||
|  // Get the bidi ordering for the given line (and cache it). Returns |  // Get the bidi ordering for the given line (and cache it). Returns | ||||||
|  // false for lines that are fully left-to-right, and an array of | @@ -210,6 +222,4 @@ | ||||||
|  // BidiSpan objects otherwise. |  // BidiSpan objects otherwise. | ||||||
|  export function getOrder(line, direction) { |  export function getOrder(line, direction) { | ||||||
| -  let order = line.order | -  let order = line.order | ||||||
| @@ -260,9 +259,9 @@ diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js | |||||||
| -  return order | -  return order | ||||||
| +  return false; | +  return false; | ||||||
|  } |  } | ||||||
| diff -NarU2 CodeMirror-orig/src/util/feature_detection.js CodeMirror-edit/src/util/feature_detection.js | diff -NarU2 codemirror-5.59.3-orig/src/util/feature_detection.js codemirror-5.59.3/src/util/feature_detection.js | ||||||
| --- CodeMirror-orig/src/util/feature_detection.js	2020-04-21 12:47:20.000000000 +0200 | --- codemirror-5.59.3-orig/src/util/feature_detection.js	2021-02-20 21:24:57.000000000 +0000 | ||||||
| +++ CodeMirror-edit/src/util/feature_detection.js	2020-05-02 03:16:21.085621400 +0200 | +++ codemirror-5.59.3/src/util/feature_detection.js	2021-02-21 20:49:22.191269270 +0000 | ||||||
| @@ -25,4 +25,5 @@ | @@ -25,4 +25,5 @@ | ||||||
|  } |  } | ||||||
|   |   | ||||||
|   | |||||||
| @@ -1,33 +1,57 @@ | |||||||
| diff -NarU2 easymde-orig/gulpfile.js easymde-mod1/gulpfile.js | diff -NarU2 easy-markdown-editor-2.14.0-orig/gulpfile.js easy-markdown-editor-2.14.0/gulpfile.js | ||||||
| --- easymde-orig/gulpfile.js	2020-04-06 14:09:36.000000000 +0200 | --- easy-markdown-editor-2.14.0-orig/gulpfile.js	2021-02-14 12:11:48.000000000 +0000 | ||||||
| +++ easymde-mod1/gulpfile.js	2020-05-01 14:33:52.260175200 +0200 | +++ easy-markdown-editor-2.14.0/gulpfile.js	2021-02-21 20:55:37.134701007 +0000 | ||||||
| @@ -25,5 +25,4 @@ | @@ -25,5 +25,4 @@ | ||||||
|      './node_modules/codemirror/lib/codemirror.css', |      './node_modules/codemirror/lib/codemirror.css', | ||||||
|      './src/css/*.css', |      './src/css/*.css', | ||||||
| -    './node_modules/codemirror-spell-checker/src/css/spell-checker.css', | -    './node_modules/codemirror-spell-checker/src/css/spell-checker.css', | ||||||
|  ]; |  ]; | ||||||
|   |   | ||||||
| diff -NarU2 easymde-orig/package.json easymde-mod1/package.json | diff -NarU2 easy-markdown-editor-2.14.0-orig/package.json easy-markdown-editor-2.14.0/package.json | ||||||
| --- easymde-orig/package.json	2020-04-06 14:09:36.000000000 +0200 | --- easy-markdown-editor-2.14.0-orig/package.json	2021-02-14 12:11:48.000000000 +0000 | ||||||
| +++ easymde-mod1/package.json	2020-05-01 14:33:57.189975800 +0200 | +++ easy-markdown-editor-2.14.0/package.json	2021-02-21 20:55:47.761190082 +0000 | ||||||
| @@ -21,5 +21,4 @@ | @@ -21,5 +21,4 @@ | ||||||
|      "dependencies": { |      "dependencies": { | ||||||
|          "codemirror": "^5.52.2", |          "codemirror": "^5.59.2", | ||||||
| -        "codemirror-spell-checker": "1.1.2", | -        "codemirror-spell-checker": "1.1.2", | ||||||
|          "marked": "^0.8.2" |          "marked": "^2.0.0" | ||||||
|      }, |      }, | ||||||
| diff -NarU2 easymde-orig/src/js/easymde.js easymde-mod1/src/js/easymde.js | diff -NarU2 easy-markdown-editor-2.14.0-orig/src/js/easymde.js easy-markdown-editor-2.14.0/src/js/easymde.js | ||||||
| --- easymde-orig/src/js/easymde.js	2020-04-06 14:09:36.000000000 +0200 | --- easy-markdown-editor-2.14.0-orig/src/js/easymde.js	2021-02-14 12:11:48.000000000 +0000 | ||||||
| +++ easymde-mod1/src/js/easymde.js	2020-05-01 14:34:19.878774400 +0200 | +++ easy-markdown-editor-2.14.0/src/js/easymde.js	2021-02-21 20:57:09.143171536 +0000 | ||||||
| @@ -11,5 +11,4 @@ | @@ -12,5 +12,4 @@ | ||||||
|  require('codemirror/mode/gfm/gfm.js'); |  require('codemirror/mode/gfm/gfm.js'); | ||||||
|  require('codemirror/mode/xml/xml.js'); |  require('codemirror/mode/xml/xml.js'); | ||||||
| -var CodeMirrorSpellChecker = require('codemirror-spell-checker'); | -var CodeMirrorSpellChecker = require('codemirror-spell-checker'); | ||||||
|  var marked = require('marked/lib/marked'); |  var marked = require('marked/lib/marked'); | ||||||
|   |   | ||||||
| @@ -1889,18 +1888,7 @@ | @@ -1762,9 +1761,4 @@ | ||||||
|  |          options.autosave.uniqueId = options.autosave.unique_id; | ||||||
|   |   | ||||||
|  | -    // If overlay mode is specified and combine is not provided, default it to true | ||||||
|  | -    if (options.overlayMode && options.overlayMode.combine === undefined) { | ||||||
|  | -      options.overlayMode.combine = true; | ||||||
|  | -    } | ||||||
|  | - | ||||||
|  |      // Update this options | ||||||
|  |      this.options = options; | ||||||
|  | @@ -2003,28 +1997,7 @@ | ||||||
|      var mode, backdrop; |      var mode, backdrop; | ||||||
|  |   | ||||||
|  | -    // CodeMirror overlay mode | ||||||
|  | -    if (options.overlayMode) { | ||||||
|  | -      CodeMirror.defineMode('overlay-mode', function(config) { | ||||||
|  | -        return CodeMirror.overlayMode(CodeMirror.getMode(config, options.spellChecker !== false ? 'spell-checker' : 'gfm'), options.overlayMode.mode, options.overlayMode.combine); | ||||||
|  | -      }); | ||||||
|  | - | ||||||
|  | -      mode = 'overlay-mode'; | ||||||
|  | -      backdrop = options.parsingConfig; | ||||||
|  | -      backdrop.gitHubSpice = false; | ||||||
|  | -    } else { | ||||||
|  |          mode = options.parsingConfig; | ||||||
|  |          mode.name = 'gfm'; | ||||||
|  |          mode.gitHubSpice = false; | ||||||
|  | -    } | ||||||
| -    if (options.spellChecker !== false) { | -    if (options.spellChecker !== false) { | ||||||
| -        mode = 'spell-checker'; | -        mode = 'spell-checker'; | ||||||
| -        backdrop = options.parsingConfig; | -        backdrop = options.parsingConfig; | ||||||
| @@ -37,16 +61,28 @@ diff -NarU2 easymde-orig/src/js/easymde.js easymde-mod1/src/js/easymde.js | |||||||
| -        CodeMirrorSpellChecker({ | -        CodeMirrorSpellChecker({ | ||||||
| -            codeMirrorInstance: CodeMirror, | -            codeMirrorInstance: CodeMirror, | ||||||
| -        }); | -        }); | ||||||
| -    } else { |  | ||||||
|          mode = options.parsingConfig; |  | ||||||
|          mode.name = 'gfm'; |  | ||||||
|          mode.gitHubSpice = false; |  | ||||||
| -    } | -    } | ||||||
|   |   | ||||||
|      // eslint-disable-next-line no-unused-vars |      // eslint-disable-next-line no-unused-vars | ||||||
| @@ -1927,5 +1915,4 @@ | diff -NarU2 easy-markdown-editor-2.14.0-orig/types/easymde.d.ts easy-markdown-editor-2.14.0/types/easymde.d.ts | ||||||
|          configureMouse: configureMouse, | --- easy-markdown-editor-2.14.0-orig/types/easymde.d.ts	2021-02-14 12:11:48.000000000 +0000 | ||||||
|          inputStyle: (options.inputStyle != undefined) ? options.inputStyle : isMobile() ? 'contenteditable' : 'textarea', | +++ easy-markdown-editor-2.14.0/types/easymde.d.ts	2021-02-21 20:57:42.492620979 +0000 | ||||||
| -        spellcheck: (options.nativeSpellcheck != undefined) ? options.nativeSpellcheck : true, | @@ -160,9 +160,4 @@ | ||||||
|      }); |      } | ||||||
|   |   | ||||||
|  | -    interface OverlayModeOptions { | ||||||
|  | -      mode: CodeMirror.Mode<any> | ||||||
|  | -      combine?: boolean | ||||||
|  | -    } | ||||||
|  | - | ||||||
|  |      interface Options { | ||||||
|  |          autoDownloadFontAwesome?: boolean; | ||||||
|  | @@ -214,7 +209,5 @@ | ||||||
|  |   | ||||||
|  |          promptTexts?: PromptTexts; | ||||||
|  | -        syncSideBySidePreviewScroll?: boolean; | ||||||
|  | - | ||||||
|  | -        overlayMode?: OverlayModeOptions | ||||||
|  | +        syncSideBySidePreviewScroll?: boolean | ||||||
|  |      } | ||||||
|  |  } | ||||||
|   | |||||||
| @@ -86,6 +86,8 @@ function have() { | |||||||
| 	python -c "import $1; $1; $1.__version__" | 	python -c "import $1; $1; $1.__version__" | ||||||
| } | } | ||||||
|  |  | ||||||
|  | mv copyparty/web/deps/marked.full.js.gz srv/ || true | ||||||
|  |  | ||||||
| . buildenv/bin/activate | . buildenv/bin/activate | ||||||
| have setuptools | have setuptools | ||||||
| have wheel | have wheel | ||||||
|   | |||||||
| @@ -28,6 +28,13 @@ gtar=$(command -v gtar || command -v gnutar) || true | |||||||
| 	unexpand() { gunexpand "$@"; } | 	unexpand() { gunexpand "$@"; } | ||||||
| 	command -v grealpath >/dev/null && | 	command -v grealpath >/dev/null && | ||||||
| 		realpath() { grealpath "$@"; } | 		realpath() { grealpath "$@"; } | ||||||
|  |  | ||||||
|  | 	[ -e /opt/local/bin/bzip2 ] && | ||||||
|  | 		bzip2() { /opt/local/bin/bzip2 "$@"; } | ||||||
|  | } | ||||||
|  | pybin=$(command -v python3 || command -v python) || { | ||||||
|  | 	echo need python | ||||||
|  | 	exit 1 | ||||||
| } | } | ||||||
|  |  | ||||||
| [ -e copyparty/__main__.py ] || cd .. | [ -e copyparty/__main__.py ] || cd .. | ||||||
| @@ -38,11 +45,17 @@ gtar=$(command -v gtar || command -v gnutar) || true | |||||||
| 	exit 1 | 	exit 1 | ||||||
| } | } | ||||||
|  |  | ||||||
|  | use_gz= | ||||||
|  | do_sh=1 | ||||||
|  | do_py=1 | ||||||
| while [ ! -z "$1" ]; do | while [ ! -z "$1" ]; do | ||||||
| 	[ "$1" = clean  ] && clean=1  && shift && continue | 	[ "$1" = clean  ] && clean=1  && shift && continue | ||||||
| 	[ "$1" = re     ] && repack=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-ogv ] && no_ogv=1 && shift && continue | ||||||
| 	[ "$1" = no-cm  ] && no_cm=1  && shift && continue | 	[ "$1" = no-cm  ] && no_cm=1  && shift && continue | ||||||
|  | 	[ "$1" = no-sh  ] && do_sh=   && shift && continue | ||||||
|  | 	[ "$1" = no-py  ] && do_py=   && shift && continue | ||||||
| 	break | 	break | ||||||
| done | done | ||||||
|  |  | ||||||
| @@ -104,7 +117,7 @@ cd sfx | |||||||
| ver= | ver= | ||||||
| git describe --tags >/dev/null 2>/dev/null && { | git describe --tags >/dev/null 2>/dev/null && { | ||||||
| 	git_ver="$(git describe --tags)";  # v0.5.5-2-gb164aa0 | 	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= | 	t_ver= | ||||||
|  |  | ||||||
| 	printf '%s\n' "$git_ver" | grep -qE '^v[0-9\.]+$' && { | 	printf '%s\n' "$git_ver" | grep -qE '^v[0-9\.]+$' && { | ||||||
| @@ -122,7 +135,7 @@ git describe --tags >/dev/null 2>/dev/null && { | |||||||
| 		exit 1 | 		exit 1 | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	dt="$(git log -1 --format=%cd --date=format:'%Y,%m,%d' | sed -E 's/,0?/, /g')" | 	dt="$(git log -1 --format=%cd --date=short | sed -E 's/-0?/, /g')" | ||||||
| 	printf 'git %3s: \033[36m%s\033[0m\n' ver "$ver" dt "$dt" | 	printf 'git %3s: \033[36m%s\033[0m\n' ver "$ver" dt "$dt" | ||||||
| 	sed -ri ' | 	sed -ri ' | ||||||
| 		s/^(VERSION =)(.*)/#\1\2\n\1 ('"$t_ver"')/; | 		s/^(VERSION =)(.*)/#\1\2\n\1 ('"$t_ver"')/; | ||||||
| @@ -150,7 +163,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 | 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 | echo use smol web deps | ||||||
| rm -f copyparty/web/deps/*.full.* | rm -f copyparty/web/deps/*.full.* copyparty/web/Makefile | ||||||
|  |  | ||||||
| # it's fine dw | # it's fine dw | ||||||
| grep -lE '\.full\.(js|css)' copyparty/web/* | | grep -lE '\.full\.(js|css)' copyparty/web/* | | ||||||
| @@ -169,10 +182,11 @@ done | |||||||
| 	sed -r '/edit2">edit \(fancy/d' <$f >t && tmv "$f" | 	sed -r '/edit2">edit \(fancy/d' <$f >t && tmv "$f" | ||||||
| } | } | ||||||
|  |  | ||||||
|  | [ $repack ] || | ||||||
| find | grep -E '\.py$' | | find | grep -E '\.py$' | | ||||||
|   grep -vE '__version__' | |   grep -vE '__version__' | | ||||||
|   tr '\n' '\0' | |   tr '\n' '\0' | | ||||||
|   xargs -0 python ../scripts/uncomment.py |   xargs -0 $pybin ../scripts/uncomment.py | ||||||
|  |  | ||||||
| f=dep-j2/jinja2/constants.py | f=dep-j2/jinja2/constants.py | ||||||
| awk '/^LOREM_IPSUM_WORDS/{o=1;print "LOREM_IPSUM_WORDS = u\"a\"";next} !o; /"""/{o=0}' <$f >t | awk '/^LOREM_IPSUM_WORDS/{o=1;print "LOREM_IPSUM_WORDS = u\"a\"";next} !o; /"""/{o=0}' <$f >t | ||||||
| @@ -180,39 +194,74 @@ tmv "$f" | |||||||
|  |  | ||||||
| # up2k goes from 28k to 22k laff | # up2k goes from 28k to 22k laff | ||||||
| echo entabbening | echo entabbening | ||||||
| find | grep -E '\.(js|css|html|py)$' | while IFS= read -r f; do | find | grep -E '\.(js|css|html)$' | while IFS= read -r f; do | ||||||
| 	unexpand -t 4 --first-only <"$f" >t | 	unexpand -t 4 --first-only <"$f" >t | ||||||
| 	tmv "$f" | 	tmv "$f" | ||||||
| done | done | ||||||
|  |  | ||||||
|  | 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$' list1; grep -E 'gz$' list1) >list | ||||||
|  |  | ||||||
| echo creating tar | echo creating tar | ||||||
| args=(--owner=1000 --group=1000) | args=(--owner=1000 --group=1000) | ||||||
| [ "$OSTYPE" = msys ] && | [ "$OSTYPE" = msys ] && | ||||||
| 	args=() | 	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 | echo compressing tar | ||||||
| # detect best level; bzip2 -7 is usually better than -9 | # detect best level; bzip2 -7 is usually better than -9 | ||||||
| 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_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; } | ||||||
| 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_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.* | rm t.* || true | ||||||
|  | exts=() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [ $do_sh ] && { | ||||||
|  | exts+=(.sh) | ||||||
| echo creating unix sfx | echo creating unix sfx | ||||||
| ( | ( | ||||||
| 	sed "s/PACK_TS/$ts/; s/PACK_HTS/$hts/; s/CPP_VER/$ver/" <../scripts/sfx.sh | | 	sed "s/PACK_TS/$ts/; s/PACK_HTS/$hts/; s/CPP_VER/$ver/" <../scripts/sfx.sh | | ||||||
| 	grep -E '^sfx_eof$' -B 9001; | 	grep -E '^sfx_eof$' -B 9001; | ||||||
| 	cat tar.xz | 	cat tar.xz | ||||||
| ) >$sfx_out.sh | ) >$sfx_out.sh | ||||||
|  | } | ||||||
|  |  | ||||||
| echo creating generic sfx |  | ||||||
| python ../scripts/sfx.py --sfx-make tar.bz2 $ver $ts | [ $do_py ] && { | ||||||
| mv sfx.out $sfx_out.py | 	echo creating generic sfx | ||||||
| chmod 755 $sfx_out.* |  | ||||||
|  | 	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" | printf "done:\n" | ||||||
| printf "  %s\n" "$(realpath $sfx_out)."{sh,py} | for ext in ${exts[@]}; do | ||||||
| # rm -rf * | 	printf "  %s\n" "$(realpath $sfx_out)"$ext | ||||||
|  | done | ||||||
|  |  | ||||||
| # tar -tvf ../sfx/tar | sed -r 's/(.* ....-..-.. ..:.. )(.*)/\2 `` \1/' | sort | sed -r 's/(.*) `` (.*)/\2 \1/'| less | # apk add bash python3 tar xz bzip2 | ||||||
| # for n in {1..9}; do tar -tf tar | grep -vE '/$' | sed -r 's/(.*)\.(.*)/\2.\1/' | sort | sed -r 's/([^\.]+)\.(.*)/\2.\1/' | tar -cT- | bzip2 -c$n | wc -c; done  | # while true; do ./make-sfx.sh; for f in ..//dist/copyparty-sfx.{sh,py}; do mv $f $f.$(wc -c <$f | awk '{print$1}'); done; done | ||||||
|   | |||||||
| @@ -35,6 +35,8 @@ ver="$1" | |||||||
| 	exit 1 | 	exit 1 | ||||||
| } | } | ||||||
|  |  | ||||||
|  | mv copyparty/web/deps/marked.full.js.gz srv/ || true | ||||||
|  |  | ||||||
| mkdir -p dist | mkdir -p dist | ||||||
| zip_path="$(pwd)/dist/copyparty-$ver.zip" | zip_path="$(pwd)/dist/copyparty-$ver.zip" | ||||||
| tgz_path="$(pwd)/dist/copyparty-$ver.tar.gz" | tgz_path="$(pwd)/dist/copyparty-$ver.tar.gz" | ||||||
|   | |||||||
							
								
								
									
										208
									
								
								scripts/sfx.py
									
									
									
									
									
								
							
							
						
						
									
										208
									
								
								scripts/sfx.py
									
									
									
									
									
								
							| @@ -1,8 +1,8 @@ | |||||||
| #!/usr/bin/env python | #!/usr/bin/env python | ||||||
| # coding: utf-8 | # coding: latin-1 | ||||||
| from __future__ import print_function, unicode_literals | from __future__ import print_function, unicode_literals | ||||||
|  |  | ||||||
| import os, sys, time, shutil, signal, tarfile, hashlib, platform, tempfile | import re, os, sys, time, shutil, signal, threading, tarfile, hashlib, platform, tempfile, traceback | ||||||
| import subprocess as sp | import subprocess as sp | ||||||
|  |  | ||||||
| """ | """ | ||||||
| @@ -27,21 +27,21 @@ CKSUM = None | |||||||
| STAMP = None | STAMP = None | ||||||
|  |  | ||||||
| PY2 = sys.version_info[0] == 2 | PY2 = sys.version_info[0] == 2 | ||||||
|  | WINDOWS = sys.platform in ["win32", "msys"] | ||||||
| sys.dont_write_bytecode = True | sys.dont_write_bytecode = True | ||||||
| me = os.path.abspath(os.path.realpath(__file__)) | me = os.path.abspath(os.path.realpath(__file__)) | ||||||
| cpp = None |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def eprint(*args, **kwargs): | def eprint(*a, **ka): | ||||||
|     kwargs["file"] = sys.stderr |     ka["file"] = sys.stderr | ||||||
|     print(*args, **kwargs) |     print(*a, **ka) | ||||||
|  |  | ||||||
|  |  | ||||||
| def msg(*args, **kwargs): | def msg(*a, **ka): | ||||||
|     if args: |     if a: | ||||||
|         args = ["[SFX]", args[0]] + list(args[1:]) |         a = ["[SFX]", a[0]] + list(a[1:]) | ||||||
|  |  | ||||||
|     eprint(*args, **kwargs) |     eprint(*a, **ka) | ||||||
|  |  | ||||||
|  |  | ||||||
| # skip 1 | # skip 1 | ||||||
| @@ -156,6 +156,9 @@ def encode(data, size, cksum, ver, ts): | |||||||
|                 skip = True |                 skip = True | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|  |             if ln.strip().startswith("# fmt: "): | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|             unpk += ln + "\n" |             unpk += ln + "\n" | ||||||
|  |  | ||||||
|         for k, v in [ |         for k, v in [ | ||||||
| @@ -209,11 +212,11 @@ def yieldfile(fn): | |||||||
|  |  | ||||||
|  |  | ||||||
| def hashfile(fn): | def hashfile(fn): | ||||||
|     hasher = hashlib.md5() |     h = hashlib.md5() | ||||||
|     for block in yieldfile(fn): |     for block in yieldfile(fn): | ||||||
|         hasher.update(block) |         h.update(block) | ||||||
|  |  | ||||||
|     return hasher.hexdigest() |     return h.hexdigest() | ||||||
|  |  | ||||||
|  |  | ||||||
| def unpack(): | def unpack(): | ||||||
| @@ -222,9 +225,10 @@ def unpack(): | |||||||
|     tag = "v" + str(STAMP) |     tag = "v" + str(STAMP) | ||||||
|     withpid = "{}.{}".format(name, os.getpid()) |     withpid = "{}.{}".format(name, os.getpid()) | ||||||
|     top = tempfile.gettempdir() |     top = tempfile.gettempdir() | ||||||
|     final = os.path.join(top, name) |     opj = os.path.join | ||||||
|     mine = os.path.join(top, withpid) |     final = opj(top, name) | ||||||
|     tar = os.path.join(mine, "tar") |     mine = opj(top, withpid) | ||||||
|  |     tar = opj(mine, "tar") | ||||||
|  |  | ||||||
|     try: |     try: | ||||||
|         if tag in os.listdir(final): |         if tag in os.listdir(final): | ||||||
| @@ -233,28 +237,24 @@ def unpack(): | |||||||
|     except: |     except: | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     nwrite = 0 |     sz = 0 | ||||||
|     os.mkdir(mine) |     os.mkdir(mine) | ||||||
|     with open(tar, "wb") as f: |     with open(tar, "wb") as f: | ||||||
|         for buf in get_payload(): |         for buf in get_payload(): | ||||||
|             nwrite += len(buf) |             sz += len(buf) | ||||||
|             f.write(buf) |             f.write(buf) | ||||||
|  |  | ||||||
|     if nwrite != SIZE: |     ck = hashfile(tar) | ||||||
|         t = "\n\n  bad file:\n    expected {} bytes, got {}\n".format(SIZE, nwrite) |     if ck != CKSUM: | ||||||
|         raise Exception(t) |         t = "\n\nexpected {} ({} byte)\nobtained {} ({} byte)\nsfx corrupt" | ||||||
|  |         raise Exception(t.format(CKSUM, SIZE, ck, sz)) | ||||||
|     cksum = hashfile(tar) |  | ||||||
|     if cksum != CKSUM: |  | ||||||
|         t = "\n\n  bad file:\n    {} expected,\n    {} obtained\n".format(CKSUM, cksum) |  | ||||||
|         raise Exception(t) |  | ||||||
|  |  | ||||||
|     with tarfile.open(tar, "r:bz2") as tf: |     with tarfile.open(tar, "r:bz2") as tf: | ||||||
|         tf.extractall(mine) |         tf.extractall(mine) | ||||||
|  |  | ||||||
|     os.remove(tar) |     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") |         f.write(b"h\n") | ||||||
|  |  | ||||||
|     try: |     try: | ||||||
| @@ -272,25 +272,25 @@ def unpack(): | |||||||
|     except: |     except: | ||||||
|         pass |         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: |     try: | ||||||
|         os.symlink(mine, final) |         os.symlink(mine, final) | ||||||
|     except: |     except: | ||||||
|         try: |         try: | ||||||
|             os.rename(mine, final) |             os.rename(mine, final) | ||||||
|  |             return final | ||||||
|         except: |         except: | ||||||
|             msg("reloc fail,", mine) |             msg("reloc fail,", mine) | ||||||
|             return mine |  | ||||||
|  |  | ||||||
|     for fn in u8(os.listdir(top)): |     return mine | ||||||
|         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 |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_payload(): | def get_payload(): | ||||||
| @@ -307,96 +307,119 @@ def get_payload(): | |||||||
|         if ofs < 0: |         if ofs < 0: | ||||||
|             raise Exception("could not find archive marker") |             raise Exception("could not find archive marker") | ||||||
|  |  | ||||||
|         # start reading from the final b"\n" |         # start at final b"\n" | ||||||
|         fpos = ofs + len(ptn) - 3 |         fpos = ofs + len(ptn) - 3 | ||||||
|         # msg("tar found at", fpos) |  | ||||||
|         f.seek(fpos) |         f.seek(fpos) | ||||||
|         dpos = 0 |         dpos = 0 | ||||||
|         leftovers = b"" |         rem = b"" | ||||||
|         while True: |         while True: | ||||||
|             rbuf = f.read(1024 * 32) |             rbuf = f.read(1024 * 32) | ||||||
|             if rbuf: |             if rbuf: | ||||||
|                 buf = leftovers + rbuf |                 buf = rem + rbuf | ||||||
|                 ofs = buf.rfind(b"\n") |                 ofs = buf.rfind(b"\n") | ||||||
|                 if len(buf) <= 4: |                 if len(buf) <= 4: | ||||||
|                     leftovers = buf |                     rem = buf | ||||||
|                     continue |                     continue | ||||||
|  |  | ||||||
|                 if ofs >= len(buf) - 4: |                 if ofs >= len(buf) - 4: | ||||||
|                     leftovers = buf[ofs:] |                     rem = buf[ofs:] | ||||||
|                     buf = buf[:ofs] |                     buf = buf[:ofs] | ||||||
|                 else: |                 else: | ||||||
|                     leftovers = b"\n# " |                     rem = b"\n# " | ||||||
|             else: |             else: | ||||||
|                 buf = leftovers |                 buf = rem | ||||||
|  |  | ||||||
|             fpos += len(buf) + 1 |             fpos += len(buf) + 1 | ||||||
|             buf = ( |             for a, b in [[b"\n# ", b""], [b"\n#r", b"\r"], [b"\n#n", b"\n"]]: | ||||||
|                 buf.replace(b"\n# ", b"") |                 buf = buf.replace(a, b) | ||||||
|                 .replace(b"\n#r", b"\r") |  | ||||||
|                 .replace(b"\n#n", b"\n") |  | ||||||
|             ) |  | ||||||
|             dpos += len(buf) - 1 |  | ||||||
|  |  | ||||||
|  |             dpos += len(buf) - 1 | ||||||
|             yield buf |             yield buf | ||||||
|  |  | ||||||
|             if not rbuf: |             if not rbuf: | ||||||
|                 break |                 break | ||||||
|  |  | ||||||
|  |  | ||||||
| def confirm(): | 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() | ||||||
|  |     msg("retcode", rv if rv else traceback.format_exc()) | ||||||
|     msg("*** hit enter to exit ***") |     msg("*** hit enter to exit ***") | ||||||
|     try: |     try: | ||||||
|         raw_input() if PY2 else input() |         raw_input() if PY2 else input() | ||||||
|     except: |     except: | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|  |     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("sfxdir:", tmp) | ||||||
|  |     msg() | ||||||
|  |  | ||||||
|     # "systemd-tmpfiles-clean.timer"?? HOW do you even come up with this shit |     # block systemd-tmpfiles-clean.timer | ||||||
|     try: |     try: | ||||||
|         import fcntl |         import fcntl | ||||||
|  |  | ||||||
|         fd = os.open(tmp, os.O_RDONLY) |         fd = os.open(tmp, os.O_RDONLY) | ||||||
|         fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB) |         fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB) | ||||||
|         tmp = os.readlink(tmp)  # can't flock a symlink, even with O_NOFOLLOW |     except Exception as ex: | ||||||
|     except: |         if not WINDOWS: | ||||||
|         pass |             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")] |     ld = [tmp, os.path.join(tmp, "dep-j2")] | ||||||
|     if j2ver: |     if j2: | ||||||
|         del ld[-1] |         del ld[-1] | ||||||
|  |  | ||||||
|     cmd = ( |     if any([re.match(r"^-.*j[0-9]", x) for x in sys.argv]): | ||||||
|         "import sys, runpy; " |         run_s(ld) | ||||||
|         + "".join(['sys.path.insert(0, r"' + x + '"); ' for x in ld]) |     else: | ||||||
|         + 'runpy.run_module("copyparty", run_name="__main__")' |         run_i(ld) | ||||||
|     ) |  | ||||||
|     cmd = [sys.executable, "-c", cmd] + list(sys.argv[1:]) |  | ||||||
|  |  | ||||||
|     cmd = [str(x) for x in cmd] |  | ||||||
|     msg("\n", cmd, "\n") |  | ||||||
|     cpp = sp.Popen(cmd) |  | ||||||
|     try: |  | ||||||
|         cpp.wait() |  | ||||||
|     except: |  | ||||||
|         cpp.wait() |  | ||||||
|  |  | ||||||
|     if cpp.returncode != 0: |  | ||||||
|         confirm() |  | ||||||
|  |  | ||||||
|     sys.exit(cpp.returncode) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def bye(sig, frame): | def run_i(ld): | ||||||
|     if cpp is not None: |     for x in ld: | ||||||
|         cpp.terminate() |         sys.path.insert(0, x) | ||||||
|  |  | ||||||
|  |     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(): | def main(): | ||||||
| @@ -430,16 +453,23 @@ def main(): | |||||||
|  |  | ||||||
|     # skip 0 |     # skip 0 | ||||||
|  |  | ||||||
|     signal.signal(signal.SIGTERM, bye) |     tmp = os.path.realpath(unpack()) | ||||||
|  |  | ||||||
|     tmp = unpack() |  | ||||||
|  |  | ||||||
|     try: |     try: | ||||||
|         from jinja2 import __version__ as j2ver |         from jinja2 import __version__ as j2 | ||||||
|     except: |     except: | ||||||
|         j2ver = None |         j2 = None | ||||||
|  |  | ||||||
|     return 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__": | if __name__ == "__main__": | ||||||
|   | |||||||
| @@ -17,14 +17,15 @@ __license__ = "MIT" | |||||||
| __url__ = "https://github.com/9001/copyparty/" | __url__ = "https://github.com/9001/copyparty/" | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_spd(nbyte, nsec): | def get_spd(nbyte, nfiles, nsec): | ||||||
|     if not 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) |     mb = nbyte / (1024 * 1024.0) | ||||||
|     spd = mb / nsec |     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): | class Inf(object): | ||||||
| @@ -36,6 +37,7 @@ class Inf(object): | |||||||
|         self.mtx_reports = threading.Lock() |         self.mtx_reports = threading.Lock() | ||||||
|  |  | ||||||
|         self.n_byte = 0 |         self.n_byte = 0 | ||||||
|  |         self.n_file = 0 | ||||||
|         self.n_sec = 0 |         self.n_sec = 0 | ||||||
|         self.n_done = 0 |         self.n_done = 0 | ||||||
|         self.t0 = t0 |         self.t0 = t0 | ||||||
| @@ -63,7 +65,8 @@ class Inf(object): | |||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             msgs = msgs[-64:] |             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)) |             print("\n".join(msgs)) | ||||||
|  |  | ||||||
|     def report(self, fn, n_byte, n_sec): |     def report(self, fn, n_byte, n_sec): | ||||||
| @@ -131,8 +134,9 @@ def main(): | |||||||
|  |  | ||||||
|     num_threads = 8 |     num_threads = 8 | ||||||
|     read_sz = 32 * 1024 |     read_sz = 32 * 1024 | ||||||
|  |     targs = (q, inf, read_sz) | ||||||
|     for _ in range(num_threads): |     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.daemon = True | ||||||
|         thr.start() |         thr.start() | ||||||
|  |  | ||||||
| @@ -151,14 +155,14 @@ def main(): | |||||||
|     log = inf.reports |     log = inf.reports | ||||||
|     log.sort() |     log.sort() | ||||||
|     for nbyte, nsec, fn in log[-64:]: |     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() | ||||||
|     print("\n".join(inf.errors)) |     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__": | if __name__ == "__main__": | ||||||
|     main() |     main() | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										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 | ||||||
|  | """ | ||||||
							
								
								
									
										202
									
								
								tests/test_httpcli.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										202
									
								
								tests/test_httpcli.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,202 @@ | |||||||
|  | #!/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 unittest | ||||||
|  |  | ||||||
|  | from argparse import Namespace | ||||||
|  | from copyparty.authsrv import AuthSrv | ||||||
|  | from copyparty.httpcli import HttpCli | ||||||
|  |  | ||||||
|  | from tests import util as tu | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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, | ||||||
|  |             ed=False, | ||||||
|  |             no_zip=False, | ||||||
|  |             no_scandir=False, | ||||||
|  |             no_sendfile=True, | ||||||
|  |             nih=True, | ||||||
|  |             mtp=[], | ||||||
|  |             mte="a", | ||||||
|  |             **{k: False for k in "e2d e2ds e2dsa e2t e2ts e2tsr".split()} | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestHttpCli(unittest.TestCase): | ||||||
|  |     def test(self): | ||||||
|  |         td = os.path.join(tu.get_ramdisk(), "vfs") | ||||||
|  |         try: | ||||||
|  |             shutil.rmtree(td) | ||||||
|  |         except OSError: | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         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.auth = 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.auth, 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.auth, 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,18 +3,26 @@ | |||||||
| from __future__ import print_function, unicode_literals | from __future__ import print_function, unicode_literals | ||||||
|  |  | ||||||
| import os | import os | ||||||
| import time |  | ||||||
| import json | import json | ||||||
| import shutil | import shutil | ||||||
| import tempfile | import tempfile | ||||||
| import unittest | import unittest | ||||||
| import subprocess as sp  # nosec |  | ||||||
|  |  | ||||||
| from textwrap import dedent | from textwrap import dedent | ||||||
| from argparse import Namespace | from argparse import Namespace | ||||||
| from copyparty.authsrv import AuthSrv | from copyparty.authsrv import AuthSrv | ||||||
| from copyparty import util | from copyparty import util | ||||||
|  |  | ||||||
|  | from tests import util as tu | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Cfg(Namespace): | ||||||
|  |     def __init__(self, a=[], v=[], c=None): | ||||||
|  |         ex = {k: False for k in "e2d e2ds e2dsa e2t e2ts e2tsr".split()} | ||||||
|  |         ex["mtp"] = [] | ||||||
|  |         ex["mte"] = "a" | ||||||
|  |         super(Cfg, self).__init__(a=a, v=v, c=c, **ex) | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestVFS(unittest.TestCase): | class TestVFS(unittest.TestCase): | ||||||
|     def dump(self, vfs): |     def dump(self, vfs): | ||||||
| @@ -35,54 +43,19 @@ class TestVFS(unittest.TestCase): | |||||||
|     def ls(self, vfs, vpath, uname): |     def ls(self, vfs, vpath, uname): | ||||||
|         """helper for resolving and listing a folder""" |         """helper for resolving and listing a folder""" | ||||||
|         vn, rem = vfs.get(vpath, uname, True, False) |         vn, rem = vfs.get(vpath, uname, True, False) | ||||||
|         return vn.ls(rem, uname) |         r1 = vn.ls(rem, uname, False) | ||||||
|  |         r2 = vn.ls(rem, uname, False) | ||||||
|  |         self.assertEqual(r1, r2) | ||||||
|  |  | ||||||
|     def runcmd(self, *argv): |         fsdir, real, virt = r1 | ||||||
|         p = sp.Popen(argv, stdout=sp.PIPE, stderr=sp.PIPE) |         real = [x[0] for x in real] | ||||||
|         stdout, stderr = p.communicate() |         return fsdir, real, virt | ||||||
|         stdout = stdout.decode("utf-8") |  | ||||||
|         stderr = stderr.decode("utf-8") |  | ||||||
|         return [p.returncode, stdout, stderr] |  | ||||||
|  |  | ||||||
|     def chkcmd(self, *argv): |     def log(self, src, msg, c=0): | ||||||
|         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): |  | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     def test(self): |     def test(self): | ||||||
|         td = os.path.join(self.get_ramdisk(), "vfs") |         td = os.path.join(tu.get_ramdisk(), "vfs") | ||||||
|         try: |         try: | ||||||
|             shutil.rmtree(td) |             shutil.rmtree(td) | ||||||
|         except OSError: |         except OSError: | ||||||
| @@ -102,7 +75,7 @@ class TestVFS(unittest.TestCase): | |||||||
|                             f.write(fn) |                             f.write(fn) | ||||||
|  |  | ||||||
|         # defaults |         # defaults | ||||||
|         vfs = AuthSrv(Namespace(c=None, a=[], v=[]), self.log).vfs |         vfs = AuthSrv(Cfg(), self.log).vfs | ||||||
|         self.assertEqual(vfs.nodes, {}) |         self.assertEqual(vfs.nodes, {}) | ||||||
|         self.assertEqual(vfs.vpath, "") |         self.assertEqual(vfs.vpath, "") | ||||||
|         self.assertEqual(vfs.realpath, td) |         self.assertEqual(vfs.realpath, td) | ||||||
| @@ -110,7 +83,7 @@ class TestVFS(unittest.TestCase): | |||||||
|         self.assertEqual(vfs.uwrite, ["*"]) |         self.assertEqual(vfs.uwrite, ["*"]) | ||||||
|  |  | ||||||
|         # single read-only rootfs (relative path) |         # single read-only rootfs (relative path) | ||||||
|         vfs = AuthSrv(Namespace(c=None, a=[], v=["a/ab/::r"]), self.log).vfs |         vfs = AuthSrv(Cfg(v=["a/ab/::r"]), self.log).vfs | ||||||
|         self.assertEqual(vfs.nodes, {}) |         self.assertEqual(vfs.nodes, {}) | ||||||
|         self.assertEqual(vfs.vpath, "") |         self.assertEqual(vfs.vpath, "") | ||||||
|         self.assertEqual(vfs.realpath, os.path.join(td, "a", "ab")) |         self.assertEqual(vfs.realpath, os.path.join(td, "a", "ab")) | ||||||
| @@ -118,9 +91,7 @@ class TestVFS(unittest.TestCase): | |||||||
|         self.assertEqual(vfs.uwrite, []) |         self.assertEqual(vfs.uwrite, []) | ||||||
|  |  | ||||||
|         # single read-only rootfs (absolute path) |         # single read-only rootfs (absolute path) | ||||||
|         vfs = AuthSrv( |         vfs = AuthSrv(Cfg(v=[td + "//a/ac/../aa//::r"]), self.log).vfs | ||||||
|             Namespace(c=None, a=[], v=[td + "//a/ac/../aa//::r"]), self.log |  | ||||||
|         ).vfs |  | ||||||
|         self.assertEqual(vfs.nodes, {}) |         self.assertEqual(vfs.nodes, {}) | ||||||
|         self.assertEqual(vfs.vpath, "") |         self.assertEqual(vfs.vpath, "") | ||||||
|         self.assertEqual(vfs.realpath, os.path.join(td, "a", "aa")) |         self.assertEqual(vfs.realpath, os.path.join(td, "a", "aa")) | ||||||
| @@ -129,7 +100,7 @@ class TestVFS(unittest.TestCase): | |||||||
|  |  | ||||||
|         # read-only rootfs with write-only subdirectory (read-write for k) |         # read-only rootfs with write-only subdirectory (read-write for k) | ||||||
|         vfs = AuthSrv( |         vfs = AuthSrv( | ||||||
|             Namespace(c=None, a=["k:k"], v=[".::r:ak", "a/ac/acb:a/ac/acb:w:ak"]), |             Cfg(a=["k:k"], v=[".::r:ak", "a/ac/acb:a/ac/acb:w:ak"]), | ||||||
|             self.log, |             self.log, | ||||||
|         ).vfs |         ).vfs | ||||||
|         self.assertEqual(len(vfs.nodes), 1) |         self.assertEqual(len(vfs.nodes), 1) | ||||||
| @@ -192,7 +163,10 @@ class TestVFS(unittest.TestCase): | |||||||
|         self.assertEqual(list(virt), []) |         self.assertEqual(list(virt), []) | ||||||
|  |  | ||||||
|         # admin-only rootfs with all-read-only subfolder |         # admin-only rootfs with all-read-only subfolder | ||||||
|         vfs = AuthSrv(Namespace(c=None, a=["k:k"], v=[".::ak", "a:a:r"]), self.log,).vfs |         vfs = AuthSrv( | ||||||
|  |             Cfg(a=["k:k"], v=[".::ak", "a:a:r"]), | ||||||
|  |             self.log, | ||||||
|  |         ).vfs | ||||||
|         self.assertEqual(len(vfs.nodes), 1) |         self.assertEqual(len(vfs.nodes), 1) | ||||||
|         self.assertEqual(vfs.vpath, "") |         self.assertEqual(vfs.vpath, "") | ||||||
|         self.assertEqual(vfs.realpath, td) |         self.assertEqual(vfs.realpath, td) | ||||||
| @@ -211,9 +185,7 @@ class TestVFS(unittest.TestCase): | |||||||
|  |  | ||||||
|         # breadth-first construction |         # breadth-first construction | ||||||
|         vfs = AuthSrv( |         vfs = AuthSrv( | ||||||
|             Namespace( |             Cfg( | ||||||
|                 c=None, |  | ||||||
|                 a=[], |  | ||||||
|                 v=[ |                 v=[ | ||||||
|                     "a/ac/acb:a/ac/acb:w", |                     "a/ac/acb:a/ac/acb:w", | ||||||
|                     "a:a:w", |                     "a:a:w", | ||||||
| @@ -234,7 +206,7 @@ class TestVFS(unittest.TestCase): | |||||||
|         self.undot(vfs, "./.././foo/..", "") |         self.undot(vfs, "./.././foo/..", "") | ||||||
|  |  | ||||||
|         # shadowing |         # shadowing | ||||||
|         vfs = AuthSrv(Namespace(c=None, a=[], v=[".::r", "b:a/ac:r"]), self.log).vfs |         vfs = AuthSrv(Cfg(v=[".::r", "b:a/ac:r"]), self.log).vfs | ||||||
|  |  | ||||||
|         fsp, r1, v1 = self.ls(vfs, "", "*") |         fsp, r1, v1 = self.ls(vfs, "", "*") | ||||||
|         self.assertEqual(fsp, td) |         self.assertEqual(fsp, td) | ||||||
| @@ -255,7 +227,7 @@ class TestVFS(unittest.TestCase): | |||||||
|         self.assertEqual(list(v1), list(v2)) |         self.assertEqual(list(v1), list(v2)) | ||||||
|  |  | ||||||
|         # config file parser |         # config file parser | ||||||
|         cfg_path = os.path.join(self.get_ramdisk(), "test.cfg") |         cfg_path = os.path.join(tu.get_ramdisk(), "test.cfg") | ||||||
|         with open(cfg_path, "wb") as f: |         with open(cfg_path, "wb") as f: | ||||||
|             f.write( |             f.write( | ||||||
|                 dedent( |                 dedent( | ||||||
| @@ -271,7 +243,7 @@ class TestVFS(unittest.TestCase): | |||||||
|                 ).encode("utf-8") |                 ).encode("utf-8") | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         au = AuthSrv(Namespace(c=[cfg_path], a=[], v=[]), self.log) |         au = AuthSrv(Cfg(c=[cfg_path]), self.log) | ||||||
|         self.assertEqual(au.user["a"], "123") |         self.assertEqual(au.user["a"], "123") | ||||||
|         self.assertEqual(au.user["asd"], "fgh:jkl") |         self.assertEqual(au.user["asd"], "fgh:jkl") | ||||||
|         n = au.vfs |         n = au.vfs | ||||||
|   | |||||||
							
								
								
									
										97
									
								
								tests/util.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								tests/util.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | |||||||
|  | import os | ||||||
|  | import time | ||||||
|  | import jinja2 | ||||||
|  | import tempfile | ||||||
|  | import subprocess as sp | ||||||
|  |  | ||||||
|  | from copyparty.util import Unrecv | ||||||
|  |  | ||||||
|  |  | ||||||
|  | J2_ENV = jinja2.Environment(loader=jinja2.BaseLoader) | ||||||
|  | J2_FILES = J2_ENV.from_string("{{ files|join('\n') }}") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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(): | ||||||
|  |     for vol in ["/dev/shm", "/Volumes/cptd"]:  # nosec (singleton test) | ||||||
|  |         if os.path.exists(vol): | ||||||
|  |             return vol | ||||||
|  |  | ||||||
|  |     if os.path.exists("/Volumes"): | ||||||
|  |         devname, _ = chkcmd("hdiutil", "attach", "-nomount", "ram://32768") | ||||||
|  |         devname = devname.strip() | ||||||
|  |         print("devname: [{}]".format(devname)) | ||||||
|  |         for _ in range(10): | ||||||
|  |             try: | ||||||
|  |                 _, _ = 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 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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, auth, log, buf): | ||||||
|  |         self.s = VSock(buf) | ||||||
|  |         self.sr = Unrecv(self.s) | ||||||
|  |         self.addr = ("127.0.0.1", "42069") | ||||||
|  |         self.args = args | ||||||
|  |         self.auth = auth | ||||||
|  |         self.log_func = log | ||||||
|  |         self.log_src = "a" | ||||||
|  |         self.hsrv = VHttpSrv() | ||||||
|  |         self.nbyte = 0 | ||||||
|  |         self.workload = 0 | ||||||
|  |         self.t0 = time.time() | ||||||
		Reference in New Issue
	
	Block a user