mirror of
				https://github.com/9001/copyparty.git
				synced 2025-10-25 00:53:47 +00:00 
			
		
		
		
	Compare commits
	
		
			150 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | e815c091b9 | ||
|  | 963529b7cf | ||
|  | 638a52374d | ||
|  | d9d42b7aa2 | ||
|  | ec7e5f36a2 | ||
|  | 56110883ea | ||
|  | 7f8d7d6006 | ||
|  | 49e4fb7e12 | ||
|  | 8dbbea473f | ||
|  | 3d375d5114 | ||
|  | f3eae67d97 | ||
|  | 40c1b19235 | ||
|  | ccaf0ab159 | ||
|  | d07f147423 | ||
|  | f5cb9f92b9 | ||
|  | f991f74983 | ||
|  | 6b3295059e | ||
|  | b18a07ae6b | ||
|  | 8ab03dabda | ||
|  | 5e760e35dc | ||
|  | afbfa04514 | ||
|  | 7aace470c5 | ||
|  | b4acb24f6a | ||
|  | bcee8a4934 | ||
|  | 36b0718542 | ||
|  | 9a92bca45d | ||
|  | b07445a363 | ||
|  | a62ec0c27e | ||
|  | 57e3a2d382 | ||
|  | b61022b374 | ||
|  | a3e2b2ec87 | ||
|  | a83d3f8801 | ||
|  | 90c5f2b9d2 | ||
|  | 4885653c07 | ||
|  | 21e1cd87ca | ||
|  | 81f82e8e9f | ||
|  | c0e31851da | ||
|  | 6599c3eced | ||
|  | 5d6c61a861 | ||
|  | 1a5c66edd3 | ||
|  | deae9fe95a | ||
|  | abd65c6334 | ||
|  | 8137a99904 | ||
|  | 6f6f9c1f74 | ||
|  | 7b575f716f | ||
|  | 6ba6ea3572 | ||
|  | 9a22ad5ea3 | ||
|  | beaab9778e | ||
|  | f327bdb6b4 | ||
|  | ae180e0f5f | ||
|  | e3f1d19756 | ||
|  | 93c2bd6ef6 | ||
|  | 4d0e5ff6db | ||
|  | 0893f06919 | ||
|  | 46b6abde3f | ||
|  | 0696610dee | ||
|  | edf0d3684c | ||
|  | 7af159f5f6 | ||
|  | 7f2cb6764a | ||
|  | 96495a9bf1 | ||
|  | b2fafec5fc | ||
|  | 0850b8ae2b | ||
|  | 8a68a96c57 | ||
|  | d3aae8ed6a | ||
|  | c62ebadda8 | ||
|  | ffcee6d390 | ||
|  | de32838346 | ||
|  | b9a4e47ea2 | ||
|  | 57d994422d | ||
|  | 6ecd745323 | ||
|  | bd769f5bdb | ||
|  | 2381692aba | ||
|  | 24fdada0a0 | ||
|  | bb5169710a | ||
|  | 9cde2352f3 | ||
|  | 482dd7a938 | ||
|  | bddcc69438 | ||
|  | 19d4540630 | ||
|  | 4f5f6c81f5 | ||
|  | 7e4c1238ba | ||
|  | f7196ac773 | ||
|  | 7a7c832000 | ||
|  | 2b4ccdbebb | ||
|  | 0d16b49489 | ||
|  | 768405b691 | ||
|  | da01413b7b | ||
|  | 914e22c53e | ||
|  | 43a23bf733 | ||
|  | 92bb00c6d2 | ||
|  | b0b97a2648 | ||
|  | 2c452fe323 | ||
|  | ad73d0c77d | ||
|  | 7f9bf1c78c | ||
|  | 61a6bc3a65 | ||
|  | 46e10b0e9f | ||
|  | 8441206e26 | ||
|  | 9fdc5ee748 | ||
|  | 00ff133387 | ||
|  | 96164cb934 | ||
|  | 82fb21ae69 | ||
|  | 89d4a2b4c4 | ||
|  | fc0c7ff374 | ||
|  | 5148c4f2e9 | ||
|  | c3b59f7bcf | ||
|  | 61e148202b | ||
|  | 8a4e0739bc | ||
|  | f75c5f2fe5 | ||
|  | 81d5859588 | ||
|  | 721886bb7a | ||
|  | b23c272820 | ||
|  | cd02bfea7a | ||
|  | 6774bd88f9 | ||
|  | 1046a4f376 | ||
|  | 8081f9ddfd | ||
|  | fa656577d1 | ||
|  | b14b86990f | ||
|  | 2a6dd7b512 | ||
|  | feebdee88b | ||
|  | 99d9277f5d | ||
|  | 9af64d6156 | ||
|  | 5e3775c1af | ||
|  | 2d2e8a3da7 | ||
|  | b2a560b76f | ||
|  | 39397a489d | ||
|  | ff593a0904 | ||
|  | f12789cf44 | ||
|  | 4f8cf2fc87 | ||
|  | fda98730ac | ||
|  | 06c6ddffb6 | ||
|  | d29f0c066c | ||
|  | c9e4de3346 | ||
|  | ca0b97f72d | ||
|  | b38f20b408 | ||
|  | 05b1dbaf56 | ||
|  | b8481e32ba | ||
|  | 9c03c65e07 | ||
|  | d8ed006b9b | ||
|  | 63c0623a5e | ||
|  | fd84506db0 | ||
|  | d8bcb44e44 | ||
|  | 56a26b0916 | ||
|  | efcf1d6b90 | ||
|  | 9f578bfec6 | ||
|  | 1f170d7d28 | ||
|  | 5ae14cf9be | ||
|  | aaf9d53be9 | ||
|  | 75c73f7ba7 | ||
|  | b6dba8beee | ||
|  | 94521cdc1a | ||
|  | 3365b1c355 | 
							
								
								
									
										15
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @@ -12,14 +12,23 @@ | ||||
|                 //"-nw", | ||||
|                 "-ed", | ||||
|                 "-emp", | ||||
|                 "-e2d", | ||||
|                 "-e2s", | ||||
|                 "-e2dsa", | ||||
|                 "-e2ts", | ||||
|                 "-a", | ||||
|                 "ed:wark", | ||||
|                 "-v", | ||||
|                 "srv::r:aed:cnodupe" | ||||
|                 "srv::r:aed:cnodupe", | ||||
|                 "-v", | ||||
|                 "dist:dist:r" | ||||
|             ] | ||||
|         }, | ||||
|         { | ||||
|             "name": "No debug", | ||||
|             "preLaunchTask": "no_dbg", | ||||
|             "type": "python", | ||||
|             //"request": "attach", "port": 42069 | ||||
|             // fork: nc -l 42069 </dev/null | ||||
|         }, | ||||
|         { | ||||
|             "name": "Run active unit test", | ||||
|             "type": "python", | ||||
|   | ||||
							
								
								
									
										12
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -50,11 +50,9 @@ | ||||
|     "files.associations": { | ||||
|         "*.makefile": "makefile" | ||||
|     }, | ||||
|     "editor.codeActionsOnSaveTimeout": 9001, | ||||
|     "editor.formatOnSaveTimeout": 9001, | ||||
|     // | ||||
|     //  things you may wanna edit: | ||||
|     // | ||||
|     "python.pythonPath": "/usr/bin/python3", | ||||
|     //"python.linting.enabled": true, | ||||
|     "python.formatting.blackArgs": [ | ||||
|         "-t", | ||||
|         "py27" | ||||
|     ], | ||||
|     "python.linting.enabled": true, | ||||
| } | ||||
							
								
								
									
										7
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							| @@ -5,6 +5,13 @@ | ||||
|             "label": "pre", | ||||
|             "command": "true;rm -rf inc/* inc/.hist/;mkdir -p inc;", | ||||
|             "type": "shell" | ||||
|         }, | ||||
|         { | ||||
|             "label": "no_dbg", | ||||
|             "type": "shell", | ||||
|             "command": "${config:python.pythonPath} -m copyparty -ed -emp -e2dsa -e2ts -a ed:wark -v srv::r:aed:cnodupe -v dist:dist:r ;exit 1" | ||||
|             // -v ~/Music/mt:mt:r:cmtp=.bpm=~/dev/copyparty/bin/mtag/audio-bpm.py:cmtp=key=~/dev/copyparty/bin/mtag/audio-key.py:ce2tsr  | ||||
|             // -v ~/Music/mt:mt:r:cmtp=.bpm=~/dev/copyparty/bin/mtag/audio-bpm.py:ce2tsr | ||||
|         } | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										183
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										183
									
								
								README.md
									
									
									
									
									
								
							| @@ -13,6 +13,30 @@ turn your phone or raspi into a portable file server with resumable uploads/down | ||||
| * code standard: `black` | ||||
|  | ||||
|  | ||||
| ## readme toc | ||||
|  | ||||
| * top | ||||
|     * [quickstart](#quickstart) | ||||
|     * [notes](#notes) | ||||
|     * [status](#status) | ||||
| * [bugs](#bugs) | ||||
| * [usage](#usage) | ||||
| * [searching](#searching) | ||||
|     * [search configuration](#search-configuration) | ||||
|     * [metadata from audio files](#metadata-from-audio-files) | ||||
|     * [file parser plugins](#file-parser-plugins) | ||||
|     * [complete examples](#complete-examples) | ||||
| * [client examples](#client-examples) | ||||
| * [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 | ||||
|  | ||||
| download [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) and you're all set! | ||||
| @@ -36,48 +60,170 @@ you may also want these, especially on servers: | ||||
|  | ||||
| ## status | ||||
|  | ||||
| * [x] sanic multipart parser | ||||
| * [x] load balancer (multiprocessing) | ||||
| * [x] upload (plain multipart, ie6 support) | ||||
| * [x] upload (js, resumable, multithreaded) | ||||
| * [x] download | ||||
| * [x] browser | ||||
| * [x] media player | ||||
| * [ ] thumbnails | ||||
| * [ ] download as zip | ||||
| * [x] volumes | ||||
| * [x] accounts | ||||
| * [x] markdown viewer | ||||
| * [x] markdown editor | ||||
| * [x] FUSE client (read-only) | ||||
| * backend stuff | ||||
|   * ☑ sanic multipart parser | ||||
|   * ☑ load balancer (multiprocessing) | ||||
|   * ☑ volumes (mountpoints) | ||||
|   * ☑ accounts | ||||
| * upload | ||||
|   * ☑ basic: plain multipart, ie6 support | ||||
|   * ☑ up2k: js, resumable, multithreaded | ||||
|   * ☑ stash: simple PUT filedropper | ||||
|   * ☑ symlink/discard existing files (content-matching) | ||||
| * download | ||||
|   * ☑ single files in browser | ||||
|   * ✖ folders as zip files | ||||
|   * ☑ 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) | ||||
|  | ||||
|  | ||||
| # 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` | ||||
| * probably more, pls let me know | ||||
|  | ||||
|  | ||||
| # usage | ||||
|  | ||||
| the browser has the following hotkeys | ||||
| * `0..9` jump to 10%..90% | ||||
| * `U/O` skip 10sec back/forward | ||||
| * `J/L` prev/next song | ||||
| * `I/K` prev/next folder | ||||
| * `P` parent folder | ||||
|  | ||||
|  | ||||
| # searching | ||||
|  | ||||
| when started with `-e2dsa` copyparty will scan/index all your files. This avoids duplicates on upload, and also makes the volumes searchable through the web-ui: | ||||
| * make search queries by `size`/`date`/`directory-path`/`filename`, or... | ||||
| * drag/drop a local file to see if the same contents exist somewhere on the server (you get the URL if it does) | ||||
|  | ||||
| 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 | ||||
|  | ||||
|  | ||||
| ## 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` | ||||
|  | ||||
|  | ||||
| # client examples | ||||
|  | ||||
| * javascript: dump some state into a file (two separate examples) | ||||
|   * `await fetch('https://127.0.0.1:3923/', {method:"PUT", body: JSON.stringify(foo)});` | ||||
|   * `var xhr = new XMLHttpRequest(); xhr.open('POST', 'https://127.0.0.1:3923/msgs?raw'); xhr.send('foo');` | ||||
|  | ||||
| * curl/wget: upload some files (post=file, chunk=stdin) | ||||
|   * `post(){ curl -b cppwd=wark http://127.0.0.1:3923/ -F act=bput -F f=@"$1";}`   | ||||
|     `post movie.mkv` | ||||
|   * `post(){ wget --header='Cookie: cppwd=wark' http://127.0.0.1:3923/?raw --post-file="$1" -O-;}`   | ||||
|     `post movie.mkv` | ||||
|   * `chunk(){ curl -b cppwd=wark http://127.0.0.1:3923/ -T-;}`   | ||||
|     `chunk <movie.mkv` | ||||
|  | ||||
| * FUSE: mount a copyparty server as a local filesystem | ||||
|   * cross-platform python client available in [./bin/](bin/) | ||||
|   * [rclone](https://rclone.org/) as client can give ~5x performance, see [./docs/rclone.md](docs/rclone.md) | ||||
|  | ||||
| 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 | ||||
|  | ||||
|  | ||||
| # 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+) | ||||
|  | ||||
|  | ||||
| ## optional gpl stuff | ||||
|  | ||||
| some bundled tools have copyleft dependencies, see [./bin/#mtag](bin/#mtag) | ||||
|  | ||||
| these are standalone and will never be imported / evaluated by copyparty | ||||
|  | ||||
|  | ||||
| # sfx | ||||
|  | ||||
| currently there are two self-contained binaries: | ||||
| * `copyparty-sfx.sh` for unix (linux and osx) -- smaller, more robust | ||||
| * `copyparty-sfx.py` for windows (unix too) -- crossplatform, beta | ||||
| * [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 | ||||
|  | ||||
| launch either of them (**use sfx.py on systemd**) and it'll unpack and run copyparty, assuming you have python installed of course | ||||
|  | ||||
| @@ -127,6 +273,7 @@ pip install black bandit pylint flake8  # vscode tooling | ||||
| in the `scripts` folder: | ||||
|  | ||||
| * run `make -C deps-docker` to build all dependencies | ||||
| * `git tag v1.2.3 && git push origin --tags` | ||||
| * create github release with `make-tgz-release.sh` | ||||
| * upload to pypi with `make-pypi-release.(sh|bat)` | ||||
| * create sfx with `make-sfx.sh` | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| # copyparty-fuse.py | ||||
| # [`copyparty-fuse.py`](copyparty-fuse.py) | ||||
| * mount a copyparty server as a local filesystem (read-only) | ||||
| * **supports Windows!** -- expect `194 MiB/s` sequential read | ||||
| * **supports Linux** -- expect `117 MiB/s` sequential read | ||||
| @@ -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) | ||||
| * does the same thing except more correct, `samba` approves | ||||
| * **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 | ||||
|  | ||||
|  | ||||
|  | ||||
| # [`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 sys | ||||
| import time | ||||
| import json | ||||
| import stat | ||||
| import errno | ||||
| import struct | ||||
| @@ -323,7 +324,7 @@ class Gateway(object): | ||||
|         if bad_good: | ||||
|             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) | ||||
|         if r.status != 200: | ||||
|             self.closeconn() | ||||
| @@ -334,12 +335,17 @@ class Gateway(object): | ||||
|             ) | ||||
|             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)) | ||||
|             raise FuseOSError(errno.ENOENT) | ||||
|  | ||||
|         try: | ||||
|             return self.parse_html(r) | ||||
|             return parser(r) | ||||
|         except: | ||||
|             info(repr(path) + "\n" + traceback.format_exc()) | ||||
|             raise | ||||
| @@ -367,6 +373,29 @@ class Gateway(object): | ||||
|  | ||||
|         return r.read() | ||||
|  | ||||
|     def parse_jls(self, datasrc): | ||||
|         rsp = b"" | ||||
|         while True: | ||||
|             buf = datasrc.read(1024 * 32) | ||||
|             if not buf: | ||||
|                 break | ||||
|  | ||||
|             rsp += buf | ||||
|  | ||||
|         rsp = json.loads(rsp.decode("utf-8")) | ||||
|         ret = [] | ||||
|         for 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): | ||||
|         ret = [] | ||||
|         remainder = b"" | ||||
| @@ -818,9 +847,9 @@ class CPPF(Operations): | ||||
|                 return cache_stat | ||||
|  | ||||
|         fun = info | ||||
|         if MACOS and path.split('/')[-1].startswith('._'): | ||||
|         if MACOS and path.split("/")[-1].startswith("._"): | ||||
|             fun = dbg | ||||
|          | ||||
|  | ||||
|         fun("=ENOENT ({})".format(hexler(path))) | ||||
|         raise FuseOSError(errno.ENOENT) | ||||
|  | ||||
| @@ -979,6 +1008,12 @@ def main(): | ||||
|         log = null_log | ||||
|         dbg = null_log | ||||
|  | ||||
|     if ar.a and ar.a.startswith("$"): | ||||
|         fn = ar.a[1:] | ||||
|         log("reading password from file [{}]".format(fn)) | ||||
|         with open(fn, "rb") as f: | ||||
|             ar.a = f.read().decode("utf-8").strip() | ||||
|  | ||||
|     if WINDOWS: | ||||
|         os.system("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 | ||||
							
								
								
									
										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}") | ||||
| @@ -10,7 +10,12 @@ | ||||
| * modify `10.13.1.1` as necessary if you wish to support browsers without javascript | ||||
|  | ||||
| ### [`explorer-nothumbs-nofoldertypes.reg`](explorer-nothumbs-nofoldertypes.reg) | ||||
| disables thumbnails and folder-type detection in windows explorer, makes it way faster (especially for slow/networked locations (such as copyparty-fuse)) | ||||
| * disables thumbnails and folder-type detection in windows explorer | ||||
| * makes it way faster (especially for slow/networked locations (such as copyparty-fuse)) | ||||
|  | ||||
| ### [`cfssl.sh`](cfssl.sh) | ||||
| * creates CA and server certificates using cfssl | ||||
| * give a 3rd argument to install it to your copyparty config | ||||
|  | ||||
| # OS integration | ||||
| init-scripts to start copyparty as a service | ||||
|   | ||||
							
								
								
									
										72
									
								
								contrib/cfssl.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										72
									
								
								contrib/cfssl.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,72 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
| # ca-name and server-name | ||||
| ca_name="$1" | ||||
| srv_name="$2" | ||||
|  | ||||
| [ -z "$srv_name" ] && { | ||||
| 	echo "need arg 1: ca name" | ||||
| 	echo "need arg 2: server name" | ||||
| 	exit 1 | ||||
| } | ||||
|  | ||||
|  | ||||
| gen_ca() { | ||||
| 	(tee /dev/stderr <<EOF | ||||
| {"CN": "$ca_name ca", | ||||
| "CA": {"expiry":"87600h", "pathlen":0}, | ||||
| "key": {"algo":"rsa", "size":4096}, | ||||
| "names": [{"O":"$ca_name ca"}]} | ||||
| EOF | ||||
| 	)| | ||||
| 	cfssl gencert -initca - | | ||||
| 	cfssljson -bare ca | ||||
| 	 | ||||
| 	mv ca-key.pem ca.key | ||||
| 	rm ca.csr | ||||
| } | ||||
|  | ||||
|  | ||||
| gen_srv() { | ||||
| 	(tee /dev/stderr <<EOF | ||||
| {"key": {"algo":"rsa", "size":4096}, | ||||
| "names": [{"O":"$ca_name - $srv_name"}]} | ||||
| EOF | ||||
| 	)| | ||||
| 	cfssl gencert -ca ca.pem -ca-key ca.key \ | ||||
| 		-profile=www -hostname="$srv_name.$ca_name" - | | ||||
| 	cfssljson -bare "$srv_name" | ||||
|  | ||||
| 	mv "$srv_name-key.pem" "$srv_name.key" | ||||
| 	rm "$srv_name.csr" | ||||
| } | ||||
|  | ||||
|  | ||||
| # create ca if not exist | ||||
| [ -e ca.key ] || | ||||
| 	gen_ca | ||||
|  | ||||
| # always create server cert | ||||
| gen_srv | ||||
|  | ||||
|  | ||||
| # dump cert info | ||||
| show() { | ||||
| 	openssl x509 -text -noout -in $1 | | ||||
| 	awk '!o; {o=0} /[0-9a-f:]{16}/{o=1}' | ||||
| } | ||||
| show ca.pem | ||||
| show "$srv_name.pem" | ||||
|  | ||||
|  | ||||
| # write cert into copyparty config | ||||
| [ -z "$3" ] || { | ||||
| 	mkdir -p ~/.config/copyparty | ||||
| 	cat "$srv_name".{key,pem} ca.pem >~/.config/copyparty/cert.pem  | ||||
| } | ||||
|  | ||||
|  | ||||
| # rm *.key *.pem | ||||
| # cfssl print-defaults config | ||||
| # cfssl print-defaults csr | ||||
| @@ -12,7 +12,7 @@ | ||||
| Description=copyparty file server | ||||
|  | ||||
| [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' | ||||
|  | ||||
| [Install] | ||||
|   | ||||
| @@ -8,18 +8,29 @@ __copyright__ = 2019 | ||||
| __license__ = "MIT" | ||||
| __url__ = "https://github.com/9001/copyparty/" | ||||
|  | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import signal | ||||
| import shutil | ||||
| import filecmp | ||||
| import locale | ||||
| import argparse | ||||
| import threading | ||||
| import traceback | ||||
| from textwrap import dedent | ||||
|  | ||||
| from .__init__ import E, WINDOWS, VT100 | ||||
| from .__init__ import E, WINDOWS, VT100, PY2 | ||||
| from .__version__ import S_VERSION, S_BUILD_DT, CODENAME | ||||
| from .svchub import SvcHub | ||||
| from .util import py_desc | ||||
| from .util import py_desc, align_tab, IMPLICATIONS | ||||
|  | ||||
| HAVE_SSL = True | ||||
| try: | ||||
|     import ssl | ||||
| except: | ||||
|     HAVE_SSL = False | ||||
|  | ||||
|  | ||||
| class RiceFormatter(argparse.HelpFormatter): | ||||
| @@ -45,6 +56,10 @@ class RiceFormatter(argparse.HelpFormatter): | ||||
|         return "".join(indent + line + "\n" for line in text.splitlines()) | ||||
|  | ||||
|  | ||||
| def warn(msg): | ||||
|     print("\033[1mwarning:\033[0;33m {}\033[0m\n".format(msg)) | ||||
|  | ||||
|  | ||||
| def ensure_locale(): | ||||
|     for x in [ | ||||
|         "en_US.UTF-8", | ||||
| @@ -85,6 +100,83 @@ def ensure_cert(): | ||||
|     # printf 'NO\n.\n.\n.\n.\ncopyparty-insecure\n.\n' | faketime '2000-01-01 00:00:00' openssl req -x509 -sha256 -newkey rsa:2048 -keyout insecure.pem -out insecure.pem -days $((($(printf %d 0x7fffffff)-$(date +%s --date=2000-01-01T00:00:00Z))/(60*60*24))) -nodes && ls -al insecure.pem && openssl x509 -in insecure.pem -text -noout | ||||
|  | ||||
|  | ||||
| def configure_ssl_ver(al): | ||||
|     def terse_sslver(txt): | ||||
|         txt = txt.lower() | ||||
|         for c in ["_", "v", "."]: | ||||
|             txt = txt.replace(c, "") | ||||
|  | ||||
|         return txt.replace("tls10", "tls1") | ||||
|  | ||||
|     # oh man i love openssl | ||||
|     # check this out | ||||
|     # hold my beer | ||||
|     ptn = re.compile(r"^OP_NO_(TLS|SSL)v") | ||||
|     sslver = terse_sslver(al.ssl_ver).split(",") | ||||
|     flags = [k for k in ssl.__dict__ if ptn.match(k)] | ||||
|     # SSLv2 SSLv3 TLSv1 TLSv1_1 TLSv1_2 TLSv1_3 | ||||
|     if "help" in sslver: | ||||
|         avail = [terse_sslver(x[6:]) for x in flags] | ||||
|         avail = " ".join(sorted(avail) + ["all"]) | ||||
|         print("\navailable ssl/tls versions:\n  " + avail) | ||||
|         sys.exit(0) | ||||
|  | ||||
|     al.ssl_flags_en = 0 | ||||
|     al.ssl_flags_de = 0 | ||||
|     for flag in sorted(flags): | ||||
|         ver = terse_sslver(flag[6:]) | ||||
|         num = getattr(ssl, flag) | ||||
|         if ver in sslver: | ||||
|             al.ssl_flags_en |= num | ||||
|         else: | ||||
|             al.ssl_flags_de |= num | ||||
|  | ||||
|     if sslver == ["all"]: | ||||
|         x = al.ssl_flags_en | ||||
|         al.ssl_flags_en = al.ssl_flags_de | ||||
|         al.ssl_flags_de = x | ||||
|  | ||||
|     for k in ["ssl_flags_en", "ssl_flags_de"]: | ||||
|         num = getattr(al, k) | ||||
|         print("{}: {:8x} ({})".format(k, num, num)) | ||||
|  | ||||
|     # think i need that beer now | ||||
|  | ||||
|  | ||||
| def configure_ssl_ciphers(al): | ||||
|     ctx = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH) | ||||
|     if al.ssl_ver: | ||||
|         ctx.options &= ~al.ssl_flags_en | ||||
|         ctx.options |= al.ssl_flags_de | ||||
|  | ||||
|     is_help = al.ciphers == "help" | ||||
|  | ||||
|     if al.ciphers and not is_help: | ||||
|         try: | ||||
|             ctx.set_ciphers(al.ciphers) | ||||
|         except: | ||||
|             print("\n\033[1;31mfailed to set ciphers\033[0m\n") | ||||
|  | ||||
|     if not hasattr(ctx, "get_ciphers"): | ||||
|         print("cannot read cipher list: openssl or python too old") | ||||
|     else: | ||||
|         ciphers = [x["description"] for x in ctx.get_ciphers()] | ||||
|         print("\n  ".join(["\nenabled ciphers:"] + align_tab(ciphers) + [""])) | ||||
|  | ||||
|     if is_help: | ||||
|         sys.exit(0) | ||||
|  | ||||
|  | ||||
| def sighandler(signal=None, frame=None): | ||||
|     msg = [""] * 5 | ||||
|     for th in threading.enumerate(): | ||||
|         msg.append(str(th)) | ||||
|         msg.extend(traceback.format_stack(sys._current_frames()[th.ident])) | ||||
|  | ||||
|     msg.append("\n") | ||||
|     print("\n".join(msg)) | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     time.strptime("19970815", "%Y%m%d")  # python#7980 | ||||
|     if WINDOWS: | ||||
| @@ -96,7 +188,20 @@ def main(): | ||||
|     print(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc)) | ||||
|  | ||||
|     ensure_locale() | ||||
|     ensure_cert() | ||||
|     if HAVE_SSL: | ||||
|         ensure_cert() | ||||
|  | ||||
|     deprecated = [["-e2s", "-e2ds"]] | ||||
|     for dk, nk in deprecated: | ||||
|         try: | ||||
|             idx = sys.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)) | ||||
|         sys.argv[idx] = nk | ||||
|         time.sleep(2) | ||||
|  | ||||
|     ap = argparse.ArgumentParser( | ||||
|         formatter_class=RiceFormatter, | ||||
| @@ -110,7 +215,7 @@ def main(): | ||||
|                and "cflag" is config flags to set on this volume | ||||
|              | ||||
|             list of cflags: | ||||
|               cnodupe rejects existing files (instead of symlinking them) | ||||
|               "cnodupe" rejects existing files (instead of symlinking them) | ||||
|  | ||||
|             example:\033[35m | ||||
|               -a ed:hunter2 -v .::r:aed -v ../inc:dump:w:aed:cnodupe  \033[36m | ||||
| @@ -133,6 +238,10 @@ def main(): | ||||
|               "save,get" dumps to file and returns the page like a GET | ||||
|               "print,get" prints the data in the log and returns GET | ||||
|               (leave out the ",get" to return an error instead) | ||||
|  | ||||
|             --ciphers help = available ssl/tls ciphers, | ||||
|             --ssl-ver help = available ssl/tls versions, | ||||
|               default is what python considers safe, usually >= TLS1 | ||||
|             """ | ||||
|         ), | ||||
|     ) | ||||
| @@ -145,19 +254,49 @@ def main(): | ||||
|     ap.add_argument("-a", metavar="ACCT", type=str, action="append", help="add account") | ||||
|     ap.add_argument("-v", metavar="VOL", type=str, action="append", help="add volume") | ||||
|     ap.add_argument("-q", action="store_true", help="quiet") | ||||
|     ap.add_argument("--log-conn", action="store_true", help="print tcp-server msgs") | ||||
|     ap.add_argument("-ed", action="store_true", help="enable ?dots") | ||||
|     ap.add_argument("-emp", action="store_true", help="enable markdown plugins") | ||||
|     ap.add_argument("-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("-nw", action="store_true", help="disable writes (benchmark)") | ||||
|     ap.add_argument("-nih", action="store_true", help="no info hostname") | ||||
|     ap.add_argument("-nid", action="store_true", help="no info disk-usage") | ||||
|     ap.add_argument("--no-sendfile", action="store_true", help="disable sendfile") | ||||
|     ap.add_argument("--urlform", type=str, default="print,get", help="how to handle url-forms") | ||||
|     ap.add_argument("--no-sendfile", action="store_true", help="disable sendfile (for debugging)") | ||||
|     ap.add_argument("--no-scandir", action="store_true", help="disable scandir (for debugging)") | ||||
|     ap.add_argument("--urlform", metavar="MODE", type=str, default="print,get", help="how to handle url-forms") | ||||
|     ap.add_argument("--salt", type=str, default="hunter2", help="up2k file-hash salt") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('database options') | ||||
|     ap2.add_argument("-e2d", action="store_true", help="enable up2k database") | ||||
|     ap2.add_argument("-e2ds", action="store_true", help="enable up2k db-scanner, sets -e2d") | ||||
|     ap2.add_argument("-e2dsa", action="store_true", help="scan all folders (for search), sets -e2ds") | ||||
|     ap2.add_argument("-e2t", action="store_true", help="enable metadata indexing") | ||||
|     ap2.add_argument("-e2ts", action="store_true", help="enable metadata scanner, sets -e2t") | ||||
|     ap2.add_argument("-e2tsr", action="store_true", help="rescan all metadata, sets -e2ts") | ||||
|     ap2.add_argument("--no-mutagen", action="store_true", help="use ffprobe for tags instead") | ||||
|     ap2.add_argument("--no-mtag-mt", action="store_true", help="disable tag-read parallelism") | ||||
|     ap2.add_argument("-mtm", metavar="M=t,t,t", action="append", type=str, help="add/replace metadata mapping") | ||||
|     ap2.add_argument("-mte", metavar="M,M,M", type=str, help="tags to index/display (comma-sep.)", | ||||
|         default="circle,album,.tn,artist,title,.bpm,key,.dur,.q") | ||||
|     ap2.add_argument("-mtp", metavar="M=[f,]bin", action="append", type=str, help="read tag M using bin") | ||||
|     ap2.add_argument("--srch-time", metavar="SEC", type=int, default=30, help="search deadline") | ||||
|  | ||||
|     ap2 = ap.add_argument_group('SSL/TLS options') | ||||
|     ap2.add_argument("--http-only", action="store_true", help="disable ssl/tls") | ||||
|     ap2.add_argument("--https-only", action="store_true", help="disable plaintext") | ||||
|     ap2.add_argument("--ssl-ver", metavar="LIST", type=str, help="ssl/tls versions to allow") | ||||
|     ap2.add_argument("--ciphers", metavar="LIST", help="set allowed ciphers") | ||||
|     ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info") | ||||
|     ap2.add_argument("--ssl-log", metavar="PATH", help="log master secrets") | ||||
|      | ||||
|     al = ap.parse_args() | ||||
|     # fmt: on | ||||
|  | ||||
|     # propagate implications | ||||
|     for k1, k2 in IMPLICATIONS: | ||||
|         if getattr(al, k1): | ||||
|             setattr(al, k2, True) | ||||
|  | ||||
|     al.i = al.i.split(",") | ||||
|     try: | ||||
|         if "-" in al.p: | ||||
| @@ -168,6 +307,23 @@ def main(): | ||||
|     except: | ||||
|         raise Exception("invalid value for -p") | ||||
|  | ||||
|     if HAVE_SSL: | ||||
|         if al.ssl_ver: | ||||
|             configure_ssl_ver(al) | ||||
|  | ||||
|         if al.ciphers: | ||||
|             configure_ssl_ciphers(al) | ||||
|     else: | ||||
|         warn("ssl module does not exist; cannot enable https") | ||||
|  | ||||
|     if PY2 and WINDOWS and al.e2d: | ||||
|         warn( | ||||
|             "windows py2 cannot do unicode filenames with -e2d\n" | ||||
|             + "  (if you crash with codec errors then that is why)" | ||||
|         ) | ||||
|  | ||||
|     # signal.signal(signal.SIGINT, sighandler) | ||||
|  | ||||
|     SvcHub(al).run() | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| # coding: utf-8 | ||||
|  | ||||
| VERSION = (0, 7, 4) | ||||
| CODENAME = "keeping track" | ||||
| BUILD_DT = (2021, 2, 4) | ||||
| VERSION = (0, 9, 10) | ||||
| CODENAME = "the strongest music server" | ||||
| BUILD_DT = (2021, 3, 21) | ||||
|  | ||||
| S_VERSION = ".".join(map(str, VERSION)) | ||||
| S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import re | ||||
| import os | ||||
| import sys | ||||
| import threading | ||||
|  | ||||
| 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): | ||||
| @@ -19,6 +20,11 @@ class VFS(object): | ||||
|         self.uwrite = uwrite  # users who can write this | ||||
|         self.flags = flags  # config switches | ||||
|         self.nodes = {}  # child nodes | ||||
|         self.all_vols = {vpath: self}  # flattened recursive | ||||
|  | ||||
|     def _trk(self, vol): | ||||
|         self.all_vols[vol.vpath] = vol | ||||
|         return vol | ||||
|  | ||||
|     def add(self, src, dst): | ||||
|         """get existing, or add new path to the vfs""" | ||||
| @@ -30,7 +36,7 @@ class VFS(object): | ||||
|             name, dst = dst.split("/", 1) | ||||
|             if name in self.nodes: | ||||
|                 # exists; do not manipulate permissions | ||||
|                 return self.nodes[name].add(src, dst) | ||||
|                 return self._trk(self.nodes[name].add(src, dst)) | ||||
|  | ||||
|             vn = VFS( | ||||
|                 "{}/{}".format(self.realpath, name), | ||||
| @@ -40,7 +46,7 @@ class VFS(object): | ||||
|                 self.flags, | ||||
|             ) | ||||
|             self.nodes[name] = vn | ||||
|             return vn.add(src, dst) | ||||
|             return self._trk(vn.add(src, dst)) | ||||
|  | ||||
|         if dst in self.nodes: | ||||
|             # leaf exists; return as-is | ||||
| @@ -50,7 +56,7 @@ class VFS(object): | ||||
|         vp = "{}/{}".format(self.vpath, dst).lstrip("/") | ||||
|         vn = VFS(src, vp) | ||||
|         self.nodes[dst] = vn | ||||
|         return vn | ||||
|         return self._trk(vn) | ||||
|  | ||||
|     def _find(self, vpath): | ||||
|         """return [vfs,remainder]""" | ||||
| @@ -97,12 +103,11 @@ class VFS(object): | ||||
|  | ||||
|         return fsdec(os.path.realpath(fsenc(rp))) | ||||
|  | ||||
|     def ls(self, rem, uname): | ||||
|     def ls(self, rem, uname, scandir, lstat=False): | ||||
|         """return user-readable [fsdir,real,virt] items at vpath""" | ||||
|         virt_vis = {}  # nodes readable by user | ||||
|         abspath = self.canonical(rem) | ||||
|         items = os.listdir(fsenc(abspath)) | ||||
|         real = [fsdec(x) for x in items] | ||||
|         real = list(statdir(nuprint, scandir, lstat, abspath)) | ||||
|         real.sort() | ||||
|         if not rem: | ||||
|             for name, vn2 in sorted(self.nodes.items()): | ||||
| @@ -110,7 +115,7 @@ class VFS(object): | ||||
|                     virt_vis[name] = vn2 | ||||
|  | ||||
|             # no vfs nodes in the list of real inodes | ||||
|             real = [x for x in real if x not in self.nodes] | ||||
|             real = [x for x in real if x[0] not in self.nodes] | ||||
|  | ||||
|         return [abspath, real, virt_vis] | ||||
|  | ||||
| @@ -143,8 +148,8 @@ class AuthSrv(object): | ||||
|         self.mutex = threading.Lock() | ||||
|         self.reload() | ||||
|  | ||||
|     def log(self, msg): | ||||
|         self.log_func("auth", msg) | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func("auth", msg, c) | ||||
|  | ||||
|     def invert(self, orig): | ||||
|         if PY2: | ||||
| @@ -196,13 +201,39 @@ class AuthSrv(object): | ||||
|                 continue | ||||
|  | ||||
|             lvl, uname = ln.split(" ") | ||||
|             if lvl in "ra": | ||||
|                 mread[vol_dst].append(uname) | ||||
|             if lvl in "wa": | ||||
|                 mwrite[vol_dst].append(uname) | ||||
|             if lvl == "c": | ||||
|                 # config option, currently switches only | ||||
|                 mflags[vol_dst][uname] = True | ||||
|             self._read_vol_str( | ||||
|                 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): | ||||
|         """ | ||||
| @@ -225,7 +256,7 @@ class AuthSrv(object): | ||||
|  | ||||
|         if self.args.v: | ||||
|             # list of src:dst:permset:permset:... | ||||
|             # permset is [rwa]username | ||||
|             # permset is [rwa]username or [c]flag | ||||
|             for v_str in self.args.v: | ||||
|                 m = self.re_vol.match(v_str) | ||||
|                 if not m: | ||||
| @@ -242,28 +273,20 @@ class AuthSrv(object): | ||||
|  | ||||
|                 perms = perms.split(":") | ||||
|                 for (lvl, uname) in [[x[0], x[1:]] for x in perms]: | ||||
|                     if lvl == "c": | ||||
|                         # 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) | ||||
|                     self._read_vol_str(lvl, uname, mread[dst], mwrite[dst], mflags[dst]) | ||||
|  | ||||
|         if self.args.c: | ||||
|             for cfg_fn in self.args.c: | ||||
|                 with open(cfg_fn, "rb") as f: | ||||
|                     self._parse_config_file(f, user, mread, mwrite, mflags, mount) | ||||
|  | ||||
|         self.all_writable = [] | ||||
|         if not mount: | ||||
|             # -h says our defaults are CWD at root and read/write for everyone | ||||
|             vfs = VFS(os.path.abspath("."), "", ["*"], ["*"]) | ||||
|         elif "" not in mount: | ||||
|             # there's volumes but no root; make root inaccessible | ||||
|             vfs = VFS(os.path.abspath("."), "") | ||||
|             vfs.flags["d2d"] = True | ||||
|  | ||||
|         maxdepth = 0 | ||||
|         for dst in sorted(mount.keys(), key=lambda x: (x.count("/"), len(x))): | ||||
| @@ -280,11 +303,6 @@ class AuthSrv(object): | ||||
|             v.uread = mread[dst] | ||||
|             v.uwrite = mwrite[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 = {} | ||||
|         for d in [mread, mwrite]: | ||||
| @@ -295,21 +313,85 @@ class AuthSrv(object): | ||||
|  | ||||
|         if missing_users: | ||||
|             self.log( | ||||
|                 "\033[31myou must -a the following users: " | ||||
|                 + ", ".join(k for k in sorted(missing_users)) | ||||
|                 + "\033[0m" | ||||
|                 "you must -a the following users: " | ||||
|                 + ", ".join(k for k in sorted(missing_users)), | ||||
|                 c=1, | ||||
|             ) | ||||
|             raise Exception("invalid config") | ||||
|  | ||||
|         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) | ||||
|  | ||||
|             # 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: | ||||
|             v, _ = vfs.get("/", "*", False, True) | ||||
|             if self.warn_anonwrite and os.getcwd() == v.realpath: | ||||
|                 self.warn_anonwrite = False | ||||
|                 self.log( | ||||
|                     "\033[31manyone can read/write the current directory: {}\033[0m".format( | ||||
|                         v.realpath | ||||
|                     ) | ||||
|                 ) | ||||
|                 msg = "anyone can read/write the current directory: {}" | ||||
|                 self.log(msg.format(v.realpath), c=1) | ||||
|         except Pebkac: | ||||
|             self.warn_anonwrite = True | ||||
|  | ||||
|   | ||||
| @@ -49,11 +49,11 @@ class MpWorker(object): | ||||
|         # print('k') | ||||
|         pass | ||||
|  | ||||
|     def log(self, src, msg): | ||||
|         self.q_yield.put([0, "log", [src, msg]]) | ||||
|     def log(self, src, msg, c=0): | ||||
|         self.q_yield.put([0, "log", [src, msg, c]]) | ||||
|  | ||||
|     def logw(self, msg): | ||||
|         self.log("mp{}".format(self.n), msg) | ||||
|     def logw(self, msg, c=0): | ||||
|         self.log("mp{}".format(self.n), msg, c) | ||||
|  | ||||
|     def httpdrop(self, addr): | ||||
|         self.q_yield.put([0, "httpdrop", [addr]]) | ||||
| @@ -73,7 +73,9 @@ class MpWorker(object): | ||||
|                 if PY2: | ||||
|                     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) | ||||
|  | ||||
|                 with self.mutex: | ||||
|   | ||||
| @@ -28,7 +28,9 @@ class BrokerThr(object): | ||||
|     def put(self, want_retval, dest, *args): | ||||
|         if dest == "httpconn": | ||||
|             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) | ||||
|  | ||||
|         else: | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import os | ||||
| import stat | ||||
| import gzip | ||||
| import time | ||||
| import copy | ||||
| import json | ||||
| import socket | ||||
| import ctypes | ||||
| @@ -34,16 +35,17 @@ class HttpCli(object): | ||||
|         self.auth = conn.auth | ||||
|         self.log_func = conn.log_func | ||||
|         self.log_src = conn.log_src | ||||
|         self.tls = hasattr(self.s, "cipher") | ||||
|  | ||||
|         self.bufsz = 1024 * 32 | ||||
|         self.absolute_urls = False | ||||
|         self.out_headers = {"Access-Control-Allow-Origin": "*"} | ||||
|  | ||||
|     def log(self, msg): | ||||
|         self.log_func(self.log_src, msg) | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func(self.log_src, msg, c) | ||||
|  | ||||
|     def _check_nonfatal(self, ex): | ||||
|         return ex.code < 400 or ex.code == 404 | ||||
|         return ex.code < 400 or ex.code in [404, 429] | ||||
|  | ||||
|     def _assert_safe_rem(self, rem): | ||||
|         # sanity check to prevent any disasters | ||||
| @@ -61,7 +63,7 @@ class HttpCli(object): | ||||
|  | ||||
|             if not headerlines[0]: | ||||
|                 # seen after login with IE6.0.2900.5512.xpsp.080413-2111 (xp-sp3) | ||||
|                 self.log("\033[1;31mBUG: trailing newline from previous request\033[0m") | ||||
|                 self.log("BUG: trailing newline from previous request", c="1;31") | ||||
|                 headerlines.pop(0) | ||||
|  | ||||
|             try: | ||||
| @@ -72,9 +74,11 @@ class HttpCli(object): | ||||
|         except Pebkac as ex: | ||||
|             # self.log("pebkac at httpcli.run #1: " + repr(ex)) | ||||
|             self.keepalive = self._check_nonfatal(ex) | ||||
|             self.loud_reply(str(ex), status=ex.code) | ||||
|             self.loud_reply(unicode(ex), status=ex.code) | ||||
|             return self.keepalive | ||||
|  | ||||
|         # time.sleep(0.4) | ||||
|  | ||||
|         # normalize incoming headers to lowercase; | ||||
|         # outgoing headers however are Correct-Case | ||||
|         for header_line in headerlines[1:]: | ||||
| @@ -124,15 +128,15 @@ class HttpCli(object): | ||||
|                     k, v = k.split("=", 1) | ||||
|                     uparam[k.lower()] = v.strip() | ||||
|                 else: | ||||
|                     uparam[k.lower()] = True | ||||
|                     uparam[k.lower()] = False | ||||
|  | ||||
|         self.uparam = uparam | ||||
|         self.vpath = unquotep(vpath) | ||||
|  | ||||
|         ua = self.headers.get("user-agent", "") | ||||
|         if ua.startswith("rclone/"): | ||||
|             uparam["raw"] = True | ||||
|             uparam["dots"] = True | ||||
|             uparam["raw"] = False | ||||
|             uparam["dots"] = False | ||||
|  | ||||
|         try: | ||||
|             if self.mode in ["GET", "HEAD"]: | ||||
| @@ -159,7 +163,7 @@ class HttpCli(object): | ||||
|         response = ["HTTP/1.1 {} {}".format(status, HTTPCODE[status])] | ||||
|  | ||||
|         if length is not None: | ||||
|             response.append("Content-Length: " + str(length)) | ||||
|             response.append("Content-Length: " + unicode(length)) | ||||
|  | ||||
|         # close if unknown length, otherwise take client's preference | ||||
|         response.append("Connection: " + ("Keep-Alive" if self.keepalive else "Close")) | ||||
| @@ -218,6 +222,9 @@ class HttpCli(object): | ||||
|             static_path = os.path.join(E.mod, "web/", self.vpath[5:]) | ||||
|             return self.tx_file(static_path) | ||||
|  | ||||
|         if "tree" in self.uparam: | ||||
|             return self.tx_tree() | ||||
|  | ||||
|         # conditional redirect to single volumes | ||||
|         if self.vpath == "" and not self.uparam: | ||||
|             nread = len(self.rvol) | ||||
| @@ -236,7 +243,7 @@ class HttpCli(object): | ||||
|         ) | ||||
|         if not self.readable and not self.writable: | ||||
|             self.log("inaccessible: [{}]".format(self.vpath)) | ||||
|             self.uparam = {"h": True} | ||||
|             self.uparam = {"h": False} | ||||
|  | ||||
|         if "h" in self.uparam: | ||||
|             self.vpath = None | ||||
| @@ -306,7 +313,7 @@ class HttpCli(object): | ||||
|                 reader, _ = self.get_body_reader() | ||||
|                 for buf in reader: | ||||
|                     buf = buf.decode("utf-8", "replace") | ||||
|                     self.log("urlform:\n  {}\n".format(buf)) | ||||
|                     self.log("urlform @ {}\n  {}\n".format(self.vpath, buf)) | ||||
|  | ||||
|             if "get" in opt: | ||||
|                 return self.handle_get() | ||||
| @@ -316,8 +323,11 @@ class HttpCli(object): | ||||
|         raise Pebkac(405, "don't know how to handle POST({})".format(ctype)) | ||||
|  | ||||
|     def get_body_reader(self): | ||||
|         remains = int(self.headers.get("content-length", None)) | ||||
|         if remains is None: | ||||
|         chunked = "chunked" in self.headers.get("transfer-encoding", "").lower() | ||||
|         remains = int(self.headers.get("content-length", -1)) | ||||
|         if chunked: | ||||
|             return read_socket_chunked(self.sr), remains | ||||
|         elif remains == -1: | ||||
|             self.keepalive = False | ||||
|             return read_socket_unbounded(self.sr), remains | ||||
|         else: | ||||
| @@ -335,6 +345,10 @@ class HttpCli(object): | ||||
|         with open(path, "wb", 512 * 1024) as f: | ||||
|             post_sz, _, sha_b64 = hashcopy(self.conn, reader, f) | ||||
|  | ||||
|         self.conn.hsrv.broker.put( | ||||
|             False, "up2k.hash_file", vfs.realpath, vfs.flags, rem, fn | ||||
|         ) | ||||
|  | ||||
|         return post_sz, sha_b64, remains, path | ||||
|  | ||||
|     def handle_stash(self): | ||||
| @@ -400,6 +414,9 @@ class HttpCli(object): | ||||
|         except: | ||||
|             raise Pebkac(422, "you POSTed invalid json") | ||||
|  | ||||
|         if "srch" in self.uparam or "srch" in body: | ||||
|             return self.handle_search(body) | ||||
|  | ||||
|         # prefer this over undot; no reason to allow traversion | ||||
|         if "/" in body["name"]: | ||||
|             raise Pebkac(400, "folders verboten") | ||||
| @@ -415,7 +432,7 @@ class HttpCli(object): | ||||
|         body["ptop"] = vfs.realpath | ||||
|         body["prel"] = rem | ||||
|         body["addr"] = self.ip | ||||
|         body["flag"] = vfs.flags | ||||
|         body["vcfg"] = vfs.flags | ||||
|  | ||||
|         x = self.conn.hsrv.broker.put(True, "up2k.handle_json", body) | ||||
|         response = x.get() | ||||
| @@ -425,6 +442,52 @@ class HttpCli(object): | ||||
|         self.reply(response.encode("utf-8"), mime="application/json") | ||||
|         return True | ||||
|  | ||||
|     def handle_search(self, body): | ||||
|         vols = [] | ||||
|         for vtop in self.rvol: | ||||
|             vfs, _ = self.conn.auth.vfs.get(vtop, self.uname, True, False) | ||||
|             vols.append([vfs.vpath, vfs.realpath, vfs.flags]) | ||||
|  | ||||
|         idx = self.conn.get_u2idx() | ||||
|         t0 = time.time() | ||||
|         if idx.p_end: | ||||
|             penalty = 0.7 | ||||
|             t_idle = t0 - idx.p_end | ||||
|             if idx.p_dur > 0.7 and t_idle < penalty: | ||||
|                 m = "rate-limit ({:.1f} sec), cost {:.2f}, idle {:.2f}" | ||||
|                 raise Pebkac(429, m.format(penalty, idx.p_dur, t_idle)) | ||||
|  | ||||
|         if "srch" in body: | ||||
|             # search by up2k hashlist | ||||
|             vbody = copy.deepcopy(body) | ||||
|             vbody["hash"] = len(vbody["hash"]) | ||||
|             self.log("qj: " + repr(vbody)) | ||||
|             hits = idx.fsearch(vols, body) | ||||
|             msg = repr(hits) | ||||
|             taglist = [] | ||||
|         else: | ||||
|             # search by query params | ||||
|             self.log("qj: " + repr(body)) | ||||
|             hits, taglist = idx.search(vols, body) | ||||
|             msg = len(hits) | ||||
|  | ||||
|         idx.p_end = time.time() | ||||
|         idx.p_dur = idx.p_end - t0 | ||||
|         self.log("q#: {} ({:.2f}s)".format(msg, idx.p_dur)) | ||||
|  | ||||
|         order = [] | ||||
|         cfg = self.args.mte.split(",") | ||||
|         for t in cfg: | ||||
|             if t in taglist: | ||||
|                 order.append(t) | ||||
|         for t in taglist: | ||||
|             if t not in order: | ||||
|                 order.append(t) | ||||
|  | ||||
|         r = json.dumps({"hits": hits, "tag_order": order}).encode("utf-8") | ||||
|         self.reply(r, mime="application/json") | ||||
|         return True | ||||
|  | ||||
|     def handle_post_binary(self): | ||||
|         try: | ||||
|             remains = int(self.headers["content-length"]) | ||||
| @@ -469,7 +532,7 @@ class HttpCli(object): | ||||
|             if len(cstart) > 1 and path != os.devnull: | ||||
|                 self.log( | ||||
|                     "clone {} to {}".format( | ||||
|                         cstart[0], " & ".join(str(x) for x in cstart[1:]) | ||||
|                         cstart[0], " & ".join(unicode(x) for x in cstart[1:]) | ||||
|                     ) | ||||
|                 ) | ||||
|                 ofs = 0 | ||||
| @@ -486,7 +549,12 @@ class HttpCli(object): | ||||
|                 self.log("clone {} done".format(cstart[0])) | ||||
|  | ||||
|         x = self.conn.hsrv.broker.put(True, "up2k.confirm_chunk", ptop, wark, chash) | ||||
|         num_left, path = x.get() | ||||
|         x = x.get() | ||||
|         try: | ||||
|             num_left, path = x | ||||
|         except: | ||||
|             self.loud_reply(x, status=500) | ||||
|             return False | ||||
|  | ||||
|         if not WINDOWS and num_left == 0: | ||||
|             times = (int(time.time()), int(lastmod)) | ||||
| @@ -622,6 +690,9 @@ class HttpCli(object): | ||||
|                             raise Pebkac(400, "empty files in post") | ||||
|  | ||||
|                         files.append([sz, sha512_hex]) | ||||
|                         self.conn.hsrv.broker.put( | ||||
|                             False, "up2k.hash_file", vfs.realpath, vfs.flags, rem, fname | ||||
|                         ) | ||||
|                         self.conn.nbyte += sz | ||||
|  | ||||
|                 except Pebkac: | ||||
| @@ -637,7 +708,7 @@ class HttpCli(object): | ||||
|                     raise | ||||
|  | ||||
|         except Pebkac as ex: | ||||
|             errmsg = str(ex) | ||||
|             errmsg = unicode(ex) | ||||
|  | ||||
|         td = max(0.1, time.time() - t0) | ||||
|         sz_total = sum(x[0] for x in files) | ||||
| @@ -927,8 +998,11 @@ class HttpCli(object): | ||||
|             open_func = open | ||||
|             # 512 kB is optimal for huge files, use 64k | ||||
|             open_args = [fsenc(fs_path), "rb", 64 * 1024] | ||||
|             if hasattr(os, "sendfile"): | ||||
|                 use_sendfile = not self.args.no_sendfile | ||||
|             use_sendfile = ( | ||||
|                 not self.tls  # | ||||
|                 and not self.args.no_sendfile | ||||
|                 and hasattr(os, "sendfile") | ||||
|             ) | ||||
|  | ||||
|         # | ||||
|         # send reply | ||||
| @@ -943,7 +1017,7 @@ class HttpCli(object): | ||||
|             mime=guess_mime(req_path)[0] or "application/octet-stream", | ||||
|         ) | ||||
|  | ||||
|         logmsg += str(status) + logtail | ||||
|         logmsg += unicode(status) + logtail | ||||
|  | ||||
|         if self.mode == "HEAD" or not do_send: | ||||
|             self.log(logmsg) | ||||
| @@ -957,7 +1031,7 @@ class HttpCli(object): | ||||
|                 remains = sendfile_py(lower, upper, f, self.s) | ||||
|  | ||||
|         if remains > 0: | ||||
|             logmsg += " \033[31m" + str(upper - remains) + "\033[0m" | ||||
|             logmsg += " \033[31m" + unicode(upper - remains) + "\033[0m" | ||||
|  | ||||
|         spd = self._spd((upper - lower) - remains) | ||||
|         self.log("{},  {}".format(logmsg, spd)) | ||||
| @@ -1004,7 +1078,7 @@ class HttpCli(object): | ||||
|         sz_html = len(template.render(**targs).encode("utf-8")) | ||||
|         self.send_headers(sz_html + sz_md, status) | ||||
|  | ||||
|         logmsg += str(status) | ||||
|         logmsg += unicode(status) | ||||
|         if self.mode == "HEAD" or not do_send: | ||||
|             self.log(logmsg) | ||||
|             return True | ||||
| @@ -1018,7 +1092,7 @@ class HttpCli(object): | ||||
|             self.log(logmsg + " \033[31md/c\033[0m") | ||||
|             return False | ||||
|  | ||||
|         self.log(logmsg + " " + str(len(html))) | ||||
|         self.log(logmsg + " " + unicode(len(html))) | ||||
|         return True | ||||
|  | ||||
|     def tx_mounts(self): | ||||
| @@ -1028,6 +1102,61 @@ class HttpCli(object): | ||||
|         self.reply(html.encode("utf-8")) | ||||
|         return True | ||||
|  | ||||
|     def tx_tree(self): | ||||
|         top = self.uparam["tree"] or "" | ||||
|         dst = self.vpath | ||||
|         if top in [".", ".."]: | ||||
|             top = undot(self.vpath + "/" + top) | ||||
|  | ||||
|         if top == dst: | ||||
|             dst = "" | ||||
|         elif top: | ||||
|             if not dst.startswith(top + "/"): | ||||
|                 raise Pebkac(400, "arg funk") | ||||
|  | ||||
|             dst = dst[len(top) + 1 :] | ||||
|  | ||||
|         ret = self.gen_tree(top, dst) | ||||
|         ret = json.dumps(ret) | ||||
|         self.reply(ret.encode("utf-8"), mime="application/json") | ||||
|         return True | ||||
|  | ||||
|     def gen_tree(self, top, target): | ||||
|         ret = {} | ||||
|         excl = None | ||||
|         if target: | ||||
|             excl, target = (target.split("/", 1) + [""])[:2] | ||||
|             sub = self.gen_tree("/".join([top, excl]).strip("/"), target) | ||||
|             ret["k" + quotep(excl)] = sub | ||||
|  | ||||
|         try: | ||||
|             vn, rem = self.auth.vfs.get(top, self.uname, True, False) | ||||
|             fsroot, vfs_ls, vfs_virt = vn.ls(rem, self.uname, not self.args.no_scandir) | ||||
|         except: | ||||
|             vfs_ls = [] | ||||
|             vfs_virt = {} | ||||
|             for v in self.rvol: | ||||
|                 d1, d2 = v.rsplit("/", 1) if "/" in v else ["", v] | ||||
|                 if d1 == top: | ||||
|                     vfs_virt[d2] = 0 | ||||
|  | ||||
|         dirs = [] | ||||
|  | ||||
|         vfs_ls = [x[0] for x in vfs_ls if stat.S_ISDIR(x[1].st_mode)] | ||||
|  | ||||
|         if not self.args.ed or "dots" not in self.uparam: | ||||
|             vfs_ls = exclude_dotfiles(vfs_ls) | ||||
|  | ||||
|         for fn in [x for x in vfs_ls if x != excl]: | ||||
|             dirs.append(quotep(fn)) | ||||
|  | ||||
|         for x in vfs_virt.keys(): | ||||
|             if x != excl: | ||||
|                 dirs.append(x) | ||||
|  | ||||
|         ret["a"] = dirs | ||||
|         return ret | ||||
|  | ||||
|     def tx_browser(self): | ||||
|         vpath = "" | ||||
|         vpnodes = [["", "/"]] | ||||
| @@ -1053,13 +1182,14 @@ class HttpCli(object): | ||||
|             if abspath.endswith(".md") and "raw" not in self.uparam: | ||||
|                 return self.tx_md(abspath) | ||||
|  | ||||
|             bad = "{0}.hist{0}up2k.".format(os.sep) | ||||
|             if abspath.endswith(bad + "db") or abspath.endswith(bad + "snap"): | ||||
|             if rem.startswith(".hist/up2k."): | ||||
|                 raise Pebkac(403) | ||||
|  | ||||
|             return self.tx_file(abspath) | ||||
|  | ||||
|         fsroot, vfs_ls, vfs_virt = vn.ls(rem, self.uname) | ||||
|         fsroot, vfs_ls, vfs_virt = vn.ls(rem, self.uname, not self.args.no_scandir) | ||||
|         stats = {k: v for k, v in vfs_ls} | ||||
|         vfs_ls = [x[0] for x in vfs_ls] | ||||
|         vfs_ls.extend(vfs_virt.keys()) | ||||
|  | ||||
|         # check for old versions of files, | ||||
| @@ -1082,22 +1212,35 @@ class HttpCli(object): | ||||
|         if not self.args.ed or "dots" not in self.uparam: | ||||
|             vfs_ls = exclude_dotfiles(vfs_ls) | ||||
|  | ||||
|         hidden = [] | ||||
|         if rem == ".hist": | ||||
|             hidden = ["up2k."] | ||||
|  | ||||
|         is_ls = "ls" in self.uparam | ||||
|  | ||||
|         icur = None | ||||
|         if "e2t" in vn.flags: | ||||
|             idx = self.conn.get_u2idx() | ||||
|             icur = idx.get_cur(vn.realpath) | ||||
|  | ||||
|         dirs = [] | ||||
|         files = [] | ||||
|         for fn in vfs_ls: | ||||
|             base = "" | ||||
|             href = fn | ||||
|             if self.absolute_urls and vpath: | ||||
|             if not is_ls and self.absolute_urls and vpath: | ||||
|                 base = "/" + vpath + "/" | ||||
|                 href = base + fn | ||||
|  | ||||
|             if fn in vfs_virt: | ||||
|                 fspath = vfs_virt[fn].realpath | ||||
|             elif hidden and any(fn.startswith(x) for x in hidden): | ||||
|                 continue | ||||
|             else: | ||||
|                 fspath = fsroot + "/" + fn | ||||
|  | ||||
|             try: | ||||
|                 inf = os.stat(fsenc(fspath)) | ||||
|                 inf = stats.get(fn) or os.stat(fsenc(fspath)) | ||||
|             except: | ||||
|                 self.log("broken symlink: {}".format(repr(fspath))) | ||||
|                 continue | ||||
| @@ -1122,35 +1265,56 @@ class HttpCli(object): | ||||
|             except: | ||||
|                 ext = "%" | ||||
|  | ||||
|             item = [margin, quotep(href), html_escape(fn), sz, ext, dt] | ||||
|             item = { | ||||
|                 "lead": margin, | ||||
|                 "href": quotep(href), | ||||
|                 "name": fn, | ||||
|                 "sz": sz, | ||||
|                 "ext": ext, | ||||
|                 "dt": dt, | ||||
|                 "ts": int(inf.st_mtime), | ||||
|             } | ||||
|             if is_dir: | ||||
|                 dirs.append(item) | ||||
|             else: | ||||
|                 files.append(item) | ||||
|                 item["rd"] = rem | ||||
|  | ||||
|         logues = [None, None] | ||||
|         for n, fn in enumerate([".prologue.html", ".epilogue.html"]): | ||||
|             fn = os.path.join(abspath, fn) | ||||
|             if os.path.exists(fsenc(fn)): | ||||
|                 with open(fsenc(fn), "rb") as f: | ||||
|                     logues[n] = f.read().decode("utf-8") | ||||
|         taglist = {} | ||||
|         for f in files: | ||||
|             fn = f["name"] | ||||
|             rd = f["rd"] | ||||
|             del f["rd"] | ||||
|             if icur: | ||||
|                 q = "select w from up where rd = ? and fn = ?" | ||||
|                 try: | ||||
|                     r = icur.execute(q, (rd, fn)).fetchone() | ||||
|                 except: | ||||
|                     args = s3enc(idx.mem_cur, rd, fn) | ||||
|                     r = icur.execute(q, args).fetchone() | ||||
|  | ||||
|         if False: | ||||
|             # this is a mistake | ||||
|             md = None | ||||
|             for fn in [x[2] for x in files]: | ||||
|                 if fn.lower() == "readme.md": | ||||
|                     fn = os.path.join(abspath, fn) | ||||
|                     with open(fn, "rb") as f: | ||||
|                         md = f.read().decode("utf-8") | ||||
|                 tags = {} | ||||
|                 f["tags"] = tags | ||||
|                  | ||||
|                 if not r: | ||||
|                     continue | ||||
|  | ||||
|                     break | ||||
|                 w = r[0][:16] | ||||
|                 q = "select k, v from mt where w = ? and k != 'x'" | ||||
|                 for k, v in icur.execute(q, (w,)): | ||||
|                     taglist[k] = True | ||||
|                     tags[k] = v | ||||
|  | ||||
|         if icur: | ||||
|             taglist = [k for k in vn.flags["mte"].split(",") if k in taglist] | ||||
|             for f in dirs: | ||||
|                 f["tags"] = {} | ||||
|  | ||||
|         srv_info = [] | ||||
|  | ||||
|         try: | ||||
|             if not self.args.nih: | ||||
|                 srv_info.append(str(socket.gethostname()).split(".")[0]) | ||||
|                 srv_info.append(unicode(socket.gethostname()).split(".")[0]) | ||||
|         except: | ||||
|             self.log("#wow #whoa") | ||||
|             pass | ||||
| @@ -1174,21 +1338,53 @@ class HttpCli(object): | ||||
|         except: | ||||
|             pass | ||||
|  | ||||
|         srv_info = "</span> /// <span>".join(srv_info) | ||||
|  | ||||
|         perms = [] | ||||
|         if self.readable: | ||||
|             perms.append("read") | ||||
|         if self.writable: | ||||
|             perms.append("write") | ||||
|  | ||||
|         logues = ["", ""] | ||||
|         for n, fn in enumerate([".prologue.html", ".epilogue.html"]): | ||||
|             fn = os.path.join(abspath, fn) | ||||
|             if os.path.exists(fsenc(fn)): | ||||
|                 with open(fsenc(fn), "rb") as f: | ||||
|                     logues[n] = f.read().decode("utf-8") | ||||
|  | ||||
|         if is_ls: | ||||
|             [x.pop(k) for k in ["name", "dt"] for y in [dirs, files] for x in y] | ||||
|             ret = { | ||||
|                 "dirs": dirs, | ||||
|                 "files": files, | ||||
|                 "srvinf": srv_info, | ||||
|                 "perms": perms, | ||||
|                 "logues": logues, | ||||
|                 "taglist": taglist, | ||||
|             } | ||||
|             ret = json.dumps(ret) | ||||
|             self.reply(ret.encode("utf-8", "replace"), mime="application/json") | ||||
|             return True | ||||
|  | ||||
|         ts = "" | ||||
|         # ts = "?{}".format(time.time()) | ||||
|  | ||||
|         dirs.extend(files) | ||||
|  | ||||
|         html = self.conn.tpl_browser.render( | ||||
|             vdir=quotep(self.vpath), | ||||
|             vpnodes=vpnodes, | ||||
|             files=dirs, | ||||
|             can_upload=self.writable, | ||||
|             can_read=self.readable, | ||||
|             ts=ts, | ||||
|             prologue=logues[0], | ||||
|             epilogue=logues[1], | ||||
|             perms=json.dumps(perms), | ||||
|             taglist=taglist, | ||||
|             tag_order=json.dumps(vn.flags["mte"].split(",")), | ||||
|             have_up2k_idx=("e2d" in vn.flags), | ||||
|             have_tags_idx=("e2t" in vn.flags), | ||||
|             logues=logues, | ||||
|             title=html_escape(self.vpath), | ||||
|             srv_info="</span> /// <span>".join(srv_info), | ||||
|             srv_info=srv_info, | ||||
|         ) | ||||
|         self.reply(html.encode("utf-8", "replace")) | ||||
|         return True | ||||
|   | ||||
| @@ -3,10 +3,15 @@ from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import ssl | ||||
| import time | ||||
| import socket | ||||
|  | ||||
| HAVE_SSL = True | ||||
| try: | ||||
|     import ssl | ||||
| except: | ||||
|     HAVE_SSL = False | ||||
|  | ||||
| try: | ||||
|     import jinja2 | ||||
| except ImportError: | ||||
| @@ -15,16 +20,19 @@ except ImportError: | ||||
|   you do not have jinja2 installed,\033[33m | ||||
|   choose one of these:\033[0m | ||||
|    * apt install python-jinja2 | ||||
|    * python3 -m pip install --user 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 | ||||
| from .util import Unrecv | ||||
| from .httpcli import HttpCli | ||||
| from .u2idx import U2idx | ||||
|  | ||||
|  | ||||
| class HttpConn(object): | ||||
| @@ -45,6 +53,7 @@ class HttpConn(object): | ||||
|         self.t0 = time.time() | ||||
|         self.nbyte = 0 | ||||
|         self.workload = 0 | ||||
|         self.u2idx = None | ||||
|         self.log_func = hsrv.log | ||||
|         self.set_rproxy() | ||||
|  | ||||
| @@ -72,12 +81,17 @@ class HttpConn(object): | ||||
|     def respath(self, res_name): | ||||
|         return os.path.join(E.mod, "web", res_name) | ||||
|  | ||||
|     def log(self, msg): | ||||
|         self.log_func(self.log_src, msg) | ||||
|     def log(self, msg, c=0): | ||||
|         self.log_func(self.log_src, msg, c) | ||||
|  | ||||
|     def run(self): | ||||
|     def get_u2idx(self): | ||||
|         if not self.u2idx: | ||||
|             self.u2idx = U2idx(self.args, self.log_func) | ||||
|  | ||||
|         return self.u2idx | ||||
|  | ||||
|     def _detect_https(self): | ||||
|         method = None | ||||
|         self.sr = None | ||||
|         if self.cert_path: | ||||
|             try: | ||||
|                 method = self.s.recv(4, socket.MSG_PEEK) | ||||
| @@ -102,16 +116,58 @@ class HttpConn(object): | ||||
|                 self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8")) | ||||
|                 return | ||||
|  | ||||
|         if method not in [None, b"GET ", b"HEAD", b"POST", b"PUT ", b"OPTI"]: | ||||
|         return method not in [None, b"GET ", b"HEAD", b"POST", b"PUT ", b"OPTI"] | ||||
|  | ||||
|     def run(self): | ||||
|         self.sr = None | ||||
|         if self.args.https_only: | ||||
|             is_https = True | ||||
|         elif self.args.http_only or not HAVE_SSL: | ||||
|             is_https = False | ||||
|         else: | ||||
|             is_https = self._detect_https() | ||||
|  | ||||
|         if is_https: | ||||
|             if self.sr: | ||||
|                 self.log("\033[1;31mTODO: cannot do https in jython\033[0m") | ||||
|                 self.log("TODO: cannot do https in jython", c="1;31") | ||||
|                 return | ||||
|  | ||||
|             self.log_src = self.log_src.replace("[36m", "[35m") | ||||
|             try: | ||||
|                 self.s = ssl.wrap_socket( | ||||
|                     self.s, server_side=True, certfile=self.cert_path | ||||
|                 ) | ||||
|                 ctx = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH) | ||||
|                 ctx.load_cert_chain(self.cert_path) | ||||
|                 if self.args.ssl_ver: | ||||
|                     ctx.options &= ~self.args.ssl_flags_en | ||||
|                     ctx.options |= self.args.ssl_flags_de | ||||
|                     # print(repr(ctx.options)) | ||||
|  | ||||
|                 if self.args.ssl_log: | ||||
|                     try: | ||||
|                         ctx.keylog_filename = self.args.ssl_log | ||||
|                     except: | ||||
|                         self.log("keylog failed; openssl or python too old") | ||||
|  | ||||
|                 if self.args.ciphers: | ||||
|                     ctx.set_ciphers(self.args.ciphers) | ||||
|  | ||||
|                 self.s = ctx.wrap_socket(self.s, server_side=True) | ||||
|                 msg = [ | ||||
|                     "\033[1;3{:d}m{}".format(c, s) | ||||
|                     for c, s in zip([0, 5, 0], self.s.cipher()) | ||||
|                 ] | ||||
|                 self.log(" ".join(msg) + "\033[0m") | ||||
|  | ||||
|                 if self.args.ssl_dbg and hasattr(self.s, "shared_ciphers"): | ||||
|                     overlap = [y[::-1] for y in self.s.shared_ciphers()] | ||||
|                     lines = [str(x) for x in (["TLS cipher overlap:"] + overlap)] | ||||
|                     self.log("\n".join(lines)) | ||||
|                     for k, v in [ | ||||
|                         ["compression", self.s.compression()], | ||||
|                         ["ALPN proto", self.s.selected_alpn_protocol()], | ||||
|                         ["NPN proto", self.s.selected_npn_protocol()], | ||||
|                     ]: | ||||
|                         self.log("TLS {}: {}".format(k, v or "nah")) | ||||
|  | ||||
|             except Exception as ex: | ||||
|                 em = str(ex) | ||||
|  | ||||
| @@ -124,7 +180,7 @@ class HttpConn(object): | ||||
|                     pass | ||||
|  | ||||
|                 else: | ||||
|                     self.log("\033[35mhandshake\033[0m " + em) | ||||
|                     self.log("handshake\033[0m " + em, c=5) | ||||
|  | ||||
|                 return | ||||
|  | ||||
|   | ||||
| @@ -38,7 +38,9 @@ class HttpSrv(object): | ||||
|  | ||||
|     def accept(self, sck, addr): | ||||
|         """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.daemon = True | ||||
|         thr.start() | ||||
| @@ -66,11 +68,15 @@ class HttpSrv(object): | ||||
|                 thr.start() | ||||
|  | ||||
|         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() | ||||
|  | ||||
|         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: | ||||
|                 sck.shutdown(socket.SHUT_RDWR) | ||||
|                 sck.close() | ||||
| @@ -78,7 +84,8 @@ class HttpSrv(object): | ||||
|                 if not MACOS: | ||||
|                     self.log( | ||||
|                         "%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]: | ||||
|                     # 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 | ||||
| @@ -9,7 +9,6 @@ from datetime import datetime, timedelta | ||||
| import calendar | ||||
|  | ||||
| from .__init__ import PY2, WINDOWS, MACOS, VT100 | ||||
| from .authsrv import AuthSrv | ||||
| from .tcpsrv import TcpSrv | ||||
| from .up2k import Up2k | ||||
| from .util import mp | ||||
| @@ -39,10 +38,6 @@ class SvcHub(object): | ||||
|         self.tcpsrv = TcpSrv(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 | ||||
|         if self.check_mp_enable(): | ||||
|             from .broker_mp import BrokerMp as Broker | ||||
| @@ -70,16 +65,16 @@ class SvcHub(object): | ||||
|             self.broker.shutdown() | ||||
|             print("nailed it") | ||||
|  | ||||
|     def _log_disabled(self, src, msg): | ||||
|     def _log_disabled(self, src, msg, c=0): | ||||
|         pass | ||||
|  | ||||
|     def _log_enabled(self, src, msg): | ||||
|     def _log_enabled(self, src, msg, c=0): | ||||
|         """handles logging from all components""" | ||||
|         with self.log_mutex: | ||||
|             now = time.time() | ||||
|             if now >= self.next_day: | ||||
|                 dt = datetime.utcfromtimestamp(now) | ||||
|                 print("\033[36m{}\033[0m".format(dt.strftime("%Y-%m-%d"))) | ||||
|                 print("\033[36m{}\033[0m\n".format(dt.strftime("%Y-%m-%d")), end="") | ||||
|  | ||||
|                 # unix timestamp of next 00:00:00 (leap-seconds safe) | ||||
|                 day_now = dt.day | ||||
| @@ -89,23 +84,30 @@ class SvcHub(object): | ||||
|                 dt = dt.replace(hour=0, minute=0, second=0) | ||||
|                 self.next_day = calendar.timegm(dt.utctimetuple()) | ||||
|  | ||||
|             fmt = "\033[36m{} \033[33m{:21} \033[0m{}" | ||||
|             fmt = "\033[36m{} \033[33m{:21} \033[0m{}\n" | ||||
|             if not VT100: | ||||
|                 fmt = "{} {:21} {}" | ||||
|                 fmt = "{} {:21} {}\n" | ||||
|                 if "\033" in msg: | ||||
|                     msg = self.ansi_re.sub("", msg) | ||||
|                 if "\033" in src: | ||||
|                     src = self.ansi_re.sub("", src) | ||||
|             elif c: | ||||
|                 if isinstance(c, int): | ||||
|                     msg = "\033[3{}m{}".format(c, msg) | ||||
|                 elif "\033" not in c: | ||||
|                     msg = "\033[{}m{}\033[0m".format(c, msg) | ||||
|                 else: | ||||
|                     msg = "{}{}\033[0m".format(c, msg) | ||||
|  | ||||
|             ts = datetime.utcfromtimestamp(now).strftime("%H:%M:%S.%f")[:-3] | ||||
|             msg = fmt.format(ts, src, msg) | ||||
|             try: | ||||
|                 print(msg) | ||||
|                 print(msg, end="") | ||||
|             except UnicodeEncodeError: | ||||
|                 try: | ||||
|                     print(msg.encode("utf-8", "replace").decode()) | ||||
|                     print(msg.encode("utf-8", "replace").decode(), end="") | ||||
|                 except: | ||||
|                     print(msg.encode("ascii", "replace").decode()) | ||||
|                     print(msg.encode("ascii", "replace").decode(), end="") | ||||
|  | ||||
|     def check_mp_support(self): | ||||
|         vmin = sys.version_info[1] | ||||
|   | ||||
| @@ -68,22 +68,29 @@ class TcpSrv(object): | ||||
|             self.log("tcpsrv", "listening @ {0}:{1}".format(ip, port)) | ||||
|  | ||||
|         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: | ||||
|                 time.sleep(0.1) | ||||
|                 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, [], []) | ||||
|             for srv in ready: | ||||
|                 sck, addr = srv.accept() | ||||
|                 sip, sport = srv.getsockname() | ||||
|                 self.log( | ||||
|                     "%s %s" % addr, | ||||
|                     "\033[1;30m|{}C-acc2 \033[0;36m{} \033[3{}m{}".format( | ||||
|                         "-" * 3, sip, sport % 8, sport | ||||
|                     ), | ||||
|                 ) | ||||
|                 if self.args.log_conn: | ||||
|                     self.log( | ||||
|                         "%s %s" % addr, | ||||
|                         "|{}C-acc2 \033[0;36m{} \033[3{}m{}".format( | ||||
|                             "-" * 3, sip, sport % 8, sport | ||||
|                         ), | ||||
|                         c="1;30", | ||||
|                     ) | ||||
|  | ||||
|                 self.num_clients.add() | ||||
|                 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 | ||||
							
								
								
									
										1116
									
								
								copyparty/up2k.py
									
									
									
									
									
								
							
							
						
						
									
										1116
									
								
								copyparty/up2k.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -10,6 +10,7 @@ import select | ||||
| import struct | ||||
| import hashlib | ||||
| import platform | ||||
| import traceback | ||||
| import threading | ||||
| import mimetypes | ||||
| import contextlib | ||||
| @@ -56,11 +57,58 @@ HTTPCODE = { | ||||
|     413: "Payload Too Large", | ||||
|     416: "Requested Range Not Satisfiable", | ||||
|     422: "Unprocessable Entity", | ||||
|     429: "Too Many Requests", | ||||
|     500: "Internal Server Error", | ||||
|     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): | ||||
|     def __init__(self, v=0): | ||||
|         self.v = v | ||||
| @@ -99,6 +147,71 @@ class Unrecv(object): | ||||
|         self.buf = buf + self.buf | ||||
|  | ||||
|  | ||||
| class ProgressPrinter(threading.Thread): | ||||
|     """ | ||||
|     periodically print progress info without linefeeds | ||||
|     """ | ||||
|  | ||||
|     def __init__(self): | ||||
|         threading.Thread.__init__(self) | ||||
|         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 | ||||
| def ren_open(fname, *args, **kwargs): | ||||
|     fdir = kwargs.pop("fdir", None) | ||||
| @@ -108,7 +221,7 @@ def ren_open(fname, *args, **kwargs): | ||||
|         with open(fname, *args, **kwargs) as f: | ||||
|             yield {"orz": [f, fname]} | ||||
|             return | ||||
|      | ||||
|  | ||||
|     orig_name = fname | ||||
|     bname = fname | ||||
|     ext = "" | ||||
| @@ -146,7 +259,7 @@ def ren_open(fname, *args, **kwargs): | ||||
|  | ||||
|         except OSError as ex_: | ||||
|             ex = ex_ | ||||
|             if ex.errno != 36: | ||||
|             if ex.errno not in [36, 63] and (not WINDOWS or ex.errno != 22): | ||||
|                 raise | ||||
|  | ||||
|         if not b64: | ||||
| @@ -437,6 +550,16 @@ def get_spd(nbyte, t0, t=None): | ||||
|     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): | ||||
|     ret = [] | ||||
|     for node in path.split("/"): | ||||
| @@ -480,10 +603,15 @@ def sanitize_fn(fn): | ||||
|     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): | ||||
|     for fpath in filepaths: | ||||
|         if not fpath.split("/")[-1].startswith("."): | ||||
|             yield fpath | ||||
|     return [x for x in filepaths if not x.split("/")[-1].startswith(".")] | ||||
|  | ||||
|  | ||||
| def html_escape(s, quote=False): | ||||
| @@ -536,6 +664,16 @@ def w8enc(txt): | ||||
|     return txt.encode(FS_ENCODING, "surrogateescape") | ||||
|  | ||||
|  | ||||
| def w8b64dec(txt): | ||||
|     """decodes base64(filesystem-bytes) to wtf8""" | ||||
|     return w8dec(base64.urlsafe_b64decode(txt.encode("ascii"))) | ||||
|  | ||||
|  | ||||
| def w8b64enc(txt): | ||||
|     """encodes wtf8 to base64(filesystem-bytes)""" | ||||
|     return base64.urlsafe_b64encode(w8enc(txt)).decode("ascii") | ||||
|  | ||||
|  | ||||
| if PY2 and WINDOWS: | ||||
|     # moonrunes become \x3f with bytestrings, | ||||
|     # losing mojibake support is worth | ||||
| @@ -549,6 +687,31 @@ else: | ||||
|     fsdec = w8dec | ||||
|  | ||||
|  | ||||
| def s3enc(mem_cur, rd, fn): | ||||
|     ret = [] | ||||
|     for v in [rd, fn]: | ||||
|         try: | ||||
|             mem_cur.execute("select * from a where b = ?", (v,)) | ||||
|             ret.append(v) | ||||
|         except: | ||||
|             ret.append("//" + w8b64enc(v)) | ||||
|             # self.log("mojien/{} [{}] {}".format(k, v, ret[-1][2:])) | ||||
|  | ||||
|     return tuple(ret) | ||||
|  | ||||
|  | ||||
| def s3dec(rd, fn): | ||||
|     ret = [] | ||||
|     for k, v in [["d", rd], ["f", fn]]: | ||||
|         if v.startswith("//"): | ||||
|             ret.append(w8b64dec(v[2:])) | ||||
|             # self.log("mojide/{} [{}] {}".format(k, ret[-1], v[2:])) | ||||
|         else: | ||||
|             ret.append(v) | ||||
|  | ||||
|     return tuple(ret) | ||||
|  | ||||
|  | ||||
| def atomic_move(src, dst): | ||||
|     if not PY2: | ||||
|         os.replace(src, dst) | ||||
| @@ -583,6 +746,40 @@ def read_socket_unbounded(sr): | ||||
|         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 hashcopy(actor, fin, fout): | ||||
|     u32_lim = int((2 ** 31) * 0.9) | ||||
|     hashobj = hashlib.sha512() | ||||
| @@ -632,16 +829,43 @@ def sendfile_kern(lower, upper, f, s): | ||||
|         except Exception as ex: | ||||
|             # print("sendfile: " + repr(ex)) | ||||
|             n = 0 | ||||
|          | ||||
|  | ||||
|         if n <= 0: | ||||
|             return upper - ofs | ||||
|          | ||||
|  | ||||
|         ofs += n | ||||
|         # print("sendfile: ok, sent {} now, {} total, {} remains".format(n, ofs - lower, upper - ofs)) | ||||
|  | ||||
|     return 0 | ||||
|  | ||||
|  | ||||
| def statdir(logger, scandir, lstat, top): | ||||
|     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): | ||||
|     # mw=idk; doot=qwe%2Crty%3Basd+fgh%2Bjkl%25zxc%26vbn  # qwe,rty;asd fgh+jkl%zxc&vbn | ||||
|     ret = "" | ||||
| @@ -696,7 +920,11 @@ def chkcmd(*argv): | ||||
| def gzip_orig_sz(fn): | ||||
|     with open(fsenc(fn), "rb") as f: | ||||
|         f.seek(-4, 2) | ||||
|         return struct.unpack(b"I", f.read(4))[0] | ||||
|         rv = f.read(4) | ||||
|         try: | ||||
|             return struct.unpack(b"I", rv)[0] | ||||
|         except: | ||||
|             return struct.unpack("I", rv)[0] | ||||
|  | ||||
|  | ||||
| def py_desc(): | ||||
| @@ -706,7 +934,11 @@ def py_desc(): | ||||
|     if ofs > 0: | ||||
|         py_ver = py_ver[:ofs] | ||||
|  | ||||
|     bitness = struct.calcsize(b"P") * 8 | ||||
|     try: | ||||
|         bitness = struct.calcsize(b"P") * 8 | ||||
|     except: | ||||
|         bitness = struct.calcsize("P") * 8 | ||||
|  | ||||
|     host_os = platform.system() | ||||
|     compiler = platform.python_compiler() | ||||
|  | ||||
| @@ -718,6 +950,22 @@ def py_desc(): | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def align_tab(lines): | ||||
|     rows = [] | ||||
|     ncols = 0 | ||||
|     for ln in lines: | ||||
|         row = [x for x in ln.split(" ") if x] | ||||
|         ncols = max(ncols, len(row)) | ||||
|         rows.append(row) | ||||
|  | ||||
|     lens = [0] * ncols | ||||
|     for row in rows: | ||||
|         for n, col in enumerate(row): | ||||
|             lens[n] = max(lens[n], len(col)) | ||||
|  | ||||
|     return ["".join(x.ljust(y + 2) for x, y in zip(row, lens)) for row in rows] | ||||
|  | ||||
|  | ||||
| class Pebkac(Exception): | ||||
|     def __init__(self, code, msg=None): | ||||
|         super(Pebkac, self).__init__(msg or HTTPCODE[code]) | ||||
|   | ||||
| @@ -39,15 +39,22 @@ body { | ||||
| 	margin: 1.3em 0 0 0; | ||||
| 	font-size: 1.4em; | ||||
| } | ||||
| #path #entree { | ||||
| 	margin-left: -.7em; | ||||
| } | ||||
| #files { | ||||
| 	border-collapse: collapse; | ||||
| 	margin-top: 2em; | ||||
| 	border-spacing: 0; | ||||
| 	z-index: 1; | ||||
| 	position: relative; | ||||
| } | ||||
| #files tbody a { | ||||
| 	display: block; | ||||
| 	padding: .3em 0; | ||||
| } | ||||
| a { | ||||
| #files tbody div a { | ||||
| 	color: #f5a; | ||||
| } | ||||
| a, #files tbody div a:last-child { | ||||
| 	color: #fc5; | ||||
| 	padding: .2em; | ||||
| 	text-decoration: none; | ||||
| @@ -55,16 +62,18 @@ a { | ||||
| #files a:hover { | ||||
| 	color: #fff; | ||||
| 	background: #161616; | ||||
| 	text-decoration: underline; | ||||
| } | ||||
| #files thead a { | ||||
| 	color: #999; | ||||
| 	font-weight: normal; | ||||
| } | ||||
| #files tr:hover { | ||||
| #files tr+tr:hover { | ||||
| 	background: #1c1c1c; | ||||
| } | ||||
| #files thead th { | ||||
| 	padding: .5em 1.3em .3em 1.3em; | ||||
| 	cursor: pointer; | ||||
| } | ||||
| #files thead th:last-child { | ||||
| 	background: #444; | ||||
| @@ -82,6 +91,16 @@ a { | ||||
| 	margin: 0; | ||||
| 	padding: 0 .5em; | ||||
| } | ||||
| #files td { | ||||
| 	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) { | ||||
| 	font-family: monospace; | ||||
| 	font-size: 1.3em; | ||||
| @@ -100,6 +119,9 @@ a { | ||||
| 	padding-bottom: 1.3em; | ||||
| 	border-bottom: .5em solid #444; | ||||
| } | ||||
| #files tbody tr td:last-child { | ||||
| 	white-space: nowrap; | ||||
| } | ||||
| #files thead th[style] { | ||||
| 	width: auto !important; | ||||
| } | ||||
| @@ -131,6 +153,15 @@ a { | ||||
| .logue { | ||||
| 	padding: .2em 1.5em; | ||||
| } | ||||
| .logue:empty { | ||||
| 	display: none; | ||||
| } | ||||
| #pro.logue { | ||||
| 	margin-bottom: .8em; | ||||
| } | ||||
| #epi.logue { | ||||
| 	margin: .8em 0; | ||||
| } | ||||
| #srv_info { | ||||
| 	opacity: .5; | ||||
| 	font-size: .8em; | ||||
| @@ -142,11 +173,14 @@ a { | ||||
| #srv_info span { | ||||
| 	color: #fff; | ||||
| } | ||||
| a.play { | ||||
| #files tbody a.play { | ||||
| 	color: #e70; | ||||
| 	padding: .2em; | ||||
| 	margin: -.2em; | ||||
| } | ||||
| a.play.act { | ||||
| 	color: #af0; | ||||
| #files tbody a.play.act { | ||||
| 	color: #840; | ||||
| 	text-shadow: 0 0 .3em #b80; | ||||
| } | ||||
| #blocked { | ||||
| 	position: fixed; | ||||
| @@ -156,7 +190,7 @@ a.play.act { | ||||
| 	height: 100%; | ||||
| 	background: #333; | ||||
| 	font-size: 2.5em; | ||||
| 	z-index:99; | ||||
| 	z-index: 99; | ||||
| } | ||||
| #blk_play, | ||||
| #blk_abrt { | ||||
| @@ -190,6 +224,7 @@ a.play.act { | ||||
| 	bottom: -6em; | ||||
| 	height: 6em; | ||||
| 	width: 100%; | ||||
| 	z-index: 3; | ||||
| 	transition: bottom 0.15s; | ||||
| } | ||||
| #widget.open { | ||||
| @@ -214,6 +249,9 @@ a.play.act { | ||||
| 	75% {cursor: url(/.cpr/dd/5.png), pointer} | ||||
| 	85% {cursor: url(/.cpr/dd/1.png), pointer} | ||||
| } | ||||
| @keyframes spin { | ||||
| 	100% {transform: rotate(360deg)} | ||||
| } | ||||
| #wtoggle { | ||||
| 	position: absolute; | ||||
| 	top: -1.2em; | ||||
| @@ -273,3 +311,344 @@ a.play.act { | ||||
| 	width: calc(100% - 10.5em); | ||||
| 	background: rgba(0,0,0,0.2); | ||||
| } | ||||
| @media (min-width: 90em) { | ||||
| 	#barpos, | ||||
| 	#barbuf { | ||||
| 		width: calc(100% - 24em); | ||||
| 		left: 9.8em; | ||||
| 		top: .7em; | ||||
| 		height: 1.6em; | ||||
| 		bottom: auto; | ||||
| 	} | ||||
| 	#widget { | ||||
| 		bottom: -3.2em; | ||||
| 		height: 3.2em; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| .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; | ||||
| } | ||||
| .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%; | ||||
| } | ||||
| #files td div a:last-child { | ||||
| 	width: 100%; | ||||
| } | ||||
| #wrap { | ||||
| 	margin-top: 2em; | ||||
| } | ||||
| #tree { | ||||
| 	display: none; | ||||
| 	position: fixed; | ||||
| 	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 { | ||||
| 	background: #333; | ||||
| } | ||||
| #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 { | ||||
| 	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; | ||||
| } | ||||
|   | ||||
| @@ -7,53 +7,104 @@ | ||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=0.8"> | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/browser.css{{ ts }}"> | ||||
|     {%- if can_upload %} | ||||
|     <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/upload.css{{ ts }}"> | ||||
|     {%- endif %} | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
|     {%- if can_upload %} | ||||
|     <div id="ops"> | ||||
|         <a href="#" data-dest="" data-desc="close submenu">---</a> | ||||
|         <a href="#" data-perm="read" data-dest="search" data-desc="search for files by attributes, path/name, music tags, or any combination of those.<br /><br /><code>foo bar</code> = must contain both foo and bar,<br /><code>foo -bar</code> = must contain foo but not bar,<br /><code>^yana .opus$</code> = must start with yana and have the opus extension">🔎</a> | ||||
|         {%- if have_up2k_idx %} | ||||
|         <a href="#" data-dest="up2k" data-desc="up2k: upload files (if you have write-access) or toggle into the search-mode and drag files onto the search button to see if they exist somewhere on the server">🚀</a> | ||||
|         {%- else %} | ||||
|         <a href="#" data-perm="write" data-dest="up2k" data-desc="up2k: upload files with resume support (close your browser and drop the same files in later)">🚀</a> | ||||
|         {%- endif %} | ||||
|         <a href="#" data-perm="write" data-dest="bup" data-desc="bup: basic uploader, even supports netscape 4.0">🎈</a> | ||||
|         <a href="#" data-perm="write" data-dest="mkdir" data-desc="mkdir: create a new directory">📂</a> | ||||
|         <a href="#" data-perm="write" data-dest="new_md" data-desc="new-md: create a new markdown document">📝</a> | ||||
|         <a href="#" data-perm="write" data-dest="msg" data-desc="msg: send a message to the server log">📟</a> | ||||
|         <a href="#" data-dest="cfg" data-desc="configuration options">⚙️</a> | ||||
|         <div id="opdesc"></div> | ||||
|     </div> | ||||
|  | ||||
|     <div id="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' %} | ||||
|     {%- endif %} | ||||
|  | ||||
|     <div id="op_cfg" class="opview opbox"> | ||||
|         <h3>key notation</h3> | ||||
|         <div id="key_notation"></div> | ||||
|         <h3>tooltips</h3> | ||||
|         <div> | ||||
|             <a id="tooltips" class="tglbtn" href="#">enable</a> | ||||
|         </div> | ||||
|     </div> | ||||
|      | ||||
|     <h1 id="path"> | ||||
|         <a href="#" id="entree">🌲</a> | ||||
|         {%- for n in vpnodes %} | ||||
|         <a href="/{{ n[0] }}">{{ n[1] }}</a> | ||||
|         {%- endfor %} | ||||
|     </h1> | ||||
|      | ||||
|     {%- if can_read %} | ||||
|     {%- if prologue %} | ||||
|     <div id="pro" class="logue">{{ prologue }}</div> | ||||
|     {%- endif %} | ||||
|     <div id="tree"> | ||||
|         <a href="#" id="detree">🍞...</a> | ||||
|         <a href="#" step="2" id="twobytwo">+</a> | ||||
|         <a href="#" step="-2" id="twig">–</a> | ||||
|         <a href="#" class="tglbtn" id="dyntree">a</a> | ||||
|         <ul id="treeul"></ul> | ||||
|         <div id="thx_ff"> </div> | ||||
|     </div> | ||||
|  | ||||
| <div id="wrap"> | ||||
|  | ||||
|     <div id="pro" class="logue">{{ logues[0] }}</div> | ||||
|  | ||||
|     <table id="files"> | ||||
|         <thead> | ||||
|             <tr> | ||||
|                 <th></th> | ||||
|                 <th>File Name</th> | ||||
|                 <th sort="int">File Size</th> | ||||
|                 <th>T</th> | ||||
|                 <th>Date</th> | ||||
|                 <th name="href"><span>File Name</span></th> | ||||
|                 <th name="sz" sort="int"><span>Size</span></th> | ||||
|                 {%- for k in taglist %} | ||||
|                     {%- if k.startswith('.') %} | ||||
|                         <th name="tags/{{ k }}" sort="int"><span>{{ k[1:] }}</span></th> | ||||
|                     {%- else %} | ||||
|                         <th name="tags/{{ k }}"><span>{{ k[0]|upper }}{{ k[1:] }}</span></th> | ||||
|                     {%- endif %} | ||||
|                 {%- endfor %} | ||||
|                 <th name="ext"><span>T</span></th> | ||||
|                 <th name="ts"><span>Date</span></th> | ||||
|             </tr> | ||||
|         </thead> | ||||
|         <tbody> | ||||
|  | ||||
| {%- for f in files %} | ||||
| <tr><td>{{ f[0] }}</td><td><a href="{{ f[1] }}">{{ f[2] }}</a></td><td>{{ f[3] }}</td><td>{{ f[4] }}</td><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 %} | ||||
|  | ||||
|         </tbody> | ||||
|     </table> | ||||
|      | ||||
|     {%- if epilogue %} | ||||
|     <div id="epi" class="logue">{{ epilogue }}</div> | ||||
|     {%- endif %} | ||||
|     {%- endif %} | ||||
|     <div id="epi" class="logue">{{ logues[1] }}</div> | ||||
|  | ||||
|     <h2><a href="?h">control-panel</a></h2> | ||||
|  | ||||
| </div> | ||||
|  | ||||
|     {%- if srv_info %} | ||||
|     <div id="srv_info"><span>{{ srv_info }}</span></div> | ||||
|     {%- endif %} | ||||
| @@ -67,16 +118,16 @@ | ||||
|             <canvas id="barbuf"></canvas> | ||||
|         </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> | ||||
|     {%- endif %} | ||||
|      | ||||
|     {%- if can_upload %} | ||||
|     <script src="/.cpr/up2k.js{{ ts }}"></script> | ||||
|     {%- endif %} | ||||
|     <script> | ||||
|         apply_perms({{ perms }}); | ||||
|     </script> | ||||
| </body> | ||||
|  | ||||
| </html> | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -65,7 +65,7 @@ function statify(obj) { | ||||
|         if (a > 0) | ||||
|             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>'); | ||||
|     } | ||||
| @@ -524,11 +524,9 @@ dom_navtgl.onclick = function () { | ||||
|     dom_navtgl.innerHTML = hidden ? 'show nav' : 'hide nav'; | ||||
|     dom_nav.style.display = hidden ? 'none' : 'block'; | ||||
|  | ||||
|     if (window.localStorage) | ||||
|         localStorage.setItem('hidenav', hidden ? 1 : 0); | ||||
|  | ||||
|     swrite('hidenav', hidden ? 1 : 0); | ||||
|     redraw(); | ||||
| }; | ||||
|  | ||||
| if (window.localStorage && localStorage.getItem('hidenav') == 1) | ||||
| if (sread('hidenav') == 1) | ||||
|     dom_navtgl.onclick(); | ||||
|   | ||||
| @@ -124,5 +124,3 @@ html.dark #toast { | ||||
|     transition: opacity 0.2s ease-in-out; | ||||
|     opacity: 1; | ||||
| } | ||||
|  | ||||
| # mt {opacity: .5;top:1px} | ||||
|   | ||||
| @@ -15,7 +15,7 @@ var dom_md = ebi('mt'); | ||||
|         if (a > 0) | ||||
|             loc.push(n[a]); | ||||
|  | ||||
|         var dec = decodeURIComponent(n[a]).replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">"); | ||||
|         var dec = uricom_dec(n[a])[0].replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">"); | ||||
|  | ||||
|         nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>'); | ||||
|     } | ||||
|   | ||||
| @@ -3,51 +3,6 @@ | ||||
| window.onerror = vis_exh; | ||||
|  | ||||
|  | ||||
| (function () { | ||||
|     var ops = document.querySelectorAll('#ops>a'); | ||||
|     for (var a = 0; a < ops.length; a++) { | ||||
|         ops[a].onclick = opclick; | ||||
|     } | ||||
| })(); | ||||
|  | ||||
|  | ||||
| function opclick(ev) { | ||||
|     if (ev) //ie | ||||
|         ev.preventDefault(); | ||||
|  | ||||
|     var dest = this.getAttribute('data-dest'); | ||||
|     goto(dest); | ||||
|  | ||||
|     // writing a blank value makes ie8 segfault w | ||||
|     if (window.localStorage) | ||||
|         localStorage.setItem('opmode', dest || '.'); | ||||
|  | ||||
|     var input = document.querySelector('.opview.act input:not([type="hidden"])') | ||||
|     if (input) | ||||
|         input.focus(); | ||||
| } | ||||
|  | ||||
|  | ||||
| function goto(dest) { | ||||
|     var obj = document.querySelectorAll('.opview.act'); | ||||
|     for (var a = obj.length - 1; a >= 0; a--) | ||||
|         obj[a].classList.remove('act'); | ||||
|  | ||||
|     obj = document.querySelectorAll('#ops>a'); | ||||
|     for (var a = obj.length - 1; a >= 0; a--) | ||||
|         obj[a].classList.remove('act'); | ||||
|  | ||||
|     if (dest) { | ||||
|         ebi('op_' + dest).classList.add('act'); | ||||
|         document.querySelector('#ops>a[data-dest=' + dest + ']').classList.add('act'); | ||||
|  | ||||
|         var fn = window['goto_' + dest]; | ||||
|         if (fn) | ||||
|             fn(); | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| function goto_up2k() { | ||||
|     if (up2k === false) | ||||
|         return goto('bup'); | ||||
| @@ -59,17 +14,6 @@ function goto_up2k() { | ||||
| } | ||||
|  | ||||
|  | ||||
| (function () { | ||||
|     goto(); | ||||
|     if (window.localStorage) { | ||||
|         var op = localStorage.getItem('opmode'); | ||||
|         if (op !== null && op !== '.') | ||||
|             goto(op); | ||||
|     } | ||||
|     ebi('ops').style.display = 'block'; | ||||
| })(); | ||||
|  | ||||
|  | ||||
| // chrome requires https to use crypto.subtle, | ||||
| // usually it's undefined but some chromes throw on invoke | ||||
| var up2k = null; | ||||
| @@ -89,6 +33,104 @@ catch (ex) { | ||||
| } | ||||
|  | ||||
|  | ||||
| function up2k_flagbus() { | ||||
|     var flag = { | ||||
|         "id": Math.floor(Math.random() * 1024 * 1024 * 1023 * 2), | ||||
|         "ch": new BroadcastChannel("up2k_flagbus"), | ||||
|         "ours": false, | ||||
|         "owner": null, | ||||
|         "wants": null, | ||||
|         "act": false, | ||||
|         "last_tx": ["x", null] | ||||
|     }; | ||||
|     var dbg = function (who, msg) { | ||||
|         console.log('flagbus(' + flag.id + '): [' + who + '] ' + msg); | ||||
|     }; | ||||
|     flag.ch.onmessage = function (e) { | ||||
|         var who = e.data[0], | ||||
|             what = e.data[1]; | ||||
|  | ||||
|         if (who == flag.id) { | ||||
|             dbg(who, 'hi me (??)'); | ||||
|             return; | ||||
|         } | ||||
|         flag.act = new Date().getTime(); | ||||
|         if (what == "want") { | ||||
|             // lowest id wins, don't care if that's us | ||||
|             if (who < flag.id) { | ||||
|                 dbg(who, 'wants (ack)'); | ||||
|                 flag.wants = [who, flag.act]; | ||||
|             } | ||||
|             else { | ||||
|                 dbg(who, 'wants (ign)'); | ||||
|             } | ||||
|         } | ||||
|         else if (what == "have") { | ||||
|             dbg(who, 'have'); | ||||
|             flag.owner = [who, flag.act]; | ||||
|         } | ||||
|         else if (what == "give") { | ||||
|             if (flag.owner && flag.owner[0] == who) { | ||||
|                 flag.owner = null; | ||||
|                 dbg(who, 'give (ok)'); | ||||
|             } | ||||
|             else { | ||||
|                 dbg(who, 'give, INVALID, ' + flag.owner); | ||||
|             } | ||||
|         } | ||||
|         else if (what == "hi") { | ||||
|             dbg(who, 'hi'); | ||||
|             flag.ch.postMessage([flag.id, "hey"]); | ||||
|         } | ||||
|         else { | ||||
|             dbg('?', e.data); | ||||
|         } | ||||
|     }; | ||||
|     var tx = function (now, msg) { | ||||
|         var td = now - flag.last_tx[1]; | ||||
|         if (td > 500 || flag.last_tx[0] != msg) { | ||||
|             dbg('*', 'tx ' + msg); | ||||
|             flag.ch.postMessage([flag.id, msg]); | ||||
|             flag.last_tx = [msg, now]; | ||||
|         } | ||||
|     }; | ||||
|     var do_take = function (now) { | ||||
|         //dbg('*', 'do_take'); | ||||
|         tx(now, "have"); | ||||
|         flag.owner = [flag.id, now]; | ||||
|         flag.ours = true; | ||||
|     }; | ||||
|     var do_want = function (now) { | ||||
|         //dbg('*', 'do_want'); | ||||
|         tx(now, "want"); | ||||
|     }; | ||||
|     flag.take = function (now) { | ||||
|         if (flag.ours) { | ||||
|             do_take(now); | ||||
|             return; | ||||
|         } | ||||
|         if (flag.owner && now - flag.owner[1] > 5000) { | ||||
|             flag.owner = null; | ||||
|         } | ||||
|         if (flag.wants && now - flag.wants[1] > 5000) { | ||||
|             flag.wants = null; | ||||
|         } | ||||
|         if (!flag.owner && !flag.wants) { | ||||
|             do_take(now); | ||||
|             return; | ||||
|         } | ||||
|         do_want(now); | ||||
|     }; | ||||
|     flag.give = function () { | ||||
|         dbg('#', 'put give'); | ||||
|         flag.ch.postMessage([flag.id, "give"]); | ||||
|         flag.owner = null; | ||||
|         flag.ours = false; | ||||
|     }; | ||||
|     flag.ch.postMessage([flag.id, 'hi']); | ||||
|     return flag; | ||||
| } | ||||
|  | ||||
| function up2k_init(have_crypto) { | ||||
|     //have_crypto = false; | ||||
|     var need_filereader_cache = undefined; | ||||
| @@ -109,10 +151,6 @@ function up2k_init(have_crypto) { | ||||
|         ebi('u2notbtn').innerHTML = ''; | ||||
|     } | ||||
|  | ||||
|     var post_url = ebi('op_bup').getElementsByTagName('form')[0].getAttribute('action'); | ||||
|     if (post_url && post_url.charAt(post_url.length - 1) !== '/') | ||||
|         post_url += '/'; | ||||
|  | ||||
|     var shame = 'your browser <a href="https://www.chromium.org/blink/webcrypto">disables sha512</a> unless you <a href="' + (window.location + '').replace(':', 's:') + '">use https</a>' | ||||
|     var is_https = (window.location + '').indexOf('https:') === 0; | ||||
|     if (is_https) | ||||
| @@ -156,8 +194,8 @@ function up2k_init(have_crypto) { | ||||
|  | ||||
|     // handle user intent to use the basic uploader instead | ||||
|     ebi('u2nope').onclick = function (e) { | ||||
|         e.preventDefault(); | ||||
|         setmsg(''); | ||||
|         ev(e); | ||||
|         setmsg(); | ||||
|         goto('bup'); | ||||
|     }; | ||||
|  | ||||
| @@ -171,37 +209,11 @@ function up2k_init(have_crypto) { | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     function cfg_get(name) { | ||||
|         var val = localStorage.getItem(name); | ||||
|         if (val === null) | ||||
|             return parseInt(ebi(name).value); | ||||
|  | ||||
|         ebi(name).value = val; | ||||
|         return val; | ||||
|     } | ||||
|  | ||||
|     function bcfg_get(name, defval) { | ||||
|         var val = localStorage.getItem(name); | ||||
|         if (val === null) | ||||
|             val = defval; | ||||
|         else | ||||
|             val = (val == '1'); | ||||
|  | ||||
|         ebi(name).checked = val; | ||||
|         return val; | ||||
|     } | ||||
|  | ||||
|     function bcfg_set(name, val) { | ||||
|         localStorage.setItem( | ||||
|             name, val ? '1' : '0'); | ||||
|  | ||||
|         ebi(name).checked = val; | ||||
|         return val; | ||||
|     } | ||||
|  | ||||
|     var parallel_uploads = cfg_get('nthread'); | ||||
|     var parallel_uploads = icfg_get('nthread'); | ||||
|     var multitask = bcfg_get('multitask', true); | ||||
|     var ask_up = bcfg_get('ask_up', true); | ||||
|     var flag_en = bcfg_get('flag_en', false); | ||||
|     var fsearch = bcfg_get('fsearch', false); | ||||
|  | ||||
|     var col_hashing = '#00bbff'; | ||||
|     var col_hashed = '#004466'; | ||||
| @@ -219,6 +231,10 @@ function up2k_init(have_crypto) { | ||||
|             "hash": [], | ||||
|             "handshake": [], | ||||
|             "upload": [] | ||||
|         }, | ||||
|         "bytes": { | ||||
|             "hashed": 0, | ||||
|             "uploaded": 0 | ||||
|         } | ||||
|     }; | ||||
|  | ||||
| @@ -229,34 +245,38 @@ function up2k_init(have_crypto) { | ||||
|     if (!bobslice || !window.FileReader || !window.FileList) | ||||
|         return un2k("this is the basic uploader; up2k needs at least<br />chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1"); | ||||
|  | ||||
|     var flag = false; | ||||
|     apply_flag_cfg(); | ||||
|     set_fsearch(); | ||||
|  | ||||
|     function nav() { | ||||
|         ebi('file' + fdom_ctr).click(); | ||||
|     } | ||||
|     ebi('u2btn').addEventListener('click', nav, false); | ||||
|  | ||||
|     function ondrag(ev) { | ||||
|         ev.stopPropagation(); | ||||
|         ev.preventDefault(); | ||||
|         ev.dataTransfer.dropEffect = 'copy'; | ||||
|         ev.dataTransfer.effectAllowed = 'copy'; | ||||
|     function ondrag(e) { | ||||
|         e.stopPropagation(); | ||||
|         e.preventDefault(); | ||||
|         e.dataTransfer.dropEffect = 'copy'; | ||||
|         e.dataTransfer.effectAllowed = 'copy'; | ||||
|     } | ||||
|     ebi('u2btn').addEventListener('dragover', ondrag, false); | ||||
|     ebi('u2btn').addEventListener('dragenter', ondrag, false); | ||||
|  | ||||
|     function gotfile(ev) { | ||||
|         ev.stopPropagation(); | ||||
|         ev.preventDefault(); | ||||
|     function gotfile(e) { | ||||
|         e.stopPropagation(); | ||||
|         e.preventDefault(); | ||||
|  | ||||
|         var files; | ||||
|         var is_itemlist = false; | ||||
|         if (ev.dataTransfer) { | ||||
|             if (ev.dataTransfer.items) { | ||||
|                 files = ev.dataTransfer.items; // DataTransferItemList | ||||
|         if (e.dataTransfer) { | ||||
|             if (e.dataTransfer.items) { | ||||
|                 files = e.dataTransfer.items; // DataTransferItemList | ||||
|                 is_itemlist = true; | ||||
|             } | ||||
|             else files = ev.dataTransfer.files; // FileList | ||||
|             else files = e.dataTransfer.files; // FileList | ||||
|         } | ||||
|         else files = ev.target.files; | ||||
|         else files = e.target.files; | ||||
|  | ||||
|         if (files.length == 0) | ||||
|             return alert('no files selected??'); | ||||
| @@ -298,7 +318,7 @@ function up2k_init(have_crypto) { | ||||
|         for (var a = 0; a < good_files.length; a++) | ||||
|             msg.push(good_files[a].name); | ||||
|  | ||||
|         if (ask_up && !confirm(msg.join('\n'))) | ||||
|         if (ask_up && !fsearch && !confirm(msg.join('\n'))) | ||||
|             return; | ||||
|  | ||||
|         for (var a = 0; a < good_files.length; a++) { | ||||
| @@ -312,6 +332,8 @@ function up2k_init(have_crypto) { | ||||
|                 "name": fobj.name, | ||||
|                 "size": fobj.size, | ||||
|                 "lmod": lmod / 1000, | ||||
|                 "purl": get_evpath(), | ||||
|                 "done": false, | ||||
|                 "hash": [] | ||||
|             }; | ||||
|  | ||||
| @@ -326,7 +348,7 @@ function up2k_init(have_crypto) { | ||||
|  | ||||
|             var tr = document.createElement('tr'); | ||||
|             tr.innerHTML = '<td id="f{0}n"></td><td id="f{0}t">hashing</td><td id="f{0}p" class="prog"></td>'.format(st.files.length); | ||||
|             tr.getElementsByTagName('td')[0].textContent = entry.name; | ||||
|             tr.getElementsByTagName('td')[0].innerHTML = fsearch ? entry.name : linksplit(esc(entry.purl + entry.name)).join(' '); | ||||
|             ebi('u2tab').appendChild(tr); | ||||
|  | ||||
|             st.files.push(entry); | ||||
| @@ -344,6 +366,19 @@ function up2k_init(have_crypto) { | ||||
|     } | ||||
|     more_one_file(); | ||||
|  | ||||
|     function u2cleanup(e) { | ||||
|         ev(e); | ||||
|         for (var a = 0; a < st.files.length; a++) { | ||||
|             var t = st.files[a]; | ||||
|             if (t.done && t.name) { | ||||
|                 var tr = ebi('f{0}p'.format(t.n)).parentNode; | ||||
|                 tr.parentNode.removeChild(tr); | ||||
|                 t.name = undefined; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     ebi('u2cleanup').onclick = u2cleanup; | ||||
|  | ||||
|     ///// | ||||
|     //// | ||||
|     ///   actuator | ||||
| @@ -357,14 +392,18 @@ function up2k_init(have_crypto) { | ||||
|     } | ||||
|  | ||||
|     function hashing_permitted() { | ||||
|         var lim = multitask ? 1 : 0; | ||||
|         return handshakes_permitted() && lim >= | ||||
|         if (multitask) { | ||||
|             var ahead = st.bytes.hashed - st.bytes.uploaded; | ||||
|             return ahead < 1024 * 1024 * 128; | ||||
|         } | ||||
|         return handshakes_permitted() && 0 == | ||||
|             st.todo.handshake.length + | ||||
|             st.busy.handshake.length; | ||||
|     } | ||||
|  | ||||
|     var tasker = (function () { | ||||
|         var mutex = false; | ||||
|         var was_busy = false; | ||||
|  | ||||
|         function taskerd() { | ||||
|             if (mutex) | ||||
| @@ -372,8 +411,63 @@ function up2k_init(have_crypto) { | ||||
|  | ||||
|             mutex = true; | ||||
|             while (true) { | ||||
|                 if (false) { | ||||
|                     ebi('srv_info').innerHTML = | ||||
|                         new Date().getTime() + ", " + | ||||
|                         st.todo.hash.length + ", " + | ||||
|                         st.todo.handshake.length + ", " + | ||||
|                         st.todo.upload.length + ", " + | ||||
|                         st.busy.hash.length + ", " + | ||||
|                         st.busy.handshake.length + ", " + | ||||
|                         st.busy.upload.length; | ||||
|                 } | ||||
|  | ||||
|                 var is_busy = 0 != | ||||
|                     st.todo.hash.length + | ||||
|                     st.todo.handshake.length + | ||||
|                     st.todo.upload.length + | ||||
|                     st.busy.hash.length + | ||||
|                     st.busy.handshake.length + | ||||
|                     st.busy.upload.length; | ||||
|  | ||||
|                 if (was_busy != is_busy) { | ||||
|                     was_busy = is_busy; | ||||
|  | ||||
|                     if (is_busy) | ||||
|                         window.addEventListener("beforeunload", warn_uploader_busy); | ||||
|                     else | ||||
|                         window.removeEventListener("beforeunload", warn_uploader_busy); | ||||
|                 } | ||||
|  | ||||
|                 if (flag) { | ||||
|                     if (is_busy) { | ||||
|                         var now = new Date().getTime(); | ||||
|                         flag.take(now); | ||||
|                         if (!flag.ours) { | ||||
|                             setTimeout(taskerd, 100); | ||||
|                             mutex = false; | ||||
|                             return; | ||||
|                         } | ||||
|                     } | ||||
|                     else if (flag.ours) { | ||||
|                         flag.give(); | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 var mou_ikkai = false; | ||||
|  | ||||
|                 if (st.todo.handshake.length > 0 && | ||||
|                     st.busy.handshake.length == 0 && ( | ||||
|                         st.todo.handshake[0].t3 || ( | ||||
|                             handshakes_permitted() && | ||||
|                             st.busy.upload.length < parallel_uploads | ||||
|                         ) | ||||
|                     ) | ||||
|                 ) { | ||||
|                     exec_handshake(); | ||||
|                     mou_ikkai = true; | ||||
|                 } | ||||
|  | ||||
|                 if (handshakes_permitted() && | ||||
|                     st.todo.handshake.length > 0 && | ||||
|                     st.busy.handshake.length == 0 && | ||||
| @@ -512,6 +606,8 @@ function up2k_init(have_crypto) { | ||||
|  | ||||
|         var t = st.todo.hash.shift(); | ||||
|         st.busy.hash.push(t); | ||||
|         st.bytes.hashed += t.size; | ||||
|         t.bytes_uploaded = 0; | ||||
|         t.t1 = new Date().getTime(); | ||||
|  | ||||
|         var nchunk = 0; | ||||
| @@ -559,8 +655,8 @@ function up2k_init(have_crypto) { | ||||
|             prog(t.n, nchunk, col_hashing); | ||||
|         }; | ||||
|  | ||||
|         var segm_load = function (ev) { | ||||
|             cache_buf = ev.target.result; | ||||
|         var segm_load = function (e) { | ||||
|             cache_buf = e.target.result; | ||||
|             cache_ofs = 0; | ||||
|             hash_calc(); | ||||
|         }; | ||||
| @@ -634,14 +730,42 @@ function up2k_init(have_crypto) { | ||||
|         st.busy.handshake.push(t); | ||||
|  | ||||
|         var xhr = new XMLHttpRequest(); | ||||
|         xhr.onload = function (ev) { | ||||
|         xhr.onload = function (e) { | ||||
|             if (xhr.status == 200) { | ||||
|                 var response = JSON.parse(xhr.responseText); | ||||
|  | ||||
|                 if (!response.name) { | ||||
|                     var msg = ''; | ||||
|                     var smsg = ''; | ||||
|                     if (!response || !response.hits || !response.hits.length) { | ||||
|                         msg = 'not found on server'; | ||||
|                         smsg = '404'; | ||||
|                     } | ||||
|                     else { | ||||
|                         smsg = 'found'; | ||||
|                         var hit = response.hits[0], | ||||
|                             msg = linksplit(hit.rp).join(''), | ||||
|                             tr = unix2iso(hit.ts), | ||||
|                             tu = unix2iso(t.lmod), | ||||
|                             diff = parseInt(t.lmod) - parseInt(hit.ts), | ||||
|                             cdiff = (Math.abs(diff) <= 2) ? '3c0' : 'f0b', | ||||
|                             sdiff = '<span style="color:#' + cdiff + '">diff ' + diff; | ||||
|  | ||||
|                         msg += '<br /><small>' + tr + ' (srv), ' + tu + ' (You), ' + sdiff + '</span></span>'; | ||||
|                     } | ||||
|                     ebi('f{0}p'.format(t.n)).innerHTML = msg; | ||||
|                     ebi('f{0}t'.format(t.n)).innerHTML = smsg; | ||||
|                     st.busy.handshake.splice(st.busy.handshake.indexOf(t), 1); | ||||
|                     st.bytes.uploaded += t.size; | ||||
|                     t.done = true; | ||||
|                     tasker(); | ||||
|                     return; | ||||
|                 } | ||||
|  | ||||
|                 if (response.name !== t.name) { | ||||
|                     // file exists; server renamed us | ||||
|                     t.name = response.name; | ||||
|                     ebi('f{0}n'.format(t.n)).textContent = t.name; | ||||
|                     ebi('f{0}n'.format(t.n)).innerHTML = linksplit(esc(t.purl + t.name)).join(' '); | ||||
|                 } | ||||
|  | ||||
|                 t.postlist = []; | ||||
| @@ -675,11 +799,15 @@ function up2k_init(have_crypto) { | ||||
|                 st.busy.handshake.splice(st.busy.handshake.indexOf(t), 1); | ||||
|  | ||||
|                 if (done) { | ||||
|                     t.done = true; | ||||
|                     st.bytes.uploaded += t.size - t.bytes_uploaded; | ||||
|                     var spd1 = (t.size / ((t.t2 - t.t1) / 1000.)) / (1024 * 1024.); | ||||
|                     var spd2 = (t.size / ((t.t3 - t.t2) / 1000.)) / (1024 * 1024.); | ||||
|                     ebi('f{0}p'.format(t.n)).innerHTML = 'hash {0}, up {1} MB/s'.format( | ||||
|                         spd1.toFixed(2), spd2.toFixed(2)); | ||||
|                 } | ||||
|                 else t.t3 = undefined; | ||||
|  | ||||
|                 tasker(); | ||||
|             } | ||||
|             else { | ||||
| @@ -691,6 +819,11 @@ function up2k_init(have_crypto) { | ||||
|                     var ofs = err.lastIndexOf(' : '); | ||||
|                     if (ofs > 0) | ||||
|                         err = err.slice(0, ofs); | ||||
|  | ||||
|                     ofs = err.indexOf('\n/'); | ||||
|                     if (ofs !== -1) { | ||||
|                         err = err.slice(0, ofs + 1) + linksplit(err.slice(ofs + 2, -1)).join(' '); | ||||
|                     } | ||||
|                 } | ||||
|                 if (err != "") { | ||||
|                     ebi('f{0}t'.format(t.n)).innerHTML = "ERROR"; | ||||
| @@ -707,14 +840,19 @@ function up2k_init(have_crypto) { | ||||
|                     "no further information")); | ||||
|             } | ||||
|         }; | ||||
|         xhr.open('POST', post_url + 'handshake.php', true); | ||||
|         xhr.responseType = 'text'; | ||||
|         xhr.send(JSON.stringify({ | ||||
|  | ||||
|         var req = { | ||||
|             "name": t.name, | ||||
|             "size": t.size, | ||||
|             "lmod": t.lmod, | ||||
|             "hash": t.hash | ||||
|         })); | ||||
|         }; | ||||
|         if (fsearch) | ||||
|             req.srch = 1; | ||||
|  | ||||
|         xhr.open('POST', t.purl + 'handshake.php', true); | ||||
|         xhr.responseType = 'text'; | ||||
|         xhr.send(JSON.stringify(req)); | ||||
|     } | ||||
|  | ||||
|     ///// | ||||
| @@ -743,7 +881,7 @@ function up2k_init(have_crypto) { | ||||
|             alert('y o u   b r o k e    i t\n\n(was that a folder? just files please)'); | ||||
|         }; | ||||
|  | ||||
|         reader.onload = function (ev) { | ||||
|         reader.onload = function (e) { | ||||
|             var xhr = new XMLHttpRequest(); | ||||
|             xhr.upload.onprogress = function (xev) { | ||||
|                 var perc = xev.loaded / (cdr - car) * 100; | ||||
| @@ -752,12 +890,14 @@ function up2k_init(have_crypto) { | ||||
|             xhr.onload = function (xev) { | ||||
|                 if (xhr.status == 200) { | ||||
|                     prog(t.n, npart, col_uploaded); | ||||
|                     st.bytes.uploaded += cdr - car; | ||||
|                     t.bytes_uploaded += cdr - car; | ||||
|                     st.busy.upload.splice(st.busy.upload.indexOf(upt), 1); | ||||
|                     t.postlist.splice(t.postlist.indexOf(npart), 1); | ||||
|                     if (t.postlist.length == 0) { | ||||
|                         t.t3 = new Date().getTime(); | ||||
|                         ebi('f{0}t'.format(t.n)).innerHTML = 'verifying'; | ||||
|                         st.todo.handshake.push(t); | ||||
|                         st.todo.handshake.unshift(t); | ||||
|                     } | ||||
|                     tasker(); | ||||
|                 } | ||||
| @@ -768,14 +908,14 @@ function up2k_init(have_crypto) { | ||||
|                         (xhr.responseText && xhr.responseText) || | ||||
|                         "no further information")); | ||||
|             }; | ||||
|             xhr.open('POST', post_url + 'chunkpit.php', true); | ||||
|             xhr.open('POST', t.purl + 'chunkpit.php', true); | ||||
|             //xhr.setRequestHeader("X-Up2k-Hash", t.hash[npart].substr(1) + "x"); | ||||
|             xhr.setRequestHeader("X-Up2k-Hash", t.hash[npart]); | ||||
|             xhr.setRequestHeader("X-Up2k-Wark", t.wark); | ||||
|             xhr.setRequestHeader('Content-Type', 'application/octet-stream'); | ||||
|             xhr.overrideMimeType('Content-Type', 'application/octet-stream'); | ||||
|             xhr.responseType = 'text'; | ||||
|             xhr.send(ev.target.result); | ||||
|             xhr.send(e.target.result); | ||||
|         }; | ||||
|  | ||||
|         reader.readAsArrayBuffer(bobslice.call(t.fobj, car, cdr)); | ||||
| @@ -804,6 +944,46 @@ function up2k_init(have_crypto) { | ||||
|     ///   config ui | ||||
|     // | ||||
|  | ||||
|     function onresize(e) { | ||||
|         var bar = ebi('ops'), | ||||
|             wpx = innerWidth, | ||||
|             fpx = parseInt(getComputedStyle(bar)['font-size']), | ||||
|             wem = wpx * 1.0 / fpx, | ||||
|             wide = wem > 54, | ||||
|             parent = ebi(wide ? 'u2btn_cw' : 'u2btn_ct'), | ||||
|             btn = ebi('u2btn'); | ||||
|  | ||||
|         //console.log([wpx, fpx, wem]); | ||||
|         if (btn.parentNode !== parent) { | ||||
|             parent.appendChild(btn); | ||||
|             ebi('u2conf').setAttribute('class', wide ? 'has_btn' : ''); | ||||
|         } | ||||
|     } | ||||
|     window.addEventListener('resize', onresize); | ||||
|     onresize(); | ||||
|  | ||||
|     function desc_show(e) { | ||||
|         var msg = this.getAttribute('alt'); | ||||
|         msg = msg.replace(/\$N/g, "<br />"); | ||||
|         var cdesc = ebi('u2cdesc'); | ||||
|         cdesc.innerHTML = msg; | ||||
|         cdesc.setAttribute('class', 'show'); | ||||
|     } | ||||
|     function desc_hide(e) { | ||||
|         ebi('u2cdesc').setAttribute('class', ''); | ||||
|     } | ||||
|     var o = document.querySelectorAll('#u2conf *[alt]'); | ||||
|     for (var a = o.length - 1; a >= 0; a--) { | ||||
|         o[a].parentNode.getElementsByTagName('input')[0].setAttribute('alt', o[a].getAttribute('alt')); | ||||
|     } | ||||
|     var o = document.querySelectorAll('#u2conf *[alt]'); | ||||
|     for (var a = 0; a < o.length; a++) { | ||||
|         o[a].onfocus = desc_show; | ||||
|         o[a].onblur = desc_hide; | ||||
|         o[a].onmouseenter = desc_show; | ||||
|         o[a].onmouseleave = desc_hide; | ||||
|     } | ||||
|  | ||||
|     function bumpthread(dir) { | ||||
|         try { | ||||
|             dir.stopPropagation(); | ||||
| @@ -818,7 +998,7 @@ function up2k_init(have_crypto) { | ||||
|                 return; | ||||
|  | ||||
|             parallel_uploads = v; | ||||
|             localStorage.setItem('nthread', v); | ||||
|             swrite('nthread', v); | ||||
|             obj.style.background = '#444'; | ||||
|             return; | ||||
|         } | ||||
| @@ -845,29 +1025,103 @@ function up2k_init(have_crypto) { | ||||
|         bcfg_set('ask_up', ask_up); | ||||
|     } | ||||
|  | ||||
|     function nop(ev) { | ||||
|         ev.preventDefault(); | ||||
|     function tgl_fsearch() { | ||||
|         set_fsearch(!fsearch); | ||||
|     } | ||||
|  | ||||
|     function set_fsearch(new_state) { | ||||
|         var perms = document.body.getAttribute('perms'); | ||||
|         var read_only = false; | ||||
|  | ||||
|         if (!ebi('fsearch')) { | ||||
|             new_state = false; | ||||
|         } | ||||
|         else if (perms && perms.indexOf('write') === -1) { | ||||
|             new_state = true; | ||||
|             read_only = true; | ||||
|         } | ||||
|  | ||||
|         if (new_state !== undefined) { | ||||
|             fsearch = new_state; | ||||
|             bcfg_set('fsearch', fsearch); | ||||
|         } | ||||
|  | ||||
|         try { | ||||
|             document.querySelector('label[for="fsearch"]').style.opacity = read_only ? '0' : '1'; | ||||
|         } | ||||
|         catch (ex) { } | ||||
|  | ||||
|         try { | ||||
|             var fun = fsearch ? 'add' : 'remove'; | ||||
|             ebi('op_up2k').classList[fun]('srch'); | ||||
|  | ||||
|             var ico = fsearch ? '🔎' : '🚀'; | ||||
|             var desc = fsearch ? 'Search' : 'Upload'; | ||||
|             ebi('u2bm').innerHTML = ico + ' <sup>' + desc + '</sup>'; | ||||
|         } | ||||
|         catch (ex) { } | ||||
|     } | ||||
|  | ||||
|     function tgl_flag_en() { | ||||
|         flag_en = !flag_en; | ||||
|         bcfg_set('flag_en', flag_en); | ||||
|         apply_flag_cfg(); | ||||
|     } | ||||
|  | ||||
|     function apply_flag_cfg() { | ||||
|         if (flag_en && !flag) { | ||||
|             try { | ||||
|                 flag = up2k_flagbus(); | ||||
|             } | ||||
|             catch (ex) { | ||||
|                 console.log("flag error: " + ex.toString()); | ||||
|                 tgl_flag_en(); | ||||
|             } | ||||
|         } | ||||
|         else if (!flag_en && flag) { | ||||
|             flag.ch.close(); | ||||
|             flag = false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function nop(e) { | ||||
|         ev(e); | ||||
|         this.click(); | ||||
|     } | ||||
|  | ||||
|     ebi('nthread_add').onclick = function (ev) { | ||||
|         ev.preventDefault(); | ||||
|     ebi('nthread_add').onclick = function (e) { | ||||
|         ev(e); | ||||
|         bumpthread(1); | ||||
|     }; | ||||
|     ebi('nthread_sub').onclick = function (ev) { | ||||
|         ev.preventDefault(); | ||||
|     ebi('nthread_sub').onclick = function (e) { | ||||
|         ev(e); | ||||
|         bumpthread(-1); | ||||
|     }; | ||||
|  | ||||
|     ebi('nthread').addEventListener('input', bumpthread, false); | ||||
|     ebi('multitask').addEventListener('click', tgl_multitask, false); | ||||
|     ebi('ask_up').addEventListener('click', tgl_ask_up, false); | ||||
|     ebi('flag_en').addEventListener('click', tgl_flag_en, false); | ||||
|     var o = ebi('fsearch'); | ||||
|     if (o) | ||||
|         o.addEventListener('click', tgl_fsearch, false); | ||||
|  | ||||
|     var nodes = ebi('u2conf').getElementsByTagName('a'); | ||||
|     for (var a = nodes.length - 1; a >= 0; a--) | ||||
|         nodes[a].addEventListener('touchend', nop, false); | ||||
|  | ||||
|     set_fsearch(); | ||||
|     bumpthread({ "target": 1 }) | ||||
|  | ||||
|     return { "init_deps": init_deps } | ||||
|     return { "init_deps": init_deps, "set_fsearch": set_fsearch } | ||||
| } | ||||
|  | ||||
|  | ||||
| function warn_uploader_busy(e) { | ||||
|     e.preventDefault(); | ||||
|     e.returnValue = ''; | ||||
|     return "upload in progress, click abort and use the file-tree to navigate instead"; | ||||
| } | ||||
|  | ||||
|  | ||||
| if (document.querySelector('#op_up2k.act')) | ||||
|     goto_up2k(); | ||||
|   | ||||
| @@ -1,92 +1,4 @@ | ||||
| .opview { | ||||
| 	display: none; | ||||
| } | ||||
| .opview.act { | ||||
| 	display: block; | ||||
| } | ||||
| #ops a { | ||||
| 	color: #fc5; | ||||
| 	font-size: 1.5em; | ||||
| 	padding: 0 .3em; | ||||
| 	margin: 0; | ||||
| 	outline: none; | ||||
| } | ||||
| #ops a.act { | ||||
| 	text-decoration: underline; | ||||
| } | ||||
| /* | ||||
| #ops a+a:after, | ||||
| #ops a:first-child:after { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #01a7e1; | ||||
| 	margin-left: .3em; | ||||
| 	position: relative; | ||||
| } | ||||
| #ops a+a:before { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #ff3f1a; | ||||
| 	margin-right: .3em; | ||||
| 	margin-left: -.3em; | ||||
| } | ||||
| #ops a:last-child:after { | ||||
| 	content: ''; | ||||
| } | ||||
| #ops a.act:before, | ||||
| #ops a.act:after { | ||||
| 	text-decoration: none !important; | ||||
| } | ||||
| */ | ||||
| #ops i { | ||||
| 	font-size: 1.5em; | ||||
| } | ||||
| #ops i:before { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #01a7e1; | ||||
| 	position: relative; | ||||
| } | ||||
| #ops i:after { | ||||
| 	content: 'x'; | ||||
| 	color: #282828; | ||||
| 	text-shadow: 0 0 .08em #ff3f1a; | ||||
| 	margin-left: -.35em; | ||||
| 	font-size: 1.05em; | ||||
| } | ||||
| #ops, | ||||
| .opbox { | ||||
| 	border: 1px solid #3a3a3a; | ||||
| 	box-shadow: 0 0 1em #222 inset; | ||||
| } | ||||
| #ops { | ||||
| 	display: none; | ||||
| 	background: #333; | ||||
| 	margin: 1.7em 1.5em 0 1.5em; | ||||
| 	padding: .3em .6em; | ||||
| 	border-radius: .3em; | ||||
| 	border-width: .15em 0; | ||||
| } | ||||
| .opbox { | ||||
| 	background: #2d2d2d; | ||||
| 	margin: 1.5em 0 0 0; | ||||
| 	padding: .5em; | ||||
| 	border-radius: 0 1em 1em 0; | ||||
| 	border-width: .15em .3em .3em 0; | ||||
| 	max-width: 40em; | ||||
| } | ||||
| .opbox input { | ||||
| 	margin: .5em; | ||||
| } | ||||
| .opbox input[type=text] { | ||||
| 	color: #fff; | ||||
| 	background: #383838; | ||||
| 	border: none; | ||||
| 	box-shadow: 0 0 .3em #222; | ||||
| 	border-bottom: 1px solid #fc5; | ||||
| 	border-radius: .2em; | ||||
| 	padding: .2em .3em; | ||||
| } | ||||
|  | ||||
| #op_up2k { | ||||
| 	padding: 0 1em 1em 1em; | ||||
| } | ||||
| @@ -94,6 +6,9 @@ | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| 	width: 2px; | ||||
| 	height: 2px; | ||||
| 	overflow: hidden; | ||||
| } | ||||
| #u2form input { | ||||
| 	background: #444; | ||||
| @@ -104,11 +19,6 @@ | ||||
| 	color: #f87; | ||||
| 	padding: .5em; | ||||
| } | ||||
| #u2form { | ||||
| 	width: 2px; | ||||
| 	height: 2px; | ||||
| 	overflow: hidden; | ||||
| } | ||||
| #u2btn { | ||||
| 	color: #eee; | ||||
| 	background: #555; | ||||
| @@ -117,17 +27,27 @@ | ||||
| 	background: linear-gradient(to bottom, #367 0%, #489 50%, #38788a 51%, #367 100%); | ||||
| 	filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#489', endColorstr='#38788a', GradientType=0); | ||||
| 	text-decoration: none; | ||||
| 	line-height: 1.5em; | ||||
| 	line-height: 1.3em; | ||||
| 	border: 1px solid #222; | ||||
| 	border-radius: .4em; | ||||
| 	text-align: center; | ||||
| 	font-size: 2em; | ||||
| 	margin: 1em auto; | ||||
| 	padding: 1em 0; | ||||
| 	width: 12em; | ||||
| 	font-size: 1.5em; | ||||
| 	margin: .5em auto; | ||||
| 	padding: .8em 0; | ||||
| 	width: 16em; | ||||
| 	cursor: pointer; | ||||
| 	box-shadow: .4em .4em 0 #111; | ||||
| } | ||||
| #op_up2k.srch #u2btn { | ||||
| 	background: linear-gradient(to bottom, #ca3 0%, #fd8 50%, #fc6 51%, #b92 100%); | ||||
| 	text-shadow: 1px 1px 1px #fc6; | ||||
| 	color: #333; | ||||
| } | ||||
| #u2conf #u2btn { | ||||
| 	margin: -1.5em 0; | ||||
| 	padding: .8em 0; | ||||
| 	width: 100%; | ||||
| } | ||||
| #u2notbtn { | ||||
| 	display: none; | ||||
| 	text-align: center; | ||||
| @@ -142,6 +62,9 @@ | ||||
| 	width: calc(100% - 2em); | ||||
| 	max-width: 100em; | ||||
| } | ||||
| #op_up2k.srch #u2tab { | ||||
| 	max-width: none; | ||||
| } | ||||
| #u2tab td { | ||||
| 	border: 1px solid #ccc; | ||||
| 	border-width: 0 0px 1px 0; | ||||
| @@ -153,12 +76,19 @@ | ||||
| #u2tab td:nth-child(3) { | ||||
| 	width: 40%; | ||||
| } | ||||
| #op_up2k.srch #u2tab td:nth-child(3) { | ||||
| 	font-family: sans-serif; | ||||
| 	width: auto; | ||||
| } | ||||
| #u2tab tr+tr:hover td { | ||||
| 	background: #222; | ||||
| } | ||||
| #u2conf { | ||||
| 	margin: 1em auto; | ||||
| 	width: 26em; | ||||
| 	width: 30em; | ||||
| } | ||||
| #u2conf.has_btn { | ||||
| 	width: 46em; | ||||
| } | ||||
| #u2conf * { | ||||
| 	text-align: center; | ||||
| @@ -194,16 +124,72 @@ | ||||
| #u2conf input+a { | ||||
| 	background: #d80; | ||||
| } | ||||
| #u2conf label { | ||||
| 	font-size: 1.6em; | ||||
| 	width: 2em; | ||||
| 	height: 1em; | ||||
| 	padding: .4em 0; | ||||
| 	display: block; | ||||
| 	user-select: none; | ||||
| 	border-radius: .25em; | ||||
| } | ||||
| #u2conf input[type="checkbox"] { | ||||
| 	position: relative; | ||||
| 	opacity: .02; | ||||
| 	top: 2em; | ||||
| } | ||||
| #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 { | ||||
| 	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; | ||||
| 	height: 0; | ||||
| 	padding: 0 1em; | ||||
| 	opacity: .1; | ||||
|     transition: all 0.14s ease-in-out; | ||||
| 	border-radius: .4em; | ||||
| 	box-shadow: 0 .2em .5em #222; | ||||
| } | ||||
| #u2cdesc.show { | ||||
| 	padding: 1em; | ||||
| 	height: auto; | ||||
| 	border-width: .2em 0; | ||||
| 	opacity: 1; | ||||
| } | ||||
| #u2foot { | ||||
| 	color: #fff; | ||||
| 	font-style: italic; | ||||
| } | ||||
| #u2footfoot { | ||||
| 	margin-bottom: -1em; | ||||
| } | ||||
| .prog { | ||||
| 	font-family: monospace; | ||||
| } | ||||
| @@ -225,3 +211,13 @@ | ||||
| 	bottom: 0; | ||||
| 	background: #0a0; | ||||
| } | ||||
| #u2tab a>span { | ||||
| 	font-weight: bold; | ||||
| 	font-style: italic; | ||||
| 	color: #fff; | ||||
| 	padding-left: .2em; | ||||
| } | ||||
| #u2cleanup { | ||||
| 	float: right; | ||||
| 	margin-bottom: -.3em; | ||||
| } | ||||
|   | ||||
| @@ -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="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"> | ||||
|             <input type="hidden" name="act" value="bput" /> | ||||
|             <input type="file" name="f" multiple><br /> | ||||
|             <input type="submit" value="start upload"> | ||||
| @@ -16,7 +9,7 @@ | ||||
|     </div> | ||||
|  | ||||
|     <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"> | ||||
|             <input type="hidden" name="act" value="mkdir" /> | ||||
|             <input type="text" name="name" size="30"> | ||||
|             <input type="submit" value="mkdir"> | ||||
| @@ -24,7 +17,7 @@ | ||||
|     </div> | ||||
|  | ||||
|     <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"> | ||||
|             <input type="hidden" name="act" value="new_md" /> | ||||
|             <input type="text" name="name" size="30"> | ||||
|             <input type="submit" value="create doc"> | ||||
| @@ -32,9 +25,9 @@ | ||||
|     </div> | ||||
|  | ||||
|     <div id="op_msg" class="opview opbox"> | ||||
|         <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"> | ||||
|             <input type="text" name="msg" size="30"> | ||||
|             <input type="submit" value="send"> | ||||
|             <input type="submit" value="send msg"> | ||||
|         </form> | ||||
|     </div> | ||||
|  | ||||
| @@ -44,6 +37,25 @@ | ||||
|             <table id="u2conf"> | ||||
|                 <tr> | ||||
|                     <td>parallel uploads</td> | ||||
|                     <td rowspan="2"> | ||||
|                         <input type="checkbox" id="multitask" /> | ||||
|                         <label for="multitask" alt="continue hashing other files while uploading">🏃</label> | ||||
|                     </td> | ||||
|                     <td rowspan="2"> | ||||
|                         <input type="checkbox" id="ask_up" /> | ||||
|                         <label for="ask_up" alt="ask for confirmation befofre upload starts">💭</label> | ||||
|                     </td> | ||||
|                     <td rowspan="2"> | ||||
|                         <input type="checkbox" id="flag_en" /> | ||||
|                         <label for="flag_en" alt="ensure only one tab is uploading at a time $N (other tabs must have this enabled too)">💤</label> | ||||
|                     </td> | ||||
|                 {%- if have_up2k_idx %} | ||||
|                     <td data-perm="read" rowspan="2"> | ||||
|                         <input type="checkbox" id="fsearch" /> | ||||
|                         <label for="fsearch" alt="don't actually upload, instead check if the files already $N exist on the server (will scan all folders you can read)">🔎</label> | ||||
|                     </td> | ||||
|                 {%- endif %} | ||||
|                     <td data-perm="read" rowspan="2" id="u2btn_cw"></td> | ||||
|                 </tr> | ||||
|                 <tr> | ||||
|                     <td> | ||||
| @@ -51,32 +63,29 @@ | ||||
|                         <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" /> | ||||
|                         <label for="multitask">hash while<br />uploading</label> | ||||
|                     </td> | ||||
|                     <td rowspan="2"> | ||||
|                         <input type="checkbox" id="ask_up" /> | ||||
|                         <label for="ask_up">ask for<br />confirmation</label> | ||||
|                     </td> | ||||
|                 </tr> | ||||
|             </table> | ||||
|  | ||||
|             <div id="u2cdesc"></div> | ||||
|  | ||||
|             <div id="u2notbtn"></div> | ||||
|  | ||||
|             <div id="u2btn"> | ||||
|                 drop files here<br /> | ||||
|                 (or click me) | ||||
|             <div id="u2btn_ct"> | ||||
|                 <div id="u2btn"> | ||||
|                     <span id="u2bm"></span><br /> | ||||
|                     drop files here<br /> | ||||
|                     (or click me) | ||||
|                 </div> | ||||
|             </div> | ||||
|  | ||||
|             <table id="u2tab"> | ||||
|                 <tr> | ||||
|                     <td>filename</td> | ||||
|                     <td>status</td> | ||||
|                     <td>progress</td> | ||||
|                     <td>progress<a href="#" id="u2cleanup">cleanup</a></td> | ||||
|                 </tr> | ||||
|             </table> | ||||
|  | ||||
|             <p id="u2foot"></p> | ||||
|             <p>( if you don't need lastmod timestamps, resumable uploads or progress bars just use the <a href="#" id="u2nope">basic uploader</a>)</p> | ||||
|             <p id="u2footfoot">( if you don't need lastmod timestamps, resumable uploads or progress bars just use the <a href="#" id="u2nope">basic uploader</a>)</p> | ||||
|     </div> | ||||
|   | ||||
| @@ -23,6 +23,7 @@ function esc(txt) { | ||||
| } | ||||
| function vis_exh(msg, url, lineNo, columnNo, error) { | ||||
|     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>', | ||||
|         esc(String(msg)), '</p><p>', esc(url + ' @' + lineNo + ':' + columnNo), '</p>']; | ||||
|  | ||||
| @@ -43,6 +44,21 @@ function ebi(id) { | ||||
|     return document.getElementById(id); | ||||
| } | ||||
|  | ||||
| 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; | ||||
| } | ||||
|  | ||||
|  | ||||
| // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/endsWith | ||||
| if (!String.prototype.endsWith) { | ||||
| @@ -75,35 +91,332 @@ function import_js(url, cb) { | ||||
| } | ||||
|  | ||||
|  | ||||
| function sortTable(table, col) { | ||||
|     var tb = table.tBodies[0], // use `<tbody>` to ignore `<thead>` and `<tfoot>` rows | ||||
| var crctab = (function () { | ||||
|     var c, tab = []; | ||||
|     for (var n = 0; n < 256; n++) { | ||||
|         c = n; | ||||
|         for (var k = 0; k < 8; k++) { | ||||
|             c = ((c & 1) ? (0xEDB88320 ^ (c >>> 1)) : (c >>> 1)); | ||||
|         } | ||||
|         tab[n] = c; | ||||
|     } | ||||
|     return tab; | ||||
| })(); | ||||
|  | ||||
|  | ||||
| function crc32(str) { | ||||
|     var crc = 0 ^ (-1); | ||||
|     for (var i = 0; i < str.length; i++) { | ||||
|         crc = (crc >>> 8) ^ crctab[(crc ^ str.charCodeAt(i)) & 0xFF]; | ||||
|     } | ||||
|     return ((crc ^ (-1)) >>> 0).toString(16); | ||||
| }; | ||||
|  | ||||
|  | ||||
| function sortTable(table, col, cb) { | ||||
|     var tb = table.tBodies[0], | ||||
|         th = table.tHead.rows[0].cells, | ||||
|         tr = Array.prototype.slice.call(tb.rows, 0), | ||||
|         i, reverse = th[col].className == 'sort1' ? -1 : 1; | ||||
|         i, reverse = th[col].className.indexOf('sort1') !== -1 ? -1 : 1; | ||||
|     for (var a = 0, thl = th.length; a < thl; a++) | ||||
|         th[a].className = ''; | ||||
|     th[col].className = 'sort' + reverse; | ||||
|         th[a].className = th[a].className.replace(/ *sort-?1 */, " "); | ||||
|     th[col].className += ' sort' + reverse; | ||||
|     var stype = th[col].getAttribute('sort'); | ||||
|     tr = tr.sort(function (a, b) { | ||||
|         var v1 = a.cells[col].textContent.trim(); | ||||
|         var v2 = b.cells[col].textContent.trim(); | ||||
|         if (stype == 'int') { | ||||
|             v1 = parseInt(v1.replace(/,/g, '')); | ||||
|             v2 = parseInt(v2.replace(/,/g, '')); | ||||
|             return reverse * (v1 - v2); | ||||
|     try { | ||||
|         var nrules = [], rules = jread("fsort", []); | ||||
|         rules.unshift([th[col].getAttribute('name'), reverse, stype || '']); | ||||
|         for (var a = 0; a < rules.length; a++) { | ||||
|             var add = true; | ||||
|             for (var b = 0; b < a; b++) | ||||
|                 if (rules[a][0] == rules[b][0]) | ||||
|                     add = false; | ||||
|  | ||||
|             if (add) | ||||
|                 nrules.push(rules[a]); | ||||
|  | ||||
|             if (nrules.length >= 10) | ||||
|                 break; | ||||
|         } | ||||
|         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; | ||||
|     th && (th = th.rows[0]) && (th = th.cells); | ||||
|     if (th) i = th.length; | ||||
|     else return; // if no `<thead>` then do nothing | ||||
|     while (--i >= 0) (function (i) { | ||||
|         th[i].onclick = function () { | ||||
|             sortTable(table, i); | ||||
|         th[i].onclick = function (e) { | ||||
|             ev(e); | ||||
|             sortTable(table, i, cb); | ||||
|         }; | ||||
|     }(i)); | ||||
| } | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| (function () { | ||||
|     var ops = document.querySelectorAll('#ops>a'); | ||||
|     for (var a = 0; a < ops.length; a++) { | ||||
|         ops[a].onclick = opclick; | ||||
|     } | ||||
| })(); | ||||
|  | ||||
|  | ||||
| function opclick(e) { | ||||
|     ev(e); | ||||
|  | ||||
|     var dest = this.getAttribute('data-dest'); | ||||
|     goto(dest); | ||||
|  | ||||
|     swrite('opmode', dest || null); | ||||
|  | ||||
|     var input = document.querySelector('.opview.act input:not([type="hidden"])') | ||||
|     if (input) | ||||
|         input.focus(); | ||||
| } | ||||
|  | ||||
|  | ||||
| function goto(dest) { | ||||
|     var obj = document.querySelectorAll('.opview.act'); | ||||
|     for (var a = obj.length - 1; a >= 0; a--) | ||||
|         obj[a].classList.remove('act'); | ||||
|  | ||||
|     obj = document.querySelectorAll('#ops>a'); | ||||
|     for (var a = obj.length - 1; a >= 0; a--) | ||||
|         obj[a].classList.remove('act'); | ||||
|  | ||||
|     if (dest) { | ||||
|         var ui = ebi('op_' + dest); | ||||
|         ui.classList.add('act'); | ||||
|         document.querySelector('#ops>a[data-dest=' + dest + ']').classList.add('act'); | ||||
|  | ||||
|         var fn = window['goto_' + dest]; | ||||
|         if (fn) | ||||
|             fn(); | ||||
|     } | ||||
|  | ||||
|     if (window['treectl']) | ||||
|         treectl.onscroll(); | ||||
| } | ||||
|  | ||||
|  | ||||
| (function () { | ||||
|     goto(); | ||||
|     var op = sread('opmode'); | ||||
|     if (op !== null && op !== '.') | ||||
|         goto(op); | ||||
| })(); | ||||
|  | ||||
|  | ||||
| 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) { | ||||
|         var fun = val ? 'add' : 'remove'; | ||||
|         o.classList[fun]('on'); | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| 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); | ||||
| } | ||||
|   | ||||
							
								
								
									
										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 | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## 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 | ||||
|  | ||||
| @@ -60,6 +67,33 @@ 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; } | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## 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 | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## 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 | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## 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 | ||||
|  | ||||
| @@ -89,6 +123,9 @@ for d in /usr /var; do find $d -type f -size +30M 2>/dev/null; done | while IFS= | ||||
| brew install python@2 | ||||
| pip install virtualenv | ||||
|  | ||||
| # readme toc | ||||
| cat README.md | awk '!/^#/{next} {lv=length($1);sub(/[^ ]+ /,"");bab=$0;gsub(/ /,"-",bab)} {printf "%" ((lv-1)*4+1) "s [%s](#%s)\n", "*",$0,bab}' | ||||
|  | ||||
|  | ||||
| ## | ||||
| ## http 206 | ||||
|   | ||||
| @@ -20,6 +20,7 @@ set -e | ||||
| # -rwxr-xr-x  0 ed ed  183808 Nov 19 00:43 copyparty-extras/sfx-lite/copyparty-sfx.py | ||||
|  | ||||
|  | ||||
| command -v gnutar && tar() { gnutar "$@"; } | ||||
| command -v gtar && tar() { gtar "$@"; } | ||||
| command -v gsed && sed() { gsed "$@"; } | ||||
| td="$(mktemp -d)" | ||||
| @@ -29,11 +30,11 @@ pwd | ||||
|  | ||||
|  | ||||
| dl_text() { | ||||
| 	command -v curl && exec curl "$@" | ||||
| 	command -v curl >/dev/null && exec curl "$@" | ||||
| 	exec wget -O- "$@" | ||||
| } | ||||
| dl_files() { | ||||
| 	command -v curl && exec curl -L --remote-name-all "$@" | ||||
| 	command -v curl >/dev/null && exec curl -L --remote-name-all "$@" | ||||
| 	exec wget "$@" | ||||
| } | ||||
| export -f dl_files | ||||
|   | ||||
| @@ -1,12 +1,10 @@ | ||||
| FROM    alpine:3.11 | ||||
| FROM    alpine:3.13 | ||||
| WORKDIR /z | ||||
| ENV     ver_asmcrypto=2821dd1dedd1196c378f5854037dda5c869313f3 \ | ||||
|         ver_markdownit=10.0.0 \ | ||||
|         ver_showdown=1.9.1 \ | ||||
| ENV     ver_asmcrypto=5b994303a9d3e27e0915f72a10b6c2c51535a4dc \ | ||||
|         ver_marked=1.1.0 \ | ||||
|         ver_ogvjs=1.6.1 \ | ||||
|         ver_mde=2.10.1 \ | ||||
|         ver_codemirror=5.53.2 \ | ||||
|         ver_ogvjs=1.8.0 \ | ||||
|         ver_mde=2.14.0 \ | ||||
|         ver_codemirror=5.59.3 \ | ||||
|         ver_fontawesome=5.13.0 \ | ||||
|         ver_zopfli=1.0.3 | ||||
|  | ||||
| @@ -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 \ | ||||
|         && apk add cmake make g++ git bash npm patch wget tar pigz brotli gzip unzip python3 python3-dev brotli py3-brotli \ | ||||
|         && wget https://github.com/brion/ogv.js/releases/download/$ver_ogvjs/ogvjs-$ver_ogvjs.zip -O ogvjs.zip \ | ||||
|         && wget https://github.com/asmcrypto/asmcrypto.js/archive/$ver_asmcrypto.tar.gz -O asmcrypto.tgz \ | ||||
|         && wget https://github.com/openpgpjs/asmcrypto.js/archive/$ver_asmcrypto.tar.gz -O asmcrypto.tgz \ | ||||
|         && wget https://github.com/markedjs/marked/archive/v$ver_marked.tar.gz -O marked.tgz \ | ||||
|         && wget https://github.com/Ionaru/easy-markdown-editor/archive/$ver_mde.tar.gz -O mde.tgz \ | ||||
|         && wget https://github.com/codemirror/CodeMirror/archive/$ver_codemirror.tar.gz -O codemirror.tgz \ | ||||
| @@ -52,6 +50,7 @@ RUN     tar -xf zopfli.tgz \ | ||||
|             -S . \ | ||||
|         && make -C build \ | ||||
|         && make -C build install \ | ||||
|         && python3 -m ensurepip \ | ||||
|         && python3 -m pip install fonttools zopfli | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| diff -NarU2 CodeMirror-orig/mode/gfm/gfm.js CodeMirror-edit/mode/gfm/gfm.js | ||||
| --- CodeMirror-orig/mode/gfm/gfm.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/mode/gfm/gfm.js	2020-05-02 02:13:32.142131800 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/mode/gfm/gfm.js codemirror-5.59.3/mode/gfm/gfm.js | ||||
| --- codemirror-5.59.3-orig/mode/gfm/gfm.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/mode/gfm/gfm.js	2021-02-21 20:42:02.166174775 +0000 | ||||
| @@ -97,5 +97,5 @@ | ||||
|          } | ||||
|        } | ||||
| @@ -15,9 +15,9 @@ diff -NarU2 CodeMirror-orig/mode/gfm/gfm.js CodeMirror-edit/mode/gfm/gfm.js | ||||
| +      }*/ | ||||
|        stream.next(); | ||||
|        return null; | ||||
| diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js | ||||
| --- CodeMirror-orig/mode/meta.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/mode/meta.js	2020-05-02 03:56:58.852408400 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/mode/meta.js codemirror-5.59.3/mode/meta.js | ||||
| --- codemirror-5.59.3-orig/mode/meta.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/mode/meta.js	2021-02-21 20:42:54.798742821 +0000 | ||||
| @@ -13,4 +13,5 @@ | ||||
|   | ||||
|    CodeMirror.modeInfo = [ | ||||
| @@ -28,7 +28,7 @@ diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js | ||||
|      {name: "Gas", mime: "text/x-gas", mode: "gas", ext: ["s"]}, | ||||
|      {name: "Gherkin", mime: "text/x-feature", mode: "gherkin", ext: ["feature"]}, | ||||
| +    */ | ||||
|      {name: "GitHub Flavored Markdown", mime: "text/x-gfm", mode: "gfm", file: /^(readme|contributing|history).md$/i}, | ||||
|      {name: "GitHub Flavored Markdown", mime: "text/x-gfm", mode: "gfm", file: /^(readme|contributing|history)\.md$/i}, | ||||
| +    /* | ||||
|      {name: "Go", mime: "text/x-go", mode: "go", ext: ["go"]}, | ||||
|      {name: "Groovy", mime: "text/x-groovy", mode: "groovy", ext: ["groovy", "gradle"], file: /^Jenkinsfile$/}, | ||||
| @@ -56,16 +56,16 @@ diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js | ||||
| +    /* | ||||
|      {name: "XQuery", mime: "application/xquery", mode: "xquery", ext: ["xy", "xquery"]}, | ||||
|      {name: "Yacas", mime: "text/x-yacas", mode: "yacas", ext: ["ys"]}, | ||||
| @@ -171,4 +180,5 @@ | ||||
|      {name: "xu", mime: "text/x-xu", mode: "mscgen", ext: ["xu"]}, | ||||
|      {name: "msgenny", mime: "text/x-msgenny", mode: "mscgen", ext: ["msgenny"]} | ||||
| @@ -172,4 +181,5 @@ | ||||
|      {name: "msgenny", mime: "text/x-msgenny", mode: "mscgen", ext: ["msgenny"]}, | ||||
|      {name: "WebAssembly", mime: "text/webassembly", mode: "wast", ext: ["wat", "wast"]}, | ||||
| +    */ | ||||
|    ]; | ||||
|    // Ensure all modes have a mime property for backwards compatibility | ||||
| diff -NarU2 CodeMirror-orig/src/display/selection.js CodeMirror-edit/src/display/selection.js | ||||
| --- CodeMirror-orig/src/display/selection.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/display/selection.js	2020-05-02 03:27:30.144662800 +0200 | ||||
| @@ -83,29 +83,21 @@ | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/display/selection.js codemirror-5.59.3/src/display/selection.js | ||||
| --- codemirror-5.59.3-orig/src/display/selection.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/display/selection.js	2021-02-21 20:44:14.860894328 +0000 | ||||
| @@ -84,29 +84,21 @@ | ||||
|      let order = getOrder(lineObj, doc.direction) | ||||
|      iterateBidiSections(order, fromArg || 0, toArg == null ? lineLen : toArg, (from, to, dir, i) => { | ||||
| -      let ltr = dir == "ltr" | ||||
| @@ -105,24 +105,24 @@ diff -NarU2 CodeMirror-orig/src/display/selection.js CodeMirror-edit/src/display | ||||
| +          botRight = openEnd && last ? rightSide : toPos.right | ||||
|          add(topLeft, fromPos.top, topRight - topLeft, fromPos.bottom) | ||||
|          if (fromPos.bottom < toPos.top) add(leftSide, fromPos.bottom, null, toPos.top) | ||||
| diff -NarU2 CodeMirror-orig/src/input/ContentEditableInput.js CodeMirror-edit/src/input/ContentEditableInput.js | ||||
| --- CodeMirror-orig/src/input/ContentEditableInput.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/input/ContentEditableInput.js	2020-05-02 03:33:05.707995500 +0200 | ||||
| @@ -391,4 +391,5 @@ | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/input/ContentEditableInput.js codemirror-5.59.3/src/input/ContentEditableInput.js | ||||
| --- codemirror-5.59.3-orig/src/input/ContentEditableInput.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/input/ContentEditableInput.js	2021-02-21 20:44:33.273953867 +0000 | ||||
| @@ -399,4 +399,5 @@ | ||||
|    let info = mapFromLineView(view, line, pos.line) | ||||
|   | ||||
| +  /* | ||||
|    let order = getOrder(line, cm.doc.direction), side = "left" | ||||
|    if (order) { | ||||
| @@ -396,4 +397,5 @@ | ||||
| @@ -404,4 +405,5 @@ | ||||
|      side = partPos % 2 ? "right" : "left" | ||||
|    } | ||||
| +  */ | ||||
|    let result = nodeAndOffsetInLineMap(info.map, pos.ch, side) | ||||
|    result.offset = result.collapse == "right" ? result.end : result.start | ||||
| diff -NarU2 CodeMirror-orig/src/input/movement.js CodeMirror-edit/src/input/movement.js | ||||
| --- CodeMirror-orig/src/input/movement.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/input/movement.js	2020-05-02 03:31:19.710773500 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/input/movement.js codemirror-5.59.3/src/input/movement.js | ||||
| --- codemirror-5.59.3-orig/src/input/movement.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/input/movement.js	2021-02-21 20:45:12.763093671 +0000 | ||||
| @@ -15,4 +15,5 @@ | ||||
|   | ||||
|  export function endOfLine(visually, cm, lineObj, lineNo, dir) { | ||||
| @@ -146,9 +146,9 @@ diff -NarU2 CodeMirror-orig/src/input/movement.js CodeMirror-edit/src/input/move | ||||
|    return null | ||||
| +  */ | ||||
|  } | ||||
| diff -NarU2 CodeMirror-orig/src/line/line_data.js CodeMirror-edit/src/line/line_data.js | ||||
| --- CodeMirror-orig/src/line/line_data.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/line/line_data.js	2020-05-02 03:17:02.785065000 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/line/line_data.js codemirror-5.59.3/src/line/line_data.js | ||||
| --- codemirror-5.59.3-orig/src/line/line_data.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/line/line_data.js	2021-02-21 20:45:36.472549599 +0000 | ||||
| @@ -79,6 +79,6 @@ | ||||
|      // Optionally wire in some hacks into the token-rendering | ||||
|      // algorithm, to deal with browser quirks. | ||||
| @@ -158,9 +158,9 @@ diff -NarU2 CodeMirror-orig/src/line/line_data.js CodeMirror-edit/src/line/line_ | ||||
| +    //  builder.addToken = buildTokenBadBidi(builder.addToken, order) | ||||
|      builder.map = [] | ||||
|      let allowFrontierUpdate = lineView != cm.display.externalMeasured && lineNo(line) | ||||
| diff -NarU2 CodeMirror-orig/src/measurement/position_measurement.js CodeMirror-edit/src/measurement/position_measurement.js | ||||
| --- CodeMirror-orig/src/measurement/position_measurement.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/measurement/position_measurement.js	2020-05-02 03:35:20.674159600 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/measurement/position_measurement.js codemirror-5.59.3/src/measurement/position_measurement.js | ||||
| --- codemirror-5.59.3-orig/src/measurement/position_measurement.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/measurement/position_measurement.js	2021-02-21 20:50:52.372945293 +0000 | ||||
| @@ -380,5 +380,6 @@ | ||||
|      sticky = "after" | ||||
|    } | ||||
| @@ -199,9 +199,9 @@ diff -NarU2 CodeMirror-orig/src/measurement/position_measurement.js CodeMirror-e | ||||
| +*/ | ||||
|   | ||||
|  let measureText | ||||
| diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js | ||||
| --- CodeMirror-orig/src/util/bidi.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/util/bidi.js	2020-05-02 03:12:44.418649800 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/util/bidi.js codemirror-5.59.3/src/util/bidi.js | ||||
| --- codemirror-5.59.3-orig/src/util/bidi.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/util/bidi.js	2021-02-21 20:52:18.168092225 +0000 | ||||
| @@ -4,5 +4,5 @@ | ||||
|   | ||||
|  export function iterateBidiSections(order, from, to, f) { | ||||
| @@ -239,20 +239,19 @@ diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js | ||||
| +  var fun = function(str, direction) { | ||||
|      let outerType = direction == "ltr" ? "L" : "R" | ||||
|   | ||||
| @@ -204,12 +210,16 @@ | ||||
| @@ -204,5 +210,11 @@ | ||||
|      return direction == "rtl" ? order.reverse() : order | ||||
|    } | ||||
| -})() | ||||
|   | ||||
| +  return function(str, direction) { | ||||
| +    var ret = fun(str, direction); | ||||
| +    console.log("bidiOrdering inner ([%s], %s) => [%s]", str, direction, ret); | ||||
| +    return ret; | ||||
| +  } | ||||
| +})() | ||||
|  })() | ||||
| +*/ | ||||
|   | ||||
|  // Get the bidi ordering for the given line (and cache it). Returns | ||||
|  // false for lines that are fully left-to-right, and an array of | ||||
| @@ -210,6 +222,4 @@ | ||||
|  // BidiSpan objects otherwise. | ||||
|  export function getOrder(line, direction) { | ||||
| -  let order = line.order | ||||
| @@ -260,9 +259,9 @@ diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js | ||||
| -  return order | ||||
| +  return false; | ||||
|  } | ||||
| diff -NarU2 CodeMirror-orig/src/util/feature_detection.js CodeMirror-edit/src/util/feature_detection.js | ||||
| --- CodeMirror-orig/src/util/feature_detection.js	2020-04-21 12:47:20.000000000 +0200 | ||||
| +++ CodeMirror-edit/src/util/feature_detection.js	2020-05-02 03:16:21.085621400 +0200 | ||||
| diff -NarU2 codemirror-5.59.3-orig/src/util/feature_detection.js codemirror-5.59.3/src/util/feature_detection.js | ||||
| --- codemirror-5.59.3-orig/src/util/feature_detection.js	2021-02-20 21:24:57.000000000 +0000 | ||||
| +++ codemirror-5.59.3/src/util/feature_detection.js	2021-02-21 20:49:22.191269270 +0000 | ||||
| @@ -25,4 +25,5 @@ | ||||
|  } | ||||
|   | ||||
|   | ||||
| @@ -1,33 +1,57 @@ | ||||
| diff -NarU2 easymde-orig/gulpfile.js easymde-mod1/gulpfile.js | ||||
| --- easymde-orig/gulpfile.js	2020-04-06 14:09:36.000000000 +0200 | ||||
| +++ easymde-mod1/gulpfile.js	2020-05-01 14:33:52.260175200 +0200 | ||||
| diff -NarU2 easy-markdown-editor-2.14.0-orig/gulpfile.js easy-markdown-editor-2.14.0/gulpfile.js | ||||
| --- easy-markdown-editor-2.14.0-orig/gulpfile.js	2021-02-14 12:11:48.000000000 +0000 | ||||
| +++ easy-markdown-editor-2.14.0/gulpfile.js	2021-02-21 20:55:37.134701007 +0000 | ||||
| @@ -25,5 +25,4 @@ | ||||
|      './node_modules/codemirror/lib/codemirror.css', | ||||
|      './src/css/*.css', | ||||
| -    './node_modules/codemirror-spell-checker/src/css/spell-checker.css', | ||||
|  ]; | ||||
|   | ||||
| diff -NarU2 easymde-orig/package.json easymde-mod1/package.json | ||||
| --- easymde-orig/package.json	2020-04-06 14:09:36.000000000 +0200 | ||||
| +++ easymde-mod1/package.json	2020-05-01 14:33:57.189975800 +0200 | ||||
| diff -NarU2 easy-markdown-editor-2.14.0-orig/package.json easy-markdown-editor-2.14.0/package.json | ||||
| --- easy-markdown-editor-2.14.0-orig/package.json	2021-02-14 12:11:48.000000000 +0000 | ||||
| +++ easy-markdown-editor-2.14.0/package.json	2021-02-21 20:55:47.761190082 +0000 | ||||
| @@ -21,5 +21,4 @@ | ||||
|      "dependencies": { | ||||
|          "codemirror": "^5.52.2", | ||||
|          "codemirror": "^5.59.2", | ||||
| -        "codemirror-spell-checker": "1.1.2", | ||||
|          "marked": "^0.8.2" | ||||
|          "marked": "^2.0.0" | ||||
|      }, | ||||
| diff -NarU2 easymde-orig/src/js/easymde.js easymde-mod1/src/js/easymde.js | ||||
| --- easymde-orig/src/js/easymde.js	2020-04-06 14:09:36.000000000 +0200 | ||||
| +++ easymde-mod1/src/js/easymde.js	2020-05-01 14:34:19.878774400 +0200 | ||||
| @@ -11,5 +11,4 @@ | ||||
| diff -NarU2 easy-markdown-editor-2.14.0-orig/src/js/easymde.js easy-markdown-editor-2.14.0/src/js/easymde.js | ||||
| --- easy-markdown-editor-2.14.0-orig/src/js/easymde.js	2021-02-14 12:11:48.000000000 +0000 | ||||
| +++ easy-markdown-editor-2.14.0/src/js/easymde.js	2021-02-21 20:57:09.143171536 +0000 | ||||
| @@ -12,5 +12,4 @@ | ||||
|  require('codemirror/mode/gfm/gfm.js'); | ||||
|  require('codemirror/mode/xml/xml.js'); | ||||
| -var CodeMirrorSpellChecker = require('codemirror-spell-checker'); | ||||
|  var marked = require('marked/lib/marked'); | ||||
|   | ||||
| @@ -1889,18 +1888,7 @@ | ||||
| @@ -1762,9 +1761,4 @@ | ||||
|          options.autosave.uniqueId = options.autosave.unique_id; | ||||
|   | ||||
| -    // If overlay mode is specified and combine is not provided, default it to true | ||||
| -    if (options.overlayMode && options.overlayMode.combine === undefined) { | ||||
| -      options.overlayMode.combine = true; | ||||
| -    } | ||||
| - | ||||
|      // Update this options | ||||
|      this.options = options; | ||||
| @@ -2003,28 +1997,7 @@ | ||||
|      var mode, backdrop; | ||||
|   | ||||
| -    // CodeMirror overlay mode | ||||
| -    if (options.overlayMode) { | ||||
| -      CodeMirror.defineMode('overlay-mode', function(config) { | ||||
| -        return CodeMirror.overlayMode(CodeMirror.getMode(config, options.spellChecker !== false ? 'spell-checker' : 'gfm'), options.overlayMode.mode, options.overlayMode.combine); | ||||
| -      }); | ||||
| - | ||||
| -      mode = 'overlay-mode'; | ||||
| -      backdrop = options.parsingConfig; | ||||
| -      backdrop.gitHubSpice = false; | ||||
| -    } else { | ||||
|          mode = options.parsingConfig; | ||||
|          mode.name = 'gfm'; | ||||
|          mode.gitHubSpice = false; | ||||
| -    } | ||||
| -    if (options.spellChecker !== false) { | ||||
| -        mode = 'spell-checker'; | ||||
| -        backdrop = options.parsingConfig; | ||||
| @@ -37,16 +61,28 @@ diff -NarU2 easymde-orig/src/js/easymde.js easymde-mod1/src/js/easymde.js | ||||
| -        CodeMirrorSpellChecker({ | ||||
| -            codeMirrorInstance: CodeMirror, | ||||
| -        }); | ||||
| -    } else { | ||||
|          mode = options.parsingConfig; | ||||
|          mode.name = 'gfm'; | ||||
|          mode.gitHubSpice = false; | ||||
| -    } | ||||
|   | ||||
|      // eslint-disable-next-line no-unused-vars | ||||
| @@ -1927,5 +1915,4 @@ | ||||
|          configureMouse: configureMouse, | ||||
|          inputStyle: (options.inputStyle != undefined) ? options.inputStyle : isMobile() ? 'contenteditable' : 'textarea', | ||||
| -        spellcheck: (options.nativeSpellcheck != undefined) ? options.nativeSpellcheck : true, | ||||
|      }); | ||||
| diff -NarU2 easy-markdown-editor-2.14.0-orig/types/easymde.d.ts easy-markdown-editor-2.14.0/types/easymde.d.ts | ||||
| --- easy-markdown-editor-2.14.0-orig/types/easymde.d.ts	2021-02-14 12:11:48.000000000 +0000 | ||||
| +++ easy-markdown-editor-2.14.0/types/easymde.d.ts	2021-02-21 20:57:42.492620979 +0000 | ||||
| @@ -160,9 +160,4 @@ | ||||
|      } | ||||
|   | ||||
| -    interface OverlayModeOptions { | ||||
| -      mode: CodeMirror.Mode<any> | ||||
| -      combine?: boolean | ||||
| -    } | ||||
| - | ||||
|      interface Options { | ||||
|          autoDownloadFontAwesome?: boolean; | ||||
| @@ -214,7 +209,5 @@ | ||||
|   | ||||
|          promptTexts?: PromptTexts; | ||||
| -        syncSideBySidePreviewScroll?: boolean; | ||||
| - | ||||
| -        overlayMode?: OverlayModeOptions | ||||
| +        syncSideBySidePreviewScroll?: boolean | ||||
|      } | ||||
|  } | ||||
|   | ||||
| @@ -86,6 +86,8 @@ function have() { | ||||
| 	python -c "import $1; $1; $1.__version__" | ||||
| } | ||||
|  | ||||
| mv copyparty/web/deps/marked.full.js.gz srv/ || true | ||||
|  | ||||
| . buildenv/bin/activate | ||||
| have setuptools | ||||
| have wheel | ||||
|   | ||||
| @@ -29,6 +29,10 @@ gtar=$(command -v gtar || command -v gnutar) || true | ||||
| 	command -v grealpath >/dev/null && | ||||
| 		realpath() { grealpath "$@"; } | ||||
| } | ||||
| pybin=$(command -v python3 || command -v python) || { | ||||
| 	echo need python | ||||
| 	exit 1 | ||||
| } | ||||
|  | ||||
| [ -e copyparty/__main__.py ] || cd .. | ||||
| [ -e copyparty/__main__.py ] || | ||||
| @@ -122,7 +126,7 @@ git describe --tags >/dev/null 2>/dev/null && { | ||||
| 		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" | ||||
| 	sed -ri ' | ||||
| 		s/^(VERSION =)(.*)/#\1\2\n\1 ('"$t_ver"')/; | ||||
| @@ -169,10 +173,11 @@ done | ||||
| 	sed -r '/edit2">edit \(fancy/d' <$f >t && tmv "$f" | ||||
| } | ||||
|  | ||||
| [ $repack ] || | ||||
| find | grep -E '\.py$' | | ||||
|   grep -vE '__version__' | | ||||
|   tr '\n' '\0' | | ||||
|   xargs -0 python ../scripts/uncomment.py | ||||
|   xargs -0 $pybin ../scripts/uncomment.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 | ||||
| @@ -180,7 +185,7 @@ tmv "$f" | ||||
|  | ||||
| # up2k goes from 28k to 22k laff | ||||
| 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 | ||||
| 	tmv "$f" | ||||
| done | ||||
| @@ -206,7 +211,7 @@ echo creating unix sfx | ||||
| ) >$sfx_out.sh | ||||
|  | ||||
| echo creating generic sfx | ||||
| python ../scripts/sfx.py --sfx-make tar.bz2 $ver $ts | ||||
| $pybin ../scripts/sfx.py --sfx-make tar.bz2 $ver $ts | ||||
| mv sfx.out $sfx_out.py | ||||
| chmod 755 $sfx_out.* | ||||
|  | ||||
| @@ -214,5 +219,5 @@ printf "done:\n" | ||||
| printf "  %s\n" "$(realpath $sfx_out)."{sh,py} | ||||
| # rm -rf * | ||||
|  | ||||
| # tar -tvf ../sfx/tar | sed -r 's/(.* ....-..-.. ..:.. )(.*)/\2 `` \1/' | sort | sed -r 's/(.*) `` (.*)/\2 \1/'| less | ||||
| # 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  | ||||
| # apk add bash python3 tar xz bzip2 | ||||
| # 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 | ||||
| } | ||||
|  | ||||
| mv copyparty/web/deps/marked.full.js.gz srv/ || true | ||||
|  | ||||
| mkdir -p dist | ||||
| zip_path="$(pwd)/dist/copyparty-$ver.zip" | ||||
| tgz_path="$(pwd)/dist/copyparty-$ver.tar.gz" | ||||
|   | ||||
| @@ -1,9 +1,8 @@ | ||||
| #!/usr/bin/env python | ||||
| # coding: utf-8 | ||||
| # coding: latin-1 | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import os, sys, time, shutil, signal, tarfile, hashlib, platform, tempfile | ||||
| import subprocess as sp | ||||
| import os, sys, time, shutil, runpy, tarfile, hashlib, platform, tempfile, traceback | ||||
|  | ||||
| """ | ||||
| run me with any version of python, i will unpack and run copyparty | ||||
| @@ -344,20 +343,24 @@ def get_payload(): | ||||
|                 break | ||||
|  | ||||
|  | ||||
| def confirm(): | ||||
| def confirm(rv): | ||||
|     msg() | ||||
|     msg(traceback.format_exc()) | ||||
|     msg("*** hit enter to exit ***") | ||||
|     try: | ||||
|         raw_input() if PY2 else input() | ||||
|     except: | ||||
|         pass | ||||
|  | ||||
|     sys.exit(rv) | ||||
|  | ||||
|  | ||||
| def run(tmp, j2ver): | ||||
|     global cpp | ||||
|  | ||||
|     msg("jinja2:", j2ver or "bundled") | ||||
|     msg("sfxdir:", tmp) | ||||
|     msg() | ||||
|  | ||||
|     # "systemd-tmpfiles-clean.timer"?? HOW do you even come up with this shit | ||||
|     try: | ||||
| @@ -373,30 +376,16 @@ def run(tmp, j2ver): | ||||
|     if j2ver: | ||||
|         del ld[-1] | ||||
|  | ||||
|     cmd = ( | ||||
|         "import sys, runpy; " | ||||
|         + "".join(['sys.path.insert(0, r"' + x + '"); ' for x in ld]) | ||||
|         + 'runpy.run_module("copyparty", run_name="__main__")' | ||||
|     ) | ||||
|     cmd = [sys.executable, "-c", cmd] + list(sys.argv[1:]) | ||||
|     for x in ld: | ||||
|         sys.path.insert(0, x) | ||||
|  | ||||
|     cmd = [str(x) for x in cmd] | ||||
|     msg("\n", cmd, "\n") | ||||
|     cpp = sp.Popen(cmd) | ||||
|     try: | ||||
|         cpp.wait() | ||||
|         runpy.run_module(str("copyparty"), run_name=str("__main__")) | ||||
|     except SystemExit as ex: | ||||
|         if ex.code: | ||||
|             confirm(ex.code) | ||||
|     except: | ||||
|         cpp.wait() | ||||
|  | ||||
|     if cpp.returncode != 0: | ||||
|         confirm() | ||||
|  | ||||
|     sys.exit(cpp.returncode) | ||||
|  | ||||
|  | ||||
| def bye(sig, frame): | ||||
|     if cpp is not None: | ||||
|         cpp.terminate() | ||||
|         confirm(1) | ||||
|  | ||||
|  | ||||
| def main(): | ||||
| @@ -430,8 +419,6 @@ def main(): | ||||
|  | ||||
|     # skip 0 | ||||
|  | ||||
|     signal.signal(signal.SIGTERM, bye) | ||||
|  | ||||
|     tmp = unpack() | ||||
|  | ||||
|     try: | ||||
| @@ -439,7 +426,7 @@ def main(): | ||||
|     except: | ||||
|         j2ver = None | ||||
|  | ||||
|     return run(tmp, j2ver) | ||||
|     run(tmp, j2ver) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
							
								
								
									
										5
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										5
									
								
								setup.py
									
									
									
									
									
								
							| @@ -2,10 +2,8 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import print_function | ||||
|  | ||||
| import io | ||||
| import os | ||||
| import sys | ||||
| from glob import glob | ||||
| from shutil import rmtree | ||||
|  | ||||
| setuptools_available = True | ||||
| @@ -49,7 +47,7 @@ with open(here + "/README.md", "rb") as f: | ||||
| about = {} | ||||
| if not VERSION: | ||||
|     with open(os.path.join(here, NAME, "__version__.py"), "rb") as f: | ||||
|         exec(f.read().decode("utf-8").split("\n\n", 1)[1], about) | ||||
|         exec (f.read().decode("utf-8").split("\n\n", 1)[1], about) | ||||
| else: | ||||
|     about["__version__"] = VERSION | ||||
|  | ||||
| @@ -116,6 +114,7 @@ args = { | ||||
|         "Programming Language :: Python :: 3.6", | ||||
|         "Programming Language :: Python :: 3.7", | ||||
|         "Programming Language :: Python :: 3.8", | ||||
|         "Programming Language :: Python :: 3.9", | ||||
|         "Programming Language :: Python :: Implementation :: CPython", | ||||
|         "Programming Language :: Python :: Implementation :: PyPy", | ||||
|         "Environment :: Console", | ||||
|   | ||||
| @@ -16,6 +16,12 @@ from copyparty.authsrv import AuthSrv | ||||
| from copyparty import util | ||||
|  | ||||
|  | ||||
| class Cfg(Namespace): | ||||
|     def __init__(self, a=[], v=[], c=None): | ||||
|         ex = {k: False for k in "e2d e2ds e2dsa e2t e2ts e2tsr mte".split()} | ||||
|         super(Cfg, self).__init__(a=a, v=v, c=c, **ex) | ||||
|  | ||||
|  | ||||
| class TestVFS(unittest.TestCase): | ||||
|     def dump(self, vfs): | ||||
|         print(json.dumps(vfs, indent=4, sort_keys=True, default=lambda o: o.__dict__)) | ||||
| @@ -35,7 +41,13 @@ class TestVFS(unittest.TestCase): | ||||
|     def ls(self, vfs, vpath, uname): | ||||
|         """helper for resolving and listing a folder""" | ||||
|         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) | ||||
|  | ||||
|         fsdir, real, virt = r1 | ||||
|         real = [x[0] for x in real] | ||||
|         return fsdir, real, virt | ||||
|  | ||||
|     def runcmd(self, *argv): | ||||
|         p = sp.Popen(argv, stdout=sp.PIPE, stderr=sp.PIPE) | ||||
| @@ -78,7 +90,7 @@ class TestVFS(unittest.TestCase): | ||||
|         finally: | ||||
|             return ret | ||||
|  | ||||
|     def log(self, src, msg): | ||||
|     def log(self, src, msg, c=0): | ||||
|         pass | ||||
|  | ||||
|     def test(self): | ||||
| @@ -102,7 +114,7 @@ class TestVFS(unittest.TestCase): | ||||
|                             f.write(fn) | ||||
|  | ||||
|         # defaults | ||||
|         vfs = AuthSrv(Namespace(c=None, a=[], v=[]), self.log).vfs | ||||
|         vfs = AuthSrv(Cfg(), self.log).vfs | ||||
|         self.assertEqual(vfs.nodes, {}) | ||||
|         self.assertEqual(vfs.vpath, "") | ||||
|         self.assertEqual(vfs.realpath, td) | ||||
| @@ -110,7 +122,7 @@ class TestVFS(unittest.TestCase): | ||||
|         self.assertEqual(vfs.uwrite, ["*"]) | ||||
|  | ||||
|         # 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.vpath, "") | ||||
|         self.assertEqual(vfs.realpath, os.path.join(td, "a", "ab")) | ||||
| @@ -118,9 +130,7 @@ class TestVFS(unittest.TestCase): | ||||
|         self.assertEqual(vfs.uwrite, []) | ||||
|  | ||||
|         # single read-only rootfs (absolute path) | ||||
|         vfs = AuthSrv( | ||||
|             Namespace(c=None, a=[], v=[td + "//a/ac/../aa//::r"]), self.log | ||||
|         ).vfs | ||||
|         vfs = AuthSrv(Cfg(v=[td + "//a/ac/../aa//::r"]), self.log).vfs | ||||
|         self.assertEqual(vfs.nodes, {}) | ||||
|         self.assertEqual(vfs.vpath, "") | ||||
|         self.assertEqual(vfs.realpath, os.path.join(td, "a", "aa")) | ||||
| @@ -129,7 +139,7 @@ class TestVFS(unittest.TestCase): | ||||
|  | ||||
|         # read-only rootfs with write-only subdirectory (read-write for k) | ||||
|         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, | ||||
|         ).vfs | ||||
|         self.assertEqual(len(vfs.nodes), 1) | ||||
| @@ -192,7 +202,10 @@ class TestVFS(unittest.TestCase): | ||||
|         self.assertEqual(list(virt), []) | ||||
|  | ||||
|         # 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(vfs.vpath, "") | ||||
|         self.assertEqual(vfs.realpath, td) | ||||
| @@ -211,9 +224,7 @@ class TestVFS(unittest.TestCase): | ||||
|  | ||||
|         # breadth-first construction | ||||
|         vfs = AuthSrv( | ||||
|             Namespace( | ||||
|                 c=None, | ||||
|                 a=[], | ||||
|             Cfg( | ||||
|                 v=[ | ||||
|                     "a/ac/acb:a/ac/acb:w", | ||||
|                     "a:a:w", | ||||
| @@ -234,7 +245,7 @@ class TestVFS(unittest.TestCase): | ||||
|         self.undot(vfs, "./.././foo/..", "") | ||||
|  | ||||
|         # 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, "", "*") | ||||
|         self.assertEqual(fsp, td) | ||||
| @@ -271,7 +282,7 @@ class TestVFS(unittest.TestCase): | ||||
|                 ).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["asd"], "fgh:jkl") | ||||
|         n = au.vfs | ||||
|   | ||||
		Reference in New Issue
	
	Block a user