mirror of
				https://github.com/9001/copyparty.git
				synced 2025-11-04 13:53:18 +00:00 
			
		
		
		
	Compare commits
	
		
			192 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					3a38dcbc05 | ||
| 
						 | 
					7ff08bce57 | ||
| 
						 | 
					fd490af434 | ||
| 
						 | 
					1195b8f17e | ||
| 
						 | 
					28dce13776 | ||
| 
						 | 
					431f20177a | ||
| 
						 | 
					87aff54d9d | ||
| 
						 | 
					f50462de82 | ||
| 
						 | 
					9bda8c7eb6 | ||
| 
						 | 
					e83c63d239 | ||
| 
						 | 
					b38533b0cc | ||
| 
						 | 
					5ccca3fbd5 | ||
| 
						 | 
					9e850fc3ab | ||
| 
						 | 
					ffbfcd7e00 | ||
| 
						 | 
					5ea7590748 | ||
| 
						 | 
					290c3bc2bb | ||
| 
						 | 
					b12131e91c | ||
| 
						 | 
					3b354447b0 | ||
| 
						 | 
					d09ec6feaa | ||
| 
						 | 
					21405c3fda | ||
| 
						 | 
					13e5c96cab | ||
| 
						 | 
					426687b75e | ||
| 
						 | 
					c8f59fb978 | ||
| 
						 | 
					871dde79a9 | ||
| 
						 | 
					e14d81bc6f | ||
| 
						 | 
					514d046d1f | ||
| 
						 | 
					4ed9528d36 | ||
| 
						 | 
					625560e642 | ||
| 
						 | 
					73ebd917d1 | ||
| 
						 | 
					cd3e0afad2 | ||
| 
						 | 
					d8d1f94a86 | ||
| 
						 | 
					00dfd8cfd1 | ||
| 
						 | 
					273de6db31 | ||
| 
						 | 
					c6c0eeb0ff | ||
| 
						 | 
					e70c74a3b5 | ||
| 
						 | 
					f7d939eeab | ||
| 
						 | 
					e815c091b9 | ||
| 
						 | 
					963529b7cf | ||
| 
						 | 
					638a52374d | ||
| 
						 | 
					d9d42b7aa2 | ||
| 
						 | 
					ec7e5f36a2 | ||
| 
						 | 
					56110883ea | ||
| 
						 | 
					7f8d7d6006 | ||
| 
						 | 
					49e4fb7e12 | ||
| 
						 | 
					8dbbea473f | ||
| 
						 | 
					3d375d5114 | ||
| 
						 | 
					f3eae67d97 | ||
| 
						 | 
					40c1b19235 | ||
| 
						 | 
					ccaf0ab159 | ||
| 
						 | 
					d07f147423 | ||
| 
						 | 
					f5cb9f92b9 | ||
| 
						 | 
					f991f74983 | ||
| 
						 | 
					6b3295059e | ||
| 
						 | 
					b18a07ae6b | ||
| 
						 | 
					8ab03dabda | ||
| 
						 | 
					5e760e35dc | ||
| 
						 | 
					afbfa04514 | ||
| 
						 | 
					7aace470c5 | ||
| 
						 | 
					b4acb24f6a | ||
| 
						 | 
					bcee8a4934 | ||
| 
						 | 
					36b0718542 | ||
| 
						 | 
					9a92bca45d | ||
| 
						 | 
					b07445a363 | ||
| 
						 | 
					a62ec0c27e | ||
| 
						 | 
					57e3a2d382 | ||
| 
						 | 
					b61022b374 | ||
| 
						 | 
					a3e2b2ec87 | ||
| 
						 | 
					a83d3f8801 | ||
| 
						 | 
					90c5f2b9d2 | ||
| 
						 | 
					4885653c07 | ||
| 
						 | 
					21e1cd87ca | ||
| 
						 | 
					81f82e8e9f | ||
| 
						 | 
					c0e31851da | ||
| 
						 | 
					6599c3eced | ||
| 
						 | 
					5d6c61a861 | ||
| 
						 | 
					1a5c66edd3 | ||
| 
						 | 
					deae9fe95a | ||
| 
						 | 
					abd65c6334 | ||
| 
						 | 
					8137a99904 | ||
| 
						 | 
					6f6f9c1f74 | ||
| 
						 | 
					7b575f716f | ||
| 
						 | 
					6ba6ea3572 | ||
| 
						 | 
					9a22ad5ea3 | ||
| 
						 | 
					beaab9778e | ||
| 
						 | 
					f327bdb6b4 | ||
| 
						 | 
					ae180e0f5f | ||
| 
						 | 
					e3f1d19756 | ||
| 
						 | 
					93c2bd6ef6 | ||
| 
						 | 
					4d0e5ff6db | ||
| 
						 | 
					0893f06919 | ||
| 
						 | 
					46b6abde3f | ||
| 
						 | 
					0696610dee | ||
| 
						 | 
					edf0d3684c | ||
| 
						 | 
					7af159f5f6 | ||
| 
						 | 
					7f2cb6764a | ||
| 
						 | 
					96495a9bf1 | ||
| 
						 | 
					b2fafec5fc | ||
| 
						 | 
					0850b8ae2b | ||
| 
						 | 
					8a68a96c57 | ||
| 
						 | 
					d3aae8ed6a | ||
| 
						 | 
					c62ebadda8 | ||
| 
						 | 
					ffcee6d390 | ||
| 
						 | 
					de32838346 | ||
| 
						 | 
					b9a4e47ea2 | ||
| 
						 | 
					57d994422d | ||
| 
						 | 
					6ecd745323 | ||
| 
						 | 
					bd769f5bdb | ||
| 
						 | 
					2381692aba | ||
| 
						 | 
					24fdada0a0 | ||
| 
						 | 
					bb5169710a | ||
| 
						 | 
					9cde2352f3 | ||
| 
						 | 
					482dd7a938 | ||
| 
						 | 
					bddcc69438 | ||
| 
						 | 
					19d4540630 | ||
| 
						 | 
					4f5f6c81f5 | ||
| 
						 | 
					7e4c1238ba | ||
| 
						 | 
					f7196ac773 | ||
| 
						 | 
					7a7c832000 | ||
| 
						 | 
					2b4ccdbebb | ||
| 
						 | 
					0d16b49489 | ||
| 
						 | 
					768405b691 | ||
| 
						 | 
					da01413b7b | ||
| 
						 | 
					914e22c53e | ||
| 
						 | 
					43a23bf733 | ||
| 
						 | 
					92bb00c6d2 | ||
| 
						 | 
					b0b97a2648 | ||
| 
						 | 
					2c452fe323 | ||
| 
						 | 
					ad73d0c77d | ||
| 
						 | 
					7f9bf1c78c | ||
| 
						 | 
					61a6bc3a65 | ||
| 
						 | 
					46e10b0e9f | ||
| 
						 | 
					8441206e26 | ||
| 
						 | 
					9fdc5ee748 | ||
| 
						 | 
					00ff133387 | ||
| 
						 | 
					96164cb934 | ||
| 
						 | 
					82fb21ae69 | ||
| 
						 | 
					89d4a2b4c4 | ||
| 
						 | 
					fc0c7ff374 | ||
| 
						 | 
					5148c4f2e9 | ||
| 
						 | 
					c3b59f7bcf | ||
| 
						 | 
					61e148202b | ||
| 
						 | 
					8a4e0739bc | ||
| 
						 | 
					f75c5f2fe5 | ||
| 
						 | 
					81d5859588 | ||
| 
						 | 
					721886bb7a | ||
| 
						 | 
					b23c272820 | ||
| 
						 | 
					cd02bfea7a | ||
| 
						 | 
					6774bd88f9 | ||
| 
						 | 
					1046a4f376 | ||
| 
						 | 
					8081f9ddfd | ||
| 
						 | 
					fa656577d1 | ||
| 
						 | 
					b14b86990f | ||
| 
						 | 
					2a6dd7b512 | ||
| 
						 | 
					feebdee88b | ||
| 
						 | 
					99d9277f5d | ||
| 
						 | 
					9af64d6156 | ||
| 
						 | 
					5e3775c1af | ||
| 
						 | 
					2d2e8a3da7 | ||
| 
						 | 
					b2a560b76f | ||
| 
						 | 
					39397a489d | ||
| 
						 | 
					ff593a0904 | ||
| 
						 | 
					f12789cf44 | ||
| 
						 | 
					4f8cf2fc87 | ||
| 
						 | 
					fda98730ac | ||
| 
						 | 
					06c6ddffb6 | ||
| 
						 | 
					d29f0c066c | ||
| 
						 | 
					c9e4de3346 | ||
| 
						 | 
					ca0b97f72d | ||
| 
						 | 
					b38f20b408 | ||
| 
						 | 
					05b1dbaf56 | ||
| 
						 | 
					b8481e32ba | ||
| 
						 | 
					9c03c65e07 | ||
| 
						 | 
					d8ed006b9b | ||
| 
						 | 
					63c0623a5e | ||
| 
						 | 
					fd84506db0 | ||
| 
						 | 
					d8bcb44e44 | ||
| 
						 | 
					56a26b0916 | ||
| 
						 | 
					efcf1d6b90 | ||
| 
						 | 
					9f578bfec6 | ||
| 
						 | 
					1f170d7d28 | ||
| 
						 | 
					5ae14cf9be | ||
| 
						 | 
					aaf9d53be9 | ||
| 
						 | 
					75c73f7ba7 | ||
| 
						 | 
					b6dba8beee | ||
| 
						 | 
					94521cdc1a | ||
| 
						 | 
					3365b1c355 | ||
| 
						 | 
					6c957c4923 | ||
| 
						 | 
					833997f04c | ||
| 
						 | 
					68d51e4037 | ||
| 
						 | 
					ce274d2011 | ||
| 
						 | 
					280778ed43 | ||
| 
						 | 
					0f558ecbbf | 
							
								
								
									
										15
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							@@ -12,14 +12,23 @@
 | 
				
			|||||||
                //"-nw",
 | 
					                //"-nw",
 | 
				
			||||||
                "-ed",
 | 
					                "-ed",
 | 
				
			||||||
                "-emp",
 | 
					                "-emp",
 | 
				
			||||||
                "-e2d",
 | 
					                "-e2dsa",
 | 
				
			||||||
                "-e2s",
 | 
					                "-e2ts",
 | 
				
			||||||
                "-a",
 | 
					                "-a",
 | 
				
			||||||
                "ed:wark",
 | 
					                "ed:wark",
 | 
				
			||||||
                "-v",
 | 
					                "-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",
 | 
					            "name": "Run active unit test",
 | 
				
			||||||
            "type": "python",
 | 
					            "type": "python",
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										12
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							@@ -50,11 +50,9 @@
 | 
				
			|||||||
    "files.associations": {
 | 
					    "files.associations": {
 | 
				
			||||||
        "*.makefile": "makefile"
 | 
					        "*.makefile": "makefile"
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "editor.codeActionsOnSaveTimeout": 9001,
 | 
					    "python.formatting.blackArgs": [
 | 
				
			||||||
    "editor.formatOnSaveTimeout": 9001,
 | 
					        "-t",
 | 
				
			||||||
    //
 | 
					        "py27"
 | 
				
			||||||
    //  things you may wanna edit:
 | 
					    ],
 | 
				
			||||||
    //
 | 
					    "python.linting.enabled": true,
 | 
				
			||||||
    "python.pythonPath": "/usr/bin/python3",
 | 
					 | 
				
			||||||
    //"python.linting.enabled": true,
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
							
								
								
									
										7
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							@@ -5,6 +5,13 @@
 | 
				
			|||||||
            "label": "pre",
 | 
					            "label": "pre",
 | 
				
			||||||
            "command": "true;rm -rf inc/* inc/.hist/;mkdir -p inc;",
 | 
					            "command": "true;rm -rf inc/* inc/.hist/;mkdir -p inc;",
 | 
				
			||||||
            "type": "shell"
 | 
					            "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
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    ]
 | 
					    ]
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
							
								
								
									
										206
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										206
									
								
								README.md
									
									
									
									
									
								
							@@ -8,11 +8,36 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
turn your phone or raspi into a portable file server with resumable uploads/downloads using IE6 or any other browser
 | 
					turn your phone or raspi into a portable file server with resumable uploads/downloads using IE6 or any other browser
 | 
				
			||||||
 | 
					
 | 
				
			||||||
* server runs on anything with `py2.7` or `py3.2+`
 | 
					* server runs on anything with `py2.7` or `py3.3+`
 | 
				
			||||||
* *resumable* uploads need `firefox 12+` / `chrome 6+` / `safari 6+` / `IE 10+`
 | 
					* *resumable* uploads need `firefox 12+` / `chrome 6+` / `safari 6+` / `IE 10+`
 | 
				
			||||||
* code standard: `black`
 | 
					* code standard: `black`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## readme toc
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* top
 | 
				
			||||||
 | 
					    * [quickstart](#quickstart)
 | 
				
			||||||
 | 
					    * [notes](#notes)
 | 
				
			||||||
 | 
					    * [status](#status)
 | 
				
			||||||
 | 
					* [bugs](#bugs)
 | 
				
			||||||
 | 
					* [usage](#usage)
 | 
				
			||||||
 | 
					    * [zip downloads](#zip-downloads)
 | 
				
			||||||
 | 
					* [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
 | 
					## quickstart
 | 
				
			||||||
 | 
					
 | 
				
			||||||
download [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) and you're all set!
 | 
					download [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) and you're all set!
 | 
				
			||||||
@@ -36,49 +61,189 @@ you may also want these, especially on servers:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
## status
 | 
					## status
 | 
				
			||||||
 | 
					
 | 
				
			||||||
* [x] sanic multipart parser
 | 
					* backend stuff
 | 
				
			||||||
* [x] load balancer (multiprocessing)
 | 
					  * ☑ sanic multipart parser
 | 
				
			||||||
* [x] upload (plain multipart, ie6 support)
 | 
					  * ☑ load balancer (multiprocessing)
 | 
				
			||||||
* [x] upload (js, resumable, multithreaded)
 | 
					  * ☑ volumes (mountpoints)
 | 
				
			||||||
* [x] download
 | 
					  * ☑ accounts
 | 
				
			||||||
* [x] browser
 | 
					* upload
 | 
				
			||||||
* [x] media player
 | 
					  * ☑ basic: plain multipart, ie6 support
 | 
				
			||||||
* [ ] thumbnails
 | 
					  * ☑ up2k: js, resumable, multithreaded
 | 
				
			||||||
* [ ] download as zip
 | 
					  * ☑ stash: simple PUT filedropper
 | 
				
			||||||
* [x] volumes
 | 
					  * ☑ symlink/discard existing files (content-matching)
 | 
				
			||||||
* [x] accounts
 | 
					* download
 | 
				
			||||||
* [x] markdown viewer
 | 
					  * ☑ single files in browser
 | 
				
			||||||
* [x] markdown editor
 | 
					  * ☑ folders as zip / tar files
 | 
				
			||||||
* [x] FUSE client (read-only)
 | 
					  * ☑ FUSE client (read-only)
 | 
				
			||||||
 | 
					* browser
 | 
				
			||||||
 | 
					  * ☑ tree-view
 | 
				
			||||||
 | 
					  * ☑ media player
 | 
				
			||||||
 | 
					  * ✖ thumbnails
 | 
				
			||||||
 | 
					  * ✖ SPA (browse while uploading)
 | 
				
			||||||
 | 
					    * currently safe using the file-tree on the left only, not folders in the file list
 | 
				
			||||||
 | 
					* server indexing
 | 
				
			||||||
 | 
					  * ☑ locate files by contents
 | 
				
			||||||
 | 
					  * ☑ search by name/path/date/size
 | 
				
			||||||
 | 
					  * ☑ search by ID3-tags etc.
 | 
				
			||||||
 | 
					* markdown
 | 
				
			||||||
 | 
					  * ☑ viewer
 | 
				
			||||||
 | 
					  * ☑ editor (sure why not)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
summary: it works! you can use it! (but technically not even close to beta)
 | 
					summary: it works! you can use it! (but technically not even close to beta)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# bugs
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* Windows: python 3.7 and older cannot read tags with ffprobe, so use mutagen or upgrade
 | 
				
			||||||
 | 
					* Windows: python 2.7 cannot index non-ascii filenames with `-e2d`
 | 
				
			||||||
 | 
					* Windows: python 2.7 cannot handle filenames with mojibake
 | 
				
			||||||
 | 
					* hiding the contents at url `/d1/d2/d3` using `-v :d1/d2/d3:cd2d` has the side-effect of creating databases (for files/tags) inside folders d1 and d2, and those databases take precedence over the main db at the top of the vfs - this means all files in d2 and below will be reindexed unless you already had a vfs entry at or below d2
 | 
				
			||||||
 | 
					* probably more, pls let me know
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# 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
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## zip downloads
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					the `zip` link next to folders can produce various types of zip/tar files using these alternatives in the browser settings tab:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					| name | url-suffix | description |
 | 
				
			||||||
 | 
					|--|--|--|
 | 
				
			||||||
 | 
					| `tar` | `?tar` | plain gnutar, works great with `curl \| tar -xv` |
 | 
				
			||||||
 | 
					| `zip` | `?zip=utf8` | works everywhere, glitchy filenames on win7 and older |
 | 
				
			||||||
 | 
					| `zip_dos` | `?zip` | traditional cp437 (no unicode) to fix glitchy filenames |
 | 
				
			||||||
 | 
					| `zip_crc` | `?zip=crc` | cp437 with crc32 computed early for truly ancient software |
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* hidden files (dotfiles) are excluded unless `-ed`
 | 
				
			||||||
 | 
					  * the up2k.db is always excluded
 | 
				
			||||||
 | 
					* `zip_crc` will take longer to download since the server has to read each file twice
 | 
				
			||||||
 | 
					  * please let me know if you find a program old enough to actually need this
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# 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
 | 
					# client examples
 | 
				
			||||||
 | 
					
 | 
				
			||||||
* javascript: dump some state into a file (two separate examples)
 | 
					* javascript: dump some state into a file (two separate examples)
 | 
				
			||||||
  * `await fetch('https://127.0.0.1:3923/', {method:"PUT", body: JSON.stringify(foo)});`
 | 
					  * `await fetch('https://127.0.0.1:3923/', {method:"PUT", body: JSON.stringify(foo)});`
 | 
				
			||||||
  * `var xhr = new XMLHttpRequest(); xhr.open('POST', 'https://127.0.0.1:3923/msgs?raw'); xhr.send('foo');`
 | 
					  * `var xhr = new XMLHttpRequest(); xhr.open('POST', 'https://127.0.0.1:3923/msgs?raw'); xhr.send('foo');`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* curl/wget: upload some files (post=file, chunk=stdin)
 | 
				
			||||||
 | 
					  * `post(){ curl -b cppwd=wark http://127.0.0.1:3923/ -F act=bput -F f=@"$1";}`  
 | 
				
			||||||
 | 
					    `post movie.mkv`
 | 
				
			||||||
 | 
					  * `post(){ wget --header='Cookie: cppwd=wark' http://127.0.0.1:3923/?raw --post-file="$1" -O-;}`  
 | 
				
			||||||
 | 
					    `post movie.mkv`
 | 
				
			||||||
 | 
					  * `chunk(){ curl -b cppwd=wark http://127.0.0.1:3923/ -T-;}`  
 | 
				
			||||||
 | 
					    `chunk <movie.mkv`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
* FUSE: mount a copyparty server as a local filesystem
 | 
					* FUSE: mount a copyparty server as a local filesystem
 | 
				
			||||||
  * cross-platform python client available in [./bin/](bin/)
 | 
					  * cross-platform python client available in [./bin/](bin/)
 | 
				
			||||||
  * [rclone](https://rclone.org/) as client can give ~5x performance, see [./docs/rclone.md](docs/rclone.md)
 | 
					  * [rclone](https://rclone.org/) as client can give ~5x performance, see [./docs/rclone.md](docs/rclone.md)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					copyparty returns a truncated sha512sum of your PUT/POST as base64; you can generate the same checksum locally to verify uplaods:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    b512(){ printf "$((sha512sum||shasum -a512)|sed -E 's/ .*//;s/(..)/\\x\1/g')"|base64|head -c43;}
 | 
				
			||||||
 | 
					    b512 <movie.mkv
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# dependencies
 | 
					# dependencies
 | 
				
			||||||
 | 
					
 | 
				
			||||||
* `jinja2`
 | 
					* `jinja2` (is built into the SFX)
 | 
				
			||||||
  * pulls in `markupsafe` as of v2.7; use jinja 2.6 on py3.2
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
optional, enables thumbnails:
 | 
					**optional,** enables music tags:
 | 
				
			||||||
 | 
					* either `mutagen` (fast, pure-python, skips a few tags, makes copyparty GPL? idk)
 | 
				
			||||||
 | 
					* or `FFprobe` (20x slower, more accurate, possibly dangerous depending on your distro and users)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					**optional,** will eventually enable thumbnails:
 | 
				
			||||||
* `Pillow` (requires py2.7 or py3.5+)
 | 
					* `Pillow` (requires py2.7 or py3.5+)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## optional gpl stuff
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					some bundled tools have copyleft dependencies, see [./bin/#mtag](bin/#mtag)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					these are standalone and will never be imported / evaluated by copyparty
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# sfx
 | 
					# sfx
 | 
				
			||||||
 | 
					
 | 
				
			||||||
currently there are two self-contained binaries:
 | 
					currently there are two self-contained binaries:
 | 
				
			||||||
* `copyparty-sfx.sh` for unix (linux and osx) -- smaller, more robust
 | 
					* [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) -- pure python, works everywhere
 | 
				
			||||||
* `copyparty-sfx.py` for windows (unix too) -- crossplatform, beta
 | 
					* [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
 | 
					launch either of them (**use sfx.py on systemd**) and it'll unpack and run copyparty, assuming you have python installed of course
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -128,6 +293,7 @@ pip install black bandit pylint flake8  # vscode tooling
 | 
				
			|||||||
in the `scripts` folder:
 | 
					in the `scripts` folder:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
* run `make -C deps-docker` to build all dependencies
 | 
					* run `make -C deps-docker` to build all dependencies
 | 
				
			||||||
 | 
					* `git tag v1.2.3 && git push origin --tags`
 | 
				
			||||||
* create github release with `make-tgz-release.sh`
 | 
					* create github release with `make-tgz-release.sh`
 | 
				
			||||||
* upload to pypi with `make-pypi-release.(sh|bat)`
 | 
					* upload to pypi with `make-pypi-release.(sh|bat)`
 | 
				
			||||||
* create sfx with `make-sfx.sh`
 | 
					* create sfx with `make-sfx.sh`
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,4 +1,4 @@
 | 
				
			|||||||
# copyparty-fuse.py
 | 
					# [`copyparty-fuse.py`](copyparty-fuse.py)
 | 
				
			||||||
* mount a copyparty server as a local filesystem (read-only)
 | 
					* mount a copyparty server as a local filesystem (read-only)
 | 
				
			||||||
* **supports Windows!** -- expect `194 MiB/s` sequential read
 | 
					* **supports Windows!** -- expect `194 MiB/s` sequential read
 | 
				
			||||||
* **supports Linux** -- expect `117 MiB/s` sequential read
 | 
					* **supports Linux** -- expect `117 MiB/s` sequential read
 | 
				
			||||||
@@ -29,7 +29,7 @@ you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releas
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# copyparty-fuse🅱️.py
 | 
					# [`copyparty-fuse🅱️.py`](copyparty-fuseb.py)
 | 
				
			||||||
* mount a copyparty server as a local filesystem (read-only)
 | 
					* mount a copyparty server as a local filesystem (read-only)
 | 
				
			||||||
* does the same thing except more correct, `samba` approves
 | 
					* does the same thing except more correct, `samba` approves
 | 
				
			||||||
* **supports Linux** -- expect `18 MiB/s` (wait what)
 | 
					* **supports Linux** -- expect `18 MiB/s` (wait what)
 | 
				
			||||||
@@ -37,5 +37,11 @@ you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releas
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# copyparty-fuse-streaming.py
 | 
					# [`copyparty-fuse-streaming.py`](copyparty-fuse-streaming.py)
 | 
				
			||||||
* pretend this doesn't exist
 | 
					* pretend this doesn't exist
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# [`mtag/`](mtag/)
 | 
				
			||||||
 | 
					* standalone programs which perform misc. file analysis
 | 
				
			||||||
 | 
					* copyparty can Popen programs like these during file indexing to collect additional metadata
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1067,7 +1067,7 @@ def main():
 | 
				
			|||||||
        dbg = null_log
 | 
					        dbg = null_log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if WINDOWS:
 | 
					    if WINDOWS:
 | 
				
			||||||
        os.system("")
 | 
					        os.system("rem")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        for ch in '<>:"\\|?*':
 | 
					        for ch in '<>:"\\|?*':
 | 
				
			||||||
            # microsoft maps illegal characters to f0xx
 | 
					            # microsoft maps illegal characters to f0xx
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -33,6 +33,7 @@ import re
 | 
				
			|||||||
import os
 | 
					import os
 | 
				
			||||||
import sys
 | 
					import sys
 | 
				
			||||||
import time
 | 
					import time
 | 
				
			||||||
 | 
					import json
 | 
				
			||||||
import stat
 | 
					import stat
 | 
				
			||||||
import errno
 | 
					import errno
 | 
				
			||||||
import struct
 | 
					import struct
 | 
				
			||||||
@@ -323,7 +324,7 @@ class Gateway(object):
 | 
				
			|||||||
        if bad_good:
 | 
					        if bad_good:
 | 
				
			||||||
            path = dewin(path)
 | 
					            path = dewin(path)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots"
 | 
					        web_path = self.quotep("/" + "/".join([self.web_root, path])) + "?dots&ls"
 | 
				
			||||||
        r = self.sendreq("GET", web_path)
 | 
					        r = self.sendreq("GET", web_path)
 | 
				
			||||||
        if r.status != 200:
 | 
					        if r.status != 200:
 | 
				
			||||||
            self.closeconn()
 | 
					            self.closeconn()
 | 
				
			||||||
@@ -334,12 +335,17 @@ class Gateway(object):
 | 
				
			|||||||
            )
 | 
					            )
 | 
				
			||||||
            raise FuseOSError(errno.ENOENT)
 | 
					            raise FuseOSError(errno.ENOENT)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if not r.getheader("Content-Type", "").startswith("text/html"):
 | 
					        ctype = r.getheader("Content-Type", "")
 | 
				
			||||||
 | 
					        if ctype == "application/json":
 | 
				
			||||||
 | 
					            parser = self.parse_jls
 | 
				
			||||||
 | 
					        elif ctype.startswith("text/html"):
 | 
				
			||||||
 | 
					            parser = self.parse_html
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
            log("listdir on file: {}".format(path))
 | 
					            log("listdir on file: {}".format(path))
 | 
				
			||||||
            raise FuseOSError(errno.ENOENT)
 | 
					            raise FuseOSError(errno.ENOENT)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            return self.parse_html(r)
 | 
					            return parser(r)
 | 
				
			||||||
        except:
 | 
					        except:
 | 
				
			||||||
            info(repr(path) + "\n" + traceback.format_exc())
 | 
					            info(repr(path) + "\n" + traceback.format_exc())
 | 
				
			||||||
            raise
 | 
					            raise
 | 
				
			||||||
@@ -367,6 +373,29 @@ class Gateway(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        return r.read()
 | 
					        return r.read()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def parse_jls(self, datasrc):
 | 
				
			||||||
 | 
					        rsp = b""
 | 
				
			||||||
 | 
					        while True:
 | 
				
			||||||
 | 
					            buf = datasrc.read(1024 * 32)
 | 
				
			||||||
 | 
					            if not buf:
 | 
				
			||||||
 | 
					                break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            rsp += buf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        rsp = json.loads(rsp.decode("utf-8"))
 | 
				
			||||||
 | 
					        ret = []
 | 
				
			||||||
 | 
					        for is_dir, nodes in [[True, rsp["dirs"]], [False, rsp["files"]]]:
 | 
				
			||||||
 | 
					            for n in nodes:
 | 
				
			||||||
 | 
					                fname = unquote(n["href"]).rstrip(b"/")
 | 
				
			||||||
 | 
					                fname = fname.decode("wtf-8")
 | 
				
			||||||
 | 
					                if bad_good:
 | 
				
			||||||
 | 
					                    fname = enwin(fname)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                fun = self.stat_dir if is_dir else self.stat_file
 | 
				
			||||||
 | 
					                ret.append([fname, fun(n["ts"], n["sz"]), 0])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return ret
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def parse_html(self, datasrc):
 | 
					    def parse_html(self, datasrc):
 | 
				
			||||||
        ret = []
 | 
					        ret = []
 | 
				
			||||||
        remainder = b""
 | 
					        remainder = b""
 | 
				
			||||||
@@ -818,7 +847,7 @@ class CPPF(Operations):
 | 
				
			|||||||
                return cache_stat
 | 
					                return cache_stat
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        fun = info
 | 
					        fun = info
 | 
				
			||||||
        if MACOS and path.split('/')[-1].startswith('._'):
 | 
					        if MACOS and path.split("/")[-1].startswith("._"):
 | 
				
			||||||
            fun = dbg
 | 
					            fun = dbg
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        fun("=ENOENT ({})".format(hexler(path)))
 | 
					        fun("=ENOENT ({})".format(hexler(path)))
 | 
				
			||||||
@@ -979,8 +1008,14 @@ def main():
 | 
				
			|||||||
        log = null_log
 | 
					        log = null_log
 | 
				
			||||||
        dbg = null_log
 | 
					        dbg = null_log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if ar.a and ar.a.startswith("$"):
 | 
				
			||||||
 | 
					        fn = ar.a[1:]
 | 
				
			||||||
 | 
					        log("reading password from file [{}]".format(fn))
 | 
				
			||||||
 | 
					        with open(fn, "rb") as f:
 | 
				
			||||||
 | 
					            ar.a = f.read().decode("utf-8").strip()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if WINDOWS:
 | 
					    if WINDOWS:
 | 
				
			||||||
        os.system("")
 | 
					        os.system("rem")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        for ch in '<>:"\\|?*':
 | 
					        for ch in '<>:"\\|?*':
 | 
				
			||||||
            # microsoft maps illegal characters to f0xx
 | 
					            # microsoft maps illegal characters to f0xx
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										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
 | 
					* modify `10.13.1.1` as necessary if you wish to support browsers without javascript
 | 
				
			||||||
 | 
					
 | 
				
			||||||
### [`explorer-nothumbs-nofoldertypes.reg`](explorer-nothumbs-nofoldertypes.reg)
 | 
					### [`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
 | 
					# OS integration
 | 
				
			||||||
init-scripts to start copyparty as a service
 | 
					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
 | 
					Description=copyparty file server
 | 
				
			||||||
 | 
					
 | 
				
			||||||
[Service]
 | 
					[Service]
 | 
				
			||||||
ExecStart=/usr/bin/python /usr/local/bin/copyparty-sfx.py -q -v /mnt::a
 | 
					ExecStart=/usr/bin/python3 /usr/local/bin/copyparty-sfx.py -q -v /mnt::a
 | 
				
			||||||
ExecStartPre=/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf'
 | 
					ExecStartPre=/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
[Install]
 | 
					[Install]
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -8,18 +8,29 @@ __copyright__ = 2019
 | 
				
			|||||||
__license__ = "MIT"
 | 
					__license__ = "MIT"
 | 
				
			||||||
__url__ = "https://github.com/9001/copyparty/"
 | 
					__url__ = "https://github.com/9001/copyparty/"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import re
 | 
				
			||||||
import os
 | 
					import os
 | 
				
			||||||
 | 
					import sys
 | 
				
			||||||
import time
 | 
					import time
 | 
				
			||||||
 | 
					import signal
 | 
				
			||||||
import shutil
 | 
					import shutil
 | 
				
			||||||
import filecmp
 | 
					import filecmp
 | 
				
			||||||
import locale
 | 
					import locale
 | 
				
			||||||
import argparse
 | 
					import argparse
 | 
				
			||||||
 | 
					import threading
 | 
				
			||||||
 | 
					import traceback
 | 
				
			||||||
from textwrap import dedent
 | 
					from textwrap import dedent
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from .__init__ import E, WINDOWS, VT100
 | 
					from .__init__ import E, WINDOWS, VT100, PY2
 | 
				
			||||||
from .__version__ import S_VERSION, S_BUILD_DT, CODENAME
 | 
					from .__version__ import S_VERSION, S_BUILD_DT, CODENAME
 | 
				
			||||||
from .svchub import SvcHub
 | 
					from .svchub import SvcHub
 | 
				
			||||||
from .util import py_desc
 | 
					from .util import py_desc, align_tab, IMPLICATIONS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					HAVE_SSL = True
 | 
				
			||||||
 | 
					try:
 | 
				
			||||||
 | 
					    import ssl
 | 
				
			||||||
 | 
					except:
 | 
				
			||||||
 | 
					    HAVE_SSL = False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class RiceFormatter(argparse.HelpFormatter):
 | 
					class RiceFormatter(argparse.HelpFormatter):
 | 
				
			||||||
@@ -45,6 +56,10 @@ class RiceFormatter(argparse.HelpFormatter):
 | 
				
			|||||||
        return "".join(indent + line + "\n" for line in text.splitlines())
 | 
					        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():
 | 
					def ensure_locale():
 | 
				
			||||||
    for x in [
 | 
					    for x in [
 | 
				
			||||||
        "en_US.UTF-8",
 | 
					        "en_US.UTF-8",
 | 
				
			||||||
@@ -85,10 +100,90 @@ 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
 | 
					    # printf 'NO\n.\n.\n.\n.\ncopyparty-insecure\n.\n' | faketime '2000-01-01 00:00:00' openssl req -x509 -sha256 -newkey rsa:2048 -keyout insecure.pem -out insecure.pem -days $((($(printf %d 0x7fffffff)-$(date +%s --date=2000-01-01T00:00:00Z))/(60*60*24))) -nodes && ls -al insecure.pem && openssl x509 -in insecure.pem -text -noout
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def main():
 | 
					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(argv=None):
 | 
				
			||||||
    time.strptime("19970815", "%Y%m%d")  # python#7980
 | 
					    time.strptime("19970815", "%Y%m%d")  # python#7980
 | 
				
			||||||
    if WINDOWS:
 | 
					    if WINDOWS:
 | 
				
			||||||
        os.system("")  # enables colors
 | 
					        os.system("rem")  # enables colors
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if argv is None:
 | 
				
			||||||
 | 
					        argv = sys.argv
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    desc = py_desc().replace("[", "\033[1;30m[")
 | 
					    desc = py_desc().replace("[", "\033[1;30m[")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -96,8 +191,21 @@ def main():
 | 
				
			|||||||
    print(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc))
 | 
					    print(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    ensure_locale()
 | 
					    ensure_locale()
 | 
				
			||||||
 | 
					    if HAVE_SSL:
 | 
				
			||||||
        ensure_cert()
 | 
					        ensure_cert()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    deprecated = [["-e2s", "-e2ds"]]
 | 
				
			||||||
 | 
					    for dk, nk in deprecated:
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            idx = argv.index(dk)
 | 
				
			||||||
 | 
					        except:
 | 
				
			||||||
 | 
					            continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        msg = "\033[1;31mWARNING:\033[0;1m\n  {} \033[0;33mwas replaced with\033[0;1m {} \033[0;33mand will be removed\n\033[0m"
 | 
				
			||||||
 | 
					        print(msg.format(dk, nk))
 | 
				
			||||||
 | 
					        argv[idx] = nk
 | 
				
			||||||
 | 
					        time.sleep(2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    ap = argparse.ArgumentParser(
 | 
					    ap = argparse.ArgumentParser(
 | 
				
			||||||
        formatter_class=RiceFormatter,
 | 
					        formatter_class=RiceFormatter,
 | 
				
			||||||
        prog="copyparty",
 | 
					        prog="copyparty",
 | 
				
			||||||
@@ -110,7 +218,7 @@ def main():
 | 
				
			|||||||
               and "cflag" is config flags to set on this volume
 | 
					               and "cflag" is config flags to set on this volume
 | 
				
			||||||
            
 | 
					            
 | 
				
			||||||
            list of cflags:
 | 
					            list of cflags:
 | 
				
			||||||
              cnodupe rejects existing files (instead of symlinking them)
 | 
					              "cnodupe" rejects existing files (instead of symlinking them)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            example:\033[35m
 | 
					            example:\033[35m
 | 
				
			||||||
              -a ed:hunter2 -v .::r:aed -v ../inc:dump:w:aed:cnodupe  \033[36m
 | 
					              -a ed:hunter2 -v .::r:aed -v ../inc:dump:w:aed:cnodupe  \033[36m
 | 
				
			||||||
@@ -127,6 +235,16 @@ def main():
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            consider the config file for more flexible account/volume management,
 | 
					            consider the config file for more flexible account/volume management,
 | 
				
			||||||
            including dynamic reload at runtime (and being more readable w)
 | 
					            including dynamic reload at runtime (and being more readable w)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            values for --urlform:
 | 
				
			||||||
 | 
					              "stash" dumps the data to file and returns length + checksum
 | 
				
			||||||
 | 
					              "save,get" dumps to file and returns the page like a GET
 | 
				
			||||||
 | 
					              "print,get" prints the data in the log and returns GET
 | 
				
			||||||
 | 
					              (leave out the ",get" to return an error instead)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            --ciphers help = available ssl/tls ciphers,
 | 
				
			||||||
 | 
					            --ssl-ver help = available ssl/tls versions,
 | 
				
			||||||
 | 
					              default is what python considers safe, usually >= TLS1
 | 
				
			||||||
            """
 | 
					            """
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
@@ -139,18 +257,50 @@ def main():
 | 
				
			|||||||
    ap.add_argument("-a", metavar="ACCT", type=str, action="append", help="add account")
 | 
					    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("-v", metavar="VOL", type=str, action="append", help="add volume")
 | 
				
			||||||
    ap.add_argument("-q", action="store_true", help="quiet")
 | 
					    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("-ed", action="store_true", help="enable ?dots")
 | 
				
			||||||
    ap.add_argument("-emp", action="store_true", help="enable markdown plugins")
 | 
					    ap.add_argument("-emp", action="store_true", help="enable markdown plugins")
 | 
				
			||||||
    ap.add_argument("-e2d", action="store_true", help="enable up2k database")
 | 
					 | 
				
			||||||
    ap.add_argument("-e2s", action="store_true", help="enable up2k db-scanner")
 | 
					 | 
				
			||||||
    ap.add_argument("-mcr", metavar="SEC", type=int, default=60, help="md-editor mod-chk rate")
 | 
					    ap.add_argument("-mcr", metavar="SEC", type=int, default=60, help="md-editor mod-chk rate")
 | 
				
			||||||
    ap.add_argument("-nw", action="store_true", help="disable writes (benchmark)")
 | 
					    ap.add_argument("-nw", action="store_true", help="disable writes (benchmark)")
 | 
				
			||||||
    ap.add_argument("-nih", action="store_true", help="no info hostname")
 | 
					    ap.add_argument("-nih", action="store_true", help="no info hostname")
 | 
				
			||||||
    ap.add_argument("-nid", action="store_true", help="no info disk-usage")
 | 
					    ap.add_argument("-nid", action="store_true", help="no info disk-usage")
 | 
				
			||||||
    ap.add_argument("--no-sendfile", action="store_true", help="disable sendfile")
 | 
					    ap.add_argument("--no-zip", action="store_true", help="disable download as zip/tar")
 | 
				
			||||||
    al = ap.parse_args()
 | 
					    ap.add_argument("--no-sendfile", action="store_true", help="disable sendfile (for debugging)")
 | 
				
			||||||
 | 
					    ap.add_argument("--no-scandir", action="store_true", help="disable scandir (for debugging)")
 | 
				
			||||||
 | 
					    ap.add_argument("--urlform", metavar="MODE", type=str, default="print,get", help="how to handle url-forms")
 | 
				
			||||||
 | 
					    ap.add_argument("--salt", type=str, default="hunter2", help="up2k file-hash salt")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ap2 = ap.add_argument_group('database options')
 | 
				
			||||||
 | 
					    ap2.add_argument("-e2d", action="store_true", help="enable up2k database")
 | 
				
			||||||
 | 
					    ap2.add_argument("-e2ds", action="store_true", help="enable up2k db-scanner, sets -e2d")
 | 
				
			||||||
 | 
					    ap2.add_argument("-e2dsa", action="store_true", help="scan all folders (for search), sets -e2ds")
 | 
				
			||||||
 | 
					    ap2.add_argument("-e2t", action="store_true", help="enable metadata indexing")
 | 
				
			||||||
 | 
					    ap2.add_argument("-e2ts", action="store_true", help="enable metadata scanner, sets -e2t")
 | 
				
			||||||
 | 
					    ap2.add_argument("-e2tsr", action="store_true", help="rescan all metadata, sets -e2ts")
 | 
				
			||||||
 | 
					    ap2.add_argument("--no-mutagen", action="store_true", help="use ffprobe for tags instead")
 | 
				
			||||||
 | 
					    ap2.add_argument("--no-mtag-mt", action="store_true", help="disable tag-read parallelism")
 | 
				
			||||||
 | 
					    ap2.add_argument("-mtm", metavar="M=t,t,t", action="append", type=str, help="add/replace metadata mapping")
 | 
				
			||||||
 | 
					    ap2.add_argument("-mte", metavar="M,M,M", type=str, help="tags to index/display (comma-sep.)",
 | 
				
			||||||
 | 
					        default="circle,album,.tn,artist,title,.bpm,key,.dur,.q")
 | 
				
			||||||
 | 
					    ap2.add_argument("-mtp", metavar="M=[f,]bin", action="append", type=str, help="read tag M using bin")
 | 
				
			||||||
 | 
					    ap2.add_argument("--srch-time", metavar="SEC", type=int, default=30, help="search deadline")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ap2 = ap.add_argument_group('SSL/TLS options')
 | 
				
			||||||
 | 
					    ap2.add_argument("--http-only", action="store_true", help="disable ssl/tls")
 | 
				
			||||||
 | 
					    ap2.add_argument("--https-only", action="store_true", help="disable plaintext")
 | 
				
			||||||
 | 
					    ap2.add_argument("--ssl-ver", metavar="LIST", type=str, help="ssl/tls versions to allow")
 | 
				
			||||||
 | 
					    ap2.add_argument("--ciphers", metavar="LIST", help="set allowed ciphers")
 | 
				
			||||||
 | 
					    ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info")
 | 
				
			||||||
 | 
					    ap2.add_argument("--ssl-log", metavar="PATH", help="log master secrets")
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    al = ap.parse_args(args=argv[1:])
 | 
				
			||||||
    # fmt: on
 | 
					    # fmt: on
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # propagate implications
 | 
				
			||||||
 | 
					    for k1, k2 in IMPLICATIONS:
 | 
				
			||||||
 | 
					        if getattr(al, k1):
 | 
				
			||||||
 | 
					            setattr(al, k2, True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    al.i = al.i.split(",")
 | 
					    al.i = al.i.split(",")
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        if "-" in al.p:
 | 
					        if "-" in al.p:
 | 
				
			||||||
@@ -161,6 +311,23 @@ def main():
 | 
				
			|||||||
    except:
 | 
					    except:
 | 
				
			||||||
        raise Exception("invalid value for -p")
 | 
					        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()
 | 
					    SvcHub(al).run()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,8 +1,8 @@
 | 
				
			|||||||
# coding: utf-8
 | 
					# coding: utf-8
 | 
				
			||||||
 | 
					
 | 
				
			||||||
VERSION = (0, 7, 3)
 | 
					VERSION = (0, 10, 4)
 | 
				
			||||||
CODENAME = "keeping track"
 | 
					CODENAME = "zip it"
 | 
				
			||||||
BUILD_DT = (2021, 2, 3)
 | 
					BUILD_DT = (2021, 3, 29)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
S_VERSION = ".".join(map(str, VERSION))
 | 
					S_VERSION = ".".join(map(str, VERSION))
 | 
				
			||||||
S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT)
 | 
					S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT)
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,12 +1,14 @@
 | 
				
			|||||||
# coding: utf-8
 | 
					# coding: utf-8
 | 
				
			||||||
from __future__ import print_function, unicode_literals
 | 
					from __future__ import print_function, unicode_literals
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import os
 | 
					 | 
				
			||||||
import re
 | 
					import re
 | 
				
			||||||
 | 
					import os
 | 
				
			||||||
 | 
					import sys
 | 
				
			||||||
 | 
					import stat
 | 
				
			||||||
import threading
 | 
					import threading
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from .__init__ import PY2, WINDOWS
 | 
					from .__init__ import PY2, WINDOWS
 | 
				
			||||||
from .util import undot, Pebkac, fsdec, fsenc
 | 
					from .util import IMPLICATIONS, undot, Pebkac, fsdec, fsenc, statdir, nuprint
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class VFS(object):
 | 
					class VFS(object):
 | 
				
			||||||
@@ -19,6 +21,19 @@ class VFS(object):
 | 
				
			|||||||
        self.uwrite = uwrite  # users who can write this
 | 
					        self.uwrite = uwrite  # users who can write this
 | 
				
			||||||
        self.flags = flags  # config switches
 | 
					        self.flags = flags  # config switches
 | 
				
			||||||
        self.nodes = {}  # child nodes
 | 
					        self.nodes = {}  # child nodes
 | 
				
			||||||
 | 
					        self.all_vols = {vpath: self}  # flattened recursive
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __repr__(self):
 | 
				
			||||||
 | 
					        return "VFS({})".format(
 | 
				
			||||||
 | 
					            ", ".join(
 | 
				
			||||||
 | 
					                "{}={!r}".format(k, self.__dict__[k])
 | 
				
			||||||
 | 
					                for k in "realpath vpath uread uwrite flags".split()
 | 
				
			||||||
 | 
					            )
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def _trk(self, vol):
 | 
				
			||||||
 | 
					        self.all_vols[vol.vpath] = vol
 | 
				
			||||||
 | 
					        return vol
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def add(self, src, dst):
 | 
					    def add(self, src, dst):
 | 
				
			||||||
        """get existing, or add new path to the vfs"""
 | 
					        """get existing, or add new path to the vfs"""
 | 
				
			||||||
@@ -30,7 +45,7 @@ class VFS(object):
 | 
				
			|||||||
            name, dst = dst.split("/", 1)
 | 
					            name, dst = dst.split("/", 1)
 | 
				
			||||||
            if name in self.nodes:
 | 
					            if name in self.nodes:
 | 
				
			||||||
                # exists; do not manipulate permissions
 | 
					                # exists; do not manipulate permissions
 | 
				
			||||||
                return self.nodes[name].add(src, dst)
 | 
					                return self._trk(self.nodes[name].add(src, dst))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            vn = VFS(
 | 
					            vn = VFS(
 | 
				
			||||||
                "{}/{}".format(self.realpath, name),
 | 
					                "{}/{}".format(self.realpath, name),
 | 
				
			||||||
@@ -39,8 +54,9 @@ class VFS(object):
 | 
				
			|||||||
                self.uwrite,
 | 
					                self.uwrite,
 | 
				
			||||||
                self.flags,
 | 
					                self.flags,
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
 | 
					            self._trk(vn)
 | 
				
			||||||
            self.nodes[name] = vn
 | 
					            self.nodes[name] = vn
 | 
				
			||||||
            return vn.add(src, dst)
 | 
					            return self._trk(vn.add(src, dst))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if dst in self.nodes:
 | 
					        if dst in self.nodes:
 | 
				
			||||||
            # leaf exists; return as-is
 | 
					            # leaf exists; return as-is
 | 
				
			||||||
@@ -50,7 +66,7 @@ class VFS(object):
 | 
				
			|||||||
        vp = "{}/{}".format(self.vpath, dst).lstrip("/")
 | 
					        vp = "{}/{}".format(self.vpath, dst).lstrip("/")
 | 
				
			||||||
        vn = VFS(src, vp)
 | 
					        vn = VFS(src, vp)
 | 
				
			||||||
        self.nodes[dst] = vn
 | 
					        self.nodes[dst] = vn
 | 
				
			||||||
        return vn
 | 
					        return self._trk(vn)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def _find(self, vpath):
 | 
					    def _find(self, vpath):
 | 
				
			||||||
        """return [vfs,remainder]"""
 | 
					        """return [vfs,remainder]"""
 | 
				
			||||||
@@ -97,12 +113,11 @@ class VFS(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        return fsdec(os.path.realpath(fsenc(rp)))
 | 
					        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"""
 | 
					        """return user-readable [fsdir,real,virt] items at vpath"""
 | 
				
			||||||
        virt_vis = {}  # nodes readable by user
 | 
					        virt_vis = {}  # nodes readable by user
 | 
				
			||||||
        abspath = self.canonical(rem)
 | 
					        abspath = self.canonical(rem)
 | 
				
			||||||
        items = os.listdir(fsenc(abspath))
 | 
					        real = list(statdir(nuprint, scandir, lstat, abspath))
 | 
				
			||||||
        real = [fsdec(x) for x in items]
 | 
					 | 
				
			||||||
        real.sort()
 | 
					        real.sort()
 | 
				
			||||||
        if not rem:
 | 
					        if not rem:
 | 
				
			||||||
            for name, vn2 in sorted(self.nodes.items()):
 | 
					            for name, vn2 in sorted(self.nodes.items()):
 | 
				
			||||||
@@ -110,10 +125,77 @@ class VFS(object):
 | 
				
			|||||||
                    virt_vis[name] = vn2
 | 
					                    virt_vis[name] = vn2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            # no vfs nodes in the list of real inodes
 | 
					            # no vfs nodes in the list of real inodes
 | 
				
			||||||
            real = [x for x in real if x not in self.nodes]
 | 
					            real = [x for x in real if x[0] not in self.nodes]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return [abspath, real, virt_vis]
 | 
					        return [abspath, real, virt_vis]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def walk(self, rel, rem, uname, dots, scandir, lstat=False):
 | 
				
			||||||
 | 
					        """
 | 
				
			||||||
 | 
					        recursively yields from ./rem;
 | 
				
			||||||
 | 
					        rel is a unix-style user-defined vpath (not vfs-related)
 | 
				
			||||||
 | 
					        """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        fsroot, vfs_ls, vfs_virt = self.ls(rem, uname, scandir, lstat)
 | 
				
			||||||
 | 
					        rfiles = [x for x in vfs_ls if not stat.S_ISDIR(x[1].st_mode)]
 | 
				
			||||||
 | 
					        rdirs = [x for x in vfs_ls if stat.S_ISDIR(x[1].st_mode)]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        rfiles.sort()
 | 
				
			||||||
 | 
					        rdirs.sort()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        yield rel, fsroot, rfiles, rdirs, vfs_virt
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for rdir, _ in rdirs:
 | 
				
			||||||
 | 
					            if not dots and rdir.startswith("."):
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            wrel = (rel + "/" + rdir).lstrip("/")
 | 
				
			||||||
 | 
					            wrem = (rem + "/" + rdir).lstrip("/")
 | 
				
			||||||
 | 
					            for x in self.walk(wrel, wrem, uname, scandir, lstat):
 | 
				
			||||||
 | 
					                yield x
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for n, vfs in sorted(vfs_virt.items()):
 | 
				
			||||||
 | 
					            if not dots and n.startswith("."):
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            wrel = (rel + "/" + n).lstrip("/")
 | 
				
			||||||
 | 
					            for x in vfs.walk(wrel, "", uname, scandir, lstat):
 | 
				
			||||||
 | 
					                yield x
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def zipgen(self, vrem, flt, uname, dots, scandir):
 | 
				
			||||||
 | 
					        if flt:
 | 
				
			||||||
 | 
					            flt = {k: True for k in flt}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for vpath, apath, files, rd, vd in self.walk("", vrem, uname, dots, scandir):
 | 
				
			||||||
 | 
					            if flt:
 | 
				
			||||||
 | 
					                files = [x for x in files if x[0] in flt]
 | 
				
			||||||
 | 
					                rd = [x for x in rd if x[0] in flt]
 | 
				
			||||||
 | 
					                vd = {x: y for x, y in vd.items() if x in flt}
 | 
				
			||||||
 | 
					                flt = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            # print(repr([vpath, apath, [x[0] for x in files]]))
 | 
				
			||||||
 | 
					            fnames = [n[0] for n in files]
 | 
				
			||||||
 | 
					            vpaths = [vpath + "/" + n for n in fnames] if vpath else fnames
 | 
				
			||||||
 | 
					            apaths = [os.path.join(apath, n) for n in fnames]
 | 
				
			||||||
 | 
					            files = list(zip(vpaths, apaths, files))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if not dots:
 | 
				
			||||||
 | 
					                # dotfile filtering based on vpath (intended visibility)
 | 
				
			||||||
 | 
					                files = [x for x in files if "/." not in "/" + x[0]]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                rm = [x for x in rd if x[0].startswith(".")]
 | 
				
			||||||
 | 
					                for x in rm:
 | 
				
			||||||
 | 
					                    rd.remove(x)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                rm = [k for k in vd.keys() if k.startswith(".")]
 | 
				
			||||||
 | 
					                for x in rm:
 | 
				
			||||||
 | 
					                    del vd[x]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            # up2k filetring based on actual abspath
 | 
				
			||||||
 | 
					            files = [x for x in files if "{0}.hist{0}up2k.".format(os.sep) not in x[1]]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            for f in [{"vp": v, "ap": a, "st": n[1]} for v, a, n in files]:
 | 
				
			||||||
 | 
					                yield f
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def user_tree(self, uname, readable=False, writable=False):
 | 
					    def user_tree(self, uname, readable=False, writable=False):
 | 
				
			||||||
        ret = []
 | 
					        ret = []
 | 
				
			||||||
        opt1 = readable and (uname in self.uread or "*" in self.uread)
 | 
					        opt1 = readable and (uname in self.uread or "*" in self.uread)
 | 
				
			||||||
@@ -143,8 +225,8 @@ class AuthSrv(object):
 | 
				
			|||||||
        self.mutex = threading.Lock()
 | 
					        self.mutex = threading.Lock()
 | 
				
			||||||
        self.reload()
 | 
					        self.reload()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def log(self, msg):
 | 
					    def log(self, msg, c=0):
 | 
				
			||||||
        self.log_func("auth", msg)
 | 
					        self.log_func("auth", msg, c)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def invert(self, orig):
 | 
					    def invert(self, orig):
 | 
				
			||||||
        if PY2:
 | 
					        if PY2:
 | 
				
			||||||
@@ -196,13 +278,39 @@ class AuthSrv(object):
 | 
				
			|||||||
                continue
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            lvl, uname = ln.split(" ")
 | 
					            lvl, uname = ln.split(" ")
 | 
				
			||||||
            if lvl in "ra":
 | 
					            self._read_vol_str(
 | 
				
			||||||
                mread[vol_dst].append(uname)
 | 
					                lvl, uname, mread[vol_dst], mwrite[vol_dst], mflags[vol_dst]
 | 
				
			||||||
            if lvl in "wa":
 | 
					            )
 | 
				
			||||||
                mwrite[vol_dst].append(uname)
 | 
					
 | 
				
			||||||
 | 
					    def _read_vol_str(self, lvl, uname, mr, mw, mf):
 | 
				
			||||||
        if lvl == "c":
 | 
					        if lvl == "c":
 | 
				
			||||||
                # config option, currently switches only
 | 
					            cval = True
 | 
				
			||||||
                mflags[vol_dst][uname] = True
 | 
					            if "=" in uname:
 | 
				
			||||||
 | 
					                uname, cval = uname.split("=", 1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            self._read_volflag(mf, uname, cval, False)
 | 
				
			||||||
 | 
					            return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if uname == "":
 | 
				
			||||||
 | 
					            uname = "*"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if lvl in "ra":
 | 
				
			||||||
 | 
					            mr.append(uname)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if lvl in "wa":
 | 
				
			||||||
 | 
					            mw.append(uname)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def _read_volflag(self, flags, name, value, is_list):
 | 
				
			||||||
 | 
					        if name not in ["mtp"]:
 | 
				
			||||||
 | 
					            flags[name] = value
 | 
				
			||||||
 | 
					            return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if not is_list:
 | 
				
			||||||
 | 
					            value = [value]
 | 
				
			||||||
 | 
					        elif not value:
 | 
				
			||||||
 | 
					            return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        flags[name] = flags.get(name, []) + value
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def reload(self):
 | 
					    def reload(self):
 | 
				
			||||||
        """
 | 
					        """
 | 
				
			||||||
@@ -225,7 +333,7 @@ class AuthSrv(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        if self.args.v:
 | 
					        if self.args.v:
 | 
				
			||||||
            # list of src:dst:permset:permset:...
 | 
					            # list of src:dst:permset:permset:...
 | 
				
			||||||
            # permset is [rwa]username
 | 
					            # permset is [rwa]username or [c]flag
 | 
				
			||||||
            for v_str in self.args.v:
 | 
					            for v_str in self.args.v:
 | 
				
			||||||
                m = self.re_vol.match(v_str)
 | 
					                m = self.re_vol.match(v_str)
 | 
				
			||||||
                if not m:
 | 
					                if not m:
 | 
				
			||||||
@@ -242,28 +350,20 @@ class AuthSrv(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
                perms = perms.split(":")
 | 
					                perms = perms.split(":")
 | 
				
			||||||
                for (lvl, uname) in [[x[0], x[1:]] for x in perms]:
 | 
					                for (lvl, uname) in [[x[0], x[1:]] for x in perms]:
 | 
				
			||||||
                    if lvl == "c":
 | 
					                    self._read_vol_str(lvl, uname, mread[dst], mwrite[dst], mflags[dst])
 | 
				
			||||||
                        # config option, currently switches only
 | 
					 | 
				
			||||||
                        mflags[dst][uname] = True
 | 
					 | 
				
			||||||
                    if uname == "":
 | 
					 | 
				
			||||||
                        uname = "*"
 | 
					 | 
				
			||||||
                    if lvl in "ra":
 | 
					 | 
				
			||||||
                        mread[dst].append(uname)
 | 
					 | 
				
			||||||
                    if lvl in "wa":
 | 
					 | 
				
			||||||
                        mwrite[dst].append(uname)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if self.args.c:
 | 
					        if self.args.c:
 | 
				
			||||||
            for cfg_fn in self.args.c:
 | 
					            for cfg_fn in self.args.c:
 | 
				
			||||||
                with open(cfg_fn, "rb") as f:
 | 
					                with open(cfg_fn, "rb") as f:
 | 
				
			||||||
                    self._parse_config_file(f, user, mread, mwrite, mflags, mount)
 | 
					                    self._parse_config_file(f, user, mread, mwrite, mflags, mount)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        self.all_writable = []
 | 
					 | 
				
			||||||
        if not mount:
 | 
					        if not mount:
 | 
				
			||||||
            # -h says our defaults are CWD at root and read/write for everyone
 | 
					            # -h says our defaults are CWD at root and read/write for everyone
 | 
				
			||||||
            vfs = VFS(os.path.abspath("."), "", ["*"], ["*"])
 | 
					            vfs = VFS(os.path.abspath("."), "", ["*"], ["*"])
 | 
				
			||||||
        elif "" not in mount:
 | 
					        elif "" not in mount:
 | 
				
			||||||
            # there's volumes but no root; make root inaccessible
 | 
					            # there's volumes but no root; make root inaccessible
 | 
				
			||||||
            vfs = VFS(os.path.abspath("."), "")
 | 
					            vfs = VFS(os.path.abspath("."), "")
 | 
				
			||||||
 | 
					            vfs.flags["d2d"] = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        maxdepth = 0
 | 
					        maxdepth = 0
 | 
				
			||||||
        for dst in sorted(mount.keys(), key=lambda x: (x.count("/"), len(x))):
 | 
					        for dst in sorted(mount.keys(), key=lambda x: (x.count("/"), len(x))):
 | 
				
			||||||
@@ -280,11 +380,6 @@ class AuthSrv(object):
 | 
				
			|||||||
            v.uread = mread[dst]
 | 
					            v.uread = mread[dst]
 | 
				
			||||||
            v.uwrite = mwrite[dst]
 | 
					            v.uwrite = mwrite[dst]
 | 
				
			||||||
            v.flags = mflags[dst]
 | 
					            v.flags = mflags[dst]
 | 
				
			||||||
            if v.uwrite:
 | 
					 | 
				
			||||||
                self.all_writable.append(v)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if vfs.uwrite and vfs not in self.all_writable:
 | 
					 | 
				
			||||||
            self.all_writable.append(vfs)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        missing_users = {}
 | 
					        missing_users = {}
 | 
				
			||||||
        for d in [mread, mwrite]:
 | 
					        for d in [mread, mwrite]:
 | 
				
			||||||
@@ -295,21 +390,100 @@ class AuthSrv(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        if missing_users:
 | 
					        if missing_users:
 | 
				
			||||||
            self.log(
 | 
					            self.log(
 | 
				
			||||||
                "\033[31myou must -a the following users: "
 | 
					                "you must -a the following users: "
 | 
				
			||||||
                + ", ".join(k for k in sorted(missing_users))
 | 
					                + ", ".join(k for k in sorted(missing_users)),
 | 
				
			||||||
                + "\033[0m"
 | 
					                c=1,
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
            raise Exception("invalid config")
 | 
					            raise Exception("invalid config")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        all_mte = {}
 | 
				
			||||||
 | 
					        errors = False
 | 
				
			||||||
 | 
					        for vol in vfs.all_vols.values():
 | 
				
			||||||
 | 
					            if (self.args.e2ds and vol.uwrite) or self.args.e2dsa:
 | 
				
			||||||
 | 
					                vol.flags["e2ds"] = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if self.args.e2d or "e2ds" in vol.flags:
 | 
				
			||||||
 | 
					                vol.flags["e2d"] = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            for k in ["e2t", "e2ts", "e2tsr"]:
 | 
				
			||||||
 | 
					                if getattr(self.args, k):
 | 
				
			||||||
 | 
					                    vol.flags[k] = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            for k1, k2 in IMPLICATIONS:
 | 
				
			||||||
 | 
					                if k1 in vol.flags:
 | 
				
			||||||
 | 
					                    vol.flags[k2] = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            # default tag-list if unset
 | 
				
			||||||
 | 
					            if "mte" not in vol.flags:
 | 
				
			||||||
 | 
					                vol.flags["mte"] = self.args.mte
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            # append parsers from argv to volume-flags
 | 
				
			||||||
 | 
					            self._read_volflag(vol.flags, "mtp", self.args.mtp, True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            # d2d drops all database features for a volume
 | 
				
			||||||
 | 
					            for grp, rm in [["d2d", "e2d"], ["d2t", "e2t"]]:
 | 
				
			||||||
 | 
					                if not vol.flags.get(grp, False):
 | 
				
			||||||
 | 
					                    continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                vol.flags["d2t"] = True
 | 
				
			||||||
 | 
					                vol.flags = {k: v for k, v in vol.flags.items() if not k.startswith(rm)}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            # mt* needs e2t so drop those too
 | 
				
			||||||
 | 
					            for grp, rm in [["e2t", "mt"]]:
 | 
				
			||||||
 | 
					                if vol.flags.get(grp, False):
 | 
				
			||||||
 | 
					                    continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                vol.flags = {k: v for k, v in vol.flags.items() if not k.startswith(rm)}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            # verify tags mentioned by -mt[mp] are used by -mte
 | 
				
			||||||
 | 
					            local_mtp = {}
 | 
				
			||||||
 | 
					            local_only_mtp = {}
 | 
				
			||||||
 | 
					            for a in vol.flags.get("mtp", []) + vol.flags.get("mtm", []):
 | 
				
			||||||
 | 
					                a = a.split("=")[0]
 | 
				
			||||||
 | 
					                local_mtp[a] = True
 | 
				
			||||||
 | 
					                local = True
 | 
				
			||||||
 | 
					                for b in self.args.mtp or []:
 | 
				
			||||||
 | 
					                    b = b.split("=")[0]
 | 
				
			||||||
 | 
					                    if a == b:
 | 
				
			||||||
 | 
					                        local = False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if local:
 | 
				
			||||||
 | 
					                    local_only_mtp[a] = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            local_mte = {}
 | 
				
			||||||
 | 
					            for a in vol.flags.get("mte", "").split(","):
 | 
				
			||||||
 | 
					                local = True
 | 
				
			||||||
 | 
					                all_mte[a] = True
 | 
				
			||||||
 | 
					                local_mte[a] = True
 | 
				
			||||||
 | 
					                for b in self.args.mte.split(","):
 | 
				
			||||||
 | 
					                    if not a or not b:
 | 
				
			||||||
 | 
					                        continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    if a == b:
 | 
				
			||||||
 | 
					                        local = False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            for mtp in local_only_mtp.keys():
 | 
				
			||||||
 | 
					                if mtp not in local_mte:
 | 
				
			||||||
 | 
					                    m = 'volume "/{}" defines metadata tag "{}", but doesnt use it in "-mte" (or with "cmte" in its volume-flags)'
 | 
				
			||||||
 | 
					                    self.log(m.format(vol.vpath, mtp), 1)
 | 
				
			||||||
 | 
					                    errors = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for mtp in self.args.mtp or []:
 | 
				
			||||||
 | 
					            mtp = mtp.split("=")[0]
 | 
				
			||||||
 | 
					            if mtp not in all_mte:
 | 
				
			||||||
 | 
					                m = 'metadata tag "{}" is defined by "-mtm" or "-mtp", but is not used by "-mte" (or by any "cmte" volume-flag)'
 | 
				
			||||||
 | 
					                self.log(m.format(mtp), 1)
 | 
				
			||||||
 | 
					                errors = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if errors:
 | 
				
			||||||
 | 
					            sys.exit(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            v, _ = vfs.get("/", "*", False, True)
 | 
					            v, _ = vfs.get("/", "*", False, True)
 | 
				
			||||||
            if self.warn_anonwrite and os.getcwd() == v.realpath:
 | 
					            if self.warn_anonwrite and os.getcwd() == v.realpath:
 | 
				
			||||||
                self.warn_anonwrite = False
 | 
					                self.warn_anonwrite = False
 | 
				
			||||||
                self.log(
 | 
					                msg = "anyone can read/write the current directory: {}"
 | 
				
			||||||
                    "\033[31manyone can read/write the current directory: {}\033[0m".format(
 | 
					                self.log(msg.format(v.realpath), c=1)
 | 
				
			||||||
                        v.realpath
 | 
					 | 
				
			||||||
                    )
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
        except Pebkac:
 | 
					        except Pebkac:
 | 
				
			||||||
            self.warn_anonwrite = True
 | 
					            self.warn_anonwrite = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -51,7 +51,7 @@ class BrokerMp(object):
 | 
				
			|||||||
            self.procs.append(proc)
 | 
					            self.procs.append(proc)
 | 
				
			||||||
            proc.start()
 | 
					            proc.start()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if True:
 | 
					        if not self.args.q:
 | 
				
			||||||
            thr = threading.Thread(target=self.debug_load_balancer)
 | 
					            thr = threading.Thread(target=self.debug_load_balancer)
 | 
				
			||||||
            thr.daemon = True
 | 
					            thr.daemon = True
 | 
				
			||||||
            thr.start()
 | 
					            thr.start()
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -49,11 +49,11 @@ class MpWorker(object):
 | 
				
			|||||||
        # print('k')
 | 
					        # print('k')
 | 
				
			||||||
        pass
 | 
					        pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def log(self, src, msg):
 | 
					    def log(self, src, msg, c=0):
 | 
				
			||||||
        self.q_yield.put([0, "log", [src, msg]])
 | 
					        self.q_yield.put([0, "log", [src, msg, c]])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def logw(self, msg):
 | 
					    def logw(self, msg, c=0):
 | 
				
			||||||
        self.log("mp{}".format(self.n), msg)
 | 
					        self.log("mp{}".format(self.n), msg, c)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def httpdrop(self, addr):
 | 
					    def httpdrop(self, addr):
 | 
				
			||||||
        self.q_yield.put([0, "httpdrop", [addr]])
 | 
					        self.q_yield.put([0, "httpdrop", [addr]])
 | 
				
			||||||
@@ -73,7 +73,9 @@ class MpWorker(object):
 | 
				
			|||||||
                if PY2:
 | 
					                if PY2:
 | 
				
			||||||
                    sck = pickle.loads(sck)  # nosec
 | 
					                    sck = pickle.loads(sck)  # nosec
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                self.log("%s %s" % addr, "\033[1;30m|%sC-qpop\033[0m" % ("-" * 4,))
 | 
					                if self.args.log_conn:
 | 
				
			||||||
 | 
					                    self.log("%s %s" % addr, "|%sC-qpop" % ("-" * 4,), c="1;30")
 | 
				
			||||||
 | 
					                
 | 
				
			||||||
                self.httpsrv.accept(sck, addr)
 | 
					                self.httpsrv.accept(sck, addr)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                with self.mutex:
 | 
					                with self.mutex:
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -28,7 +28,9 @@ class BrokerThr(object):
 | 
				
			|||||||
    def put(self, want_retval, dest, *args):
 | 
					    def put(self, want_retval, dest, *args):
 | 
				
			||||||
        if dest == "httpconn":
 | 
					        if dest == "httpconn":
 | 
				
			||||||
            sck, addr = args
 | 
					            sck, addr = args
 | 
				
			||||||
            self.log("%s %s" % addr, "\033[1;30m|%sC-qpop\033[0m" % ("-" * 4,))
 | 
					            if self.args.log_conn:
 | 
				
			||||||
 | 
					                self.log("%s %s" % addr, "|%sC-qpop" % ("-" * 4,), c="1;30")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            self.httpsrv.accept(sck, addr)
 | 
					            self.httpsrv.accept(sck, addr)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        else:
 | 
					        else:
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -5,7 +5,9 @@ import os
 | 
				
			|||||||
import stat
 | 
					import stat
 | 
				
			||||||
import gzip
 | 
					import gzip
 | 
				
			||||||
import time
 | 
					import time
 | 
				
			||||||
 | 
					import copy
 | 
				
			||||||
import json
 | 
					import json
 | 
				
			||||||
 | 
					import string
 | 
				
			||||||
import socket
 | 
					import socket
 | 
				
			||||||
import ctypes
 | 
					import ctypes
 | 
				
			||||||
from datetime import datetime
 | 
					from datetime import datetime
 | 
				
			||||||
@@ -13,6 +15,8 @@ import calendar
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
from .__init__ import E, PY2, WINDOWS
 | 
					from .__init__ import E, PY2, WINDOWS
 | 
				
			||||||
from .util import *  # noqa  # pylint: disable=unused-wildcard-import
 | 
					from .util import *  # noqa  # pylint: disable=unused-wildcard-import
 | 
				
			||||||
 | 
					from .szip import StreamZip
 | 
				
			||||||
 | 
					from .star import StreamTar
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if not PY2:
 | 
					if not PY2:
 | 
				
			||||||
    unicode = str
 | 
					    unicode = str
 | 
				
			||||||
@@ -34,22 +38,27 @@ class HttpCli(object):
 | 
				
			|||||||
        self.auth = conn.auth
 | 
					        self.auth = conn.auth
 | 
				
			||||||
        self.log_func = conn.log_func
 | 
					        self.log_func = conn.log_func
 | 
				
			||||||
        self.log_src = conn.log_src
 | 
					        self.log_src = conn.log_src
 | 
				
			||||||
 | 
					        self.tls = hasattr(self.s, "cipher")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        self.bufsz = 1024 * 32
 | 
					        self.bufsz = 1024 * 32
 | 
				
			||||||
        self.absolute_urls = False
 | 
					        self.absolute_urls = False
 | 
				
			||||||
        self.out_headers = {"Access-Control-Allow-Origin": "*"}
 | 
					        self.out_headers = {"Access-Control-Allow-Origin": "*"}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def log(self, msg):
 | 
					    def log(self, msg, c=0):
 | 
				
			||||||
        self.log_func(self.log_src, msg)
 | 
					        self.log_func(self.log_src, msg, c)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def _check_nonfatal(self, ex):
 | 
					    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):
 | 
					    def _assert_safe_rem(self, rem):
 | 
				
			||||||
        # sanity check to prevent any disasters
 | 
					        # sanity check to prevent any disasters
 | 
				
			||||||
        if rem.startswith("/") or rem.startswith("../") or "/../" in rem:
 | 
					        if rem.startswith("/") or rem.startswith("../") or "/../" in rem:
 | 
				
			||||||
            raise Exception("that was close")
 | 
					            raise Exception("that was close")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def j2(self, name, **kwargs):
 | 
				
			||||||
 | 
					        tpl = self.conn.hsrv.j2[name]
 | 
				
			||||||
 | 
					        return tpl.render(**kwargs) if kwargs else tpl
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def run(self):
 | 
					    def run(self):
 | 
				
			||||||
        """returns true if connection can be reused"""
 | 
					        """returns true if connection can be reused"""
 | 
				
			||||||
        self.keepalive = False
 | 
					        self.keepalive = False
 | 
				
			||||||
@@ -61,7 +70,7 @@ class HttpCli(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            if not headerlines[0]:
 | 
					            if not headerlines[0]:
 | 
				
			||||||
                # seen after login with IE6.0.2900.5512.xpsp.080413-2111 (xp-sp3)
 | 
					                # 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)
 | 
					                headerlines.pop(0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            try:
 | 
					            try:
 | 
				
			||||||
@@ -72,9 +81,11 @@ class HttpCli(object):
 | 
				
			|||||||
        except Pebkac as ex:
 | 
					        except Pebkac as ex:
 | 
				
			||||||
            # self.log("pebkac at httpcli.run #1: " + repr(ex))
 | 
					            # self.log("pebkac at httpcli.run #1: " + repr(ex))
 | 
				
			||||||
            self.keepalive = self._check_nonfatal(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
 | 
					            return self.keepalive
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # time.sleep(0.4)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # normalize incoming headers to lowercase;
 | 
					        # normalize incoming headers to lowercase;
 | 
				
			||||||
        # outgoing headers however are Correct-Case
 | 
					        # outgoing headers however are Correct-Case
 | 
				
			||||||
        for header_line in headerlines[1:]:
 | 
					        for header_line in headerlines[1:]:
 | 
				
			||||||
@@ -124,15 +135,15 @@ class HttpCli(object):
 | 
				
			|||||||
                    k, v = k.split("=", 1)
 | 
					                    k, v = k.split("=", 1)
 | 
				
			||||||
                    uparam[k.lower()] = v.strip()
 | 
					                    uparam[k.lower()] = v.strip()
 | 
				
			||||||
                else:
 | 
					                else:
 | 
				
			||||||
                    uparam[k.lower()] = True
 | 
					                    uparam[k.lower()] = False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        self.uparam = uparam
 | 
					        self.uparam = uparam
 | 
				
			||||||
        self.vpath = unquotep(vpath)
 | 
					        self.vpath = unquotep(vpath)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        ua = self.headers.get("user-agent", "")
 | 
					        ua = self.headers.get("user-agent", "")
 | 
				
			||||||
        if ua.startswith("rclone/"):
 | 
					        if ua.startswith("rclone/"):
 | 
				
			||||||
            uparam["raw"] = True
 | 
					            uparam["raw"] = False
 | 
				
			||||||
            uparam["dots"] = True
 | 
					            uparam["dots"] = False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            if self.mode in ["GET", "HEAD"]:
 | 
					            if self.mode in ["GET", "HEAD"]:
 | 
				
			||||||
@@ -150,7 +161,9 @@ class HttpCli(object):
 | 
				
			|||||||
            try:
 | 
					            try:
 | 
				
			||||||
                # self.log("pebkac at httpcli.run #2: " + repr(ex))
 | 
					                # self.log("pebkac at httpcli.run #2: " + repr(ex))
 | 
				
			||||||
                self.keepalive = self._check_nonfatal(ex)
 | 
					                self.keepalive = self._check_nonfatal(ex)
 | 
				
			||||||
                self.loud_reply("{}: {}".format(str(ex), self.vpath), status=ex.code)
 | 
					                self.log("{}\033[0m, {}".format(str(ex), self.vpath), 3)
 | 
				
			||||||
 | 
					                msg = "<pre>{}\r\nURL: {}\r\n".format(str(ex), self.vpath)
 | 
				
			||||||
 | 
					                self.reply(msg.encode("utf-8", "replace"), status=ex.code)
 | 
				
			||||||
                return self.keepalive
 | 
					                return self.keepalive
 | 
				
			||||||
            except Pebkac:
 | 
					            except Pebkac:
 | 
				
			||||||
                return False
 | 
					                return False
 | 
				
			||||||
@@ -159,7 +172,7 @@ class HttpCli(object):
 | 
				
			|||||||
        response = ["HTTP/1.1 {} {}".format(status, HTTPCODE[status])]
 | 
					        response = ["HTTP/1.1 {} {}".format(status, HTTPCODE[status])]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if length is not None:
 | 
					        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
 | 
					        # close if unknown length, otherwise take client's preference
 | 
				
			||||||
        response.append("Connection: " + ("Keep-Alive" if self.keepalive else "Close"))
 | 
					        response.append("Connection: " + ("Keep-Alive" if self.keepalive else "Close"))
 | 
				
			||||||
@@ -218,6 +231,9 @@ class HttpCli(object):
 | 
				
			|||||||
            static_path = os.path.join(E.mod, "web/", self.vpath[5:])
 | 
					            static_path = os.path.join(E.mod, "web/", self.vpath[5:])
 | 
				
			||||||
            return self.tx_file(static_path)
 | 
					            return self.tx_file(static_path)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if "tree" in self.uparam:
 | 
				
			||||||
 | 
					            return self.tx_tree()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # conditional redirect to single volumes
 | 
					        # conditional redirect to single volumes
 | 
				
			||||||
        if self.vpath == "" and not self.uparam:
 | 
					        if self.vpath == "" and not self.uparam:
 | 
				
			||||||
            nread = len(self.rvol)
 | 
					            nread = len(self.rvol)
 | 
				
			||||||
@@ -236,7 +252,7 @@ class HttpCli(object):
 | 
				
			|||||||
        )
 | 
					        )
 | 
				
			||||||
        if not self.readable and not self.writable:
 | 
					        if not self.readable and not self.writable:
 | 
				
			||||||
            self.log("inaccessible: [{}]".format(self.vpath))
 | 
					            self.log("inaccessible: [{}]".format(self.vpath))
 | 
				
			||||||
            self.uparam = {"h": True}
 | 
					            self.uparam = {"h": False}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if "h" in self.uparam:
 | 
					        if "h" in self.uparam:
 | 
				
			||||||
            self.vpath = None
 | 
					            self.vpath = None
 | 
				
			||||||
@@ -294,16 +310,40 @@ class HttpCli(object):
 | 
				
			|||||||
        if "application/octet-stream" in ctype:
 | 
					        if "application/octet-stream" in ctype:
 | 
				
			||||||
            return self.handle_post_binary()
 | 
					            return self.handle_post_binary()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        raise Pebkac(405, "don't know how to handle {} POST".format(ctype))
 | 
					        if "application/x-www-form-urlencoded" in ctype:
 | 
				
			||||||
 | 
					            opt = self.args.urlform
 | 
				
			||||||
 | 
					            if "stash" in opt:
 | 
				
			||||||
 | 
					                return self.handle_stash()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def handle_stash(self):
 | 
					            if "save" in opt:
 | 
				
			||||||
        remains = int(self.headers.get("content-length", None))
 | 
					                post_sz, _, _, path = self.dump_to_file()
 | 
				
			||||||
        if remains is None:
 | 
					                self.log("urlform: {} bytes, {}".format(post_sz, path))
 | 
				
			||||||
            reader = read_socket_unbounded(self.sr)
 | 
					            elif "print" in opt:
 | 
				
			||||||
 | 
					                reader, _ = self.get_body_reader()
 | 
				
			||||||
 | 
					                for buf in reader:
 | 
				
			||||||
 | 
					                    buf = buf.decode("utf-8", "replace")
 | 
				
			||||||
 | 
					                    self.log("urlform @ {}\n  {}\n".format(self.vpath, buf))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if "get" in opt:
 | 
				
			||||||
 | 
					                return self.handle_get()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            raise Pebkac(405, "POST({}) is disabled".format(ctype))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        raise Pebkac(405, "don't know how to handle POST({})".format(ctype))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def get_body_reader(self):
 | 
				
			||||||
 | 
					        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
 | 
					            self.keepalive = False
 | 
				
			||||||
 | 
					            return read_socket_unbounded(self.sr), remains
 | 
				
			||||||
        else:
 | 
					        else:
 | 
				
			||||||
            reader = read_socket(self.sr, remains)
 | 
					            return read_socket(self.sr, remains), remains
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def dump_to_file(self):
 | 
				
			||||||
 | 
					        reader, remains = self.get_body_reader()
 | 
				
			||||||
        vfs, rem = self.conn.auth.vfs.get(self.vpath, self.uname, False, True)
 | 
					        vfs, rem = self.conn.auth.vfs.get(self.vpath, self.uname, False, True)
 | 
				
			||||||
        fdir = os.path.join(vfs.realpath, rem)
 | 
					        fdir = os.path.join(vfs.realpath, rem)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -314,6 +354,14 @@ class HttpCli(object):
 | 
				
			|||||||
        with open(path, "wb", 512 * 1024) as f:
 | 
					        with open(path, "wb", 512 * 1024) as f:
 | 
				
			||||||
            post_sz, _, sha_b64 = hashcopy(self.conn, reader, 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):
 | 
				
			||||||
 | 
					        post_sz, sha_b64, remains, path = self.dump_to_file()
 | 
				
			||||||
        spd = self._spd(post_sz)
 | 
					        spd = self._spd(post_sz)
 | 
				
			||||||
        self.log("{} wrote {}/{} bytes to {}".format(spd, post_sz, remains, path))
 | 
					        self.log("{} wrote {}/{} bytes to {}".format(spd, post_sz, remains, path))
 | 
				
			||||||
        self.reply("{}\n{}\n".format(post_sz, sha_b64).encode("utf-8"))
 | 
					        self.reply("{}\n{}\n".format(post_sz, sha_b64).encode("utf-8"))
 | 
				
			||||||
@@ -349,8 +397,30 @@ class HttpCli(object):
 | 
				
			|||||||
        if act == "tput":
 | 
					        if act == "tput":
 | 
				
			||||||
            return self.handle_text_upload()
 | 
					            return self.handle_text_upload()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if act == "zip":
 | 
				
			||||||
 | 
					            return self.handle_zip_post()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        raise Pebkac(422, 'invalid action "{}"'.format(act))
 | 
					        raise Pebkac(422, 'invalid action "{}"'.format(act))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def handle_zip_post(self):
 | 
				
			||||||
 | 
					        for k in ["zip", "tar"]:
 | 
				
			||||||
 | 
					            v = self.uparam.get(k)
 | 
				
			||||||
 | 
					            if v is not None:
 | 
				
			||||||
 | 
					                break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if v is None:
 | 
				
			||||||
 | 
					            raise Pebkac(422, "need zip or tar keyword")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        vn, rem = self.auth.vfs.get(self.vpath, self.uname, True, False)
 | 
				
			||||||
 | 
					        items = self.parser.require("files", 1024 * 1024)
 | 
				
			||||||
 | 
					        if not items:
 | 
				
			||||||
 | 
					            raise Pebkac(422, "need files list")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        items = items.replace("\r", "").split("\n")
 | 
				
			||||||
 | 
					        items = [unquotep(x) for x in items if items]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return self.tx_zip(k, v, vn, rem, items, self.args.ed)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def handle_post_json(self):
 | 
					    def handle_post_json(self):
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            remains = int(self.headers["content-length"])
 | 
					            remains = int(self.headers["content-length"])
 | 
				
			||||||
@@ -375,29 +445,91 @@ class HttpCli(object):
 | 
				
			|||||||
        except:
 | 
					        except:
 | 
				
			||||||
            raise Pebkac(422, "you POSTed invalid json")
 | 
					            raise Pebkac(422, "you POSTed invalid json")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # prefer this over undot; no reason to allow traversion
 | 
					        if "srch" in self.uparam or "srch" in body:
 | 
				
			||||||
        if "/" in body["name"]:
 | 
					            return self.handle_search(body)
 | 
				
			||||||
            raise Pebkac(400, "folders verboten")
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # up2k-php compat
 | 
					        # up2k-php compat
 | 
				
			||||||
        for k in "chunkpit.php", "handshake.php":
 | 
					        for k in "chunkpit.php", "handshake.php":
 | 
				
			||||||
            if self.vpath.endswith(k):
 | 
					            if self.vpath.endswith(k):
 | 
				
			||||||
                self.vpath = self.vpath[: -len(k)]
 | 
					                self.vpath = self.vpath[: -len(k)]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        sub = None
 | 
				
			||||||
 | 
					        name = undot(body["name"])
 | 
				
			||||||
 | 
					        if "/" in name:
 | 
				
			||||||
 | 
					            sub, name = name.rsplit("/", 1)
 | 
				
			||||||
 | 
					            self.vpath = "/".join([self.vpath, sub]).strip("/")
 | 
				
			||||||
 | 
					            body["name"] = name
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        vfs, rem = self.conn.auth.vfs.get(self.vpath, self.uname, False, True)
 | 
					        vfs, rem = self.conn.auth.vfs.get(self.vpath, self.uname, False, True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        body["vtop"] = vfs.vpath
 | 
					        body["vtop"] = vfs.vpath
 | 
				
			||||||
        body["ptop"] = vfs.realpath
 | 
					        body["ptop"] = vfs.realpath
 | 
				
			||||||
        body["prel"] = rem
 | 
					        body["prel"] = rem
 | 
				
			||||||
        body["addr"] = self.ip
 | 
					        body["addr"] = self.ip
 | 
				
			||||||
        body["flag"] = vfs.flags
 | 
					        body["vcfg"] = vfs.flags
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if sub:
 | 
				
			||||||
 | 
					            try:
 | 
				
			||||||
 | 
					                dst = os.path.join(vfs.realpath, rem)
 | 
				
			||||||
 | 
					                os.makedirs(dst)
 | 
				
			||||||
 | 
					            except:
 | 
				
			||||||
 | 
					                if not os.path.isdir(dst):
 | 
				
			||||||
 | 
					                    raise Pebkac(400, "some file got your folder name")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        x = self.conn.hsrv.broker.put(True, "up2k.handle_json", body)
 | 
					        x = self.conn.hsrv.broker.put(True, "up2k.handle_json", body)
 | 
				
			||||||
        response = x.get()
 | 
					        ret = x.get()
 | 
				
			||||||
        response = json.dumps(response)
 | 
					        if sub:
 | 
				
			||||||
 | 
					            ret["name"] = "/".join([sub, ret["name"]])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        self.log(response)
 | 
					        ret = json.dumps(ret)
 | 
				
			||||||
        self.reply(response.encode("utf-8"), mime="application/json")
 | 
					        self.log(ret)
 | 
				
			||||||
 | 
					        self.reply(ret.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
 | 
					        return True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def handle_post_binary(self):
 | 
					    def handle_post_binary(self):
 | 
				
			||||||
@@ -444,7 +576,7 @@ class HttpCli(object):
 | 
				
			|||||||
            if len(cstart) > 1 and path != os.devnull:
 | 
					            if len(cstart) > 1 and path != os.devnull:
 | 
				
			||||||
                self.log(
 | 
					                self.log(
 | 
				
			||||||
                    "clone {} to {}".format(
 | 
					                    "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
 | 
					                ofs = 0
 | 
				
			||||||
@@ -461,7 +593,12 @@ class HttpCli(object):
 | 
				
			|||||||
                self.log("clone {} done".format(cstart[0]))
 | 
					                self.log("clone {} done".format(cstart[0]))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        x = self.conn.hsrv.broker.put(True, "up2k.confirm_chunk", ptop, wark, chash)
 | 
					        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:
 | 
					        if not WINDOWS and num_left == 0:
 | 
				
			||||||
            times = (int(time.time()), int(lastmod))
 | 
					            times = (int(time.time()), int(lastmod))
 | 
				
			||||||
@@ -487,7 +624,7 @@ class HttpCli(object):
 | 
				
			|||||||
            pwd = "x"  # nosec
 | 
					            pwd = "x"  # nosec
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        h = {"Set-Cookie": "cppwd={}; Path=/; SameSite=Lax".format(pwd)}
 | 
					        h = {"Set-Cookie": "cppwd={}; Path=/; SameSite=Lax".format(pwd)}
 | 
				
			||||||
        html = self.conn.tpl_msg.render(h1=msg, h2='<a href="/">ack</a>', redir="/")
 | 
					        html = self.j2("msg", h1=msg, h2='<a href="/">ack</a>', redir="/")
 | 
				
			||||||
        self.reply(html.encode("utf-8"), headers=h)
 | 
					        self.reply(html.encode("utf-8"), headers=h)
 | 
				
			||||||
        return True
 | 
					        return True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -517,10 +654,10 @@ class HttpCli(object):
 | 
				
			|||||||
                raise Pebkac(500, "mkdir failed, check the logs")
 | 
					                raise Pebkac(500, "mkdir failed, check the logs")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        vpath = "{}/{}".format(self.vpath, sanitized).lstrip("/")
 | 
					        vpath = "{}/{}".format(self.vpath, sanitized).lstrip("/")
 | 
				
			||||||
        html = self.conn.tpl_msg.render(
 | 
					        esc_paths = [quotep(vpath), html_escape(vpath)]
 | 
				
			||||||
            h2='<a href="/{}">go to /{}</a>'.format(
 | 
					        html = self.j2(
 | 
				
			||||||
                quotep(vpath), html_escape(vpath)
 | 
					            "msg",
 | 
				
			||||||
            ),
 | 
					            h2='<a href="/{}">go to /{}</a>'.format(*esc_paths),
 | 
				
			||||||
            pre="aight",
 | 
					            pre="aight",
 | 
				
			||||||
            click=True,
 | 
					            click=True,
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
@@ -551,7 +688,8 @@ class HttpCli(object):
 | 
				
			|||||||
                f.write(b"`GRUNNUR`\n")
 | 
					                f.write(b"`GRUNNUR`\n")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        vpath = "{}/{}".format(self.vpath, sanitized).lstrip("/")
 | 
					        vpath = "{}/{}".format(self.vpath, sanitized).lstrip("/")
 | 
				
			||||||
        html = self.conn.tpl_msg.render(
 | 
					        html = self.j2(
 | 
				
			||||||
 | 
					            "msg",
 | 
				
			||||||
            h2='<a href="/{}?edit">go to /{}?edit</a>'.format(
 | 
					            h2='<a href="/{}?edit">go to /{}?edit</a>'.format(
 | 
				
			||||||
                quotep(vpath), html_escape(vpath)
 | 
					                quotep(vpath), html_escape(vpath)
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
@@ -598,6 +736,9 @@ class HttpCli(object):
 | 
				
			|||||||
                            raise Pebkac(400, "empty files in post")
 | 
					                            raise Pebkac(400, "empty files in post")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                        files.append([sz, sha512_hex])
 | 
					                        files.append([sz, sha512_hex])
 | 
				
			||||||
 | 
					                        self.conn.hsrv.broker.put(
 | 
				
			||||||
 | 
					                            False, "up2k.hash_file", vfs.realpath, vfs.flags, rem, fname
 | 
				
			||||||
 | 
					                        )
 | 
				
			||||||
                        self.conn.nbyte += sz
 | 
					                        self.conn.nbyte += sz
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                except Pebkac:
 | 
					                except Pebkac:
 | 
				
			||||||
@@ -613,7 +754,7 @@ class HttpCli(object):
 | 
				
			|||||||
                    raise
 | 
					                    raise
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        except Pebkac as ex:
 | 
					        except Pebkac as ex:
 | 
				
			||||||
            errmsg = str(ex)
 | 
					            errmsg = unicode(ex)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        td = max(0.1, time.time() - t0)
 | 
					        td = max(0.1, time.time() - t0)
 | 
				
			||||||
        sz_total = sum(x[0] for x in files)
 | 
					        sz_total = sum(x[0] for x in files)
 | 
				
			||||||
@@ -654,7 +795,8 @@ class HttpCli(object):
 | 
				
			|||||||
                    ).encode("utf-8")
 | 
					                    ).encode("utf-8")
 | 
				
			||||||
                )
 | 
					                )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        html = self.conn.tpl_msg.render(
 | 
					        html = self.j2(
 | 
				
			||||||
 | 
					            "msg",
 | 
				
			||||||
            h2='<a href="/{}">return to /{}</a>'.format(
 | 
					            h2='<a href="/{}">return to /{}</a>'.format(
 | 
				
			||||||
                quotep(self.vpath), html_escape(self.vpath)
 | 
					                quotep(self.vpath), html_escape(self.vpath)
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
@@ -903,8 +1045,11 @@ class HttpCli(object):
 | 
				
			|||||||
            open_func = open
 | 
					            open_func = open
 | 
				
			||||||
            # 512 kB is optimal for huge files, use 64k
 | 
					            # 512 kB is optimal for huge files, use 64k
 | 
				
			||||||
            open_args = [fsenc(fs_path), "rb", 64 * 1024]
 | 
					            open_args = [fsenc(fs_path), "rb", 64 * 1024]
 | 
				
			||||||
            if hasattr(os, "sendfile"):
 | 
					            use_sendfile = (
 | 
				
			||||||
                use_sendfile = not self.args.no_sendfile
 | 
					                not self.tls  #
 | 
				
			||||||
 | 
					                and not self.args.no_sendfile
 | 
				
			||||||
 | 
					                and hasattr(os, "sendfile")
 | 
				
			||||||
 | 
					            )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        #
 | 
					        #
 | 
				
			||||||
        # send reply
 | 
					        # send reply
 | 
				
			||||||
@@ -919,7 +1064,7 @@ class HttpCli(object):
 | 
				
			|||||||
            mime=guess_mime(req_path)[0] or "application/octet-stream",
 | 
					            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:
 | 
					        if self.mode == "HEAD" or not do_send:
 | 
				
			||||||
            self.log(logmsg)
 | 
					            self.log(logmsg)
 | 
				
			||||||
@@ -933,22 +1078,81 @@ class HttpCli(object):
 | 
				
			|||||||
                remains = sendfile_py(lower, upper, f, self.s)
 | 
					                remains = sendfile_py(lower, upper, f, self.s)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if remains > 0:
 | 
					        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)
 | 
					        spd = self._spd((upper - lower) - remains)
 | 
				
			||||||
        self.log("{},  {}".format(logmsg, spd))
 | 
					        self.log("{},  {}".format(logmsg, spd))
 | 
				
			||||||
        return ret
 | 
					        return ret
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def tx_zip(self, fmt, uarg, vn, rem, items, dots):
 | 
				
			||||||
 | 
					        if self.args.no_zip:
 | 
				
			||||||
 | 
					            raise Pebkac(400, "not enabled")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        logmsg = "{:4} {} ".format("", self.req)
 | 
				
			||||||
 | 
					        self.keepalive = False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if not uarg:
 | 
				
			||||||
 | 
					            uarg = ""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if fmt == "tar":
 | 
				
			||||||
 | 
					            mime = "application/x-tar"
 | 
				
			||||||
 | 
					            packer = StreamTar
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            mime = "application/zip"
 | 
				
			||||||
 | 
					            packer = StreamZip
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        fn = items[0] if items and items[0] else self.vpath
 | 
				
			||||||
 | 
					        if fn:
 | 
				
			||||||
 | 
					            fn = fn.rstrip("/").split("/")[-1]
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            fn = self.headers.get("host", "hey")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        afn = "".join(
 | 
				
			||||||
 | 
					            [x if x in (string.ascii_letters + string.digits) else "_" for x in fn]
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        bascii = unicode(string.ascii_letters + string.digits).encode("utf-8")
 | 
				
			||||||
 | 
					        ufn = fn.encode("utf-8", "xmlcharrefreplace")
 | 
				
			||||||
 | 
					        if PY2:
 | 
				
			||||||
 | 
					            ufn = [unicode(x) if x in bascii else "%{:02x}".format(ord(x)) for x in ufn]
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            ufn = [
 | 
				
			||||||
 | 
					                chr(x).encode("utf-8")
 | 
				
			||||||
 | 
					                if x in bascii
 | 
				
			||||||
 | 
					                else "%{:02x}".format(x).encode("ascii")
 | 
				
			||||||
 | 
					                for x in ufn
 | 
				
			||||||
 | 
					            ]
 | 
				
			||||||
 | 
					        ufn = b"".join(ufn).decode("ascii")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        cdis = "attachment; filename=\"{}.{}\"; filename*=UTF-8''{}.{}"
 | 
				
			||||||
 | 
					        cdis = cdis.format(afn, fmt, ufn, fmt)
 | 
				
			||||||
 | 
					        self.send_headers(None, mime=mime, headers={"Content-Disposition": cdis})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        fgen = vn.zipgen(rem, items, self.uname, dots, not self.args.no_scandir)
 | 
				
			||||||
 | 
					        # for f in fgen: print(repr({k: f[k] for k in ["vp", "ap"]}))
 | 
				
			||||||
 | 
					        bgen = packer(fgen, utf8="utf" in uarg, pre_crc="crc" in uarg)
 | 
				
			||||||
 | 
					        bsent = 0
 | 
				
			||||||
 | 
					        for buf in bgen.gen():
 | 
				
			||||||
 | 
					            if not buf:
 | 
				
			||||||
 | 
					                break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            try:
 | 
				
			||||||
 | 
					                self.s.sendall(buf)
 | 
				
			||||||
 | 
					                bsent += len(buf)
 | 
				
			||||||
 | 
					            except:
 | 
				
			||||||
 | 
					                logmsg += " \033[31m" + unicode(bsent) + "\033[0m"
 | 
				
			||||||
 | 
					                break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        spd = self._spd(bsent)
 | 
				
			||||||
 | 
					        self.log("{},  {}".format(logmsg, spd))
 | 
				
			||||||
 | 
					        return True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def tx_md(self, fs_path):
 | 
					    def tx_md(self, fs_path):
 | 
				
			||||||
        logmsg = "{:4} {} ".format("", self.req)
 | 
					        logmsg = "{:4} {} ".format("", self.req)
 | 
				
			||||||
        if "edit2" in self.uparam:
 | 
					 | 
				
			||||||
            html_path = "web/mde.html"
 | 
					 | 
				
			||||||
            template = self.conn.tpl_mde
 | 
					 | 
				
			||||||
        else:
 | 
					 | 
				
			||||||
            html_path = "web/md.html"
 | 
					 | 
				
			||||||
            template = self.conn.tpl_md
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        html_path = os.path.join(E.mod, html_path)
 | 
					        tpl = "mde" if "edit2" in self.uparam else "md"
 | 
				
			||||||
 | 
					        html_path = os.path.join(E.mod, "web", "{}.html".format(tpl))
 | 
				
			||||||
 | 
					        template = self.j2(tpl)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        st = os.stat(fsenc(fs_path))
 | 
					        st = os.stat(fsenc(fs_path))
 | 
				
			||||||
        # sz_md = st.st_size
 | 
					        # sz_md = st.st_size
 | 
				
			||||||
@@ -980,7 +1184,7 @@ class HttpCli(object):
 | 
				
			|||||||
        sz_html = len(template.render(**targs).encode("utf-8"))
 | 
					        sz_html = len(template.render(**targs).encode("utf-8"))
 | 
				
			||||||
        self.send_headers(sz_html + sz_md, status)
 | 
					        self.send_headers(sz_html + sz_md, status)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        logmsg += str(status)
 | 
					        logmsg += unicode(status)
 | 
				
			||||||
        if self.mode == "HEAD" or not do_send:
 | 
					        if self.mode == "HEAD" or not do_send:
 | 
				
			||||||
            self.log(logmsg)
 | 
					            self.log(logmsg)
 | 
				
			||||||
            return True
 | 
					            return True
 | 
				
			||||||
@@ -994,16 +1198,71 @@ class HttpCli(object):
 | 
				
			|||||||
            self.log(logmsg + " \033[31md/c\033[0m")
 | 
					            self.log(logmsg + " \033[31md/c\033[0m")
 | 
				
			||||||
            return False
 | 
					            return False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        self.log(logmsg + " " + str(len(html)))
 | 
					        self.log(logmsg + " " + unicode(len(html)))
 | 
				
			||||||
        return True
 | 
					        return True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def tx_mounts(self):
 | 
					    def tx_mounts(self):
 | 
				
			||||||
        rvol = [x + "/" if x else x for x in self.rvol]
 | 
					        rvol = [x + "/" if x else x for x in self.rvol]
 | 
				
			||||||
        wvol = [x + "/" if x else x for x in self.wvol]
 | 
					        wvol = [x + "/" if x else x for x in self.wvol]
 | 
				
			||||||
        html = self.conn.tpl_mounts.render(this=self, rvol=rvol, wvol=wvol)
 | 
					        html = self.j2("splash", this=self, rvol=rvol, wvol=wvol)
 | 
				
			||||||
        self.reply(html.encode("utf-8"))
 | 
					        self.reply(html.encode("utf-8"))
 | 
				
			||||||
        return True
 | 
					        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):
 | 
					    def tx_browser(self):
 | 
				
			||||||
        vpath = ""
 | 
					        vpath = ""
 | 
				
			||||||
        vpnodes = [["", "/"]]
 | 
					        vpnodes = [["", "/"]]
 | 
				
			||||||
@@ -1029,13 +1288,19 @@ class HttpCli(object):
 | 
				
			|||||||
            if abspath.endswith(".md") and "raw" not in self.uparam:
 | 
					            if abspath.endswith(".md") and "raw" not in self.uparam:
 | 
				
			||||||
                return self.tx_md(abspath)
 | 
					                return self.tx_md(abspath)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            bad = "{0}.hist{0}up2k.".format(os.sep)
 | 
					            if rem.startswith(".hist/up2k."):
 | 
				
			||||||
            if abspath.endswith(bad + "db") or abspath.endswith(bad + "snap"):
 | 
					 | 
				
			||||||
                raise Pebkac(403)
 | 
					                raise Pebkac(403)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            return self.tx_file(abspath)
 | 
					            return self.tx_file(abspath)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        fsroot, vfs_ls, vfs_virt = vn.ls(rem, self.uname)
 | 
					        for k in ["zip", "tar"]:
 | 
				
			||||||
 | 
					            v = self.uparam.get(k)
 | 
				
			||||||
 | 
					            if v is not None:
 | 
				
			||||||
 | 
					                return self.tx_zip(k, v, vn, rem, [], self.args.ed)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        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())
 | 
					        vfs_ls.extend(vfs_virt.keys())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # check for old versions of files,
 | 
					        # check for old versions of files,
 | 
				
			||||||
@@ -1058,30 +1323,46 @@ class HttpCli(object):
 | 
				
			|||||||
        if not self.args.ed or "dots" not in self.uparam:
 | 
					        if not self.args.ed or "dots" not in self.uparam:
 | 
				
			||||||
            vfs_ls = exclude_dotfiles(vfs_ls)
 | 
					            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 = []
 | 
					        dirs = []
 | 
				
			||||||
        files = []
 | 
					        files = []
 | 
				
			||||||
        for fn in vfs_ls:
 | 
					        for fn in vfs_ls:
 | 
				
			||||||
            base = ""
 | 
					            base = ""
 | 
				
			||||||
            href = fn
 | 
					            href = fn
 | 
				
			||||||
            if self.absolute_urls and vpath:
 | 
					            if not is_ls and self.absolute_urls and vpath:
 | 
				
			||||||
                base = "/" + vpath + "/"
 | 
					                base = "/" + vpath + "/"
 | 
				
			||||||
                href = base + fn
 | 
					                href = base + fn
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            if fn in vfs_virt:
 | 
					            if fn in vfs_virt:
 | 
				
			||||||
                fspath = vfs_virt[fn].realpath
 | 
					                fspath = vfs_virt[fn].realpath
 | 
				
			||||||
 | 
					            elif hidden and any(fn.startswith(x) for x in hidden):
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
            else:
 | 
					            else:
 | 
				
			||||||
                fspath = fsroot + "/" + fn
 | 
					                fspath = fsroot + "/" + fn
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            try:
 | 
					            try:
 | 
				
			||||||
                inf = os.stat(fsenc(fspath))
 | 
					                inf = stats.get(fn) or os.stat(fsenc(fspath))
 | 
				
			||||||
            except:
 | 
					            except:
 | 
				
			||||||
                self.log("broken symlink: {}".format(repr(fspath)))
 | 
					                self.log("broken symlink: {}".format(repr(fspath)))
 | 
				
			||||||
                continue
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            is_dir = stat.S_ISDIR(inf.st_mode)
 | 
					            is_dir = stat.S_ISDIR(inf.st_mode)
 | 
				
			||||||
            if is_dir:
 | 
					            if is_dir:
 | 
				
			||||||
                margin = "DIR"
 | 
					 | 
				
			||||||
                href += "/"
 | 
					                href += "/"
 | 
				
			||||||
 | 
					                if self.args.no_zip:
 | 
				
			||||||
 | 
					                    margin = "DIR"
 | 
				
			||||||
 | 
					                else:
 | 
				
			||||||
 | 
					                    margin = '<a href="{}?zip">zip</a>'.format(quotep(href))
 | 
				
			||||||
            elif fn in hist:
 | 
					            elif fn in hist:
 | 
				
			||||||
                margin = '<a href="{}.hist/{}">#{}</a>'.format(
 | 
					                margin = '<a href="{}.hist/{}">#{}</a>'.format(
 | 
				
			||||||
                    base, html_escape(hist[fn][2], quote=True), hist[fn][0]
 | 
					                    base, html_escape(hist[fn][2], quote=True), hist[fn][0]
 | 
				
			||||||
@@ -1098,35 +1379,56 @@ class HttpCli(object):
 | 
				
			|||||||
            except:
 | 
					            except:
 | 
				
			||||||
                ext = "%"
 | 
					                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:
 | 
					            if is_dir:
 | 
				
			||||||
                dirs.append(item)
 | 
					                dirs.append(item)
 | 
				
			||||||
            else:
 | 
					            else:
 | 
				
			||||||
                files.append(item)
 | 
					                files.append(item)
 | 
				
			||||||
 | 
					                item["rd"] = rem
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        logues = [None, None]
 | 
					        taglist = {}
 | 
				
			||||||
        for n, fn in enumerate([".prologue.html", ".epilogue.html"]):
 | 
					        for f in files:
 | 
				
			||||||
            fn = os.path.join(abspath, fn)
 | 
					            fn = f["name"]
 | 
				
			||||||
            if os.path.exists(fsenc(fn)):
 | 
					            rd = f["rd"]
 | 
				
			||||||
                with open(fsenc(fn), "rb") as f:
 | 
					            del f["rd"]
 | 
				
			||||||
                    logues[n] = f.read().decode("utf-8")
 | 
					            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:
 | 
					                tags = {}
 | 
				
			||||||
            # this is a mistake
 | 
					                f["tags"] = tags
 | 
				
			||||||
            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")
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
                    break
 | 
					                if not r:
 | 
				
			||||||
 | 
					                    continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                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.get("mte", "").split(",") if k in taglist]
 | 
				
			||||||
 | 
					            for f in dirs:
 | 
				
			||||||
 | 
					                f["tags"] = {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        srv_info = []
 | 
					        srv_info = []
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            if not self.args.nih:
 | 
					            if not self.args.nih:
 | 
				
			||||||
                srv_info.append(str(socket.gethostname()).split(".")[0])
 | 
					                srv_info.append(unicode(socket.gethostname()).split(".")[0])
 | 
				
			||||||
        except:
 | 
					        except:
 | 
				
			||||||
            self.log("#wow #whoa")
 | 
					            self.log("#wow #whoa")
 | 
				
			||||||
            pass
 | 
					            pass
 | 
				
			||||||
@@ -1150,21 +1452,57 @@ class HttpCli(object):
 | 
				
			|||||||
        except:
 | 
					        except:
 | 
				
			||||||
            pass
 | 
					            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 = ""
 | 
				
			||||||
        # ts = "?{}".format(time.time())
 | 
					        # ts = "?{}".format(time.time())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        dirs.extend(files)
 | 
					        dirs.extend(files)
 | 
				
			||||||
        html = self.conn.tpl_browser.render(
 | 
					
 | 
				
			||||||
 | 
					        html = self.j2(
 | 
				
			||||||
 | 
					            "browser",
 | 
				
			||||||
            vdir=quotep(self.vpath),
 | 
					            vdir=quotep(self.vpath),
 | 
				
			||||||
            vpnodes=vpnodes,
 | 
					            vpnodes=vpnodes,
 | 
				
			||||||
            files=dirs,
 | 
					            files=dirs,
 | 
				
			||||||
            can_upload=self.writable,
 | 
					 | 
				
			||||||
            can_read=self.readable,
 | 
					 | 
				
			||||||
            ts=ts,
 | 
					            ts=ts,
 | 
				
			||||||
            prologue=logues[0],
 | 
					            perms=json.dumps(perms),
 | 
				
			||||||
            epilogue=logues[1],
 | 
					            taglist=taglist,
 | 
				
			||||||
 | 
					            tag_order=json.dumps(
 | 
				
			||||||
 | 
					                vn.flags["mte"].split(",") if "mte" in vn.flags else []
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
 | 
					            have_up2k_idx=("e2d" in vn.flags),
 | 
				
			||||||
 | 
					            have_tags_idx=("e2t" in vn.flags),
 | 
				
			||||||
 | 
					            have_zip=(not self.args.no_zip),
 | 
				
			||||||
 | 
					            logues=logues,
 | 
				
			||||||
            title=html_escape(self.vpath),
 | 
					            title=html_escape(self.vpath),
 | 
				
			||||||
            srv_info="</span> /// <span>".join(srv_info),
 | 
					            srv_info=srv_info,
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        self.reply(html.encode("utf-8", "replace"))
 | 
					        self.reply(html.encode("utf-8", "replace"))
 | 
				
			||||||
        return True
 | 
					        return True
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -3,28 +3,19 @@ from __future__ import print_function, unicode_literals
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
import os
 | 
					import os
 | 
				
			||||||
import sys
 | 
					import sys
 | 
				
			||||||
import ssl
 | 
					 | 
				
			||||||
import time
 | 
					import time
 | 
				
			||||||
import socket
 | 
					import socket
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					HAVE_SSL = True
 | 
				
			||||||
try:
 | 
					try:
 | 
				
			||||||
    import jinja2
 | 
					    import ssl
 | 
				
			||||||
except ImportError:
 | 
					except:
 | 
				
			||||||
    print(
 | 
					    HAVE_SSL = False
 | 
				
			||||||
        """\033[1;31m
 | 
					 | 
				
			||||||
  you do not have jinja2 installed,\033[33m
 | 
					 | 
				
			||||||
  choose one of these:\033[0m
 | 
					 | 
				
			||||||
   * apt install python-jinja2
 | 
					 | 
				
			||||||
   * python3 -m pip install --user jinja2
 | 
					 | 
				
			||||||
   * (try another python version, if you have one)
 | 
					 | 
				
			||||||
   * (try copyparty.sfx instead)
 | 
					 | 
				
			||||||
"""
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    sys.exit(1)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
from .__init__ import E
 | 
					from .__init__ import E
 | 
				
			||||||
from .util import Unrecv
 | 
					from .util import Unrecv
 | 
				
			||||||
from .httpcli import HttpCli
 | 
					from .httpcli import HttpCli
 | 
				
			||||||
 | 
					from .u2idx import U2idx
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class HttpConn(object):
 | 
					class HttpConn(object):
 | 
				
			||||||
@@ -45,17 +36,10 @@ class HttpConn(object):
 | 
				
			|||||||
        self.t0 = time.time()
 | 
					        self.t0 = time.time()
 | 
				
			||||||
        self.nbyte = 0
 | 
					        self.nbyte = 0
 | 
				
			||||||
        self.workload = 0
 | 
					        self.workload = 0
 | 
				
			||||||
 | 
					        self.u2idx = None
 | 
				
			||||||
        self.log_func = hsrv.log
 | 
					        self.log_func = hsrv.log
 | 
				
			||||||
        self.set_rproxy()
 | 
					        self.set_rproxy()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        env = jinja2.Environment()
 | 
					 | 
				
			||||||
        env.loader = jinja2.FileSystemLoader(os.path.join(E.mod, "web"))
 | 
					 | 
				
			||||||
        self.tpl_mounts = env.get_template("splash.html")
 | 
					 | 
				
			||||||
        self.tpl_browser = env.get_template("browser.html")
 | 
					 | 
				
			||||||
        self.tpl_msg = env.get_template("msg.html")
 | 
					 | 
				
			||||||
        self.tpl_md = env.get_template("md.html")
 | 
					 | 
				
			||||||
        self.tpl_mde = env.get_template("mde.html")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def set_rproxy(self, ip=None):
 | 
					    def set_rproxy(self, ip=None):
 | 
				
			||||||
        if ip is None:
 | 
					        if ip is None:
 | 
				
			||||||
            color = 36
 | 
					            color = 36
 | 
				
			||||||
@@ -72,12 +56,17 @@ class HttpConn(object):
 | 
				
			|||||||
    def respath(self, res_name):
 | 
					    def respath(self, res_name):
 | 
				
			||||||
        return os.path.join(E.mod, "web", res_name)
 | 
					        return os.path.join(E.mod, "web", res_name)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def log(self, msg):
 | 
					    def log(self, msg, c=0):
 | 
				
			||||||
        self.log_func(self.log_src, msg)
 | 
					        self.log_func(self.log_src, msg, c)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def 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
 | 
					        method = None
 | 
				
			||||||
        self.sr = None
 | 
					 | 
				
			||||||
        if self.cert_path:
 | 
					        if self.cert_path:
 | 
				
			||||||
            try:
 | 
					            try:
 | 
				
			||||||
                method = self.s.recv(4, socket.MSG_PEEK)
 | 
					                method = self.s.recv(4, socket.MSG_PEEK)
 | 
				
			||||||
@@ -98,20 +87,64 @@ class HttpConn(object):
 | 
				
			|||||||
                err = "need at least 4 bytes in the first packet; got {}".format(
 | 
					                err = "need at least 4 bytes in the first packet; got {}".format(
 | 
				
			||||||
                    len(method)
 | 
					                    len(method)
 | 
				
			||||||
                )
 | 
					                )
 | 
				
			||||||
 | 
					                if method:
 | 
				
			||||||
                    self.log(err)
 | 
					                    self.log(err)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8"))
 | 
					                self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8"))
 | 
				
			||||||
                return
 | 
					                return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        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:
 | 
					            if self.sr:
 | 
				
			||||||
                self.log("\033[1;31mTODO: cannot do https in jython\033[0m")
 | 
					                self.log("TODO: cannot do https in jython", c="1;31")
 | 
				
			||||||
                return
 | 
					                return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            self.log_src = self.log_src.replace("[36m", "[35m")
 | 
					            self.log_src = self.log_src.replace("[36m", "[35m")
 | 
				
			||||||
            try:
 | 
					            try:
 | 
				
			||||||
                self.s = ssl.wrap_socket(
 | 
					                ctx = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH)
 | 
				
			||||||
                    self.s, server_side=True, certfile=self.cert_path
 | 
					                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:
 | 
					            except Exception as ex:
 | 
				
			||||||
                em = str(ex)
 | 
					                em = str(ex)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -124,7 +157,7 @@ class HttpConn(object):
 | 
				
			|||||||
                    pass
 | 
					                    pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                else:
 | 
					                else:
 | 
				
			||||||
                    self.log("\033[35mhandshake\033[0m " + em)
 | 
					                    self.log("handshake\033[0m " + em, c=5)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                return
 | 
					                return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -2,10 +2,28 @@
 | 
				
			|||||||
from __future__ import print_function, unicode_literals
 | 
					from __future__ import print_function, unicode_literals
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import os
 | 
					import os
 | 
				
			||||||
 | 
					import sys
 | 
				
			||||||
import time
 | 
					import time
 | 
				
			||||||
import socket
 | 
					import socket
 | 
				
			||||||
import threading
 | 
					import threading
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					try:
 | 
				
			||||||
 | 
					    import jinja2
 | 
				
			||||||
 | 
					except ImportError:
 | 
				
			||||||
 | 
					    print(
 | 
				
			||||||
 | 
					        """\033[1;31m
 | 
				
			||||||
 | 
					  you do not have jinja2 installed,\033[33m
 | 
				
			||||||
 | 
					  choose one of these:\033[0m
 | 
				
			||||||
 | 
					   * apt install python-jinja2
 | 
				
			||||||
 | 
					   * {} -m pip install --user jinja2
 | 
				
			||||||
 | 
					   * (try another python version, if you have one)
 | 
				
			||||||
 | 
					   * (try copyparty.sfx instead)
 | 
				
			||||||
 | 
					""".format(
 | 
				
			||||||
 | 
					            os.path.basename(sys.executable)
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					    sys.exit(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from .__init__ import E, MACOS
 | 
					from .__init__ import E, MACOS
 | 
				
			||||||
from .httpconn import HttpConn
 | 
					from .httpconn import HttpConn
 | 
				
			||||||
from .authsrv import AuthSrv
 | 
					from .authsrv import AuthSrv
 | 
				
			||||||
@@ -30,6 +48,13 @@ class HttpSrv(object):
 | 
				
			|||||||
        self.workload_thr_alive = False
 | 
					        self.workload_thr_alive = False
 | 
				
			||||||
        self.auth = AuthSrv(self.args, self.log)
 | 
					        self.auth = AuthSrv(self.args, self.log)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        env = jinja2.Environment()
 | 
				
			||||||
 | 
					        env.loader = jinja2.FileSystemLoader(os.path.join(E.mod, "web"))
 | 
				
			||||||
 | 
					        self.j2 = {
 | 
				
			||||||
 | 
					            x: env.get_template(x + ".html")
 | 
				
			||||||
 | 
					            for x in ["splash", "browser", "msg", "md", "mde"]
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        cert_path = os.path.join(E.cfg, "cert.pem")
 | 
					        cert_path = os.path.join(E.cfg, "cert.pem")
 | 
				
			||||||
        if os.path.exists(cert_path):
 | 
					        if os.path.exists(cert_path):
 | 
				
			||||||
            self.cert_path = cert_path
 | 
					            self.cert_path = cert_path
 | 
				
			||||||
@@ -38,7 +63,9 @@ class HttpSrv(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    def accept(self, sck, addr):
 | 
					    def accept(self, sck, addr):
 | 
				
			||||||
        """takes an incoming tcp connection and creates a thread to handle it"""
 | 
					        """takes an incoming tcp connection and creates a thread to handle it"""
 | 
				
			||||||
        self.log("%s %s" % addr, "\033[1;30m|%sC-cthr\033[0m" % ("-" * 5,))
 | 
					        if self.args.log_conn:
 | 
				
			||||||
 | 
					            self.log("%s %s" % addr, "|%sC-cthr" % ("-" * 5,), c="1;30")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        thr = threading.Thread(target=self.thr_client, args=(sck, addr))
 | 
					        thr = threading.Thread(target=self.thr_client, args=(sck, addr))
 | 
				
			||||||
        thr.daemon = True
 | 
					        thr.daemon = True
 | 
				
			||||||
        thr.start()
 | 
					        thr.start()
 | 
				
			||||||
@@ -66,11 +93,15 @@ class HttpSrv(object):
 | 
				
			|||||||
                thr.start()
 | 
					                thr.start()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            self.log("%s %s" % addr, "\033[1;30m|%sC-crun\033[0m" % ("-" * 6,))
 | 
					            if self.args.log_conn:
 | 
				
			||||||
 | 
					                self.log("%s %s" % addr, "|%sC-crun" % ("-" * 6,), c="1;30")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            cli.run()
 | 
					            cli.run()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        finally:
 | 
					        finally:
 | 
				
			||||||
            self.log("%s %s" % addr, "\033[1;30m|%sC-cdone\033[0m" % ("-" * 7,))
 | 
					            if self.args.log_conn:
 | 
				
			||||||
 | 
					                self.log("%s %s" % addr, "|%sC-cdone" % ("-" * 7,), c="1;30")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            try:
 | 
					            try:
 | 
				
			||||||
                sck.shutdown(socket.SHUT_RDWR)
 | 
					                sck.shutdown(socket.SHUT_RDWR)
 | 
				
			||||||
                sck.close()
 | 
					                sck.close()
 | 
				
			||||||
@@ -78,7 +109,8 @@ class HttpSrv(object):
 | 
				
			|||||||
                if not MACOS:
 | 
					                if not MACOS:
 | 
				
			||||||
                    self.log(
 | 
					                    self.log(
 | 
				
			||||||
                        "%s %s" % addr,
 | 
					                        "%s %s" % addr,
 | 
				
			||||||
                        "shut_rdwr err:\n  {}\n  {}".format(repr(sck), ex),
 | 
					                        "shut({}): {}".format(sck.fileno(), ex),
 | 
				
			||||||
 | 
					                        c="1;30",
 | 
				
			||||||
                    )
 | 
					                    )
 | 
				
			||||||
                if ex.errno not in [10038, 10054, 107, 57, 9]:
 | 
					                if ex.errno not in [10038, 10054, 107, 57, 9]:
 | 
				
			||||||
                    # 10038 No longer considered a socket
 | 
					                    # 10038 No longer considered a socket
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										347
									
								
								copyparty/mtag.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										347
									
								
								copyparty/mtag.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,347 @@
 | 
				
			|||||||
 | 
					# coding: utf-8
 | 
				
			||||||
 | 
					from __future__ import print_function, unicode_literals
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import re
 | 
				
			||||||
 | 
					import os
 | 
				
			||||||
 | 
					import sys
 | 
				
			||||||
 | 
					import shutil
 | 
				
			||||||
 | 
					import subprocess as sp
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from .__init__ import PY2, WINDOWS
 | 
				
			||||||
 | 
					from .util import fsenc, fsdec, REKOBO_LKEY
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if not PY2:
 | 
				
			||||||
 | 
					    unicode = str
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class MTag(object):
 | 
				
			||||||
 | 
					    def __init__(self, log_func, args):
 | 
				
			||||||
 | 
					        self.log_func = log_func
 | 
				
			||||||
 | 
					        self.usable = True
 | 
				
			||||||
 | 
					        self.prefer_mt = False
 | 
				
			||||||
 | 
					        mappings = args.mtm
 | 
				
			||||||
 | 
					        self.backend = "ffprobe" if args.no_mutagen else "mutagen"
 | 
				
			||||||
 | 
					        or_ffprobe = " or ffprobe"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if self.backend == "mutagen":
 | 
				
			||||||
 | 
					            self.get = self.get_mutagen
 | 
				
			||||||
 | 
					            try:
 | 
				
			||||||
 | 
					                import mutagen
 | 
				
			||||||
 | 
					            except:
 | 
				
			||||||
 | 
					                self.log("could not load mutagen, trying ffprobe instead", c=3)
 | 
				
			||||||
 | 
					                self.backend = "ffprobe"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if self.backend == "ffprobe":
 | 
				
			||||||
 | 
					            self.get = self.get_ffprobe
 | 
				
			||||||
 | 
					            self.prefer_mt = True
 | 
				
			||||||
 | 
					            # about 20x slower
 | 
				
			||||||
 | 
					            if PY2:
 | 
				
			||||||
 | 
					                cmd = [b"ffprobe", b"-version"]
 | 
				
			||||||
 | 
					                try:
 | 
				
			||||||
 | 
					                    sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE)
 | 
				
			||||||
 | 
					                except:
 | 
				
			||||||
 | 
					                    self.usable = False
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                if not shutil.which("ffprobe"):
 | 
				
			||||||
 | 
					                    self.usable = False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if self.usable and WINDOWS and sys.version_info < (3, 8):
 | 
				
			||||||
 | 
					                self.usable = False
 | 
				
			||||||
 | 
					                or_ffprobe = " or python >= 3.8"
 | 
				
			||||||
 | 
					                msg = "found ffprobe but your python is too old; need 3.8 or newer"
 | 
				
			||||||
 | 
					                self.log(msg, c=1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if not self.usable:
 | 
				
			||||||
 | 
					            msg = "need mutagen{} to read media tags so please run this:\n  {} -m pip install --user mutagen"
 | 
				
			||||||
 | 
					            self.log(msg.format(or_ffprobe, os.path.basename(sys.executable)), c=1)
 | 
				
			||||||
 | 
					            return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # https://picard-docs.musicbrainz.org/downloads/MusicBrainz_Picard_Tag_Map.html
 | 
				
			||||||
 | 
					        tagmap = {
 | 
				
			||||||
 | 
					            "album": ["album", "talb", "\u00a9alb", "original-album", "toal"],
 | 
				
			||||||
 | 
					            "artist": [
 | 
				
			||||||
 | 
					                "artist",
 | 
				
			||||||
 | 
					                "tpe1",
 | 
				
			||||||
 | 
					                "\u00a9art",
 | 
				
			||||||
 | 
					                "composer",
 | 
				
			||||||
 | 
					                "performer",
 | 
				
			||||||
 | 
					                "arranger",
 | 
				
			||||||
 | 
					                "\u00a9wrt",
 | 
				
			||||||
 | 
					                "tcom",
 | 
				
			||||||
 | 
					                "tpe3",
 | 
				
			||||||
 | 
					                "original-artist",
 | 
				
			||||||
 | 
					                "tope",
 | 
				
			||||||
 | 
					            ],
 | 
				
			||||||
 | 
					            "title": ["title", "tit2", "\u00a9nam"],
 | 
				
			||||||
 | 
					            "circle": [
 | 
				
			||||||
 | 
					                "album-artist",
 | 
				
			||||||
 | 
					                "tpe2",
 | 
				
			||||||
 | 
					                "aart",
 | 
				
			||||||
 | 
					                "conductor",
 | 
				
			||||||
 | 
					                "organization",
 | 
				
			||||||
 | 
					                "band",
 | 
				
			||||||
 | 
					            ],
 | 
				
			||||||
 | 
					            ".tn": ["tracknumber", "trck", "trkn", "track"],
 | 
				
			||||||
 | 
					            "genre": ["genre", "tcon", "\u00a9gen"],
 | 
				
			||||||
 | 
					            "date": [
 | 
				
			||||||
 | 
					                "original-release-date",
 | 
				
			||||||
 | 
					                "release-date",
 | 
				
			||||||
 | 
					                "date",
 | 
				
			||||||
 | 
					                "tdrc",
 | 
				
			||||||
 | 
					                "\u00a9day",
 | 
				
			||||||
 | 
					                "original-date",
 | 
				
			||||||
 | 
					                "original-year",
 | 
				
			||||||
 | 
					                "tyer",
 | 
				
			||||||
 | 
					                "tdor",
 | 
				
			||||||
 | 
					                "tory",
 | 
				
			||||||
 | 
					                "year",
 | 
				
			||||||
 | 
					                "creation-time",
 | 
				
			||||||
 | 
					            ],
 | 
				
			||||||
 | 
					            ".bpm": ["bpm", "tbpm", "tmpo", "tbp"],
 | 
				
			||||||
 | 
					            "key": ["initial-key", "tkey", "key"],
 | 
				
			||||||
 | 
					            "comment": ["comment", "comm", "\u00a9cmt", "comments", "description"],
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if mappings:
 | 
				
			||||||
 | 
					            for k, v in [x.split("=") for x in mappings]:
 | 
				
			||||||
 | 
					                tagmap[k] = v.split(",")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.tagmap = {}
 | 
				
			||||||
 | 
					        for k, vs in tagmap.items():
 | 
				
			||||||
 | 
					            vs2 = []
 | 
				
			||||||
 | 
					            for v in vs:
 | 
				
			||||||
 | 
					                if "-" not in v:
 | 
				
			||||||
 | 
					                    vs2.append(v)
 | 
				
			||||||
 | 
					                    continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                vs2.append(v.replace("-", " "))
 | 
				
			||||||
 | 
					                vs2.append(v.replace("-", "_"))
 | 
				
			||||||
 | 
					                vs2.append(v.replace("-", ""))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            self.tagmap[k] = vs2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.rmap = {
 | 
				
			||||||
 | 
					            v: [n, k] for k, vs in self.tagmap.items() for n, v in enumerate(vs)
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        # self.get = self.compare
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def log(self, msg, c=0):
 | 
				
			||||||
 | 
					        self.log_func("mtag", msg, c)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def normalize_tags(self, ret, md):
 | 
				
			||||||
 | 
					        for k, v in dict(md).items():
 | 
				
			||||||
 | 
					            if not v:
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            k = k.lower().split("::")[0].strip()
 | 
				
			||||||
 | 
					            mk = self.rmap.get(k)
 | 
				
			||||||
 | 
					            if not mk:
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            pref, mk = mk
 | 
				
			||||||
 | 
					            if mk not in ret or ret[mk][0] > pref:
 | 
				
			||||||
 | 
					                ret[mk] = [pref, v[0]]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # take first value
 | 
				
			||||||
 | 
					        ret = {k: unicode(v[1]).strip() for k, v in ret.items()}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # track 3/7 => track 3
 | 
				
			||||||
 | 
					        for k, v in ret.items():
 | 
				
			||||||
 | 
					            if k[0] == ".":
 | 
				
			||||||
 | 
					                v = v.split("/")[0].strip().lstrip("0")
 | 
				
			||||||
 | 
					                ret[k] = v or 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # normalize key notation to rkeobo
 | 
				
			||||||
 | 
					        okey = ret.get("key")
 | 
				
			||||||
 | 
					        if okey:
 | 
				
			||||||
 | 
					            key = okey.replace(" ", "").replace("maj", "").replace("min", "m")
 | 
				
			||||||
 | 
					            ret["key"] = REKOBO_LKEY.get(key.lower(), okey)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return ret
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def compare(self, abspath):
 | 
				
			||||||
 | 
					        if abspath.endswith(".au"):
 | 
				
			||||||
 | 
					            return {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        print("\n" + abspath)
 | 
				
			||||||
 | 
					        r1 = self.get_mutagen(abspath)
 | 
				
			||||||
 | 
					        r2 = self.get_ffprobe(abspath)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        keys = {}
 | 
				
			||||||
 | 
					        for d in [r1, r2]:
 | 
				
			||||||
 | 
					            for k in d.keys():
 | 
				
			||||||
 | 
					                keys[k] = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        diffs = []
 | 
				
			||||||
 | 
					        l1 = []
 | 
				
			||||||
 | 
					        l2 = []
 | 
				
			||||||
 | 
					        for k in sorted(keys.keys()):
 | 
				
			||||||
 | 
					            if k in [".q", ".dur"]:
 | 
				
			||||||
 | 
					                continue  # lenient
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            v1 = r1.get(k)
 | 
				
			||||||
 | 
					            v2 = r2.get(k)
 | 
				
			||||||
 | 
					            if v1 == v2:
 | 
				
			||||||
 | 
					                print("  ", k, v1)
 | 
				
			||||||
 | 
					            elif v1 != "0000":  # ffprobe date=0
 | 
				
			||||||
 | 
					                diffs.append(k)
 | 
				
			||||||
 | 
					                print(" 1", k, v1)
 | 
				
			||||||
 | 
					                print(" 2", k, v2)
 | 
				
			||||||
 | 
					                if v1:
 | 
				
			||||||
 | 
					                    l1.append(k)
 | 
				
			||||||
 | 
					                if v2:
 | 
				
			||||||
 | 
					                    l2.append(k)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if diffs:
 | 
				
			||||||
 | 
					            raise Exception()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return r1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def get_mutagen(self, abspath):
 | 
				
			||||||
 | 
					        import mutagen
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            md = mutagen.File(abspath, easy=True)
 | 
				
			||||||
 | 
					            x = md.info.length
 | 
				
			||||||
 | 
					        except Exception as ex:
 | 
				
			||||||
 | 
					            return {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ret = {}
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            dur = int(md.info.length)
 | 
				
			||||||
 | 
					            try:
 | 
				
			||||||
 | 
					                q = int(md.info.bitrate / 1024)
 | 
				
			||||||
 | 
					            except:
 | 
				
			||||||
 | 
					                q = int((os.path.getsize(abspath) / dur) / 128)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            ret[".dur"] = [0, dur]
 | 
				
			||||||
 | 
					            ret[".q"] = [0, q]
 | 
				
			||||||
 | 
					        except:
 | 
				
			||||||
 | 
					            pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return self.normalize_tags(ret, md)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def get_ffprobe(self, abspath):
 | 
				
			||||||
 | 
					        cmd = [b"ffprobe", b"-hide_banner", b"--", fsenc(abspath)]
 | 
				
			||||||
 | 
					        p = sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE)
 | 
				
			||||||
 | 
					        r = p.communicate()
 | 
				
			||||||
 | 
					        txt = r[1].decode("utf-8", "replace")
 | 
				
			||||||
 | 
					        txt = [x.rstrip("\r") for x in txt.split("\n")]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        """
 | 
				
			||||||
 | 
					        note:
 | 
				
			||||||
 | 
					          tags which contain newline will be truncated on first \n,
 | 
				
			||||||
 | 
					          ffprobe emits \n and spacepads the : to align visually
 | 
				
			||||||
 | 
					        note:
 | 
				
			||||||
 | 
					          the Stream ln always mentions Audio: if audio
 | 
				
			||||||
 | 
					          the Stream ln usually has kb/s, is more accurate
 | 
				
			||||||
 | 
					          the Duration ln always has kb/s
 | 
				
			||||||
 | 
					          the Metadata: after Chapter may contain BPM info,
 | 
				
			||||||
 | 
					            title : Tempo: 126.0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Input #0, wav,
 | 
				
			||||||
 | 
					          Metadata:
 | 
				
			||||||
 | 
					            date : <OK>
 | 
				
			||||||
 | 
					          Duration:
 | 
				
			||||||
 | 
					            Chapter #
 | 
				
			||||||
 | 
					            Metadata:
 | 
				
			||||||
 | 
					              title : <NG>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Input #0, mp3,
 | 
				
			||||||
 | 
					          Metadata:
 | 
				
			||||||
 | 
					            album : <OK>
 | 
				
			||||||
 | 
					          Duration:
 | 
				
			||||||
 | 
					            Stream #0:0: Audio:
 | 
				
			||||||
 | 
					            Stream #0:1: Video:
 | 
				
			||||||
 | 
					            Metadata:
 | 
				
			||||||
 | 
					              comment : <NG>
 | 
				
			||||||
 | 
					        """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ptn_md_beg = re.compile("^( +)Metadata:$")
 | 
				
			||||||
 | 
					        ptn_md_kv = re.compile("^( +)([^:]+) *: (.*)")
 | 
				
			||||||
 | 
					        ptn_dur = re.compile("^ *Duration: ([^ ]+)(, |$)")
 | 
				
			||||||
 | 
					        ptn_br1 = re.compile("^ *Duration: .*, bitrate: ([0-9]+) kb/s(, |$)")
 | 
				
			||||||
 | 
					        ptn_br2 = re.compile("^ *Stream.*: Audio:.* ([0-9]+) kb/s(, |$)")
 | 
				
			||||||
 | 
					        ptn_audio = re.compile("^ *Stream .*: Audio: ")
 | 
				
			||||||
 | 
					        ptn_au_parent = re.compile("^ *(Input #|Stream .*: Audio: )")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ret = {}
 | 
				
			||||||
 | 
					        md = {}
 | 
				
			||||||
 | 
					        in_md = False
 | 
				
			||||||
 | 
					        is_audio = False
 | 
				
			||||||
 | 
					        au_parent = False
 | 
				
			||||||
 | 
					        for ln in txt:
 | 
				
			||||||
 | 
					            m = ptn_md_kv.match(ln)
 | 
				
			||||||
 | 
					            if m and in_md and len(m.group(1)) == in_md:
 | 
				
			||||||
 | 
					                _, k, v = [x.strip() for x in m.groups()]
 | 
				
			||||||
 | 
					                if k != "" and v != "":
 | 
				
			||||||
 | 
					                    md[k] = [v]
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                in_md = False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            m = ptn_md_beg.match(ln)
 | 
				
			||||||
 | 
					            if m and au_parent:
 | 
				
			||||||
 | 
					                in_md = len(m.group(1)) + 2
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            au_parent = bool(ptn_au_parent.search(ln))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if ptn_audio.search(ln):
 | 
				
			||||||
 | 
					                is_audio = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            m = ptn_dur.search(ln)
 | 
				
			||||||
 | 
					            if m:
 | 
				
			||||||
 | 
					                sec = 0
 | 
				
			||||||
 | 
					                tstr = m.group(1)
 | 
				
			||||||
 | 
					                if tstr.lower() != "n/a":
 | 
				
			||||||
 | 
					                    try:
 | 
				
			||||||
 | 
					                        tf = tstr.split(",")[0].split(".")[0].split(":")
 | 
				
			||||||
 | 
					                        for f in tf:
 | 
				
			||||||
 | 
					                            sec *= 60
 | 
				
			||||||
 | 
					                            sec += int(f)
 | 
				
			||||||
 | 
					                    except:
 | 
				
			||||||
 | 
					                        self.log("invalid timestr from ffprobe: [{}]".format(tstr), c=3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                ret[".dur"] = sec
 | 
				
			||||||
 | 
					                m = ptn_br1.search(ln)
 | 
				
			||||||
 | 
					                if m:
 | 
				
			||||||
 | 
					                    ret[".q"] = m.group(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            m = ptn_br2.search(ln)
 | 
				
			||||||
 | 
					            if m:
 | 
				
			||||||
 | 
					                ret[".q"] = m.group(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if not is_audio:
 | 
				
			||||||
 | 
					            return {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ret = {k: [0, v] for k, v in ret.items()}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return self.normalize_tags(ret, md)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def get_bin(self, parsers, abspath):
 | 
				
			||||||
 | 
					        pypath = os.path.abspath(os.path.dirname(os.path.dirname(__file__)))
 | 
				
			||||||
 | 
					        pypath = [str(pypath)] + [str(x) for x in sys.path if x]
 | 
				
			||||||
 | 
					        pypath = str(os.pathsep.join(pypath))
 | 
				
			||||||
 | 
					        env = os.environ.copy()
 | 
				
			||||||
 | 
					        env["PYTHONPATH"] = pypath
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ret = {}
 | 
				
			||||||
 | 
					        for tagname, (binpath, timeout) in parsers.items():
 | 
				
			||||||
 | 
					            try:
 | 
				
			||||||
 | 
					                cmd = [sys.executable, binpath, abspath]
 | 
				
			||||||
 | 
					                args = {"env": env, "timeout": timeout}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if WINDOWS:
 | 
				
			||||||
 | 
					                    args["creationflags"] = 0x4000
 | 
				
			||||||
 | 
					                else:
 | 
				
			||||||
 | 
					                    cmd = ["nice"] + cmd
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                cmd = [fsenc(x) for x in cmd]
 | 
				
			||||||
 | 
					                v = sp.check_output(cmd, **args).strip()
 | 
				
			||||||
 | 
					                if v:
 | 
				
			||||||
 | 
					                    ret[tagname] = v.decode("utf-8")
 | 
				
			||||||
 | 
					            except:
 | 
				
			||||||
 | 
					                pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return ret
 | 
				
			||||||
							
								
								
									
										95
									
								
								copyparty/star.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								copyparty/star.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,95 @@
 | 
				
			|||||||
 | 
					import os
 | 
				
			||||||
 | 
					import tarfile
 | 
				
			||||||
 | 
					import threading
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from .sutil import errdesc
 | 
				
			||||||
 | 
					from .util import Queue, fsenc
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class QFile(object):
 | 
				
			||||||
 | 
					    """file-like object which buffers writes into a queue"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self):
 | 
				
			||||||
 | 
					        self.q = Queue(64)
 | 
				
			||||||
 | 
					        self.bq = []
 | 
				
			||||||
 | 
					        self.nq = 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def write(self, buf):
 | 
				
			||||||
 | 
					        if buf is None or self.nq >= 240 * 1024:
 | 
				
			||||||
 | 
					            self.q.put(b"".join(self.bq))
 | 
				
			||||||
 | 
					            self.bq = []
 | 
				
			||||||
 | 
					            self.nq = 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if buf is None:
 | 
				
			||||||
 | 
					            self.q.put(None)
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            self.bq.append(buf)
 | 
				
			||||||
 | 
					            self.nq += len(buf)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class StreamTar(object):
 | 
				
			||||||
 | 
					    """construct in-memory tar file from the given path"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self, fgen, **kwargs):
 | 
				
			||||||
 | 
					        self.ci = 0
 | 
				
			||||||
 | 
					        self.co = 0
 | 
				
			||||||
 | 
					        self.qfile = QFile()
 | 
				
			||||||
 | 
					        self.fgen = fgen
 | 
				
			||||||
 | 
					        self.errf = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # python 3.8 changed to PAX_FORMAT as default,
 | 
				
			||||||
 | 
					        # waste of space and don't care about the new features
 | 
				
			||||||
 | 
					        fmt = tarfile.GNU_FORMAT
 | 
				
			||||||
 | 
					        self.tar = tarfile.open(fileobj=self.qfile, mode="w|", format=fmt)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        w = threading.Thread(target=self._gen)
 | 
				
			||||||
 | 
					        w.daemon = True
 | 
				
			||||||
 | 
					        w.start()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def gen(self):
 | 
				
			||||||
 | 
					        while True:
 | 
				
			||||||
 | 
					            buf = self.qfile.q.get()
 | 
				
			||||||
 | 
					            if not buf:
 | 
				
			||||||
 | 
					                break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            self.co += len(buf)
 | 
				
			||||||
 | 
					            yield buf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        yield None
 | 
				
			||||||
 | 
					        if self.errf:
 | 
				
			||||||
 | 
					            os.unlink(self.errf["ap"])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def ser(self, f):
 | 
				
			||||||
 | 
					        name = f["vp"]
 | 
				
			||||||
 | 
					        src = f["ap"]
 | 
				
			||||||
 | 
					        fsi = f["st"]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        inf = tarfile.TarInfo(name=name)
 | 
				
			||||||
 | 
					        inf.mode = fsi.st_mode
 | 
				
			||||||
 | 
					        inf.size = fsi.st_size
 | 
				
			||||||
 | 
					        inf.mtime = fsi.st_mtime
 | 
				
			||||||
 | 
					        inf.uid = 0
 | 
				
			||||||
 | 
					        inf.gid = 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.ci += inf.size
 | 
				
			||||||
 | 
					        with open(fsenc(src), "rb", 512 * 1024) as f:
 | 
				
			||||||
 | 
					            self.tar.addfile(inf, f)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def _gen(self):
 | 
				
			||||||
 | 
					        errors = []
 | 
				
			||||||
 | 
					        for f in self.fgen:
 | 
				
			||||||
 | 
					            if "err" in f:
 | 
				
			||||||
 | 
					                errors.append([f["vp"], f["err"]])
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            try:
 | 
				
			||||||
 | 
					                self.ser(f)
 | 
				
			||||||
 | 
					            except Exception as ex:
 | 
				
			||||||
 | 
					                errors.append([f["vp"], repr(ex)])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if errors:
 | 
				
			||||||
 | 
					            self.errf = errdesc(errors)
 | 
				
			||||||
 | 
					            self.ser(self.errf)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.tar.close()
 | 
				
			||||||
 | 
					        self.qfile.write(None)
 | 
				
			||||||
							
								
								
									
										25
									
								
								copyparty/sutil.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								copyparty/sutil.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,25 @@
 | 
				
			|||||||
 | 
					import os
 | 
				
			||||||
 | 
					import time
 | 
				
			||||||
 | 
					import tempfile
 | 
				
			||||||
 | 
					from datetime import datetime
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def errdesc(errors):
 | 
				
			||||||
 | 
					    report = ["copyparty failed to add the following files to the archive:", ""]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for fn, err in errors:
 | 
				
			||||||
 | 
					        report.extend([" file: {}".format(fn), "error: {}".format(err), ""])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    with tempfile.NamedTemporaryFile(prefix="copyparty-", delete=False) as tf:
 | 
				
			||||||
 | 
					        tf_path = tf.name
 | 
				
			||||||
 | 
					        tf.write("\r\n".join(report).encode("utf-8", "replace"))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    dt = datetime.utcfromtimestamp(time.time())
 | 
				
			||||||
 | 
					    dt = dt.strftime("%Y-%m%d-%H%M%S")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    os.chmod(tf_path, 0o444)
 | 
				
			||||||
 | 
					    return {
 | 
				
			||||||
 | 
					        "vp": "archive-errors-{}.txt".format(dt),
 | 
				
			||||||
 | 
					        "ap": tf_path,
 | 
				
			||||||
 | 
					        "st": os.stat(tf_path),
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
@@ -9,7 +9,6 @@ from datetime import datetime, timedelta
 | 
				
			|||||||
import calendar
 | 
					import calendar
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from .__init__ import PY2, WINDOWS, MACOS, VT100
 | 
					from .__init__ import PY2, WINDOWS, MACOS, VT100
 | 
				
			||||||
from .authsrv import AuthSrv
 | 
					 | 
				
			||||||
from .tcpsrv import TcpSrv
 | 
					from .tcpsrv import TcpSrv
 | 
				
			||||||
from .up2k import Up2k
 | 
					from .up2k import Up2k
 | 
				
			||||||
from .util import mp
 | 
					from .util import mp
 | 
				
			||||||
@@ -39,10 +38,6 @@ class SvcHub(object):
 | 
				
			|||||||
        self.tcpsrv = TcpSrv(self)
 | 
					        self.tcpsrv = TcpSrv(self)
 | 
				
			||||||
        self.up2k = Up2k(self)
 | 
					        self.up2k = Up2k(self)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if self.args.e2d and self.args.e2s:
 | 
					 | 
				
			||||||
            auth = AuthSrv(self.args, self.log, False)
 | 
					 | 
				
			||||||
            self.up2k.build_indexes(auth.all_writable)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        # decide which worker impl to use
 | 
					        # decide which worker impl to use
 | 
				
			||||||
        if self.check_mp_enable():
 | 
					        if self.check_mp_enable():
 | 
				
			||||||
            from .broker_mp import BrokerMp as Broker
 | 
					            from .broker_mp import BrokerMp as Broker
 | 
				
			||||||
@@ -70,16 +65,16 @@ class SvcHub(object):
 | 
				
			|||||||
            self.broker.shutdown()
 | 
					            self.broker.shutdown()
 | 
				
			||||||
            print("nailed it")
 | 
					            print("nailed it")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def _log_disabled(self, src, msg):
 | 
					    def _log_disabled(self, src, msg, c=0):
 | 
				
			||||||
        pass
 | 
					        pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def _log_enabled(self, src, msg):
 | 
					    def _log_enabled(self, src, msg, c=0):
 | 
				
			||||||
        """handles logging from all components"""
 | 
					        """handles logging from all components"""
 | 
				
			||||||
        with self.log_mutex:
 | 
					        with self.log_mutex:
 | 
				
			||||||
            now = time.time()
 | 
					            now = time.time()
 | 
				
			||||||
            if now >= self.next_day:
 | 
					            if now >= self.next_day:
 | 
				
			||||||
                dt = datetime.utcfromtimestamp(now)
 | 
					                dt = datetime.utcfromtimestamp(now)
 | 
				
			||||||
                print("\033[36m{}\033[0m".format(dt.strftime("%Y-%m-%d")))
 | 
					                print("\033[36m{}\033[0m\n".format(dt.strftime("%Y-%m-%d")), end="")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                # unix timestamp of next 00:00:00 (leap-seconds safe)
 | 
					                # unix timestamp of next 00:00:00 (leap-seconds safe)
 | 
				
			||||||
                day_now = dt.day
 | 
					                day_now = dt.day
 | 
				
			||||||
@@ -89,23 +84,30 @@ class SvcHub(object):
 | 
				
			|||||||
                dt = dt.replace(hour=0, minute=0, second=0)
 | 
					                dt = dt.replace(hour=0, minute=0, second=0)
 | 
				
			||||||
                self.next_day = calendar.timegm(dt.utctimetuple())
 | 
					                self.next_day = calendar.timegm(dt.utctimetuple())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            fmt = "\033[36m{} \033[33m{:21} \033[0m{}"
 | 
					            fmt = "\033[36m{} \033[33m{:21} \033[0m{}\n"
 | 
				
			||||||
            if not VT100:
 | 
					            if not VT100:
 | 
				
			||||||
                fmt = "{} {:21} {}"
 | 
					                fmt = "{} {:21} {}\n"
 | 
				
			||||||
                if "\033" in msg:
 | 
					                if "\033" in msg:
 | 
				
			||||||
                    msg = self.ansi_re.sub("", msg)
 | 
					                    msg = self.ansi_re.sub("", msg)
 | 
				
			||||||
                if "\033" in src:
 | 
					                if "\033" in src:
 | 
				
			||||||
                    src = self.ansi_re.sub("", src)
 | 
					                    src = self.ansi_re.sub("", src)
 | 
				
			||||||
 | 
					            elif c:
 | 
				
			||||||
 | 
					                if isinstance(c, int):
 | 
				
			||||||
 | 
					                    msg = "\033[3{}m{}".format(c, msg)
 | 
				
			||||||
 | 
					                elif "\033" not in c:
 | 
				
			||||||
 | 
					                    msg = "\033[{}m{}\033[0m".format(c, msg)
 | 
				
			||||||
 | 
					                else:
 | 
				
			||||||
 | 
					                    msg = "{}{}\033[0m".format(c, msg)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            ts = datetime.utcfromtimestamp(now).strftime("%H:%M:%S.%f")[:-3]
 | 
					            ts = datetime.utcfromtimestamp(now).strftime("%H:%M:%S.%f")[:-3]
 | 
				
			||||||
            msg = fmt.format(ts, src, msg)
 | 
					            msg = fmt.format(ts, src, msg)
 | 
				
			||||||
            try:
 | 
					            try:
 | 
				
			||||||
                print(msg)
 | 
					                print(msg, end="")
 | 
				
			||||||
            except UnicodeEncodeError:
 | 
					            except UnicodeEncodeError:
 | 
				
			||||||
                try:
 | 
					                try:
 | 
				
			||||||
                    print(msg.encode("utf-8", "replace").decode())
 | 
					                    print(msg.encode("utf-8", "replace").decode(), end="")
 | 
				
			||||||
                except:
 | 
					                except:
 | 
				
			||||||
                    print(msg.encode("ascii", "replace").decode())
 | 
					                    print(msg.encode("ascii", "replace").decode(), end="")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def check_mp_support(self):
 | 
					    def check_mp_support(self):
 | 
				
			||||||
        vmin = sys.version_info[1]
 | 
					        vmin = sys.version_info[1]
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										271
									
								
								copyparty/szip.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										271
									
								
								copyparty/szip.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,271 @@
 | 
				
			|||||||
 | 
					import os
 | 
				
			||||||
 | 
					import time
 | 
				
			||||||
 | 
					import zlib
 | 
				
			||||||
 | 
					import struct
 | 
				
			||||||
 | 
					from datetime import datetime
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from .sutil import errdesc
 | 
				
			||||||
 | 
					from .util import yieldfile, sanitize_fn
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def dostime2unix(buf):
 | 
				
			||||||
 | 
					    t, d = struct.unpack("<HH", buf)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ts = (t & 0x1F) * 2
 | 
				
			||||||
 | 
					    tm = (t >> 5) & 0x3F
 | 
				
			||||||
 | 
					    th = t >> 11
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    dd = d & 0x1F
 | 
				
			||||||
 | 
					    dm = (d >> 5) & 0xF
 | 
				
			||||||
 | 
					    dy = (d >> 9) + 1980
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    tt = (dy, dm, dd, th, tm, ts)
 | 
				
			||||||
 | 
					    tf = "{:04d}-{:02d}-{:02d} {:02d}:{:02d}:{:02d}"
 | 
				
			||||||
 | 
					    iso = tf.format(*tt)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    dt = datetime.strptime(iso, "%Y-%m-%d %H:%M:%S")
 | 
				
			||||||
 | 
					    return int(dt.timestamp())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def unixtime2dos(ts):
 | 
				
			||||||
 | 
					    tt = time.gmtime(ts)
 | 
				
			||||||
 | 
					    dy, dm, dd, th, tm, ts = list(tt)[:6]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    bd = ((dy - 1980) << 9) + (dm << 5) + dd
 | 
				
			||||||
 | 
					    bt = (th << 11) + (tm << 5) + ts // 2
 | 
				
			||||||
 | 
					    return struct.pack("<HH", bt, bd)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def gen_fdesc(sz, crc32, z64):
 | 
				
			||||||
 | 
					    ret = b"\x50\x4b\x07\x08"
 | 
				
			||||||
 | 
					    fmt = "<LQQ" if z64 else "<LLL"
 | 
				
			||||||
 | 
					    ret += struct.pack(fmt, crc32, sz, sz)
 | 
				
			||||||
 | 
					    return ret
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def gen_hdr(h_pos, fn, sz, lastmod, utf8, crc32, pre_crc):
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					    does regular file headers
 | 
				
			||||||
 | 
					    and the central directory meme if h_pos is set
 | 
				
			||||||
 | 
					    (h_pos = absolute position of the regular header)
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # appnote 4.5 / zip 3.0 (2008) / unzip 6.0 (2009) says to add z64
 | 
				
			||||||
 | 
					    # extinfo for values which exceed H, but that becomes an off-by-one
 | 
				
			||||||
 | 
					    # (can't tell if it was clamped or exactly maxval), make it obvious
 | 
				
			||||||
 | 
					    z64 = sz >= 0xFFFFFFFF
 | 
				
			||||||
 | 
					    z64v = [sz, sz] if z64 else []
 | 
				
			||||||
 | 
					    if h_pos and h_pos >= 0xFFFFFFFF:
 | 
				
			||||||
 | 
					        # central, also consider ptr to original header
 | 
				
			||||||
 | 
					        z64v.append(h_pos)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # confusingly this doesn't bump if h_pos
 | 
				
			||||||
 | 
					    req_ver = b"\x2d\x00" if z64 else b"\x0a\x00"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if crc32:
 | 
				
			||||||
 | 
					        crc32 = struct.pack("<L", crc32)
 | 
				
			||||||
 | 
					    else:
 | 
				
			||||||
 | 
					        crc32 = b"\x00" * 4
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if h_pos is None:
 | 
				
			||||||
 | 
					        # 4b magic, 2b min-ver
 | 
				
			||||||
 | 
					        ret = b"\x50\x4b\x03\x04" + req_ver
 | 
				
			||||||
 | 
					    else:
 | 
				
			||||||
 | 
					        # 4b magic, 2b spec-ver, 2b min-ver
 | 
				
			||||||
 | 
					        ret = b"\x50\x4b\x01\x02\x1e\x03" + req_ver
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ret += b"\x00" if pre_crc else b"\x08"  # streaming
 | 
				
			||||||
 | 
					    ret += b"\x08" if utf8 else b"\x00"  # appnote 6.3.2 (2007)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 2b compression, 4b time, 4b crc
 | 
				
			||||||
 | 
					    ret += b"\x00\x00" + unixtime2dos(lastmod) + crc32
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # spec says to put zeros when !crc if bit3 (streaming)
 | 
				
			||||||
 | 
					    # however infozip does actual sz and it even works on winxp
 | 
				
			||||||
 | 
					    # (same reasning for z64 extradata later)
 | 
				
			||||||
 | 
					    vsz = 0xFFFFFFFF if z64 else sz
 | 
				
			||||||
 | 
					    ret += struct.pack("<LL", vsz, vsz)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # windows support (the "?" replace below too)
 | 
				
			||||||
 | 
					    fn = sanitize_fn(fn, "/")
 | 
				
			||||||
 | 
					    bfn = fn.encode("utf-8" if utf8 else "cp437", "replace").replace(b"?", b"_")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    z64_len = len(z64v) * 8 + 4 if z64v else 0
 | 
				
			||||||
 | 
					    ret += struct.pack("<HH", len(bfn), z64_len)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if h_pos is not None:
 | 
				
			||||||
 | 
					        # 2b comment, 2b diskno
 | 
				
			||||||
 | 
					        ret += b"\x00" * 4
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # 2b internal.attr, 4b external.attr
 | 
				
			||||||
 | 
					        # infozip-macos: 0100 0000 a481 file:644
 | 
				
			||||||
 | 
					        # infozip-macos: 0100 0100 0080 file:000
 | 
				
			||||||
 | 
					        ret += b"\x01\x00\x00\x00\xa4\x81"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # 4b local-header-ofs
 | 
				
			||||||
 | 
					        ret += struct.pack("<L", min(h_pos, 0xFFFFFFFF))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ret += bfn
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if z64v:
 | 
				
			||||||
 | 
					        ret += struct.pack("<HH" + "Q" * len(z64v), 1, len(z64v) * 8, *z64v)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return ret
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def gen_ecdr(items, cdir_pos, cdir_end):
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					    summary of all file headers,
 | 
				
			||||||
 | 
					    usually the zipfile footer unless something clamps
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ret = b"\x50\x4b\x05\x06"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 2b ndisk, 2b disk0
 | 
				
			||||||
 | 
					    ret += b"\x00" * 4
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    cdir_sz = cdir_end - cdir_pos
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    nitems = min(0xFFFF, len(items))
 | 
				
			||||||
 | 
					    csz = min(0xFFFFFFFF, cdir_sz)
 | 
				
			||||||
 | 
					    cpos = min(0xFFFFFFFF, cdir_pos)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    need_64 = nitems == 0xFFFF or 0xFFFFFFFF in [csz, cpos]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 2b tnfiles, 2b dnfiles, 4b dir sz, 4b dir pos
 | 
				
			||||||
 | 
					    ret += struct.pack("<HHLL", nitems, nitems, csz, cpos)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 2b comment length
 | 
				
			||||||
 | 
					    ret += b"\x00\x00"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return [ret, need_64]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def gen_ecdr64(items, cdir_pos, cdir_end):
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					    z64 end of central directory
 | 
				
			||||||
 | 
					    added when numfiles or a headerptr clamps
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ret = b"\x50\x4b\x06\x06"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 8b own length from hereon
 | 
				
			||||||
 | 
					    ret += b"\x2c" + b"\x00" * 7
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 2b spec-ver, 2b min-ver
 | 
				
			||||||
 | 
					    ret += b"\x1e\x03\x2d\x00"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 4b ndisk, 4b disk0
 | 
				
			||||||
 | 
					    ret += b"\x00" * 8
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 8b tnfiles, 8b dnfiles, 8b dir sz, 8b dir pos
 | 
				
			||||||
 | 
					    cdir_sz = cdir_end - cdir_pos
 | 
				
			||||||
 | 
					    ret += struct.pack("<QQQQ", len(items), len(items), cdir_sz, cdir_pos)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return ret
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def gen_ecdr64_loc(ecdr64_pos):
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					    z64 end of central directory locator
 | 
				
			||||||
 | 
					    points to ecdr64
 | 
				
			||||||
 | 
					    why
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ret = b"\x50\x4b\x06\x07"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 4b cdisk, 8b start of ecdr64, 4b ndisks
 | 
				
			||||||
 | 
					    ret += struct.pack("<LQL", 0, ecdr64_pos, 1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return ret
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class StreamZip(object):
 | 
				
			||||||
 | 
					    def __init__(self, fgen, utf8=False, pre_crc=False):
 | 
				
			||||||
 | 
					        self.fgen = fgen
 | 
				
			||||||
 | 
					        self.utf8 = utf8
 | 
				
			||||||
 | 
					        self.pre_crc = pre_crc
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.pos = 0
 | 
				
			||||||
 | 
					        self.items = []
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def _ct(self, buf):
 | 
				
			||||||
 | 
					        self.pos += len(buf)
 | 
				
			||||||
 | 
					        return buf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def ser(self, f):
 | 
				
			||||||
 | 
					        name = f["vp"]
 | 
				
			||||||
 | 
					        src = f["ap"]
 | 
				
			||||||
 | 
					        st = f["st"]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        sz = st.st_size
 | 
				
			||||||
 | 
					        ts = st.st_mtime + 1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        crc = None
 | 
				
			||||||
 | 
					        if self.pre_crc:
 | 
				
			||||||
 | 
					            crc = 0
 | 
				
			||||||
 | 
					            for buf in yieldfile(src):
 | 
				
			||||||
 | 
					                crc = zlib.crc32(buf, crc)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            crc &= 0xFFFFFFFF
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        h_pos = self.pos
 | 
				
			||||||
 | 
					        buf = gen_hdr(None, name, sz, ts, self.utf8, crc, self.pre_crc)
 | 
				
			||||||
 | 
					        yield self._ct(buf)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        crc = crc or 0
 | 
				
			||||||
 | 
					        for buf in yieldfile(src):
 | 
				
			||||||
 | 
					            if not self.pre_crc:
 | 
				
			||||||
 | 
					                crc = zlib.crc32(buf, crc)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            yield self._ct(buf)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        crc &= 0xFFFFFFFF
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.items.append([name, sz, ts, crc, h_pos])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        z64 = sz >= 4 * 1024 * 1024 * 1024
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if z64 or not self.pre_crc:
 | 
				
			||||||
 | 
					            buf = gen_fdesc(sz, crc, z64)
 | 
				
			||||||
 | 
					            yield self._ct(buf)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def gen(self):
 | 
				
			||||||
 | 
					        errors = []
 | 
				
			||||||
 | 
					        for f in self.fgen:
 | 
				
			||||||
 | 
					            if "err" in f:
 | 
				
			||||||
 | 
					                errors.append([f["vp"], f["err"]])
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            try:
 | 
				
			||||||
 | 
					                for x in self.ser(f):
 | 
				
			||||||
 | 
					                    yield x
 | 
				
			||||||
 | 
					            except Exception as ex:
 | 
				
			||||||
 | 
					                errors.append([f["vp"], repr(ex)])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if errors:
 | 
				
			||||||
 | 
					            errf = errdesc(errors)
 | 
				
			||||||
 | 
					            print(repr(errf))
 | 
				
			||||||
 | 
					            for x in self.ser(errf):
 | 
				
			||||||
 | 
					                yield x
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        cdir_pos = self.pos
 | 
				
			||||||
 | 
					        for name, sz, ts, crc, h_pos in self.items:
 | 
				
			||||||
 | 
					            buf = gen_hdr(h_pos, name, sz, ts, self.utf8, crc, self.pre_crc)
 | 
				
			||||||
 | 
					            yield self._ct(buf)
 | 
				
			||||||
 | 
					        cdir_end = self.pos
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        _, need_64 = gen_ecdr(self.items, cdir_pos, cdir_end)
 | 
				
			||||||
 | 
					        if need_64:
 | 
				
			||||||
 | 
					            ecdir64_pos = self.pos
 | 
				
			||||||
 | 
					            buf = gen_ecdr64(self.items, cdir_pos, cdir_end)
 | 
				
			||||||
 | 
					            yield self._ct(buf)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            buf = gen_ecdr64_loc(ecdir64_pos)
 | 
				
			||||||
 | 
					            yield self._ct(buf)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ecdr, _ = gen_ecdr(self.items, cdir_pos, cdir_end)
 | 
				
			||||||
 | 
					        yield self._ct(ecdr)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if errors:
 | 
				
			||||||
 | 
					            os.unlink(errf["ap"])
 | 
				
			||||||
@@ -53,15 +53,13 @@ class TcpSrv(object):
 | 
				
			|||||||
            srv.bind((ip, port))
 | 
					            srv.bind((ip, port))
 | 
				
			||||||
            return srv
 | 
					            return srv
 | 
				
			||||||
        except (OSError, socket.error) as ex:
 | 
					        except (OSError, socket.error) as ex:
 | 
				
			||||||
            if ex.errno == 98:
 | 
					            if ex.errno in [98, 48]:
 | 
				
			||||||
                raise Exception(
 | 
					                e = "\033[1;31mport {} is busy on interface {}\033[0m".format(port, ip)
 | 
				
			||||||
                    "\033[1;31mport {} is busy on interface {}\033[0m".format(port, ip)
 | 
					            elif ex.errno in [99, 49]:
 | 
				
			||||||
                )
 | 
					                e = "\033[1;31minterface {} does not exist\033[0m".format(ip)
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
            if ex.errno == 99:
 | 
					                raise
 | 
				
			||||||
                raise Exception(
 | 
					            raise Exception(e)
 | 
				
			||||||
                    "\033[1;31minterface {} does not exist\033[0m".format(ip)
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def run(self):
 | 
					    def run(self):
 | 
				
			||||||
        for srv in self.srv:
 | 
					        for srv in self.srv:
 | 
				
			||||||
@@ -70,22 +68,29 @@ class TcpSrv(object):
 | 
				
			|||||||
            self.log("tcpsrv", "listening @ {0}:{1}".format(ip, port))
 | 
					            self.log("tcpsrv", "listening @ {0}:{1}".format(ip, port))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        while True:
 | 
					        while True:
 | 
				
			||||||
            self.log("tcpsrv", "\033[1;30m|%sC-ncli\033[0m" % ("-" * 1,))
 | 
					            if self.args.log_conn:
 | 
				
			||||||
 | 
					                self.log("tcpsrv", "|%sC-ncli" % ("-" * 1,), c="1;30")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            if self.num_clients.v >= self.args.nc:
 | 
					            if self.num_clients.v >= self.args.nc:
 | 
				
			||||||
                time.sleep(0.1)
 | 
					                time.sleep(0.1)
 | 
				
			||||||
                continue
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            self.log("tcpsrv", "\033[1;30m|%sC-acc1\033[0m" % ("-" * 2,))
 | 
					            if self.args.log_conn:
 | 
				
			||||||
 | 
					                self.log("tcpsrv", "|%sC-acc1" % ("-" * 2,), c="1;30")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            ready, _, _ = select.select(self.srv, [], [])
 | 
					            ready, _, _ = select.select(self.srv, [], [])
 | 
				
			||||||
            for srv in ready:
 | 
					            for srv in ready:
 | 
				
			||||||
                sck, addr = srv.accept()
 | 
					                sck, addr = srv.accept()
 | 
				
			||||||
                sip, sport = srv.getsockname()
 | 
					                sip, sport = srv.getsockname()
 | 
				
			||||||
 | 
					                if self.args.log_conn:
 | 
				
			||||||
                    self.log(
 | 
					                    self.log(
 | 
				
			||||||
                        "%s %s" % addr,
 | 
					                        "%s %s" % addr,
 | 
				
			||||||
                    "\033[1;30m|{}C-acc2 \033[0;36m{} \033[3{}m{}".format(
 | 
					                        "|{}C-acc2 \033[0;36m{} \033[3{}m{}".format(
 | 
				
			||||||
                            "-" * 3, sip, sport % 8, sport
 | 
					                            "-" * 3, sip, sport % 8, sport
 | 
				
			||||||
                        ),
 | 
					                        ),
 | 
				
			||||||
 | 
					                        c="1;30",
 | 
				
			||||||
                    )
 | 
					                    )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                self.num_clients.add()
 | 
					                self.num_clients.add()
 | 
				
			||||||
                self.hub.broker.put(False, "httpconn", sck, addr)
 | 
					                self.hub.broker.put(False, "httpconn", sck, addr)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										281
									
								
								copyparty/u2idx.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										281
									
								
								copyparty/u2idx.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,281 @@
 | 
				
			|||||||
 | 
					# coding: utf-8
 | 
				
			||||||
 | 
					from __future__ import print_function, unicode_literals
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import re
 | 
				
			||||||
 | 
					import os
 | 
				
			||||||
 | 
					import time
 | 
				
			||||||
 | 
					import threading
 | 
				
			||||||
 | 
					from datetime import datetime
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from .util import u8safe, s3dec, html_escape, Pebkac
 | 
				
			||||||
 | 
					from .up2k import up2k_wark_from_hashlist
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					try:
 | 
				
			||||||
 | 
					    HAVE_SQLITE3 = True
 | 
				
			||||||
 | 
					    import sqlite3
 | 
				
			||||||
 | 
					except:
 | 
				
			||||||
 | 
					    HAVE_SQLITE3 = False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class U2idx(object):
 | 
				
			||||||
 | 
					    def __init__(self, args, log_func):
 | 
				
			||||||
 | 
					        self.args = args
 | 
				
			||||||
 | 
					        self.log_func = log_func
 | 
				
			||||||
 | 
					        self.timeout = args.srch_time
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if not HAVE_SQLITE3:
 | 
				
			||||||
 | 
					            self.log("could not load sqlite3; searchign wqill be disabled")
 | 
				
			||||||
 | 
					            return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.cur = {}
 | 
				
			||||||
 | 
					        self.mem_cur = sqlite3.connect(":memory:")
 | 
				
			||||||
 | 
					        self.mem_cur.execute(r"create table a (b text)")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.p_end = None
 | 
				
			||||||
 | 
					        self.p_dur = 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def log(self, msg, c=0):
 | 
				
			||||||
 | 
					        self.log_func("u2idx", msg, c)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def fsearch(self, vols, body):
 | 
				
			||||||
 | 
					        """search by up2k hashlist"""
 | 
				
			||||||
 | 
					        if not HAVE_SQLITE3:
 | 
				
			||||||
 | 
					            return []
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        fsize = body["size"]
 | 
				
			||||||
 | 
					        fhash = body["hash"]
 | 
				
			||||||
 | 
					        wark = up2k_wark_from_hashlist(self.args.salt, fsize, fhash)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        uq = "substr(w,1,16) = ? and w = ?"
 | 
				
			||||||
 | 
					        uv = [wark[:16], wark]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            return self.run_query(vols, uq, uv, {})[0]
 | 
				
			||||||
 | 
					        except Exception as ex:
 | 
				
			||||||
 | 
					            raise Pebkac(500, repr(ex))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def get_cur(self, ptop):
 | 
				
			||||||
 | 
					        cur = self.cur.get(ptop)
 | 
				
			||||||
 | 
					        if cur:
 | 
				
			||||||
 | 
					            return cur
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        cur = _open(ptop)
 | 
				
			||||||
 | 
					        if not cur:
 | 
				
			||||||
 | 
					            return None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.cur[ptop] = cur
 | 
				
			||||||
 | 
					        return cur
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def search(self, vols, body):
 | 
				
			||||||
 | 
					        """search by query params"""
 | 
				
			||||||
 | 
					        if not HAVE_SQLITE3:
 | 
				
			||||||
 | 
					            return []
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        qobj = {}
 | 
				
			||||||
 | 
					        _conv_sz(qobj, body, "sz_min", "up.sz >= ?")
 | 
				
			||||||
 | 
					        _conv_sz(qobj, body, "sz_max", "up.sz <= ?")
 | 
				
			||||||
 | 
					        _conv_dt(qobj, body, "dt_min", "up.mt >= ?")
 | 
				
			||||||
 | 
					        _conv_dt(qobj, body, "dt_max", "up.mt <= ?")
 | 
				
			||||||
 | 
					        for seg, dk in [["path", "up.rd"], ["name", "up.fn"]]:
 | 
				
			||||||
 | 
					            if seg in body:
 | 
				
			||||||
 | 
					                _conv_txt(qobj, body, seg, dk)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        uq, uv = _sqlize(qobj)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        qobj = {}
 | 
				
			||||||
 | 
					        if "tags" in body:
 | 
				
			||||||
 | 
					            _conv_txt(qobj, body, "tags", "mt.v")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if "adv" in body:
 | 
				
			||||||
 | 
					            _conv_adv(qobj, body, "adv")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            return self.run_query(vols, uq, uv, qobj)
 | 
				
			||||||
 | 
					        except Exception as ex:
 | 
				
			||||||
 | 
					            raise Pebkac(500, repr(ex))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def run_query(self, vols, uq, uv, targs):
 | 
				
			||||||
 | 
					        self.log("qs: {} {} ,  {}".format(uq, repr(uv), repr(targs)))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        done_flag = []
 | 
				
			||||||
 | 
					        self.active_id = "{:.6f}_{}".format(
 | 
				
			||||||
 | 
					            time.time(), threading.current_thread().ident
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					        thr = threading.Thread(
 | 
				
			||||||
 | 
					            target=self.terminator,
 | 
				
			||||||
 | 
					            args=(
 | 
				
			||||||
 | 
					                self.active_id,
 | 
				
			||||||
 | 
					                done_flag,
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					        thr.daemon = True
 | 
				
			||||||
 | 
					        thr.start()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if not targs:
 | 
				
			||||||
 | 
					            if not uq:
 | 
				
			||||||
 | 
					                q = "select * from up"
 | 
				
			||||||
 | 
					                v = ()
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                q = "select * from up where " + uq
 | 
				
			||||||
 | 
					                v = tuple(uv)
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            q = "select up.* from up"
 | 
				
			||||||
 | 
					            keycmp = "substr(up.w,1,16)"
 | 
				
			||||||
 | 
					            where = []
 | 
				
			||||||
 | 
					            v = []
 | 
				
			||||||
 | 
					            ctr = 0
 | 
				
			||||||
 | 
					            for tq, tv in sorted(targs.items()):
 | 
				
			||||||
 | 
					                ctr += 1
 | 
				
			||||||
 | 
					                tq = tq.split("\n")[0]
 | 
				
			||||||
 | 
					                keycmp2 = "mt{}.w".format(ctr)
 | 
				
			||||||
 | 
					                q += " inner join mt mt{} on {} = {}".format(ctr, keycmp, keycmp2)
 | 
				
			||||||
 | 
					                keycmp = keycmp2
 | 
				
			||||||
 | 
					                where.append(tq.replace("mt.", keycmp[:-1]))
 | 
				
			||||||
 | 
					                v.append(tv)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if uq:
 | 
				
			||||||
 | 
					                where.append(uq)
 | 
				
			||||||
 | 
					                v.extend(uv)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            q += " where " + (" and ".join(where))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # self.log("q2: {} {}".format(q, repr(v)))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ret = []
 | 
				
			||||||
 | 
					        lim = 1000
 | 
				
			||||||
 | 
					        taglist = {}
 | 
				
			||||||
 | 
					        for (vtop, ptop, flags) in vols:
 | 
				
			||||||
 | 
					            cur = self.get_cur(ptop)
 | 
				
			||||||
 | 
					            if not cur:
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            self.active_cur = cur
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            sret = []
 | 
				
			||||||
 | 
					            c = cur.execute(q, v)
 | 
				
			||||||
 | 
					            for hit in c:
 | 
				
			||||||
 | 
					                w, ts, sz, rd, fn = hit
 | 
				
			||||||
 | 
					                lim -= 1
 | 
				
			||||||
 | 
					                if lim <= 0:
 | 
				
			||||||
 | 
					                    break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if rd.startswith("//") or fn.startswith("//"):
 | 
				
			||||||
 | 
					                    rd, fn = s3dec(rd, fn)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                rp = os.path.join(vtop, rd, fn).replace("\\", "/")
 | 
				
			||||||
 | 
					                sret.append({"ts": int(ts), "sz": sz, "rp": rp, "w": w[:16]})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            for hit in sret:
 | 
				
			||||||
 | 
					                w = hit["w"]
 | 
				
			||||||
 | 
					                del hit["w"]
 | 
				
			||||||
 | 
					                tags = {}
 | 
				
			||||||
 | 
					                q2 = "select k, v from mt where w = ? and k != 'x'"
 | 
				
			||||||
 | 
					                for k, v2 in cur.execute(q2, (w,)):
 | 
				
			||||||
 | 
					                    taglist[k] = True
 | 
				
			||||||
 | 
					                    tags[k] = v2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                hit["tags"] = tags
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            ret.extend(sret)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        done_flag.append(True)
 | 
				
			||||||
 | 
					        self.active_id = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # undupe hits from multiple metadata keys
 | 
				
			||||||
 | 
					        if len(ret) > 1:
 | 
				
			||||||
 | 
					            ret = [ret[0]] + [
 | 
				
			||||||
 | 
					                y for x, y in zip(ret[:-1], ret[1:]) if x["rp"] != y["rp"]
 | 
				
			||||||
 | 
					            ]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return ret, list(taglist.keys())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def terminator(self, identifier, done_flag):
 | 
				
			||||||
 | 
					        for _ in range(self.timeout):
 | 
				
			||||||
 | 
					            time.sleep(1)
 | 
				
			||||||
 | 
					            if done_flag:
 | 
				
			||||||
 | 
					                return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if identifier == self.active_id:
 | 
				
			||||||
 | 
					            self.active_cur.connection.interrupt()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def _open(ptop):
 | 
				
			||||||
 | 
					    db_path = os.path.join(ptop, ".hist", "up2k.db")
 | 
				
			||||||
 | 
					    if os.path.exists(db_path):
 | 
				
			||||||
 | 
					        return sqlite3.connect(db_path).cursor()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def _conv_sz(q, body, k, sql):
 | 
				
			||||||
 | 
					    if k in body:
 | 
				
			||||||
 | 
					        q[sql] = int(float(body[k]) * 1024 * 1024)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def _conv_dt(q, body, k, sql):
 | 
				
			||||||
 | 
					    if k not in body:
 | 
				
			||||||
 | 
					        return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    v = body[k].upper().rstrip("Z").replace(",", " ").replace("T", " ")
 | 
				
			||||||
 | 
					    while "  " in v:
 | 
				
			||||||
 | 
					        v = v.replace("  ", " ")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for fmt in ["%Y-%m-%d %H:%M:%S", "%Y-%m-%d %H:%M", "%Y-%m-%d %H", "%Y-%m-%d"]:
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            ts = datetime.strptime(v, fmt).timestamp()
 | 
				
			||||||
 | 
					            break
 | 
				
			||||||
 | 
					        except:
 | 
				
			||||||
 | 
					            ts = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if ts:
 | 
				
			||||||
 | 
					        q[sql] = ts
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def _conv_txt(q, body, k, sql):
 | 
				
			||||||
 | 
					    for v in body[k].split(" "):
 | 
				
			||||||
 | 
					        inv = ""
 | 
				
			||||||
 | 
					        if v.startswith("-"):
 | 
				
			||||||
 | 
					            inv = "not"
 | 
				
			||||||
 | 
					            v = v[1:]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if not v:
 | 
				
			||||||
 | 
					            continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        head = "'%'||"
 | 
				
			||||||
 | 
					        if v.startswith("^"):
 | 
				
			||||||
 | 
					            head = ""
 | 
				
			||||||
 | 
					            v = v[1:]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        tail = "||'%'"
 | 
				
			||||||
 | 
					        if v.endswith("$"):
 | 
				
			||||||
 | 
					            tail = ""
 | 
				
			||||||
 | 
					            v = v[:-1]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        qk = "{} {} like {}?{}".format(sql, inv, head, tail)
 | 
				
			||||||
 | 
					        q[qk + "\n" + v] = u8safe(v)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def _conv_adv(q, body, k):
 | 
				
			||||||
 | 
					    ptn = re.compile(r"^(\.?[a-z]+) *(==?|!=|<=?|>=?) *(.*)$")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    parts = body[k].split(" ")
 | 
				
			||||||
 | 
					    parts = [x.strip() for x in parts if x.strip()]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for part in parts:
 | 
				
			||||||
 | 
					        m = ptn.match(part)
 | 
				
			||||||
 | 
					        if not m:
 | 
				
			||||||
 | 
					            p = html_escape(part)
 | 
				
			||||||
 | 
					            raise Pebkac(400, "invalid argument [" + p + "]")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        k, op, v = m.groups()
 | 
				
			||||||
 | 
					        qk = "mt.k = '{}' and mt.v {} ?".format(k, op)
 | 
				
			||||||
 | 
					        q[qk + "\n" + v] = u8safe(v)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def _sqlize(qobj):
 | 
				
			||||||
 | 
					    keys = []
 | 
				
			||||||
 | 
					    values = []
 | 
				
			||||||
 | 
					    for k, v in sorted(qobj.items()):
 | 
				
			||||||
 | 
					        keys.append(k.split("\n")[0])
 | 
				
			||||||
 | 
					        values.append(v)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return " and ".join(keys), values
 | 
				
			||||||
							
								
								
									
										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 struct
 | 
				
			||||||
import hashlib
 | 
					import hashlib
 | 
				
			||||||
import platform
 | 
					import platform
 | 
				
			||||||
 | 
					import traceback
 | 
				
			||||||
import threading
 | 
					import threading
 | 
				
			||||||
import mimetypes
 | 
					import mimetypes
 | 
				
			||||||
import contextlib
 | 
					import contextlib
 | 
				
			||||||
@@ -56,11 +57,58 @@ HTTPCODE = {
 | 
				
			|||||||
    413: "Payload Too Large",
 | 
					    413: "Payload Too Large",
 | 
				
			||||||
    416: "Requested Range Not Satisfiable",
 | 
					    416: "Requested Range Not Satisfiable",
 | 
				
			||||||
    422: "Unprocessable Entity",
 | 
					    422: "Unprocessable Entity",
 | 
				
			||||||
 | 
					    429: "Too Many Requests",
 | 
				
			||||||
    500: "Internal Server Error",
 | 
					    500: "Internal Server Error",
 | 
				
			||||||
    501: "Not Implemented",
 | 
					    501: "Not Implemented",
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					IMPLICATIONS = [
 | 
				
			||||||
 | 
					    ["e2dsa", "e2ds"],
 | 
				
			||||||
 | 
					    ["e2ds", "e2d"],
 | 
				
			||||||
 | 
					    ["e2tsr", "e2ts"],
 | 
				
			||||||
 | 
					    ["e2ts", "e2t"],
 | 
				
			||||||
 | 
					    ["e2t", "e2d"],
 | 
				
			||||||
 | 
					]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					REKOBO_KEY = {
 | 
				
			||||||
 | 
					    v: ln.split(" ", 1)[0]
 | 
				
			||||||
 | 
					    for ln in """
 | 
				
			||||||
 | 
					1B 6d B
 | 
				
			||||||
 | 
					2B 7d Gb F#
 | 
				
			||||||
 | 
					3B 8d Db C#
 | 
				
			||||||
 | 
					4B 9d Ab G#
 | 
				
			||||||
 | 
					5B 10d Eb D#
 | 
				
			||||||
 | 
					6B 11d Bb A#
 | 
				
			||||||
 | 
					7B 12d F
 | 
				
			||||||
 | 
					8B 1d C
 | 
				
			||||||
 | 
					9B 2d G
 | 
				
			||||||
 | 
					10B 3d D
 | 
				
			||||||
 | 
					11B 4d A
 | 
				
			||||||
 | 
					12B 5d E
 | 
				
			||||||
 | 
					1A 6m Abm G#m
 | 
				
			||||||
 | 
					2A 7m Ebm D#m
 | 
				
			||||||
 | 
					3A 8m Bbm A#m
 | 
				
			||||||
 | 
					4A 9m Fm
 | 
				
			||||||
 | 
					5A 10m Cm
 | 
				
			||||||
 | 
					6A 11m Gm
 | 
				
			||||||
 | 
					7A 12m Dm
 | 
				
			||||||
 | 
					8A 1m Am
 | 
				
			||||||
 | 
					9A 2m Em
 | 
				
			||||||
 | 
					10A 3m Bm
 | 
				
			||||||
 | 
					11A 4m Gbm F#m
 | 
				
			||||||
 | 
					12A 5m Dbm C#m
 | 
				
			||||||
 | 
					""".strip().split(
 | 
				
			||||||
 | 
					        "\n"
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					    for v in ln.strip().split(" ")[1:]
 | 
				
			||||||
 | 
					    if v
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					REKOBO_LKEY = {k.lower(): v for k, v in REKOBO_KEY.items()}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class Counter(object):
 | 
					class Counter(object):
 | 
				
			||||||
    def __init__(self, v=0):
 | 
					    def __init__(self, v=0):
 | 
				
			||||||
        self.v = v
 | 
					        self.v = v
 | 
				
			||||||
@@ -99,6 +147,71 @@ class Unrecv(object):
 | 
				
			|||||||
        self.buf = buf + self.buf
 | 
					        self.buf = buf + self.buf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class ProgressPrinter(threading.Thread):
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					    periodically print progress info without linefeeds
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self):
 | 
				
			||||||
 | 
					        threading.Thread.__init__(self)
 | 
				
			||||||
 | 
					        self.daemon = True
 | 
				
			||||||
 | 
					        self.msg = None
 | 
				
			||||||
 | 
					        self.end = False
 | 
				
			||||||
 | 
					        self.start()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def run(self):
 | 
				
			||||||
 | 
					        msg = None
 | 
				
			||||||
 | 
					        while not self.end:
 | 
				
			||||||
 | 
					            time.sleep(0.1)
 | 
				
			||||||
 | 
					            if msg == self.msg or self.end:
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            msg = self.msg
 | 
				
			||||||
 | 
					            uprint(" {}\033[K\r".format(msg))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        print("\033[K", end="")
 | 
				
			||||||
 | 
					        sys.stdout.flush()  # necessary on win10 even w/ stderr btw
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def uprint(msg):
 | 
				
			||||||
 | 
					    try:
 | 
				
			||||||
 | 
					        print(msg, end="")
 | 
				
			||||||
 | 
					    except UnicodeEncodeError:
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            print(msg.encode("utf-8", "replace").decode(), end="")
 | 
				
			||||||
 | 
					        except:
 | 
				
			||||||
 | 
					            print(msg.encode("ascii", "replace").decode(), end="")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def nuprint(msg):
 | 
				
			||||||
 | 
					    uprint("{}\n".format(msg))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def rice_tid():
 | 
				
			||||||
 | 
					    tid = threading.current_thread().ident
 | 
				
			||||||
 | 
					    c = struct.unpack(b"B" * 5, struct.pack(b">Q", tid)[-5:])
 | 
				
			||||||
 | 
					    return "".join("\033[1;37;48;5;{}m{:02x}".format(x, x) for x in c) + "\033[0m"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def trace(*args, **kwargs):
 | 
				
			||||||
 | 
					    t = time.time()
 | 
				
			||||||
 | 
					    stack = "".join(
 | 
				
			||||||
 | 
					        "\033[36m{}\033[33m{}".format(x[0].split(os.sep)[-1][:-3], x[1])
 | 
				
			||||||
 | 
					        for x in traceback.extract_stack()[3:-1]
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					    parts = ["{:.6f}".format(t), rice_tid(), stack]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if args:
 | 
				
			||||||
 | 
					        parts.append(repr(args))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if kwargs:
 | 
				
			||||||
 | 
					        parts.append(repr(kwargs))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    msg = "\033[0m ".join(parts)
 | 
				
			||||||
 | 
					    # _tracebuf.append(msg)
 | 
				
			||||||
 | 
					    nuprint(msg)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@contextlib.contextmanager
 | 
					@contextlib.contextmanager
 | 
				
			||||||
def ren_open(fname, *args, **kwargs):
 | 
					def ren_open(fname, *args, **kwargs):
 | 
				
			||||||
    fdir = kwargs.pop("fdir", None)
 | 
					    fdir = kwargs.pop("fdir", None)
 | 
				
			||||||
@@ -146,7 +259,7 @@ def ren_open(fname, *args, **kwargs):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        except OSError as ex_:
 | 
					        except OSError as ex_:
 | 
				
			||||||
            ex = ex_
 | 
					            ex = ex_
 | 
				
			||||||
            if ex.errno != 36:
 | 
					            if ex.errno not in [36, 63] and (not WINDOWS or ex.errno != 22):
 | 
				
			||||||
                raise
 | 
					                raise
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if not b64:
 | 
					        if not b64:
 | 
				
			||||||
@@ -437,6 +550,16 @@ def get_spd(nbyte, t0, t=None):
 | 
				
			|||||||
    return "{} \033[0m{}/s\033[0m".format(s1, s2)
 | 
					    return "{} \033[0m{}/s\033[0m".format(s1, s2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def s2hms(s, optional_h=False):
 | 
				
			||||||
 | 
					    s = int(s)
 | 
				
			||||||
 | 
					    h, s = divmod(s, 3600)
 | 
				
			||||||
 | 
					    m, s = divmod(s, 60)
 | 
				
			||||||
 | 
					    if not h and optional_h:
 | 
				
			||||||
 | 
					        return "{}:{:02}".format(m, s)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return "{}:{:02}:{:02}".format(h, m, s)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def undot(path):
 | 
					def undot(path):
 | 
				
			||||||
    ret = []
 | 
					    ret = []
 | 
				
			||||||
    for node in path.split("/"):
 | 
					    for node in path.split("/"):
 | 
				
			||||||
@@ -453,11 +576,12 @@ def undot(path):
 | 
				
			|||||||
    return "/".join(ret)
 | 
					    return "/".join(ret)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def sanitize_fn(fn):
 | 
					def sanitize_fn(fn, ok=""):
 | 
				
			||||||
 | 
					    if "/" not in ok:
 | 
				
			||||||
        fn = fn.replace("\\", "/").split("/")[-1]
 | 
					        fn = fn.replace("\\", "/").split("/")[-1]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if WINDOWS:
 | 
					    if WINDOWS:
 | 
				
			||||||
        for bad, good in [
 | 
					        for bad, good in [x for x in [
 | 
				
			||||||
            ["<", "<"],
 | 
					            ["<", "<"],
 | 
				
			||||||
            [">", ">"],
 | 
					            [">", ">"],
 | 
				
			||||||
            [":", ":"],
 | 
					            [":", ":"],
 | 
				
			||||||
@@ -467,7 +591,7 @@ def sanitize_fn(fn):
 | 
				
			|||||||
            ["|", "|"],
 | 
					            ["|", "|"],
 | 
				
			||||||
            ["?", "?"],
 | 
					            ["?", "?"],
 | 
				
			||||||
            ["*", "*"],
 | 
					            ["*", "*"],
 | 
				
			||||||
        ]:
 | 
					        ] if x[0] not in ok]:
 | 
				
			||||||
            fn = fn.replace(bad, good)
 | 
					            fn = fn.replace(bad, good)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        bad = ["con", "prn", "aux", "nul"]
 | 
					        bad = ["con", "prn", "aux", "nul"]
 | 
				
			||||||
@@ -480,10 +604,15 @@ def sanitize_fn(fn):
 | 
				
			|||||||
    return fn.strip()
 | 
					    return fn.strip()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def u8safe(txt):
 | 
				
			||||||
 | 
					    try:
 | 
				
			||||||
 | 
					        return txt.encode("utf-8", "xmlcharrefreplace").decode("utf-8", "replace")
 | 
				
			||||||
 | 
					    except:
 | 
				
			||||||
 | 
					        return txt.encode("utf-8", "replace").decode("utf-8", "replace")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def exclude_dotfiles(filepaths):
 | 
					def exclude_dotfiles(filepaths):
 | 
				
			||||||
    for fpath in filepaths:
 | 
					    return [x for x in filepaths if not x.split("/")[-1].startswith(".")]
 | 
				
			||||||
        if not fpath.split("/")[-1].startswith("."):
 | 
					 | 
				
			||||||
            yield fpath
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def html_escape(s, quote=False):
 | 
					def html_escape(s, quote=False):
 | 
				
			||||||
@@ -536,6 +665,16 @@ def w8enc(txt):
 | 
				
			|||||||
    return txt.encode(FS_ENCODING, "surrogateescape")
 | 
					    return txt.encode(FS_ENCODING, "surrogateescape")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def w8b64dec(txt):
 | 
				
			||||||
 | 
					    """decodes base64(filesystem-bytes) to wtf8"""
 | 
				
			||||||
 | 
					    return w8dec(base64.urlsafe_b64decode(txt.encode("ascii")))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def w8b64enc(txt):
 | 
				
			||||||
 | 
					    """encodes wtf8 to base64(filesystem-bytes)"""
 | 
				
			||||||
 | 
					    return base64.urlsafe_b64encode(w8enc(txt)).decode("ascii")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if PY2 and WINDOWS:
 | 
					if PY2 and WINDOWS:
 | 
				
			||||||
    # moonrunes become \x3f with bytestrings,
 | 
					    # moonrunes become \x3f with bytestrings,
 | 
				
			||||||
    # losing mojibake support is worth
 | 
					    # losing mojibake support is worth
 | 
				
			||||||
@@ -549,6 +688,31 @@ else:
 | 
				
			|||||||
    fsdec = w8dec
 | 
					    fsdec = w8dec
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def s3enc(mem_cur, rd, fn):
 | 
				
			||||||
 | 
					    ret = []
 | 
				
			||||||
 | 
					    for v in [rd, fn]:
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            mem_cur.execute("select * from a where b = ?", (v,))
 | 
				
			||||||
 | 
					            ret.append(v)
 | 
				
			||||||
 | 
					        except:
 | 
				
			||||||
 | 
					            ret.append("//" + w8b64enc(v))
 | 
				
			||||||
 | 
					            # self.log("mojien/{} [{}] {}".format(k, v, ret[-1][2:]))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return tuple(ret)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def s3dec(rd, fn):
 | 
				
			||||||
 | 
					    ret = []
 | 
				
			||||||
 | 
					    for k, v in [["d", rd], ["f", fn]]:
 | 
				
			||||||
 | 
					        if v.startswith("//"):
 | 
				
			||||||
 | 
					            ret.append(w8b64dec(v[2:]))
 | 
				
			||||||
 | 
					            # self.log("mojide/{} [{}] {}".format(k, ret[-1], v[2:]))
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            ret.append(v)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return tuple(ret)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def atomic_move(src, dst):
 | 
					def atomic_move(src, dst):
 | 
				
			||||||
    if not PY2:
 | 
					    if not PY2:
 | 
				
			||||||
        os.replace(src, dst)
 | 
					        os.replace(src, dst)
 | 
				
			||||||
@@ -583,6 +747,50 @@ def read_socket_unbounded(sr):
 | 
				
			|||||||
        yield buf
 | 
					        yield buf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def read_socket_chunked(sr, log=None):
 | 
				
			||||||
 | 
					    err = "expected chunk length, got [{}] |{}| instead"
 | 
				
			||||||
 | 
					    while True:
 | 
				
			||||||
 | 
					        buf = b""
 | 
				
			||||||
 | 
					        while b"\r" not in buf:
 | 
				
			||||||
 | 
					            rbuf = sr.recv(2)
 | 
				
			||||||
 | 
					            if not rbuf or len(buf) > 16:
 | 
				
			||||||
 | 
					                err = err.format(buf.decode("utf-8", "replace"), len(buf))
 | 
				
			||||||
 | 
					                raise Pebkac(400, err)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            buf += rbuf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if not buf.endswith(b"\n"):
 | 
				
			||||||
 | 
					            sr.recv(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            chunklen = int(buf.rstrip(b"\r\n"), 16)
 | 
				
			||||||
 | 
					        except:
 | 
				
			||||||
 | 
					            err = err.format(buf.decode("utf-8", "replace"), len(buf))
 | 
				
			||||||
 | 
					            raise Pebkac(400, err)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if chunklen == 0:
 | 
				
			||||||
 | 
					            sr.recv(2)  # \r\n after final chunk
 | 
				
			||||||
 | 
					            return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if log:
 | 
				
			||||||
 | 
					            log("receiving {} byte chunk".format(chunklen))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for chunk in read_socket(sr, chunklen):
 | 
				
			||||||
 | 
					            yield chunk
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        sr.recv(2)  # \r\n after each chunk too
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def yieldfile(fn):
 | 
				
			||||||
 | 
					    with open(fsenc(fn), "rb", 512 * 1024) as f:
 | 
				
			||||||
 | 
					        while True:
 | 
				
			||||||
 | 
					            buf = f.read(64 * 1024)
 | 
				
			||||||
 | 
					            if not buf:
 | 
				
			||||||
 | 
					                break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            yield buf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def hashcopy(actor, fin, fout):
 | 
					def hashcopy(actor, fin, fout):
 | 
				
			||||||
    u32_lim = int((2 ** 31) * 0.9)
 | 
					    u32_lim = int((2 ** 31) * 0.9)
 | 
				
			||||||
    hashobj = hashlib.sha512()
 | 
					    hashobj = hashlib.sha512()
 | 
				
			||||||
@@ -642,6 +850,33 @@ def sendfile_kern(lower, upper, f, s):
 | 
				
			|||||||
    return 0
 | 
					    return 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def statdir(logger, scandir, lstat, top):
 | 
				
			||||||
 | 
					    try:
 | 
				
			||||||
 | 
					        btop = fsenc(top)
 | 
				
			||||||
 | 
					        if scandir and hasattr(os, "scandir"):
 | 
				
			||||||
 | 
					            src = "scandir"
 | 
				
			||||||
 | 
					            with os.scandir(btop) as dh:
 | 
				
			||||||
 | 
					                for fh in dh:
 | 
				
			||||||
 | 
					                    try:
 | 
				
			||||||
 | 
					                        yield [fsdec(fh.name), fh.stat(follow_symlinks=not lstat)]
 | 
				
			||||||
 | 
					                    except Exception as ex:
 | 
				
			||||||
 | 
					                        msg = "scan-stat: \033[36m{} @ {}"
 | 
				
			||||||
 | 
					                        logger(msg.format(repr(ex), fsdec(fh.path)))
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            src = "listdir"
 | 
				
			||||||
 | 
					            fun = os.lstat if lstat else os.stat
 | 
				
			||||||
 | 
					            for name in os.listdir(btop):
 | 
				
			||||||
 | 
					                abspath = os.path.join(btop, name)
 | 
				
			||||||
 | 
					                try:
 | 
				
			||||||
 | 
					                    yield [fsdec(name), fun(abspath)]
 | 
				
			||||||
 | 
					                except Exception as ex:
 | 
				
			||||||
 | 
					                    msg = "list-stat: \033[36m{} @ {}"
 | 
				
			||||||
 | 
					                    logger(msg.format(repr(ex), fsdec(abspath)))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    except Exception as ex:
 | 
				
			||||||
 | 
					        logger("{}: \033[31m{} @ {}".format(src, repr(ex), top))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def unescape_cookie(orig):
 | 
					def unescape_cookie(orig):
 | 
				
			||||||
    # mw=idk; doot=qwe%2Crty%3Basd+fgh%2Bjkl%25zxc%26vbn  # qwe,rty;asd fgh+jkl%zxc&vbn
 | 
					    # mw=idk; doot=qwe%2Crty%3Basd+fgh%2Bjkl%25zxc%26vbn  # qwe,rty;asd fgh+jkl%zxc&vbn
 | 
				
			||||||
    ret = ""
 | 
					    ret = ""
 | 
				
			||||||
@@ -696,7 +931,11 @@ def chkcmd(*argv):
 | 
				
			|||||||
def gzip_orig_sz(fn):
 | 
					def gzip_orig_sz(fn):
 | 
				
			||||||
    with open(fsenc(fn), "rb") as f:
 | 
					    with open(fsenc(fn), "rb") as f:
 | 
				
			||||||
        f.seek(-4, 2)
 | 
					        f.seek(-4, 2)
 | 
				
			||||||
        return struct.unpack(b"I", f.read(4))[0]
 | 
					        rv = f.read(4)
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            return struct.unpack(b"I", rv)[0]
 | 
				
			||||||
 | 
					        except:
 | 
				
			||||||
 | 
					            return struct.unpack("I", rv)[0]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def py_desc():
 | 
					def py_desc():
 | 
				
			||||||
@@ -706,7 +945,11 @@ def py_desc():
 | 
				
			|||||||
    if ofs > 0:
 | 
					    if ofs > 0:
 | 
				
			||||||
        py_ver = py_ver[:ofs]
 | 
					        py_ver = py_ver[:ofs]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    try:
 | 
				
			||||||
        bitness = struct.calcsize(b"P") * 8
 | 
					        bitness = struct.calcsize(b"P") * 8
 | 
				
			||||||
 | 
					    except:
 | 
				
			||||||
 | 
					        bitness = struct.calcsize("P") * 8
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    host_os = platform.system()
 | 
					    host_os = platform.system()
 | 
				
			||||||
    compiler = platform.python_compiler()
 | 
					    compiler = platform.python_compiler()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -718,6 +961,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):
 | 
					class Pebkac(Exception):
 | 
				
			||||||
    def __init__(self, code, msg=None):
 | 
					    def __init__(self, code, msg=None):
 | 
				
			||||||
        super(Pebkac, self).__init__(msg or HTTPCODE[code])
 | 
					        super(Pebkac, self).__init__(msg or HTTPCODE[code])
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -39,15 +39,22 @@ body {
 | 
				
			|||||||
	margin: 1.3em 0 0 0;
 | 
						margin: 1.3em 0 0 0;
 | 
				
			||||||
	font-size: 1.4em;
 | 
						font-size: 1.4em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#path #entree {
 | 
				
			||||||
 | 
						margin-left: -.7em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#files {
 | 
					#files {
 | 
				
			||||||
	border-collapse: collapse;
 | 
						border-spacing: 0;
 | 
				
			||||||
	margin-top: 2em;
 | 
						z-index: 1;
 | 
				
			||||||
 | 
						position: relative;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#files tbody a {
 | 
					#files tbody a {
 | 
				
			||||||
	display: block;
 | 
						display: block;
 | 
				
			||||||
	padding: .3em 0;
 | 
						padding: .3em 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
a {
 | 
					#files tbody div a {
 | 
				
			||||||
 | 
						color: #f5a;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					a, #files tbody div a:last-child {
 | 
				
			||||||
	color: #fc5;
 | 
						color: #fc5;
 | 
				
			||||||
	padding: .2em;
 | 
						padding: .2em;
 | 
				
			||||||
	text-decoration: none;
 | 
						text-decoration: none;
 | 
				
			||||||
@@ -55,16 +62,18 @@ a {
 | 
				
			|||||||
#files a:hover {
 | 
					#files a:hover {
 | 
				
			||||||
	color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
	background: #161616;
 | 
						background: #161616;
 | 
				
			||||||
 | 
						text-decoration: underline;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#files thead a {
 | 
					#files thead a {
 | 
				
			||||||
	color: #999;
 | 
						color: #999;
 | 
				
			||||||
	font-weight: normal;
 | 
						font-weight: normal;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#files tr:hover {
 | 
					#files tr+tr:hover {
 | 
				
			||||||
	background: #1c1c1c;
 | 
						background: #1c1c1c;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#files thead th {
 | 
					#files thead th {
 | 
				
			||||||
	padding: .5em 1.3em .3em 1.3em;
 | 
						padding: .5em 1.3em .3em 1.3em;
 | 
				
			||||||
 | 
						cursor: pointer;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#files thead th:last-child {
 | 
					#files thead th:last-child {
 | 
				
			||||||
	background: #444;
 | 
						background: #444;
 | 
				
			||||||
@@ -82,6 +91,16 @@ a {
 | 
				
			|||||||
	margin: 0;
 | 
						margin: 0;
 | 
				
			||||||
	padding: 0 .5em;
 | 
						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) {
 | 
					#files tbody td:nth-child(3) {
 | 
				
			||||||
	font-family: monospace;
 | 
						font-family: monospace;
 | 
				
			||||||
	font-size: 1.3em;
 | 
						font-size: 1.3em;
 | 
				
			||||||
@@ -100,6 +119,9 @@ a {
 | 
				
			|||||||
	padding-bottom: 1.3em;
 | 
						padding-bottom: 1.3em;
 | 
				
			||||||
	border-bottom: .5em solid #444;
 | 
						border-bottom: .5em solid #444;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#files tbody tr td:last-child {
 | 
				
			||||||
 | 
						white-space: nowrap;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#files thead th[style] {
 | 
					#files thead th[style] {
 | 
				
			||||||
	width: auto !important;
 | 
						width: auto !important;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -131,6 +153,15 @@ a {
 | 
				
			|||||||
.logue {
 | 
					.logue {
 | 
				
			||||||
	padding: .2em 1.5em;
 | 
						padding: .2em 1.5em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					.logue:empty {
 | 
				
			||||||
 | 
						display: none;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#pro.logue {
 | 
				
			||||||
 | 
						margin-bottom: .8em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#epi.logue {
 | 
				
			||||||
 | 
						margin: .8em 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#srv_info {
 | 
					#srv_info {
 | 
				
			||||||
	opacity: .5;
 | 
						opacity: .5;
 | 
				
			||||||
	font-size: .8em;
 | 
						font-size: .8em;
 | 
				
			||||||
@@ -142,11 +173,19 @@ a {
 | 
				
			|||||||
#srv_info span {
 | 
					#srv_info span {
 | 
				
			||||||
	color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
a.play {
 | 
					#files tbody a.play {
 | 
				
			||||||
	color: #e70;
 | 
						color: #e70;
 | 
				
			||||||
 | 
						padding: .2em;
 | 
				
			||||||
 | 
						margin: -.2em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
a.play.act {
 | 
					#files tbody a.play.act {
 | 
				
			||||||
	color: #af0;
 | 
						color: #840;
 | 
				
			||||||
 | 
						text-shadow: 0 0 .3em #b80;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#files tbody tr.sel td {
 | 
				
			||||||
 | 
						background: #80b;
 | 
				
			||||||
 | 
						color: #fff;
 | 
				
			||||||
 | 
						border-color: #a3d;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#blocked {
 | 
					#blocked {
 | 
				
			||||||
	position: fixed;
 | 
						position: fixed;
 | 
				
			||||||
@@ -190,6 +229,7 @@ a.play.act {
 | 
				
			|||||||
	bottom: -6em;
 | 
						bottom: -6em;
 | 
				
			||||||
	height: 6em;
 | 
						height: 6em;
 | 
				
			||||||
	width: 100%;
 | 
						width: 100%;
 | 
				
			||||||
 | 
						z-index: 3;
 | 
				
			||||||
	transition: bottom 0.15s;
 | 
						transition: bottom 0.15s;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#widget.open {
 | 
					#widget.open {
 | 
				
			||||||
@@ -214,6 +254,9 @@ a.play.act {
 | 
				
			|||||||
	75% {cursor: url(/.cpr/dd/5.png), pointer}
 | 
						75% {cursor: url(/.cpr/dd/5.png), pointer}
 | 
				
			||||||
	85% {cursor: url(/.cpr/dd/1.png), pointer}
 | 
						85% {cursor: url(/.cpr/dd/1.png), pointer}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					@keyframes spin {
 | 
				
			||||||
 | 
						100% {transform: rotate(360deg)}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#wtoggle {
 | 
					#wtoggle {
 | 
				
			||||||
	position: absolute;
 | 
						position: absolute;
 | 
				
			||||||
	top: -1.2em;
 | 
						top: -1.2em;
 | 
				
			||||||
@@ -230,6 +273,25 @@ a.play.act {
 | 
				
			|||||||
	padding: .2em 0 0 .07em;
 | 
						padding: .2em 0 0 .07em;
 | 
				
			||||||
	color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#wtoggle>span {
 | 
				
			||||||
 | 
						display: none;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#wtoggle.sel {
 | 
				
			||||||
 | 
						width: 4.27em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#wtoggle.sel>span {
 | 
				
			||||||
 | 
						display: inline-block;
 | 
				
			||||||
 | 
						line-height: 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#wtoggle.sel>span a {
 | 
				
			||||||
 | 
						font-size: .4em;
 | 
				
			||||||
 | 
						margin: -.3em 0;
 | 
				
			||||||
 | 
						position: relative;
 | 
				
			||||||
 | 
						display: inline-block;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#wtoggle.sel>span #selzip {
 | 
				
			||||||
 | 
						top: -.6em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#barpos,
 | 
					#barpos,
 | 
				
			||||||
#barbuf {
 | 
					#barbuf {
 | 
				
			||||||
	position: absolute;
 | 
						position: absolute;
 | 
				
			||||||
@@ -273,3 +335,344 @@ a.play.act {
 | 
				
			|||||||
	width: calc(100% - 10.5em);
 | 
						width: calc(100% - 10.5em);
 | 
				
			||||||
	background: rgba(0,0,0,0.2);
 | 
						background: rgba(0,0,0,0.2);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					@media (min-width: 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,59 +7,119 @@
 | 
				
			|||||||
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
 | 
					    <meta http-equiv="X-UA-Compatible" content="IE=edge">
 | 
				
			||||||
    <meta name="viewport" content="width=device-width, initial-scale=0.8">
 | 
					    <meta name="viewport" content="width=device-width, initial-scale=0.8">
 | 
				
			||||||
    <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/browser.css{{ ts }}">
 | 
					    <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/browser.css{{ ts }}">
 | 
				
			||||||
    {%- if can_upload %}
 | 
					 | 
				
			||||||
    <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/upload.css{{ ts }}">
 | 
					    <link rel="stylesheet" type="text/css" media="screen" href="/.cpr/upload.css{{ ts }}">
 | 
				
			||||||
    {%- endif %}
 | 
					 | 
				
			||||||
</head>
 | 
					</head>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<body>
 | 
					<body>
 | 
				
			||||||
    {%- if can_upload %}
 | 
					    <div id="ops">
 | 
				
			||||||
    {%- include 'upload.html' %}
 | 
					        <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 %}
 | 
					        {%- 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' %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <div id="op_cfg" class="opview opbox">
 | 
				
			||||||
 | 
					        <h3>key notation</h3>
 | 
				
			||||||
 | 
					        <div id="key_notation"></div>
 | 
				
			||||||
 | 
					        {%- if have_zip %}
 | 
				
			||||||
 | 
					        <h3>folder download</h3>
 | 
				
			||||||
 | 
					        <div id="arc_fmt"></div>
 | 
				
			||||||
 | 
					        {%- endif %}
 | 
				
			||||||
 | 
					        <h3>tooltips</h3>
 | 
				
			||||||
 | 
					        <div><a id="tooltips" class="tglbtn" href="#">enable</a></div>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
    <h1 id="path">
 | 
					    <h1 id="path">
 | 
				
			||||||
 | 
					        <a href="#" id="entree">🌲</a>
 | 
				
			||||||
        {%- for n in vpnodes %}
 | 
					        {%- for n in vpnodes %}
 | 
				
			||||||
        <a href="/{{ n[0] }}">{{ n[1] }}</a>
 | 
					        <a href="/{{ n[0] }}">{{ n[1] }}</a>
 | 
				
			||||||
        {%- endfor %}
 | 
					        {%- endfor %}
 | 
				
			||||||
    </h1>
 | 
					    </h1>
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
    {%- if can_read %}
 | 
					    <div id="tree">
 | 
				
			||||||
    {%- if prologue %}
 | 
					        <a href="#" id="detree">🍞...</a>
 | 
				
			||||||
    <div id="pro" class="logue">{{ prologue }}</div>
 | 
					        <a href="#" step="2" id="twobytwo">+</a>
 | 
				
			||||||
    {%- endif %}
 | 
					        <a href="#" step="-2" id="twig">–</a>
 | 
				
			||||||
 | 
					        <a href="#" class="tglbtn" id="dyntree">a</a>
 | 
				
			||||||
 | 
					        <ul id="treeul"></ul>
 | 
				
			||||||
 | 
					        <div id="thx_ff"> </div>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<div id="wrap">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <div id="pro" class="logue">{{ logues[0] }}</div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <table id="files">
 | 
					    <table id="files">
 | 
				
			||||||
        <thead>
 | 
					        <thead>
 | 
				
			||||||
            <tr>
 | 
					            <tr>
 | 
				
			||||||
                <th></th>
 | 
					                <th name="lead"><span>c</span></th>
 | 
				
			||||||
                <th>File Name</th>
 | 
					                <th name="href"><span>File Name</span></th>
 | 
				
			||||||
                <th sort="int">File Size</th>
 | 
					                <th name="sz" sort="int"><span>Size</span></th>
 | 
				
			||||||
                <th>T</th>
 | 
					                {%- for k in taglist %}
 | 
				
			||||||
                <th>Date</th>
 | 
					                    {%- if k.startswith('.') %}
 | 
				
			||||||
 | 
					                        <th name="tags/{{ k }}" sort="int"><span>{{ k[1:] }}</span></th>
 | 
				
			||||||
 | 
					                    {%- else %}
 | 
				
			||||||
 | 
					                        <th name="tags/{{ k }}"><span>{{ k[0]|upper }}{{ k[1:] }}</span></th>
 | 
				
			||||||
 | 
					                    {%- endif %}
 | 
				
			||||||
 | 
					                {%- endfor %}
 | 
				
			||||||
 | 
					                <th name="ext"><span>T</span></th>
 | 
				
			||||||
 | 
					                <th name="ts"><span>Date</span></th>
 | 
				
			||||||
            </tr>
 | 
					            </tr>
 | 
				
			||||||
        </thead>
 | 
					        </thead>
 | 
				
			||||||
        <tbody>
 | 
					        <tbody>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
{%- for f in files %}
 | 
					{%- for f in files %}
 | 
				
			||||||
<tr><td>{{ f[0] }}</td><td><a href="{{ f[1] }}">{{ f[2] }}</a></td><td>{{ f[3] }}</td><td>{{ f[4] }}</td><td>{{ f[5] }}</td></tr>
 | 
					    <tr><td>{{ f.lead }}</td><td><a href="{{ f.href }}">{{ f.name|e }}</a></td><td>{{ f.sz }}</td>
 | 
				
			||||||
 | 
					    {%- if f.tags is defined %}
 | 
				
			||||||
 | 
					        {%- for k in taglist %}
 | 
				
			||||||
 | 
					            <td>{{ f.tags[k] }}</td>
 | 
				
			||||||
 | 
					        {%- endfor %}
 | 
				
			||||||
 | 
					    {%- endif %}
 | 
				
			||||||
 | 
					    <td>{{ f.ext }}</td><td>{{ f.dt }}</td></tr>
 | 
				
			||||||
{%- endfor %}
 | 
					{%- endfor %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        </tbody>
 | 
					        </tbody>
 | 
				
			||||||
    </table>
 | 
					    </table>
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
    {%- if epilogue %}
 | 
					    <div id="epi" class="logue">{{ logues[1] }}</div>
 | 
				
			||||||
    <div id="epi" class="logue">{{ epilogue }}</div>
 | 
					 | 
				
			||||||
    {%- endif %}
 | 
					 | 
				
			||||||
    {%- endif %}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <h2><a href="?h">control-panel</a></h2>
 | 
					    <h2><a href="?h">control-panel</a></h2>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					</div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    {%- if srv_info %}
 | 
					    {%- if srv_info %}
 | 
				
			||||||
    <div id="srv_info"><span>{{ srv_info }}</span></div>
 | 
					    <div id="srv_info"><span>{{ srv_info }}</span></div>
 | 
				
			||||||
    {%- endif %}
 | 
					    {%- endif %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <div id="widget">
 | 
					    <div id="widget">
 | 
				
			||||||
        <div id="wtoggle">♫</div>
 | 
					        <div id="wtoggle">
 | 
				
			||||||
 | 
					            <span>
 | 
				
			||||||
 | 
					                <a href="#" id="selall">sel.<br />all</a>
 | 
				
			||||||
 | 
					                <a href="#" id="selinv">sel.<br />inv.</a>
 | 
				
			||||||
 | 
					                <a href="#" id="selzip">zip</a>
 | 
				
			||||||
 | 
					            </span>
 | 
				
			||||||
 | 
					            ♫
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
        <div id="widgeti">
 | 
					        <div id="widgeti">
 | 
				
			||||||
            <div id="pctl"><a href="#" id="bprev">⏮</a><a href="#" id="bplay">▶</a><a href="#" id="bnext">⏭</a></div>
 | 
					            <div id="pctl"><a href="#" id="bprev">⏮</a><a href="#" id="bplay">▶</a><a href="#" id="bnext">⏭</a></div>
 | 
				
			||||||
            <canvas id="pvol" width="288" height="38"></canvas>
 | 
					            <canvas id="pvol" width="288" height="38"></canvas>
 | 
				
			||||||
@@ -68,15 +128,15 @@
 | 
				
			|||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <script>
 | 
				
			||||||
 | 
					        var tag_order_cfg = {{ tag_order }};
 | 
				
			||||||
 | 
					    </script>
 | 
				
			||||||
    <script src="/.cpr/util.js{{ ts }}"></script>
 | 
					    <script src="/.cpr/util.js{{ ts }}"></script>
 | 
				
			||||||
 | 
					 | 
				
			||||||
    {%- if can_read %}
 | 
					 | 
				
			||||||
    <script src="/.cpr/browser.js{{ ts }}"></script>
 | 
					    <script src="/.cpr/browser.js{{ ts }}"></script>
 | 
				
			||||||
    {%- endif %}
 | 
					 | 
				
			||||||
    
 | 
					 | 
				
			||||||
    {%- if can_upload %}
 | 
					 | 
				
			||||||
    <script src="/.cpr/up2k.js{{ ts }}"></script>
 | 
					    <script src="/.cpr/up2k.js{{ ts }}"></script>
 | 
				
			||||||
    {%- endif %}
 | 
					    <script>
 | 
				
			||||||
 | 
					        apply_perms({{ perms }});
 | 
				
			||||||
 | 
					    </script>
 | 
				
			||||||
</body>
 | 
					</body>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
</html>
 | 
					</html>
 | 
				
			||||||
 
 | 
				
			|||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							@@ -147,7 +147,7 @@ var md_opt = {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	</script>
 | 
						</script>
 | 
				
			||||||
    <script src="/.cpr/util.js"></script>
 | 
					    <script src="/.cpr/util.js"></script>
 | 
				
			||||||
	<script src="/.cpr/deps/marked.full.js"></script>
 | 
						<script src="/.cpr/deps/marked.js"></script>
 | 
				
			||||||
	<script src="/.cpr/md.js"></script>
 | 
						<script src="/.cpr/md.js"></script>
 | 
				
			||||||
	{%- if edit %}
 | 
						{%- if edit %}
 | 
				
			||||||
	<script src="/.cpr/md2.js"></script>
 | 
						<script src="/.cpr/md2.js"></script>
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -65,7 +65,7 @@ function statify(obj) {
 | 
				
			|||||||
        if (a > 0)
 | 
					        if (a > 0)
 | 
				
			||||||
            loc.push(n[a]);
 | 
					            loc.push(n[a]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var dec = hesc(decodeURIComponent(n[a]));
 | 
					        var dec = hesc(uricom_dec(n[a])[0]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>');
 | 
					        nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>');
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -524,11 +524,9 @@ dom_navtgl.onclick = function () {
 | 
				
			|||||||
    dom_navtgl.innerHTML = hidden ? 'show nav' : 'hide nav';
 | 
					    dom_navtgl.innerHTML = hidden ? 'show nav' : 'hide nav';
 | 
				
			||||||
    dom_nav.style.display = hidden ? 'none' : 'block';
 | 
					    dom_nav.style.display = hidden ? 'none' : 'block';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (window.localStorage)
 | 
					    swrite('hidenav', hidden ? 1 : 0);
 | 
				
			||||||
        localStorage.setItem('hidenav', hidden ? 1 : 0);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    redraw();
 | 
					    redraw();
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if (window.localStorage && localStorage.getItem('hidenav') == 1)
 | 
					if (sread('hidenav') == 1)
 | 
				
			||||||
    dom_navtgl.onclick();
 | 
					    dom_navtgl.onclick();
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -124,5 +124,3 @@ html.dark #toast {
 | 
				
			|||||||
    transition: opacity 0.2s ease-in-out;
 | 
					    transition: opacity 0.2s ease-in-out;
 | 
				
			||||||
    opacity: 1;
 | 
					    opacity: 1;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					 | 
				
			||||||
# mt {opacity: .5;top:1px}
 | 
					 | 
				
			||||||
 
 | 
				
			|||||||
@@ -15,7 +15,7 @@ var dom_md = ebi('mt');
 | 
				
			|||||||
        if (a > 0)
 | 
					        if (a > 0)
 | 
				
			||||||
            loc.push(n[a]);
 | 
					            loc.push(n[a]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var dec = decodeURIComponent(n[a]).replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">");
 | 
					        var dec = uricom_dec(n[a])[0].replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>');
 | 
					        nav.push('<a href="/' + loc.join('/') + '">' + dec + '</a>');
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -3,51 +3,6 @@
 | 
				
			|||||||
window.onerror = vis_exh;
 | 
					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() {
 | 
					function goto_up2k() {
 | 
				
			||||||
    if (up2k === false)
 | 
					    if (up2k === false)
 | 
				
			||||||
        return goto('bup');
 | 
					        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,
 | 
					// chrome requires https to use crypto.subtle,
 | 
				
			||||||
// usually it's undefined but some chromes throw on invoke
 | 
					// usually it's undefined but some chromes throw on invoke
 | 
				
			||||||
var up2k = null;
 | 
					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) {
 | 
					function up2k_init(have_crypto) {
 | 
				
			||||||
    //have_crypto = false;
 | 
					    //have_crypto = false;
 | 
				
			||||||
    var need_filereader_cache = undefined;
 | 
					    var need_filereader_cache = undefined;
 | 
				
			||||||
@@ -109,10 +151,6 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        ebi('u2notbtn').innerHTML = '';
 | 
					        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 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;
 | 
					    var is_https = (window.location + '').indexOf('https:') === 0;
 | 
				
			||||||
    if (is_https)
 | 
					    if (is_https)
 | 
				
			||||||
@@ -156,8 +194,8 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    // handle user intent to use the basic uploader instead
 | 
					    // handle user intent to use the basic uploader instead
 | 
				
			||||||
    ebi('u2nope').onclick = function (e) {
 | 
					    ebi('u2nope').onclick = function (e) {
 | 
				
			||||||
        e.preventDefault();
 | 
					        ev(e);
 | 
				
			||||||
        setmsg('');
 | 
					        setmsg();
 | 
				
			||||||
        goto('bup');
 | 
					        goto('bup');
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -171,37 +209,11 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        };
 | 
					        };
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function cfg_get(name) {
 | 
					    var parallel_uploads = icfg_get('nthread');
 | 
				
			||||||
        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 multitask = bcfg_get('multitask', true);
 | 
					    var multitask = bcfg_get('multitask', true);
 | 
				
			||||||
    var ask_up = bcfg_get('ask_up', 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_hashing = '#00bbff';
 | 
				
			||||||
    var col_hashed = '#004466';
 | 
					    var col_hashed = '#004466';
 | 
				
			||||||
@@ -219,6 +231,10 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
            "hash": [],
 | 
					            "hash": [],
 | 
				
			||||||
            "handshake": [],
 | 
					            "handshake": [],
 | 
				
			||||||
            "upload": []
 | 
					            "upload": []
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "bytes": {
 | 
				
			||||||
 | 
					            "hashed": 0,
 | 
				
			||||||
 | 
					            "uploaded": 0
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -229,47 +245,60 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
    if (!bobslice || !window.FileReader || !window.FileList)
 | 
					    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");
 | 
					        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() {
 | 
					    function nav() {
 | 
				
			||||||
        ebi('file' + fdom_ctr).click();
 | 
					        ebi('file' + fdom_ctr).click();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    ebi('u2btn').addEventListener('click', nav, false);
 | 
					    ebi('u2btn').addEventListener('click', nav, false);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function ondrag(ev) {
 | 
					    function ondrag(e) {
 | 
				
			||||||
        ev.stopPropagation();
 | 
					        e.stopPropagation();
 | 
				
			||||||
        ev.preventDefault();
 | 
					        e.preventDefault();
 | 
				
			||||||
        ev.dataTransfer.dropEffect = 'copy';
 | 
					        e.dataTransfer.dropEffect = 'copy';
 | 
				
			||||||
        ev.dataTransfer.effectAllowed = 'copy';
 | 
					        e.dataTransfer.effectAllowed = 'copy';
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    ebi('u2btn').addEventListener('dragover', ondrag, false);
 | 
					    ebi('u2btn').addEventListener('dragover', ondrag, false);
 | 
				
			||||||
    ebi('u2btn').addEventListener('dragenter', ondrag, false);
 | 
					    ebi('u2btn').addEventListener('dragenter', ondrag, false);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function gotfile(ev) {
 | 
					    function gotfile(e) {
 | 
				
			||||||
        ev.stopPropagation();
 | 
					        e.stopPropagation();
 | 
				
			||||||
        ev.preventDefault();
 | 
					        e.preventDefault();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var files;
 | 
					        var files;
 | 
				
			||||||
        var is_itemlist = false;
 | 
					        var is_itemlist = false;
 | 
				
			||||||
        if (ev.dataTransfer) {
 | 
					        if (e.dataTransfer) {
 | 
				
			||||||
            if (ev.dataTransfer.items) {
 | 
					            if (e.dataTransfer.items) {
 | 
				
			||||||
                files = ev.dataTransfer.items; // DataTransferItemList
 | 
					                files = e.dataTransfer.items; // DataTransferItemList
 | 
				
			||||||
                is_itemlist = true;
 | 
					                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)
 | 
					        if (!files || files.length == 0)
 | 
				
			||||||
            return alert('no files selected??');
 | 
					            return alert('no files selected??');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        more_one_file();
 | 
					        more_one_file();
 | 
				
			||||||
        var bad_files = [];
 | 
					        var bad_files = [];
 | 
				
			||||||
        var good_files = [];
 | 
					        var good_files = [];
 | 
				
			||||||
 | 
					        var dirs = [];
 | 
				
			||||||
        for (var a = 0; a < files.length; a++) {
 | 
					        for (var a = 0; a < files.length; a++) {
 | 
				
			||||||
            var fobj = files[a];
 | 
					            var fobj = files[a];
 | 
				
			||||||
            if (is_itemlist) {
 | 
					            if (is_itemlist) {
 | 
				
			||||||
                if (fobj.kind !== 'file')
 | 
					                if (fobj.kind !== 'file')
 | 
				
			||||||
                    continue;
 | 
					                    continue;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    var wi = fobj.webkitGetAsEntry();
 | 
				
			||||||
 | 
					                    if (wi.isDirectory) {
 | 
				
			||||||
 | 
					                        dirs.push(wi);
 | 
				
			||||||
 | 
					                        continue;
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                catch (ex) { }
 | 
				
			||||||
                fobj = fobj.getAsFile();
 | 
					                fobj = fobj.getAsFile();
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
            try {
 | 
					            try {
 | 
				
			||||||
@@ -280,12 +309,69 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                bad_files.push(fobj.name);
 | 
					                bad_files.push(fobj.name);
 | 
				
			||||||
                continue;
 | 
					                continue;
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
            good_files.push(fobj);
 | 
					            good_files.push([fobj, fobj.name]);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (dirs) {
 | 
				
			||||||
 | 
					            return read_dirs(null, [], dirs, good_files, bad_files);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    function read_dirs(rd, pf, dirs, good, bad) {
 | 
				
			||||||
 | 
					        if (!dirs.length) {
 | 
				
			||||||
 | 
					            if (!pf.length)
 | 
				
			||||||
 | 
					                return gotallfiles(good, bad);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            console.log("retry pf, " + pf.length);
 | 
				
			||||||
 | 
					            setTimeout(function () {
 | 
				
			||||||
 | 
					                read_dirs(rd, pf, dirs, good, bad);
 | 
				
			||||||
 | 
					            }, 50);
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (!rd)
 | 
				
			||||||
 | 
					            rd = dirs[0].createReader();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        rd.readEntries(function (ents) {
 | 
				
			||||||
 | 
					            var ngot = 0;
 | 
				
			||||||
 | 
					            ents.forEach(function (dn) {
 | 
				
			||||||
 | 
					                if (dn.isDirectory) {
 | 
				
			||||||
 | 
					                    dirs.push(dn);
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                else {
 | 
				
			||||||
 | 
					                    var name = dn.fullPath;
 | 
				
			||||||
 | 
					                    if (name.indexOf('/') === 0)
 | 
				
			||||||
 | 
					                        name = name.slice(1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    pf.push(name);
 | 
				
			||||||
 | 
					                    dn.file(function (fobj) {
 | 
				
			||||||
 | 
					                        var idx = pf.indexOf(name);
 | 
				
			||||||
 | 
					                        pf.splice(idx, 1);
 | 
				
			||||||
 | 
					                        try {
 | 
				
			||||||
 | 
					                            if (fobj.size > 0) {
 | 
				
			||||||
 | 
					                                good.push([fobj, name]);
 | 
				
			||||||
 | 
					                                return;
 | 
				
			||||||
 | 
					                            }
 | 
				
			||||||
 | 
					                        }
 | 
				
			||||||
 | 
					                        catch (ex) { }
 | 
				
			||||||
 | 
					                        bad.push(name);
 | 
				
			||||||
 | 
					                    });
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                ngot += 1;
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					            // console.log("ngot: " + ngot);
 | 
				
			||||||
 | 
					            if (!ngot) {
 | 
				
			||||||
 | 
					                dirs.shift();
 | 
				
			||||||
 | 
					                rd = null;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            return read_dirs(rd, pf, dirs, good, bad);
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    function gotallfiles(good_files, bad_files) {
 | 
				
			||||||
        if (bad_files.length > 0) {
 | 
					        if (bad_files.length > 0) {
 | 
				
			||||||
            var msg = 'These {0} files (of {1} total) were skipped because they are empty:\n'.format(bad_files.length, files.length);
 | 
					            var ntot = bad_files.length + good_files.length;
 | 
				
			||||||
            for (var a = 0; a < bad_files.length; a++)
 | 
					            var msg = 'These {0} files (of {1} total) were skipped because they are empty:\n'.format(bad_files.length, ntot);
 | 
				
			||||||
 | 
					            for (var a = 0, aa = Math.min(20, bad_files.length); a < aa; a++)
 | 
				
			||||||
                msg += '-- ' + bad_files[a] + '\n';
 | 
					                msg += '-- ' + bad_files[a] + '\n';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            if (files.length - bad_files.length <= 1 && /(android)/i.test(navigator.userAgent))
 | 
					            if (files.length - bad_files.length <= 1 && /(android)/i.test(navigator.userAgent))
 | 
				
			||||||
@@ -295,23 +381,25 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var msg = ['upload these ' + good_files.length + ' files?'];
 | 
					        var msg = ['upload these ' + good_files.length + ' files?'];
 | 
				
			||||||
        for (var a = 0; a < good_files.length; a++)
 | 
					        for (var a = 0, aa = Math.min(20, good_files.length); a < aa; a++)
 | 
				
			||||||
            msg.push(good_files[a].name);
 | 
					            msg.push(good_files[a][1]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (ask_up && !confirm(msg.join('\n')))
 | 
					        if (ask_up && !fsearch && !confirm(msg.join('\n')))
 | 
				
			||||||
            return;
 | 
					            return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        for (var a = 0; a < good_files.length; a++) {
 | 
					        for (var a = 0; a < good_files.length; a++) {
 | 
				
			||||||
            var fobj = good_files[a];
 | 
					            var fobj = good_files[a][0];
 | 
				
			||||||
            var now = new Date().getTime();
 | 
					            var now = new Date().getTime();
 | 
				
			||||||
            var lmod = fobj.lastModified || now;
 | 
					            var lmod = fobj.lastModified || now;
 | 
				
			||||||
            var entry = {
 | 
					            var entry = {
 | 
				
			||||||
                "n": parseInt(st.files.length.toString()),
 | 
					                "n": parseInt(st.files.length.toString()),
 | 
				
			||||||
                "t0": now,  // TODO remove probably
 | 
					                "t0": now,
 | 
				
			||||||
                "fobj": fobj,
 | 
					                "fobj": fobj,
 | 
				
			||||||
                "name": fobj.name,
 | 
					                "name": good_files[a][1],
 | 
				
			||||||
                "size": fobj.size,
 | 
					                "size": fobj.size,
 | 
				
			||||||
                "lmod": lmod / 1000,
 | 
					                "lmod": lmod / 1000,
 | 
				
			||||||
 | 
					                "purl": get_evpath(),
 | 
				
			||||||
 | 
					                "done": false,
 | 
				
			||||||
                "hash": []
 | 
					                "hash": []
 | 
				
			||||||
            };
 | 
					            };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -326,7 +414,7 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            var tr = document.createElement('tr');
 | 
					            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.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);
 | 
					            ebi('u2tab').appendChild(tr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            st.files.push(entry);
 | 
					            st.files.push(entry);
 | 
				
			||||||
@@ -344,6 +432,19 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
    more_one_file();
 | 
					    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
 | 
					    ///   actuator
 | 
				
			||||||
@@ -357,14 +458,18 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function hashing_permitted() {
 | 
					    function hashing_permitted() {
 | 
				
			||||||
        var lim = multitask ? 1 : 0;
 | 
					        if (multitask) {
 | 
				
			||||||
        return handshakes_permitted() && lim >=
 | 
					            var ahead = st.bytes.hashed - st.bytes.uploaded;
 | 
				
			||||||
 | 
					            return ahead < 1024 * 1024 * 128;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return handshakes_permitted() && 0 ==
 | 
				
			||||||
            st.todo.handshake.length +
 | 
					            st.todo.handshake.length +
 | 
				
			||||||
            st.busy.handshake.length;
 | 
					            st.busy.handshake.length;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    var tasker = (function () {
 | 
					    var tasker = (function () {
 | 
				
			||||||
        var mutex = false;
 | 
					        var mutex = false;
 | 
				
			||||||
 | 
					        var was_busy = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        function taskerd() {
 | 
					        function taskerd() {
 | 
				
			||||||
            if (mutex)
 | 
					            if (mutex)
 | 
				
			||||||
@@ -372,8 +477,63 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            mutex = true;
 | 
					            mutex = true;
 | 
				
			||||||
            while (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;
 | 
					                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() &&
 | 
					                if (handshakes_permitted() &&
 | 
				
			||||||
                    st.todo.handshake.length > 0 &&
 | 
					                    st.todo.handshake.length > 0 &&
 | 
				
			||||||
                    st.busy.handshake.length == 0 &&
 | 
					                    st.busy.handshake.length == 0 &&
 | 
				
			||||||
@@ -512,6 +672,8 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        var t = st.todo.hash.shift();
 | 
					        var t = st.todo.hash.shift();
 | 
				
			||||||
        st.busy.hash.push(t);
 | 
					        st.busy.hash.push(t);
 | 
				
			||||||
 | 
					        st.bytes.hashed += t.size;
 | 
				
			||||||
 | 
					        t.bytes_uploaded = 0;
 | 
				
			||||||
        t.t1 = new Date().getTime();
 | 
					        t.t1 = new Date().getTime();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var nchunk = 0;
 | 
					        var nchunk = 0;
 | 
				
			||||||
@@ -559,8 +721,8 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
            prog(t.n, nchunk, col_hashing);
 | 
					            prog(t.n, nchunk, col_hashing);
 | 
				
			||||||
        };
 | 
					        };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var segm_load = function (ev) {
 | 
					        var segm_load = function (e) {
 | 
				
			||||||
            cache_buf = ev.target.result;
 | 
					            cache_buf = e.target.result;
 | 
				
			||||||
            cache_ofs = 0;
 | 
					            cache_ofs = 0;
 | 
				
			||||||
            hash_calc();
 | 
					            hash_calc();
 | 
				
			||||||
        };
 | 
					        };
 | 
				
			||||||
@@ -634,14 +796,42 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        st.busy.handshake.push(t);
 | 
					        st.busy.handshake.push(t);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var xhr = new XMLHttpRequest();
 | 
					        var xhr = new XMLHttpRequest();
 | 
				
			||||||
        xhr.onload = function (ev) {
 | 
					        xhr.onload = function (e) {
 | 
				
			||||||
            if (xhr.status == 200) {
 | 
					            if (xhr.status == 200) {
 | 
				
			||||||
                var response = JSON.parse(xhr.responseText);
 | 
					                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) {
 | 
					                if (response.name !== t.name) {
 | 
				
			||||||
                    // file exists; server renamed us
 | 
					                    // file exists; server renamed us
 | 
				
			||||||
                    t.name = response.name;
 | 
					                    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 = [];
 | 
					                t.postlist = [];
 | 
				
			||||||
@@ -675,11 +865,15 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                st.busy.handshake.splice(st.busy.handshake.indexOf(t), 1);
 | 
					                st.busy.handshake.splice(st.busy.handshake.indexOf(t), 1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                if (done) {
 | 
					                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 spd1 = (t.size / ((t.t2 - t.t1) / 1000.)) / (1024 * 1024.);
 | 
				
			||||||
                    var spd2 = (t.size / ((t.t3 - t.t2) / 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(
 | 
					                    ebi('f{0}p'.format(t.n)).innerHTML = 'hash {0}, up {1} MB/s'.format(
 | 
				
			||||||
                        spd1.toFixed(2), spd2.toFixed(2));
 | 
					                        spd1.toFixed(2), spd2.toFixed(2));
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
 | 
					                else t.t3 = undefined;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                tasker();
 | 
					                tasker();
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
            else {
 | 
					            else {
 | 
				
			||||||
@@ -691,6 +885,11 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                    var ofs = err.lastIndexOf(' : ');
 | 
					                    var ofs = err.lastIndexOf(' : ');
 | 
				
			||||||
                    if (ofs > 0)
 | 
					                    if (ofs > 0)
 | 
				
			||||||
                        err = err.slice(0, ofs);
 | 
					                        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 != "") {
 | 
					                if (err != "") {
 | 
				
			||||||
                    ebi('f{0}t'.format(t.n)).innerHTML = "ERROR";
 | 
					                    ebi('f{0}t'.format(t.n)).innerHTML = "ERROR";
 | 
				
			||||||
@@ -707,14 +906,19 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                    "no further information"));
 | 
					                    "no further information"));
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        };
 | 
					        };
 | 
				
			||||||
        xhr.open('POST', post_url + 'handshake.php', true);
 | 
					
 | 
				
			||||||
        xhr.responseType = 'text';
 | 
					        var req = {
 | 
				
			||||||
        xhr.send(JSON.stringify({
 | 
					 | 
				
			||||||
            "name": t.name,
 | 
					            "name": t.name,
 | 
				
			||||||
            "size": t.size,
 | 
					            "size": t.size,
 | 
				
			||||||
            "lmod": t.lmod,
 | 
					            "lmod": t.lmod,
 | 
				
			||||||
            "hash": t.hash
 | 
					            "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 +947,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)');
 | 
					            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();
 | 
					            var xhr = new XMLHttpRequest();
 | 
				
			||||||
            xhr.upload.onprogress = function (xev) {
 | 
					            xhr.upload.onprogress = function (xev) {
 | 
				
			||||||
                var perc = xev.loaded / (cdr - car) * 100;
 | 
					                var perc = xev.loaded / (cdr - car) * 100;
 | 
				
			||||||
@@ -752,12 +956,14 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
            xhr.onload = function (xev) {
 | 
					            xhr.onload = function (xev) {
 | 
				
			||||||
                if (xhr.status == 200) {
 | 
					                if (xhr.status == 200) {
 | 
				
			||||||
                    prog(t.n, npart, col_uploaded);
 | 
					                    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);
 | 
					                    st.busy.upload.splice(st.busy.upload.indexOf(upt), 1);
 | 
				
			||||||
                    t.postlist.splice(t.postlist.indexOf(npart), 1);
 | 
					                    t.postlist.splice(t.postlist.indexOf(npart), 1);
 | 
				
			||||||
                    if (t.postlist.length == 0) {
 | 
					                    if (t.postlist.length == 0) {
 | 
				
			||||||
                        t.t3 = new Date().getTime();
 | 
					                        t.t3 = new Date().getTime();
 | 
				
			||||||
                        ebi('f{0}t'.format(t.n)).innerHTML = 'verifying';
 | 
					                        ebi('f{0}t'.format(t.n)).innerHTML = 'verifying';
 | 
				
			||||||
                        st.todo.handshake.push(t);
 | 
					                        st.todo.handshake.unshift(t);
 | 
				
			||||||
                    }
 | 
					                    }
 | 
				
			||||||
                    tasker();
 | 
					                    tasker();
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
@@ -768,14 +974,14 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                        (xhr.responseText && xhr.responseText) ||
 | 
					                        (xhr.responseText && xhr.responseText) ||
 | 
				
			||||||
                        "no further information"));
 | 
					                        "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].substr(1) + "x");
 | 
				
			||||||
            xhr.setRequestHeader("X-Up2k-Hash", t.hash[npart]);
 | 
					            xhr.setRequestHeader("X-Up2k-Hash", t.hash[npart]);
 | 
				
			||||||
            xhr.setRequestHeader("X-Up2k-Wark", t.wark);
 | 
					            xhr.setRequestHeader("X-Up2k-Wark", t.wark);
 | 
				
			||||||
            xhr.setRequestHeader('Content-Type', 'application/octet-stream');
 | 
					            xhr.setRequestHeader('Content-Type', 'application/octet-stream');
 | 
				
			||||||
            xhr.overrideMimeType('Content-Type', 'application/octet-stream');
 | 
					            xhr.overrideMimeType('Content-Type', 'application/octet-stream');
 | 
				
			||||||
            xhr.responseType = 'text';
 | 
					            xhr.responseType = 'text';
 | 
				
			||||||
            xhr.send(ev.target.result);
 | 
					            xhr.send(e.target.result);
 | 
				
			||||||
        };
 | 
					        };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        reader.readAsArrayBuffer(bobslice.call(t.fobj, car, cdr));
 | 
					        reader.readAsArrayBuffer(bobslice.call(t.fobj, car, cdr));
 | 
				
			||||||
@@ -804,6 +1010,46 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
    ///   config ui
 | 
					    ///   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) {
 | 
					    function bumpthread(dir) {
 | 
				
			||||||
        try {
 | 
					        try {
 | 
				
			||||||
            dir.stopPropagation();
 | 
					            dir.stopPropagation();
 | 
				
			||||||
@@ -818,7 +1064,7 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                return;
 | 
					                return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            parallel_uploads = v;
 | 
					            parallel_uploads = v;
 | 
				
			||||||
            localStorage.setItem('nthread', v);
 | 
					            swrite('nthread', v);
 | 
				
			||||||
            obj.style.background = '#444';
 | 
					            obj.style.background = '#444';
 | 
				
			||||||
            return;
 | 
					            return;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@@ -845,29 +1091,103 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        bcfg_set('ask_up', ask_up);
 | 
					        bcfg_set('ask_up', ask_up);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function nop(ev) {
 | 
					    function tgl_fsearch() {
 | 
				
			||||||
        ev.preventDefault();
 | 
					        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();
 | 
					        this.click();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    ebi('nthread_add').onclick = function (ev) {
 | 
					    ebi('nthread_add').onclick = function (e) {
 | 
				
			||||||
        ev.preventDefault();
 | 
					        ev(e);
 | 
				
			||||||
        bumpthread(1);
 | 
					        bumpthread(1);
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
    ebi('nthread_sub').onclick = function (ev) {
 | 
					    ebi('nthread_sub').onclick = function (e) {
 | 
				
			||||||
        ev.preventDefault();
 | 
					        ev(e);
 | 
				
			||||||
        bumpthread(-1);
 | 
					        bumpthread(-1);
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    ebi('nthread').addEventListener('input', bumpthread, false);
 | 
					    ebi('nthread').addEventListener('input', bumpthread, false);
 | 
				
			||||||
    ebi('multitask').addEventListener('click', tgl_multitask, false);
 | 
					    ebi('multitask').addEventListener('click', tgl_multitask, false);
 | 
				
			||||||
    ebi('ask_up').addEventListener('click', tgl_ask_up, 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');
 | 
					    var nodes = ebi('u2conf').getElementsByTagName('a');
 | 
				
			||||||
    for (var a = nodes.length - 1; a >= 0; a--)
 | 
					    for (var a = nodes.length - 1; a >= 0; a--)
 | 
				
			||||||
        nodes[a].addEventListener('touchend', nop, false);
 | 
					        nodes[a].addEventListener('touchend', nop, false);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set_fsearch();
 | 
				
			||||||
    bumpthread({ "target": 1 })
 | 
					    bumpthread({ "target": 1 })
 | 
				
			||||||
 | 
					    return { "init_deps": init_deps, "set_fsearch": set_fsearch }
 | 
				
			||||||
    return { "init_deps": init_deps }
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					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 {
 | 
					#op_up2k {
 | 
				
			||||||
	padding: 0 1em 1em 1em;
 | 
						padding: 0 1em 1em 1em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -94,6 +6,9 @@
 | 
				
			|||||||
	position: absolute;
 | 
						position: absolute;
 | 
				
			||||||
	top: 0;
 | 
						top: 0;
 | 
				
			||||||
	left: 0;
 | 
						left: 0;
 | 
				
			||||||
 | 
						width: 2px;
 | 
				
			||||||
 | 
						height: 2px;
 | 
				
			||||||
 | 
						overflow: hidden;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2form input {
 | 
					#u2form input {
 | 
				
			||||||
	background: #444;
 | 
						background: #444;
 | 
				
			||||||
@@ -104,11 +19,6 @@
 | 
				
			|||||||
	color: #f87;
 | 
						color: #f87;
 | 
				
			||||||
	padding: .5em;
 | 
						padding: .5em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2form {
 | 
					 | 
				
			||||||
	width: 2px;
 | 
					 | 
				
			||||||
	height: 2px;
 | 
					 | 
				
			||||||
	overflow: hidden;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
#u2btn {
 | 
					#u2btn {
 | 
				
			||||||
	color: #eee;
 | 
						color: #eee;
 | 
				
			||||||
	background: #555;
 | 
						background: #555;
 | 
				
			||||||
@@ -117,17 +27,27 @@
 | 
				
			|||||||
	background: linear-gradient(to bottom, #367 0%, #489 50%, #38788a 51%, #367 100%);
 | 
						background: linear-gradient(to bottom, #367 0%, #489 50%, #38788a 51%, #367 100%);
 | 
				
			||||||
	filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#489', endColorstr='#38788a', GradientType=0);
 | 
						filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#489', endColorstr='#38788a', GradientType=0);
 | 
				
			||||||
	text-decoration: none;
 | 
						text-decoration: none;
 | 
				
			||||||
	line-height: 1.5em;
 | 
						line-height: 1.3em;
 | 
				
			||||||
	border: 1px solid #222;
 | 
						border: 1px solid #222;
 | 
				
			||||||
	border-radius: .4em;
 | 
						border-radius: .4em;
 | 
				
			||||||
	text-align: center;
 | 
						text-align: center;
 | 
				
			||||||
	font-size: 2em;
 | 
						font-size: 1.5em;
 | 
				
			||||||
	margin: 1em auto;
 | 
						margin: .5em auto;
 | 
				
			||||||
	padding: 1em 0;
 | 
						padding: .8em 0;
 | 
				
			||||||
	width: 12em;
 | 
						width: 16em;
 | 
				
			||||||
	cursor: pointer;
 | 
						cursor: pointer;
 | 
				
			||||||
	box-shadow: .4em .4em 0 #111;
 | 
						box-shadow: .4em .4em 0 #111;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#op_up2k.srch #u2btn {
 | 
				
			||||||
 | 
						background: linear-gradient(to bottom, #ca3 0%, #fd8 50%, #fc6 51%, #b92 100%);
 | 
				
			||||||
 | 
						text-shadow: 1px 1px 1px #fc6;
 | 
				
			||||||
 | 
						color: #333;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#u2conf #u2btn {
 | 
				
			||||||
 | 
						margin: -1.5em 0;
 | 
				
			||||||
 | 
						padding: .8em 0;
 | 
				
			||||||
 | 
						width: 100%;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#u2notbtn {
 | 
					#u2notbtn {
 | 
				
			||||||
	display: none;
 | 
						display: none;
 | 
				
			||||||
	text-align: center;
 | 
						text-align: center;
 | 
				
			||||||
@@ -142,6 +62,9 @@
 | 
				
			|||||||
	width: calc(100% - 2em);
 | 
						width: calc(100% - 2em);
 | 
				
			||||||
	max-width: 100em;
 | 
						max-width: 100em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#op_up2k.srch #u2tab {
 | 
				
			||||||
 | 
						max-width: none;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#u2tab td {
 | 
					#u2tab td {
 | 
				
			||||||
	border: 1px solid #ccc;
 | 
						border: 1px solid #ccc;
 | 
				
			||||||
	border-width: 0 0px 1px 0;
 | 
						border-width: 0 0px 1px 0;
 | 
				
			||||||
@@ -153,12 +76,19 @@
 | 
				
			|||||||
#u2tab td:nth-child(3) {
 | 
					#u2tab td:nth-child(3) {
 | 
				
			||||||
	width: 40%;
 | 
						width: 40%;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#op_up2k.srch #u2tab td:nth-child(3) {
 | 
				
			||||||
 | 
						font-family: sans-serif;
 | 
				
			||||||
 | 
						width: auto;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#u2tab tr+tr:hover td {
 | 
					#u2tab tr+tr:hover td {
 | 
				
			||||||
	background: #222;
 | 
						background: #222;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2conf {
 | 
					#u2conf {
 | 
				
			||||||
	margin: 1em auto;
 | 
						margin: 1em auto;
 | 
				
			||||||
	width: 26em;
 | 
						width: 30em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#u2conf.has_btn {
 | 
				
			||||||
 | 
						width: 48em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2conf * {
 | 
					#u2conf * {
 | 
				
			||||||
	text-align: center;
 | 
						text-align: center;
 | 
				
			||||||
@@ -194,16 +124,72 @@
 | 
				
			|||||||
#u2conf input+a {
 | 
					#u2conf input+a {
 | 
				
			||||||
	background: #d80;
 | 
						background: #d80;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#u2conf label {
 | 
				
			||||||
 | 
						font-size: 1.6em;
 | 
				
			||||||
 | 
						width: 2em;
 | 
				
			||||||
 | 
						height: 1em;
 | 
				
			||||||
 | 
						padding: .4em 0;
 | 
				
			||||||
 | 
						display: block;
 | 
				
			||||||
 | 
						user-select: none;
 | 
				
			||||||
 | 
						border-radius: .25em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#u2conf input[type="checkbox"] {
 | 
				
			||||||
 | 
						position: relative;
 | 
				
			||||||
 | 
						opacity: .02;
 | 
				
			||||||
 | 
						top: 2em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#u2conf input[type="checkbox"]+label {
 | 
					#u2conf input[type="checkbox"]+label {
 | 
				
			||||||
	color: #f5a;
 | 
						position: relative;
 | 
				
			||||||
 | 
						background: #603;
 | 
				
			||||||
 | 
						border-bottom: .2em solid #a16;
 | 
				
			||||||
 | 
						box-shadow: 0 .1em .3em #a00 inset;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2conf input[type="checkbox"]:checked+label {
 | 
					#u2conf input[type="checkbox"]:checked+label {
 | 
				
			||||||
	color: #fc5;
 | 
						background: #6a1;
 | 
				
			||||||
 | 
						border-bottom: .2em solid #efa;
 | 
				
			||||||
 | 
						box-shadow: 0 .1em .5em #0c0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#u2conf input[type="checkbox"]+label:hover {
 | 
				
			||||||
 | 
						box-shadow: 0 .1em .3em #fb0;
 | 
				
			||||||
 | 
						border-color: #fb0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#op_up2k.srch #u2conf td:nth-child(1)>*,
 | 
				
			||||||
 | 
					#op_up2k.srch #u2conf td:nth-child(2)>*,
 | 
				
			||||||
 | 
					#op_up2k.srch #u2conf td:nth-child(3)>* {
 | 
				
			||||||
 | 
						background: #777;
 | 
				
			||||||
 | 
						border-color: #ccc;
 | 
				
			||||||
 | 
						box-shadow: none;
 | 
				
			||||||
 | 
						opacity: .2;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#u2cdesc {
 | 
				
			||||||
 | 
						position: absolute;
 | 
				
			||||||
 | 
						width: 34em;
 | 
				
			||||||
 | 
						left: calc(50% - 15em);
 | 
				
			||||||
 | 
						background: #222;
 | 
				
			||||||
 | 
						border: 0 solid #555;
 | 
				
			||||||
 | 
						text-align: center;
 | 
				
			||||||
 | 
						overflow: hidden;
 | 
				
			||||||
 | 
						margin: 0 -2em;
 | 
				
			||||||
 | 
						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 {
 | 
					#u2foot {
 | 
				
			||||||
	color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
	font-style: italic;
 | 
						font-style: italic;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#u2footfoot {
 | 
				
			||||||
 | 
						margin-bottom: -1em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
.prog {
 | 
					.prog {
 | 
				
			||||||
	font-family: monospace;
 | 
						font-family: monospace;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -225,3 +211,13 @@
 | 
				
			|||||||
	bottom: 0;
 | 
						bottom: 0;
 | 
				
			||||||
	background: #0a0;
 | 
						background: #0a0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#u2tab a>span {
 | 
				
			||||||
 | 
						font-weight: bold;
 | 
				
			||||||
 | 
						font-style: italic;
 | 
				
			||||||
 | 
						color: #fff;
 | 
				
			||||||
 | 
						padding-left: .2em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#u2cleanup {
 | 
				
			||||||
 | 
						float: right;
 | 
				
			||||||
 | 
						margin-bottom: -.3em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,13 +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></div>
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <div id="op_bup" class="opview opbox act">
 | 
					    <div id="op_bup" class="opview opbox act">
 | 
				
			||||||
        <div id="u2err"></div>
 | 
					        <div id="u2err"></div>
 | 
				
			||||||
        <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}">
 | 
					        <form method="post" enctype="multipart/form-data" accept-charset="utf-8">
 | 
				
			||||||
            <input type="hidden" name="act" value="bput" />
 | 
					            <input type="hidden" name="act" value="bput" />
 | 
				
			||||||
            <input type="file" name="f" multiple><br />
 | 
					            <input type="file" name="f" multiple><br />
 | 
				
			||||||
            <input type="submit" value="start upload">
 | 
					            <input type="submit" value="start upload">
 | 
				
			||||||
@@ -15,7 +9,7 @@
 | 
				
			|||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <div id="op_mkdir" class="opview opbox act">
 | 
					    <div id="op_mkdir" class="opview opbox act">
 | 
				
			||||||
        <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}">
 | 
					        <form method="post" enctype="multipart/form-data" accept-charset="utf-8">
 | 
				
			||||||
            <input type="hidden" name="act" value="mkdir" />
 | 
					            <input type="hidden" name="act" value="mkdir" />
 | 
				
			||||||
            <input type="text" name="name" size="30">
 | 
					            <input type="text" name="name" size="30">
 | 
				
			||||||
            <input type="submit" value="mkdir">
 | 
					            <input type="submit" value="mkdir">
 | 
				
			||||||
@@ -23,19 +17,45 @@
 | 
				
			|||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <div id="op_new_md" class="opview opbox">
 | 
					    <div id="op_new_md" class="opview opbox">
 | 
				
			||||||
        <form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="/{{ vdir }}">
 | 
					        <form method="post" enctype="multipart/form-data" accept-charset="utf-8">
 | 
				
			||||||
            <input type="hidden" name="act" value="new_md" />
 | 
					            <input type="hidden" name="act" value="new_md" />
 | 
				
			||||||
            <input type="text" name="name" size="30">
 | 
					            <input type="text" name="name" size="30">
 | 
				
			||||||
            <input type="submit" value="create doc">
 | 
					            <input type="submit" value="create doc">
 | 
				
			||||||
        </form>
 | 
					        </form>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <div id="op_msg" class="opview opbox">
 | 
				
			||||||
 | 
					        <form method="post" enctype="application/x-www-form-urlencoded" accept-charset="utf-8">
 | 
				
			||||||
 | 
					            <input type="text" name="msg" size="30">
 | 
				
			||||||
 | 
					            <input type="submit" value="send msg">
 | 
				
			||||||
 | 
					        </form>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <div id="op_up2k" class="opview">
 | 
					    <div id="op_up2k" class="opview">
 | 
				
			||||||
        <form id="u2form" method="post" enctype="multipart/form-data" onsubmit="return false;"></form>
 | 
					        <form id="u2form" method="post" enctype="multipart/form-data" onsubmit="return false;"></form>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            <table id="u2conf">
 | 
					            <table id="u2conf">
 | 
				
			||||||
                <tr>
 | 
					                <tr>
 | 
				
			||||||
                    <td>parallel uploads</td>
 | 
					                    <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>
 | 
				
			||||||
                <tr>
 | 
					                <tr>
 | 
				
			||||||
                    <td>
 | 
					                    <td>
 | 
				
			||||||
@@ -43,32 +63,30 @@
 | 
				
			|||||||
                        <input class="txtbox" id="nthread" value="2" />
 | 
					                        <input class="txtbox" id="nthread" value="2" />
 | 
				
			||||||
                        <a href="#" id="nthread_add">+</a>
 | 
					                        <a href="#" id="nthread_add">+</a>
 | 
				
			||||||
                    </td>
 | 
					                    </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>
 | 
					                </tr>
 | 
				
			||||||
            </table>
 | 
					            </table>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <div id="u2cdesc"></div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            <div id="u2notbtn"></div>
 | 
					            <div id="u2notbtn"></div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <div id="u2btn_ct">
 | 
				
			||||||
                <div id="u2btn">
 | 
					                <div id="u2btn">
 | 
				
			||||||
                drop files here<br />
 | 
					                    <span id="u2bm"></span><br />
 | 
				
			||||||
 | 
					                    drag/drop files<br />
 | 
				
			||||||
 | 
					                    and folders here<br />
 | 
				
			||||||
                    (or click me)
 | 
					                    (or click me)
 | 
				
			||||||
                </div>
 | 
					                </div>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            <table id="u2tab">
 | 
					            <table id="u2tab">
 | 
				
			||||||
                <tr>
 | 
					                <tr>
 | 
				
			||||||
                    <td>filename</td>
 | 
					                    <td>filename</td>
 | 
				
			||||||
                    <td>status</td>
 | 
					                    <td>status</td>
 | 
				
			||||||
                    <td>progress</td>
 | 
					                    <td>progress<a href="#" id="u2cleanup">cleanup</a></td>
 | 
				
			||||||
                </tr>
 | 
					                </tr>
 | 
				
			||||||
            </table>
 | 
					            </table>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            <p id="u2foot"></p>
 | 
					            <p id="u2foot"></p>
 | 
				
			||||||
            <p>( if you don't need lastmod timestamps, resumable uploads or progress bars just use the <a href="#" id="u2nope">basic uploader</a>)</p>
 | 
					            <p id="u2footfoot">( if you don't need lastmod timestamps, resumable uploads or progress bars just use the <a href="#" id="u2nope">basic uploader</a>)</p>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -23,6 +23,7 @@ function esc(txt) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
function vis_exh(msg, url, lineNo, columnNo, error) {
 | 
					function vis_exh(msg, url, lineNo, columnNo, error) {
 | 
				
			||||||
    window.onerror = undefined;
 | 
					    window.onerror = undefined;
 | 
				
			||||||
 | 
					    window['vis_exh'] = null;
 | 
				
			||||||
    var html = ['<h1>you hit a bug!</h1><p>please screenshot this error and send me a copy arigathanks gozaimuch (ed/irc.rizon.net or ed#2644)</p><p>',
 | 
					    var html = ['<h1>you hit a bug!</h1><p>please screenshot this error and send me a copy arigathanks gozaimuch (ed/irc.rizon.net or ed#2644)</p><p>',
 | 
				
			||||||
        esc(String(msg)), '</p><p>', esc(url + ' @' + lineNo + ':' + columnNo), '</p>'];
 | 
					        esc(String(msg)), '</p><p>', esc(url + ' @' + lineNo + ':' + columnNo), '</p>'];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -43,6 +44,21 @@ function ebi(id) {
 | 
				
			|||||||
    return document.getElementById(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
 | 
					// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/endsWith
 | 
				
			||||||
if (!String.prototype.endsWith) {
 | 
					if (!String.prototype.endsWith) {
 | 
				
			||||||
@@ -75,35 +91,335 @@ function import_js(url, cb) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function sortTable(table, col) {
 | 
					var crctab = (function () {
 | 
				
			||||||
    var tb = table.tBodies[0], // use `<tbody>` to ignore `<thead>` and `<tfoot>` rows
 | 
					    var c, tab = [];
 | 
				
			||||||
 | 
					    for (var n = 0; n < 256; n++) {
 | 
				
			||||||
 | 
					        c = n;
 | 
				
			||||||
 | 
					        for (var k = 0; k < 8; k++) {
 | 
				
			||||||
 | 
					            c = ((c & 1) ? (0xEDB88320 ^ (c >>> 1)) : (c >>> 1));
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        tab[n] = c;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return tab;
 | 
				
			||||||
 | 
					})();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function crc32(str) {
 | 
				
			||||||
 | 
					    var crc = 0 ^ (-1);
 | 
				
			||||||
 | 
					    for (var i = 0; i < str.length; i++) {
 | 
				
			||||||
 | 
					        crc = (crc >>> 8) ^ crctab[(crc ^ str.charCodeAt(i)) & 0xFF];
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return ((crc ^ (-1)) >>> 0).toString(16);
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function sortTable(table, col, cb) {
 | 
				
			||||||
 | 
					    var tb = table.tBodies[0],
 | 
				
			||||||
        th = table.tHead.rows[0].cells,
 | 
					        th = table.tHead.rows[0].cells,
 | 
				
			||||||
        tr = Array.prototype.slice.call(tb.rows, 0),
 | 
					        tr = Array.prototype.slice.call(tb.rows, 0),
 | 
				
			||||||
        i, reverse = th[col].className == 'sort1' ? -1 : 1;
 | 
					        i, reverse = th[col].className.indexOf('sort1') !== -1 ? -1 : 1;
 | 
				
			||||||
    for (var a = 0, thl = th.length; a < thl; a++)
 | 
					    for (var a = 0, thl = th.length; a < thl; a++)
 | 
				
			||||||
        th[a].className = '';
 | 
					        th[a].className = th[a].className.replace(/ *sort-?1 */, " ");
 | 
				
			||||||
    th[col].className = 'sort' + reverse;
 | 
					    th[col].className += ' sort' + reverse;
 | 
				
			||||||
    var stype = th[col].getAttribute('sort');
 | 
					    var stype = th[col].getAttribute('sort');
 | 
				
			||||||
    tr = tr.sort(function (a, b) {
 | 
					    try {
 | 
				
			||||||
        var v1 = a.cells[col].textContent.trim();
 | 
					        var nrules = [], rules = jread("fsort", []);
 | 
				
			||||||
        var v2 = b.cells[col].textContent.trim();
 | 
					        rules.unshift([th[col].getAttribute('name'), reverse, stype || '']);
 | 
				
			||||||
 | 
					        for (var a = 0; a < rules.length; a++) {
 | 
				
			||||||
 | 
					            var add = true;
 | 
				
			||||||
 | 
					            for (var b = 0; b < a; b++)
 | 
				
			||||||
 | 
					                if (rules[a][0] == rules[b][0])
 | 
				
			||||||
 | 
					                    add = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (add)
 | 
				
			||||||
 | 
					                nrules.push(rules[a]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (nrules.length >= 10)
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        jwrite("fsort", nrules);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    catch (ex) {
 | 
				
			||||||
 | 
					        console.log("failed to persist sort rules, resetting: " + ex);
 | 
				
			||||||
 | 
					        jwrite("fsort", null);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    var vl = [];
 | 
				
			||||||
 | 
					    for (var a = 0; a < tr.length; a++) {
 | 
				
			||||||
 | 
					        var cell = tr[a].cells[col];
 | 
				
			||||||
 | 
					        if (!cell) {
 | 
				
			||||||
 | 
					            vl.push([null, a]);
 | 
				
			||||||
 | 
					            continue;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        var v = cell.getAttribute('sortv') || cell.textContent.trim();
 | 
				
			||||||
        if (stype == 'int') {
 | 
					        if (stype == 'int') {
 | 
				
			||||||
            v1 = parseInt(v1.replace(/,/g, ''));
 | 
					            v = parseInt(v.replace(/[, ]/g, '')) || 0;
 | 
				
			||||||
            v2 = parseInt(v2.replace(/,/g, ''));
 | 
					 | 
				
			||||||
            return reverse * (v1 - v2);
 | 
					 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        return reverse * (v1.localeCompare(v2));
 | 
					        vl.push([v, a]);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    vl.sort(function (a, b) {
 | 
				
			||||||
 | 
					        a = a[0];
 | 
				
			||||||
 | 
					        b = b[0];
 | 
				
			||||||
 | 
					        if (a === null)
 | 
				
			||||||
 | 
					            return -1;
 | 
				
			||||||
 | 
					        if (b === null)
 | 
				
			||||||
 | 
					            return 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (stype == 'int') {
 | 
				
			||||||
 | 
					            return reverse * (a - b);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return reverse * (a.localeCompare(b));
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
    for (i = 0; i < tr.length; ++i) tb.appendChild(tr[i]);
 | 
					    for (i = 0; i < tr.length; ++i) tb.appendChild(tr[vl[i][1]]);
 | 
				
			||||||
 | 
					    if (cb) cb();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
function makeSortable(table) {
 | 
					function makeSortable(table, cb) {
 | 
				
			||||||
    var th = table.tHead, i;
 | 
					    var th = table.tHead, i;
 | 
				
			||||||
    th && (th = th.rows[0]) && (th = th.cells);
 | 
					    th && (th = th.rows[0]) && (th = th.cells);
 | 
				
			||||||
    if (th) i = th.length;
 | 
					    if (th) i = th.length;
 | 
				
			||||||
    else return; // if no `<thead>` then do nothing
 | 
					    else return; // if no `<thead>` then do nothing
 | 
				
			||||||
    while (--i >= 0) (function (i) {
 | 
					    while (--i >= 0) (function (i) {
 | 
				
			||||||
        th[i].onclick = function () {
 | 
					        th[i].onclick = function (e) {
 | 
				
			||||||
            sortTable(table, i);
 | 
					            ev(e);
 | 
				
			||||||
 | 
					            sortTable(table, i, cb);
 | 
				
			||||||
        };
 | 
					        };
 | 
				
			||||||
    }(i));
 | 
					    }(i));
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					(function () {
 | 
				
			||||||
 | 
					    var ops = document.querySelectorAll('#ops>a');
 | 
				
			||||||
 | 
					    for (var a = 0; a < ops.length; a++) {
 | 
				
			||||||
 | 
					        ops[a].onclick = opclick;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					})();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function opclick(e) {
 | 
				
			||||||
 | 
					    ev(e);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    var dest = this.getAttribute('data-dest');
 | 
				
			||||||
 | 
					    goto(dest);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    swrite('opmode', dest || null);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    var input = document.querySelector('.opview.act input:not([type="hidden"])')
 | 
				
			||||||
 | 
					    if (input)
 | 
				
			||||||
 | 
					        input.focus();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function goto(dest) {
 | 
				
			||||||
 | 
					    var obj = document.querySelectorAll('.opview.act');
 | 
				
			||||||
 | 
					    for (var a = obj.length - 1; a >= 0; a--)
 | 
				
			||||||
 | 
					        obj[a].classList.remove('act');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    obj = document.querySelectorAll('#ops>a');
 | 
				
			||||||
 | 
					    for (var a = obj.length - 1; a >= 0; a--)
 | 
				
			||||||
 | 
					        obj[a].classList.remove('act');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (dest) {
 | 
				
			||||||
 | 
					        var ui = ebi('op_' + dest);
 | 
				
			||||||
 | 
					        ui.classList.add('act');
 | 
				
			||||||
 | 
					        document.querySelector('#ops>a[data-dest=' + dest + ']').classList.add('act');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        var fn = window['goto_' + dest];
 | 
				
			||||||
 | 
					        if (fn)
 | 
				
			||||||
 | 
					            fn();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (window['treectl'])
 | 
				
			||||||
 | 
					        treectl.onscroll();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					(function () {
 | 
				
			||||||
 | 
					    goto();
 | 
				
			||||||
 | 
					    var op = sread('opmode');
 | 
				
			||||||
 | 
					    if (op !== null && op !== '.')
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            goto(op);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        catch (ex) { }
 | 
				
			||||||
 | 
					})();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function linksplit(rp) {
 | 
				
			||||||
 | 
					    var ret = [];
 | 
				
			||||||
 | 
					    var apath = '/';
 | 
				
			||||||
 | 
					    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
 | 
					gzip -d < .hist/up2k.snap | jq -r '.[].name' | while IFS= read -r f; do wc -c -- "$f" | grep -qiE '^[^0-9a-z]*0' && rm -f -- "$f"; done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					##
 | 
				
			||||||
 | 
					## detect partial uploads based on file contents
 | 
				
			||||||
 | 
					##  (in case of context loss or old copyparties)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo; find -type f | while IFS= read -r x; do printf '\033[A\033[36m%s\033[K\033[0m\n' "$x"; tail -c$((1024*1024)) <"$x" | xxd -a | awk 'NR==1&&/^[0: ]+.{16}$/{next} NR==2&&/^\*$/{next} NR==3&&/^[0f]+: [0 ]+65 +.{16}$/{next} {e=1} END {exit e}' || continue; printf '\033[A\033[31msus:\033[33m %s \033[0m\n\n' "$x"; done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
##
 | 
					##
 | 
				
			||||||
## create a test payload
 | 
					## create a test payload
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -60,6 +67,36 @@ wget -S --header='Accept-Encoding: gzip' -U 'MSIE 6.0; SV1' http://127.0.0.1:392
 | 
				
			|||||||
shab64() { sp=$1; f="$2"; v=0; sz=$(stat -c%s "$f"); while true; do w=$((v+sp*1024*1024)); printf $(tail -c +$((v+1)) "$f" | head -c $((w-v)) | sha512sum | cut -c-64 | sed -r 's/ .*//;s/(..)/\\x\1/g') | base64 -w0 | cut -c-43 | tr '+/' '-_'; v=$w; [ $v -lt $sz ] || break; done; }
 | 
					shab64() { sp=$1; f="$2"; v=0; sz=$(stat -c%s "$f"); while true; do w=$((v+sp*1024*1024)); printf $(tail -c +$((v+1)) "$f" | head -c $((w-v)) | sha512sum | cut -c-64 | sed -r 's/ .*//;s/(..)/\\x\1/g') | base64 -w0 | cut -c-43 | tr '+/' '-_'; v=$w; [ $v -lt $sz ] || break; done; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					##
 | 
				
			||||||
 | 
					## poll url for performance issues
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					command -v gdate && date() { gdate "$@"; }; while true; do t=$(date +%s.%N); (time wget http://127.0.0.1:3923/?ls -qO- | jq -C '.files[]|{sz:.sz,ta:.tags.artist,tb:.tags.".bpm"}|del(.[]|select(.==null))' | awk -F\" '/"/{t[$2]++} END {for (k in t){v=t[k];p=sprintf("%" (v+1) "s",v);gsub(/ /,"#",p);printf "\033[36m%s\033[33m%s   ",k,p}}') 2>&1 | awk -v ts=$t 'NR==1{t1=$0} NR==2{sub(/.*0m/,"");sub(/s$/,"");t2=$0;c=2; if(t2>0.3){c=3} if(t2>0.8){c=1} } END{sub(/[0-9]{6}$/,"",ts);printf "%s   \033[3%dm%s   %s\033[0m\n",ts,c,t2,t1}'; sleep 0.1 || break; done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					##
 | 
				
			||||||
 | 
					## sqlite3 stuff
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# find dupe metadata keys
 | 
				
			||||||
 | 
					sqlite3 up2k.db 'select mt1.w, mt1.k, mt1.v, mt2.v from mt mt1 inner join mt mt2 on mt1.w = mt2.w where mt1.k = mt2.k and mt1.rowid != mt2.rowid'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# partial reindex by deleting all tags for a list of files
 | 
				
			||||||
 | 
					time sqlite3 up2k.db 'select mt1.w from mt mt1 inner join mt mt2 on mt1.w = mt2.w where mt1.k = +mt2.k and mt1.rowid != mt2.rowid'  > warks
 | 
				
			||||||
 | 
					cat warks | while IFS= read -r x; do sqlite3 up2k.db "delete from mt where w = '$x'"; done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# dump all dbs
 | 
				
			||||||
 | 
					find -iname up2k.db | while IFS= read -r x; do sqlite3 "$x" 'select substr(w,1,12), rd, fn from up' | sed -r 's/\|/ \| /g' | while IFS= read -r y; do printf '%s | %s\n' "$x" "$y"; done; done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					##
 | 
				
			||||||
 | 
					## media
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# split track into test files
 | 
				
			||||||
 | 
					e=6; s=10; d=~/dev/copyparty/srv/aus; n=1; p=0; e=$((e*60)); rm -rf $d; mkdir $d; while true; do ffmpeg -hide_banner -ss $p -i 'nervous_testpilot - office.mp3' -c copy -t $s $d/$(printf %04d $n).mp3; n=$((n+1)); p=$((p+s)); [ $p -gt $e ] && break; done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-v srv/aus:aus:r:ce2dsa:ce2ts:cmtp=fgsfds=bin/mtag/sleep.py
 | 
				
			||||||
 | 
					sqlite3 .hist/up2k.db 'select * from mt where k="fgsfds" or k="t:mtp"' | tee /dev/stderr | wc -l
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
##
 | 
					##
 | 
				
			||||||
## vscode
 | 
					## vscode
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -89,6 +126,18 @@ for d in /usr /var; do find $d -type f -size +30M 2>/dev/null; done | while IFS=
 | 
				
			|||||||
brew install python@2
 | 
					brew install python@2
 | 
				
			||||||
pip install virtualenv
 | 
					pip install virtualenv
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# readme toc
 | 
				
			||||||
 | 
					cat README.md | awk '!/^#/{next} {lv=length($1);sub(/[^ ]+ /,"");bab=$0;gsub(/ /,"-",bab)} {printf "%" ((lv-1)*4+1) "s [%s](#%s)\n", "*",$0,bab}'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# fix firefox phantom breakpoints,
 | 
				
			||||||
 | 
					# suggestions from bugtracker, doesnt work (debugger is not attachable)
 | 
				
			||||||
 | 
					devtools settings >> advanced >> enable browser chrome debugging + enable remote debugging
 | 
				
			||||||
 | 
					burger > developer >> browser toolbox  (ctrl-alt-shift-i)
 | 
				
			||||||
 | 
					iframe btn topright >> chrome://devtools/content/debugger/index.html
 | 
				
			||||||
 | 
					dbg.asyncStore.pendingBreakpoints = {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# fix firefox phantom breakpoints
 | 
				
			||||||
 | 
					about:config >> devtools.debugger.prefs-schema-version = -1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
##
 | 
					##
 | 
				
			||||||
## http 206
 | 
					## http 206
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -20,6 +20,7 @@ set -e
 | 
				
			|||||||
# -rwxr-xr-x  0 ed ed  183808 Nov 19 00:43 copyparty-extras/sfx-lite/copyparty-sfx.py
 | 
					# -rwxr-xr-x  0 ed ed  183808 Nov 19 00:43 copyparty-extras/sfx-lite/copyparty-sfx.py
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					command -v gnutar && tar() { gnutar "$@"; }
 | 
				
			||||||
command -v gtar && tar() { gtar "$@"; }
 | 
					command -v gtar && tar() { gtar "$@"; }
 | 
				
			||||||
command -v gsed && sed() { gsed "$@"; }
 | 
					command -v gsed && sed() { gsed "$@"; }
 | 
				
			||||||
td="$(mktemp -d)"
 | 
					td="$(mktemp -d)"
 | 
				
			||||||
@@ -29,11 +30,11 @@ pwd
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
dl_text() {
 | 
					dl_text() {
 | 
				
			||||||
	command -v curl && exec curl "$@"
 | 
						command -v curl >/dev/null && exec curl "$@"
 | 
				
			||||||
	exec wget -O- "$@"
 | 
						exec wget -O- "$@"
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
dl_files() {
 | 
					dl_files() {
 | 
				
			||||||
	command -v curl && exec curl -L --remote-name-all "$@"
 | 
						command -v curl >/dev/null && exec curl -L --remote-name-all "$@"
 | 
				
			||||||
	exec wget "$@"
 | 
						exec wget "$@"
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
export -f dl_files
 | 
					export -f dl_files
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,12 +1,10 @@
 | 
				
			|||||||
FROM    alpine:3.11
 | 
					FROM    alpine:3.13
 | 
				
			||||||
WORKDIR /z
 | 
					WORKDIR /z
 | 
				
			||||||
ENV     ver_asmcrypto=2821dd1dedd1196c378f5854037dda5c869313f3 \
 | 
					ENV     ver_asmcrypto=5b994303a9d3e27e0915f72a10b6c2c51535a4dc \
 | 
				
			||||||
        ver_markdownit=10.0.0 \
 | 
					 | 
				
			||||||
        ver_showdown=1.9.1 \
 | 
					 | 
				
			||||||
        ver_marked=1.1.0 \
 | 
					        ver_marked=1.1.0 \
 | 
				
			||||||
        ver_ogvjs=1.6.1 \
 | 
					        ver_ogvjs=1.8.0 \
 | 
				
			||||||
        ver_mde=2.10.1 \
 | 
					        ver_mde=2.14.0 \
 | 
				
			||||||
        ver_codemirror=5.53.2 \
 | 
					        ver_codemirror=5.59.3 \
 | 
				
			||||||
        ver_fontawesome=5.13.0 \
 | 
					        ver_fontawesome=5.13.0 \
 | 
				
			||||||
        ver_zopfli=1.0.3
 | 
					        ver_zopfli=1.0.3
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -17,7 +15,7 @@ RUN     mkdir -p /z/dist/no-pk \
 | 
				
			|||||||
        && wget https://fonts.gstatic.com/s/sourcecodepro/v11/HI_SiYsKILxRpg3hIP6sJ7fM7PqlPevW.woff2 -O scp.woff2 \
 | 
					        && wget https://fonts.gstatic.com/s/sourcecodepro/v11/HI_SiYsKILxRpg3hIP6sJ7fM7PqlPevW.woff2 -O scp.woff2 \
 | 
				
			||||||
        && apk add cmake make g++ git bash npm patch wget tar pigz brotli gzip unzip python3 python3-dev brotli py3-brotli \
 | 
					        && apk add cmake make g++ git bash npm patch wget tar pigz brotli gzip unzip python3 python3-dev brotli py3-brotli \
 | 
				
			||||||
        && wget https://github.com/brion/ogv.js/releases/download/$ver_ogvjs/ogvjs-$ver_ogvjs.zip -O ogvjs.zip \
 | 
					        && wget https://github.com/brion/ogv.js/releases/download/$ver_ogvjs/ogvjs-$ver_ogvjs.zip -O ogvjs.zip \
 | 
				
			||||||
        && wget https://github.com/asmcrypto/asmcrypto.js/archive/$ver_asmcrypto.tar.gz -O asmcrypto.tgz \
 | 
					        && wget https://github.com/openpgpjs/asmcrypto.js/archive/$ver_asmcrypto.tar.gz -O asmcrypto.tgz \
 | 
				
			||||||
        && wget https://github.com/markedjs/marked/archive/v$ver_marked.tar.gz -O marked.tgz \
 | 
					        && wget https://github.com/markedjs/marked/archive/v$ver_marked.tar.gz -O marked.tgz \
 | 
				
			||||||
        && wget https://github.com/Ionaru/easy-markdown-editor/archive/$ver_mde.tar.gz -O mde.tgz \
 | 
					        && wget https://github.com/Ionaru/easy-markdown-editor/archive/$ver_mde.tar.gz -O mde.tgz \
 | 
				
			||||||
        && wget https://github.com/codemirror/CodeMirror/archive/$ver_codemirror.tar.gz -O codemirror.tgz \
 | 
					        && wget https://github.com/codemirror/CodeMirror/archive/$ver_codemirror.tar.gz -O codemirror.tgz \
 | 
				
			||||||
@@ -52,6 +50,7 @@ RUN     tar -xf zopfli.tgz \
 | 
				
			|||||||
            -S . \
 | 
					            -S . \
 | 
				
			||||||
        && make -C build \
 | 
					        && make -C build \
 | 
				
			||||||
        && make -C build install \
 | 
					        && make -C build install \
 | 
				
			||||||
 | 
					        && python3 -m ensurepip \
 | 
				
			||||||
        && python3 -m pip install fonttools zopfli
 | 
					        && python3 -m pip install fonttools zopfli
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,6 @@
 | 
				
			|||||||
diff -NarU2 CodeMirror-orig/mode/gfm/gfm.js CodeMirror-edit/mode/gfm/gfm.js
 | 
					diff -NarU2 codemirror-5.59.3-orig/mode/gfm/gfm.js codemirror-5.59.3/mode/gfm/gfm.js
 | 
				
			||||||
--- CodeMirror-orig/mode/gfm/gfm.js	2020-04-21 12:47:20.000000000 +0200
 | 
					--- codemirror-5.59.3-orig/mode/gfm/gfm.js	2021-02-20 21:24:57.000000000 +0000
 | 
				
			||||||
+++ CodeMirror-edit/mode/gfm/gfm.js	2020-05-02 02:13:32.142131800 +0200
 | 
					+++ codemirror-5.59.3/mode/gfm/gfm.js	2021-02-21 20:42:02.166174775 +0000
 | 
				
			||||||
@@ -97,5 +97,5 @@
 | 
					@@ -97,5 +97,5 @@
 | 
				
			||||||
         }
 | 
					         }
 | 
				
			||||||
       }
 | 
					       }
 | 
				
			||||||
@@ -15,9 +15,9 @@ diff -NarU2 CodeMirror-orig/mode/gfm/gfm.js CodeMirror-edit/mode/gfm/gfm.js
 | 
				
			|||||||
+      }*/
 | 
					+      }*/
 | 
				
			||||||
       stream.next();
 | 
					       stream.next();
 | 
				
			||||||
       return null;
 | 
					       return null;
 | 
				
			||||||
diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js
 | 
					diff -NarU2 codemirror-5.59.3-orig/mode/meta.js codemirror-5.59.3/mode/meta.js
 | 
				
			||||||
--- CodeMirror-orig/mode/meta.js	2020-04-21 12:47:20.000000000 +0200
 | 
					--- codemirror-5.59.3-orig/mode/meta.js	2021-02-20 21:24:57.000000000 +0000
 | 
				
			||||||
+++ CodeMirror-edit/mode/meta.js	2020-05-02 03:56:58.852408400 +0200
 | 
					+++ codemirror-5.59.3/mode/meta.js	2021-02-21 20:42:54.798742821 +0000
 | 
				
			||||||
@@ -13,4 +13,5 @@
 | 
					@@ -13,4 +13,5 @@
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
   CodeMirror.modeInfo = [
 | 
					   CodeMirror.modeInfo = [
 | 
				
			||||||
@@ -28,7 +28,7 @@ diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js
 | 
				
			|||||||
     {name: "Gas", mime: "text/x-gas", mode: "gas", ext: ["s"]},
 | 
					     {name: "Gas", mime: "text/x-gas", mode: "gas", ext: ["s"]},
 | 
				
			||||||
     {name: "Gherkin", mime: "text/x-feature", mode: "gherkin", ext: ["feature"]},
 | 
					     {name: "Gherkin", mime: "text/x-feature", mode: "gherkin", ext: ["feature"]},
 | 
				
			||||||
+    */
 | 
					+    */
 | 
				
			||||||
     {name: "GitHub Flavored Markdown", mime: "text/x-gfm", mode: "gfm", file: /^(readme|contributing|history).md$/i},
 | 
					     {name: "GitHub Flavored Markdown", mime: "text/x-gfm", mode: "gfm", file: /^(readme|contributing|history)\.md$/i},
 | 
				
			||||||
+    /*
 | 
					+    /*
 | 
				
			||||||
     {name: "Go", mime: "text/x-go", mode: "go", ext: ["go"]},
 | 
					     {name: "Go", mime: "text/x-go", mode: "go", ext: ["go"]},
 | 
				
			||||||
     {name: "Groovy", mime: "text/x-groovy", mode: "groovy", ext: ["groovy", "gradle"], file: /^Jenkinsfile$/},
 | 
					     {name: "Groovy", mime: "text/x-groovy", mode: "groovy", ext: ["groovy", "gradle"], file: /^Jenkinsfile$/},
 | 
				
			||||||
@@ -56,16 +56,16 @@ diff -NarU2 CodeMirror-orig/mode/meta.js CodeMirror-edit/mode/meta.js
 | 
				
			|||||||
+    /*
 | 
					+    /*
 | 
				
			||||||
     {name: "XQuery", mime: "application/xquery", mode: "xquery", ext: ["xy", "xquery"]},
 | 
					     {name: "XQuery", mime: "application/xquery", mode: "xquery", ext: ["xy", "xquery"]},
 | 
				
			||||||
     {name: "Yacas", mime: "text/x-yacas", mode: "yacas", ext: ["ys"]},
 | 
					     {name: "Yacas", mime: "text/x-yacas", mode: "yacas", ext: ["ys"]},
 | 
				
			||||||
@@ -171,4 +180,5 @@
 | 
					@@ -172,4 +181,5 @@
 | 
				
			||||||
     {name: "xu", mime: "text/x-xu", mode: "mscgen", ext: ["xu"]},
 | 
					     {name: "msgenny", mime: "text/x-msgenny", mode: "mscgen", ext: ["msgenny"]},
 | 
				
			||||||
     {name: "msgenny", mime: "text/x-msgenny", mode: "mscgen", ext: ["msgenny"]}
 | 
					     {name: "WebAssembly", mime: "text/webassembly", mode: "wast", ext: ["wat", "wast"]},
 | 
				
			||||||
+    */
 | 
					+    */
 | 
				
			||||||
   ];
 | 
					   ];
 | 
				
			||||||
   // Ensure all modes have a mime property for backwards compatibility
 | 
					   // Ensure all modes have a mime property for backwards compatibility
 | 
				
			||||||
diff -NarU2 CodeMirror-orig/src/display/selection.js CodeMirror-edit/src/display/selection.js
 | 
					diff -NarU2 codemirror-5.59.3-orig/src/display/selection.js codemirror-5.59.3/src/display/selection.js
 | 
				
			||||||
--- CodeMirror-orig/src/display/selection.js	2020-04-21 12:47:20.000000000 +0200
 | 
					--- codemirror-5.59.3-orig/src/display/selection.js	2021-02-20 21:24:57.000000000 +0000
 | 
				
			||||||
+++ CodeMirror-edit/src/display/selection.js	2020-05-02 03:27:30.144662800 +0200
 | 
					+++ codemirror-5.59.3/src/display/selection.js	2021-02-21 20:44:14.860894328 +0000
 | 
				
			||||||
@@ -83,29 +83,21 @@
 | 
					@@ -84,29 +84,21 @@
 | 
				
			||||||
     let order = getOrder(lineObj, doc.direction)
 | 
					     let order = getOrder(lineObj, doc.direction)
 | 
				
			||||||
     iterateBidiSections(order, fromArg || 0, toArg == null ? lineLen : toArg, (from, to, dir, i) => {
 | 
					     iterateBidiSections(order, fromArg || 0, toArg == null ? lineLen : toArg, (from, to, dir, i) => {
 | 
				
			||||||
-      let ltr = dir == "ltr"
 | 
					-      let ltr = dir == "ltr"
 | 
				
			||||||
@@ -105,24 +105,24 @@ diff -NarU2 CodeMirror-orig/src/display/selection.js CodeMirror-edit/src/display
 | 
				
			|||||||
+          botRight = openEnd && last ? rightSide : toPos.right
 | 
					+          botRight = openEnd && last ? rightSide : toPos.right
 | 
				
			||||||
         add(topLeft, fromPos.top, topRight - topLeft, fromPos.bottom)
 | 
					         add(topLeft, fromPos.top, topRight - topLeft, fromPos.bottom)
 | 
				
			||||||
         if (fromPos.bottom < toPos.top) add(leftSide, fromPos.bottom, null, toPos.top)
 | 
					         if (fromPos.bottom < toPos.top) add(leftSide, fromPos.bottom, null, toPos.top)
 | 
				
			||||||
diff -NarU2 CodeMirror-orig/src/input/ContentEditableInput.js CodeMirror-edit/src/input/ContentEditableInput.js
 | 
					diff -NarU2 codemirror-5.59.3-orig/src/input/ContentEditableInput.js codemirror-5.59.3/src/input/ContentEditableInput.js
 | 
				
			||||||
--- CodeMirror-orig/src/input/ContentEditableInput.js	2020-04-21 12:47:20.000000000 +0200
 | 
					--- codemirror-5.59.3-orig/src/input/ContentEditableInput.js	2021-02-20 21:24:57.000000000 +0000
 | 
				
			||||||
+++ CodeMirror-edit/src/input/ContentEditableInput.js	2020-05-02 03:33:05.707995500 +0200
 | 
					+++ codemirror-5.59.3/src/input/ContentEditableInput.js	2021-02-21 20:44:33.273953867 +0000
 | 
				
			||||||
@@ -391,4 +391,5 @@
 | 
					@@ -399,4 +399,5 @@
 | 
				
			||||||
   let info = mapFromLineView(view, line, pos.line)
 | 
					   let info = mapFromLineView(view, line, pos.line)
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
+  /*
 | 
					+  /*
 | 
				
			||||||
   let order = getOrder(line, cm.doc.direction), side = "left"
 | 
					   let order = getOrder(line, cm.doc.direction), side = "left"
 | 
				
			||||||
   if (order) {
 | 
					   if (order) {
 | 
				
			||||||
@@ -396,4 +397,5 @@
 | 
					@@ -404,4 +405,5 @@
 | 
				
			||||||
     side = partPos % 2 ? "right" : "left"
 | 
					     side = partPos % 2 ? "right" : "left"
 | 
				
			||||||
   }
 | 
					   }
 | 
				
			||||||
+  */
 | 
					+  */
 | 
				
			||||||
   let result = nodeAndOffsetInLineMap(info.map, pos.ch, side)
 | 
					   let result = nodeAndOffsetInLineMap(info.map, pos.ch, side)
 | 
				
			||||||
   result.offset = result.collapse == "right" ? result.end : result.start
 | 
					   result.offset = result.collapse == "right" ? result.end : result.start
 | 
				
			||||||
diff -NarU2 CodeMirror-orig/src/input/movement.js CodeMirror-edit/src/input/movement.js
 | 
					diff -NarU2 codemirror-5.59.3-orig/src/input/movement.js codemirror-5.59.3/src/input/movement.js
 | 
				
			||||||
--- CodeMirror-orig/src/input/movement.js	2020-04-21 12:47:20.000000000 +0200
 | 
					--- codemirror-5.59.3-orig/src/input/movement.js	2021-02-20 21:24:57.000000000 +0000
 | 
				
			||||||
+++ CodeMirror-edit/src/input/movement.js	2020-05-02 03:31:19.710773500 +0200
 | 
					+++ codemirror-5.59.3/src/input/movement.js	2021-02-21 20:45:12.763093671 +0000
 | 
				
			||||||
@@ -15,4 +15,5 @@
 | 
					@@ -15,4 +15,5 @@
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
 export function endOfLine(visually, cm, lineObj, lineNo, dir) {
 | 
					 export function endOfLine(visually, cm, lineObj, lineNo, dir) {
 | 
				
			||||||
@@ -146,9 +146,9 @@ diff -NarU2 CodeMirror-orig/src/input/movement.js CodeMirror-edit/src/input/move
 | 
				
			|||||||
   return null
 | 
					   return null
 | 
				
			||||||
+  */
 | 
					+  */
 | 
				
			||||||
 }
 | 
					 }
 | 
				
			||||||
diff -NarU2 CodeMirror-orig/src/line/line_data.js CodeMirror-edit/src/line/line_data.js
 | 
					diff -NarU2 codemirror-5.59.3-orig/src/line/line_data.js codemirror-5.59.3/src/line/line_data.js
 | 
				
			||||||
--- CodeMirror-orig/src/line/line_data.js	2020-04-21 12:47:20.000000000 +0200
 | 
					--- codemirror-5.59.3-orig/src/line/line_data.js	2021-02-20 21:24:57.000000000 +0000
 | 
				
			||||||
+++ CodeMirror-edit/src/line/line_data.js	2020-05-02 03:17:02.785065000 +0200
 | 
					+++ codemirror-5.59.3/src/line/line_data.js	2021-02-21 20:45:36.472549599 +0000
 | 
				
			||||||
@@ -79,6 +79,6 @@
 | 
					@@ -79,6 +79,6 @@
 | 
				
			||||||
     // Optionally wire in some hacks into the token-rendering
 | 
					     // Optionally wire in some hacks into the token-rendering
 | 
				
			||||||
     // algorithm, to deal with browser quirks.
 | 
					     // algorithm, to deal with browser quirks.
 | 
				
			||||||
@@ -158,9 +158,9 @@ diff -NarU2 CodeMirror-orig/src/line/line_data.js CodeMirror-edit/src/line/line_
 | 
				
			|||||||
+    //  builder.addToken = buildTokenBadBidi(builder.addToken, order)
 | 
					+    //  builder.addToken = buildTokenBadBidi(builder.addToken, order)
 | 
				
			||||||
     builder.map = []
 | 
					     builder.map = []
 | 
				
			||||||
     let allowFrontierUpdate = lineView != cm.display.externalMeasured && lineNo(line)
 | 
					     let allowFrontierUpdate = lineView != cm.display.externalMeasured && lineNo(line)
 | 
				
			||||||
diff -NarU2 CodeMirror-orig/src/measurement/position_measurement.js CodeMirror-edit/src/measurement/position_measurement.js
 | 
					diff -NarU2 codemirror-5.59.3-orig/src/measurement/position_measurement.js codemirror-5.59.3/src/measurement/position_measurement.js
 | 
				
			||||||
--- CodeMirror-orig/src/measurement/position_measurement.js	2020-04-21 12:47:20.000000000 +0200
 | 
					--- codemirror-5.59.3-orig/src/measurement/position_measurement.js	2021-02-20 21:24:57.000000000 +0000
 | 
				
			||||||
+++ CodeMirror-edit/src/measurement/position_measurement.js	2020-05-02 03:35:20.674159600 +0200
 | 
					+++ codemirror-5.59.3/src/measurement/position_measurement.js	2021-02-21 20:50:52.372945293 +0000
 | 
				
			||||||
@@ -380,5 +380,6 @@
 | 
					@@ -380,5 +380,6 @@
 | 
				
			||||||
     sticky = "after"
 | 
					     sticky = "after"
 | 
				
			||||||
   }
 | 
					   }
 | 
				
			||||||
@@ -199,9 +199,9 @@ diff -NarU2 CodeMirror-orig/src/measurement/position_measurement.js CodeMirror-e
 | 
				
			|||||||
+*/
 | 
					+*/
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
 let measureText
 | 
					 let measureText
 | 
				
			||||||
diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js
 | 
					diff -NarU2 codemirror-5.59.3-orig/src/util/bidi.js codemirror-5.59.3/src/util/bidi.js
 | 
				
			||||||
--- CodeMirror-orig/src/util/bidi.js	2020-04-21 12:47:20.000000000 +0200
 | 
					--- codemirror-5.59.3-orig/src/util/bidi.js	2021-02-20 21:24:57.000000000 +0000
 | 
				
			||||||
+++ CodeMirror-edit/src/util/bidi.js	2020-05-02 03:12:44.418649800 +0200
 | 
					+++ codemirror-5.59.3/src/util/bidi.js	2021-02-21 20:52:18.168092225 +0000
 | 
				
			||||||
@@ -4,5 +4,5 @@
 | 
					@@ -4,5 +4,5 @@
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
 export function iterateBidiSections(order, from, to, f) {
 | 
					 export function iterateBidiSections(order, from, to, f) {
 | 
				
			||||||
@@ -239,20 +239,19 @@ diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js
 | 
				
			|||||||
+  var fun = function(str, direction) {
 | 
					+  var fun = function(str, direction) {
 | 
				
			||||||
     let outerType = direction == "ltr" ? "L" : "R"
 | 
					     let outerType = direction == "ltr" ? "L" : "R"
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
@@ -204,12 +210,16 @@
 | 
					@@ -204,5 +210,11 @@
 | 
				
			||||||
     return direction == "rtl" ? order.reverse() : order
 | 
					     return direction == "rtl" ? order.reverse() : order
 | 
				
			||||||
   }
 | 
					   }
 | 
				
			||||||
-})()
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
+  return function(str, direction) {
 | 
					+  return function(str, direction) {
 | 
				
			||||||
+    var ret = fun(str, direction);
 | 
					+    var ret = fun(str, direction);
 | 
				
			||||||
+    console.log("bidiOrdering inner ([%s], %s) => [%s]", str, direction, ret);
 | 
					+    console.log("bidiOrdering inner ([%s], %s) => [%s]", str, direction, ret);
 | 
				
			||||||
+    return ret;
 | 
					+    return ret;
 | 
				
			||||||
+  }
 | 
					+  }
 | 
				
			||||||
+})()
 | 
					 })()
 | 
				
			||||||
+*/
 | 
					+*/
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 // Get the bidi ordering for the given line (and cache it). Returns
 | 
					 // Get the bidi ordering for the given line (and cache it). Returns
 | 
				
			||||||
 // false for lines that are fully left-to-right, and an array of
 | 
					@@ -210,6 +222,4 @@
 | 
				
			||||||
 // BidiSpan objects otherwise.
 | 
					 // BidiSpan objects otherwise.
 | 
				
			||||||
 export function getOrder(line, direction) {
 | 
					 export function getOrder(line, direction) {
 | 
				
			||||||
-  let order = line.order
 | 
					-  let order = line.order
 | 
				
			||||||
@@ -260,9 +259,9 @@ diff -NarU2 CodeMirror-orig/src/util/bidi.js CodeMirror-edit/src/util/bidi.js
 | 
				
			|||||||
-  return order
 | 
					-  return order
 | 
				
			||||||
+  return false;
 | 
					+  return false;
 | 
				
			||||||
 }
 | 
					 }
 | 
				
			||||||
diff -NarU2 CodeMirror-orig/src/util/feature_detection.js CodeMirror-edit/src/util/feature_detection.js
 | 
					diff -NarU2 codemirror-5.59.3-orig/src/util/feature_detection.js codemirror-5.59.3/src/util/feature_detection.js
 | 
				
			||||||
--- CodeMirror-orig/src/util/feature_detection.js	2020-04-21 12:47:20.000000000 +0200
 | 
					--- codemirror-5.59.3-orig/src/util/feature_detection.js	2021-02-20 21:24:57.000000000 +0000
 | 
				
			||||||
+++ CodeMirror-edit/src/util/feature_detection.js	2020-05-02 03:16:21.085621400 +0200
 | 
					+++ codemirror-5.59.3/src/util/feature_detection.js	2021-02-21 20:49:22.191269270 +0000
 | 
				
			||||||
@@ -25,4 +25,5 @@
 | 
					@@ -25,4 +25,5 @@
 | 
				
			||||||
 }
 | 
					 }
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,33 +1,57 @@
 | 
				
			|||||||
diff -NarU2 easymde-orig/gulpfile.js easymde-mod1/gulpfile.js
 | 
					diff -NarU2 easy-markdown-editor-2.14.0-orig/gulpfile.js easy-markdown-editor-2.14.0/gulpfile.js
 | 
				
			||||||
--- easymde-orig/gulpfile.js	2020-04-06 14:09:36.000000000 +0200
 | 
					--- easy-markdown-editor-2.14.0-orig/gulpfile.js	2021-02-14 12:11:48.000000000 +0000
 | 
				
			||||||
+++ easymde-mod1/gulpfile.js	2020-05-01 14:33:52.260175200 +0200
 | 
					+++ easy-markdown-editor-2.14.0/gulpfile.js	2021-02-21 20:55:37.134701007 +0000
 | 
				
			||||||
@@ -25,5 +25,4 @@
 | 
					@@ -25,5 +25,4 @@
 | 
				
			||||||
     './node_modules/codemirror/lib/codemirror.css',
 | 
					     './node_modules/codemirror/lib/codemirror.css',
 | 
				
			||||||
     './src/css/*.css',
 | 
					     './src/css/*.css',
 | 
				
			||||||
-    './node_modules/codemirror-spell-checker/src/css/spell-checker.css',
 | 
					-    './node_modules/codemirror-spell-checker/src/css/spell-checker.css',
 | 
				
			||||||
 ];
 | 
					 ];
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
diff -NarU2 easymde-orig/package.json easymde-mod1/package.json
 | 
					diff -NarU2 easy-markdown-editor-2.14.0-orig/package.json easy-markdown-editor-2.14.0/package.json
 | 
				
			||||||
--- easymde-orig/package.json	2020-04-06 14:09:36.000000000 +0200
 | 
					--- easy-markdown-editor-2.14.0-orig/package.json	2021-02-14 12:11:48.000000000 +0000
 | 
				
			||||||
+++ easymde-mod1/package.json	2020-05-01 14:33:57.189975800 +0200
 | 
					+++ easy-markdown-editor-2.14.0/package.json	2021-02-21 20:55:47.761190082 +0000
 | 
				
			||||||
@@ -21,5 +21,4 @@
 | 
					@@ -21,5 +21,4 @@
 | 
				
			||||||
     "dependencies": {
 | 
					     "dependencies": {
 | 
				
			||||||
         "codemirror": "^5.52.2",
 | 
					         "codemirror": "^5.59.2",
 | 
				
			||||||
-        "codemirror-spell-checker": "1.1.2",
 | 
					-        "codemirror-spell-checker": "1.1.2",
 | 
				
			||||||
         "marked": "^0.8.2"
 | 
					         "marked": "^2.0.0"
 | 
				
			||||||
     },
 | 
					     },
 | 
				
			||||||
diff -NarU2 easymde-orig/src/js/easymde.js easymde-mod1/src/js/easymde.js
 | 
					diff -NarU2 easy-markdown-editor-2.14.0-orig/src/js/easymde.js easy-markdown-editor-2.14.0/src/js/easymde.js
 | 
				
			||||||
--- easymde-orig/src/js/easymde.js	2020-04-06 14:09:36.000000000 +0200
 | 
					--- easy-markdown-editor-2.14.0-orig/src/js/easymde.js	2021-02-14 12:11:48.000000000 +0000
 | 
				
			||||||
+++ easymde-mod1/src/js/easymde.js	2020-05-01 14:34:19.878774400 +0200
 | 
					+++ easy-markdown-editor-2.14.0/src/js/easymde.js	2021-02-21 20:57:09.143171536 +0000
 | 
				
			||||||
@@ -11,5 +11,4 @@
 | 
					@@ -12,5 +12,4 @@
 | 
				
			||||||
 require('codemirror/mode/gfm/gfm.js');
 | 
					 require('codemirror/mode/gfm/gfm.js');
 | 
				
			||||||
 require('codemirror/mode/xml/xml.js');
 | 
					 require('codemirror/mode/xml/xml.js');
 | 
				
			||||||
-var CodeMirrorSpellChecker = require('codemirror-spell-checker');
 | 
					-var CodeMirrorSpellChecker = require('codemirror-spell-checker');
 | 
				
			||||||
 var marked = require('marked/lib/marked');
 | 
					 var marked = require('marked/lib/marked');
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
@@ -1889,18 +1888,7 @@
 | 
					@@ -1762,9 +1761,4 @@
 | 
				
			||||||
 | 
					         options.autosave.uniqueId = options.autosave.unique_id;
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
 | 
					-    // If overlay mode is specified and combine is not provided, default it to true
 | 
				
			||||||
 | 
					-    if (options.overlayMode && options.overlayMode.combine === undefined) {
 | 
				
			||||||
 | 
					-      options.overlayMode.combine = true;
 | 
				
			||||||
 | 
					-    }
 | 
				
			||||||
 | 
					-
 | 
				
			||||||
 | 
					     // Update this options
 | 
				
			||||||
 | 
					     this.options = options;
 | 
				
			||||||
 | 
					@@ -2003,28 +1997,7 @@
 | 
				
			||||||
     var mode, backdrop;
 | 
					     var mode, backdrop;
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					-    // CodeMirror overlay mode
 | 
				
			||||||
 | 
					-    if (options.overlayMode) {
 | 
				
			||||||
 | 
					-      CodeMirror.defineMode('overlay-mode', function(config) {
 | 
				
			||||||
 | 
					-        return CodeMirror.overlayMode(CodeMirror.getMode(config, options.spellChecker !== false ? 'spell-checker' : 'gfm'), options.overlayMode.mode, options.overlayMode.combine);
 | 
				
			||||||
 | 
					-      });
 | 
				
			||||||
 | 
					-
 | 
				
			||||||
 | 
					-      mode = 'overlay-mode';
 | 
				
			||||||
 | 
					-      backdrop = options.parsingConfig;
 | 
				
			||||||
 | 
					-      backdrop.gitHubSpice = false;
 | 
				
			||||||
 | 
					-    } else {
 | 
				
			||||||
 | 
					         mode = options.parsingConfig;
 | 
				
			||||||
 | 
					         mode.name = 'gfm';
 | 
				
			||||||
 | 
					         mode.gitHubSpice = false;
 | 
				
			||||||
 | 
					-    }
 | 
				
			||||||
-    if (options.spellChecker !== false) {
 | 
					-    if (options.spellChecker !== false) {
 | 
				
			||||||
-        mode = 'spell-checker';
 | 
					-        mode = 'spell-checker';
 | 
				
			||||||
-        backdrop = options.parsingConfig;
 | 
					-        backdrop = options.parsingConfig;
 | 
				
			||||||
@@ -37,16 +61,28 @@ diff -NarU2 easymde-orig/src/js/easymde.js easymde-mod1/src/js/easymde.js
 | 
				
			|||||||
-        CodeMirrorSpellChecker({
 | 
					-        CodeMirrorSpellChecker({
 | 
				
			||||||
-            codeMirrorInstance: CodeMirror,
 | 
					-            codeMirrorInstance: CodeMirror,
 | 
				
			||||||
-        });
 | 
					-        });
 | 
				
			||||||
-    } else {
 | 
					 | 
				
			||||||
         mode = options.parsingConfig;
 | 
					 | 
				
			||||||
         mode.name = 'gfm';
 | 
					 | 
				
			||||||
         mode.gitHubSpice = false;
 | 
					 | 
				
			||||||
-    }
 | 
					-    }
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
     // eslint-disable-next-line no-unused-vars
 | 
					     // eslint-disable-next-line no-unused-vars
 | 
				
			||||||
@@ -1927,5 +1915,4 @@
 | 
					diff -NarU2 easy-markdown-editor-2.14.0-orig/types/easymde.d.ts easy-markdown-editor-2.14.0/types/easymde.d.ts
 | 
				
			||||||
         configureMouse: configureMouse,
 | 
					--- easy-markdown-editor-2.14.0-orig/types/easymde.d.ts	2021-02-14 12:11:48.000000000 +0000
 | 
				
			||||||
         inputStyle: (options.inputStyle != undefined) ? options.inputStyle : isMobile() ? 'contenteditable' : 'textarea',
 | 
					+++ easy-markdown-editor-2.14.0/types/easymde.d.ts	2021-02-21 20:57:42.492620979 +0000
 | 
				
			||||||
-        spellcheck: (options.nativeSpellcheck != undefined) ? options.nativeSpellcheck : true,
 | 
					@@ -160,9 +160,4 @@
 | 
				
			||||||
     });
 | 
					     }
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
 | 
					-    interface OverlayModeOptions {
 | 
				
			||||||
 | 
					-      mode: CodeMirror.Mode<any>
 | 
				
			||||||
 | 
					-      combine?: boolean
 | 
				
			||||||
 | 
					-    }
 | 
				
			||||||
 | 
					-
 | 
				
			||||||
 | 
					     interface Options {
 | 
				
			||||||
 | 
					         autoDownloadFontAwesome?: boolean;
 | 
				
			||||||
 | 
					@@ -214,7 +209,5 @@
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					         promptTexts?: PromptTexts;
 | 
				
			||||||
 | 
					-        syncSideBySidePreviewScroll?: boolean;
 | 
				
			||||||
 | 
					-
 | 
				
			||||||
 | 
					-        overlayMode?: OverlayModeOptions
 | 
				
			||||||
 | 
					+        syncSideBySidePreviewScroll?: boolean
 | 
				
			||||||
 | 
					     }
 | 
				
			||||||
 | 
					 }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -86,6 +86,8 @@ function have() {
 | 
				
			|||||||
	python -c "import $1; $1; $1.__version__"
 | 
						python -c "import $1; $1; $1.__version__"
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					mv copyparty/web/deps/marked.full.js.gz srv/ || true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
. buildenv/bin/activate
 | 
					. buildenv/bin/activate
 | 
				
			||||||
have setuptools
 | 
					have setuptools
 | 
				
			||||||
have wheel
 | 
					have wheel
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -28,6 +28,13 @@ gtar=$(command -v gtar || command -v gnutar) || true
 | 
				
			|||||||
	unexpand() { gunexpand "$@"; }
 | 
						unexpand() { gunexpand "$@"; }
 | 
				
			||||||
	command -v grealpath >/dev/null &&
 | 
						command -v grealpath >/dev/null &&
 | 
				
			||||||
		realpath() { grealpath "$@"; }
 | 
							realpath() { grealpath "$@"; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						[ -e /opt/local/bin/bzip2 ] &&
 | 
				
			||||||
 | 
							bzip2() { /opt/local/bin/bzip2 "$@"; }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					pybin=$(command -v python3 || command -v python) || {
 | 
				
			||||||
 | 
						echo need python
 | 
				
			||||||
 | 
						exit 1
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
[ -e copyparty/__main__.py ] || cd ..
 | 
					[ -e copyparty/__main__.py ] || cd ..
 | 
				
			||||||
@@ -38,11 +45,15 @@ gtar=$(command -v gtar || command -v gnutar) || true
 | 
				
			|||||||
	exit 1
 | 
						exit 1
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					do_sh=1
 | 
				
			||||||
 | 
					do_py=1
 | 
				
			||||||
while [ ! -z "$1" ]; do
 | 
					while [ ! -z "$1" ]; do
 | 
				
			||||||
	[ "$1" = clean  ] && clean=1  && shift && continue
 | 
						[ "$1" = clean  ] && clean=1  && shift && continue
 | 
				
			||||||
	[ "$1" = re     ] && repack=1 && shift && continue
 | 
						[ "$1" = re     ] && repack=1 && shift && continue
 | 
				
			||||||
	[ "$1" = no-ogv ] && no_ogv=1 && shift && continue
 | 
						[ "$1" = no-ogv ] && no_ogv=1 && shift && continue
 | 
				
			||||||
	[ "$1" = no-cm  ] && no_cm=1  && shift && continue
 | 
						[ "$1" = no-cm  ] && no_cm=1  && shift && continue
 | 
				
			||||||
 | 
						[ "$1" = no-sh  ] && do_sh=   && shift && continue
 | 
				
			||||||
 | 
						[ "$1" = no-py  ] && do_py=   && shift && continue
 | 
				
			||||||
	break
 | 
						break
 | 
				
			||||||
done
 | 
					done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -62,28 +73,32 @@ cd sfx
 | 
				
			|||||||
	)/pe-copyparty"
 | 
						)/pe-copyparty"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	echo "repack of files in $old"
 | 
						echo "repack of files in $old"
 | 
				
			||||||
	cp -pR "$old/"*{jinja2,copyparty} .
 | 
						cp -pR "$old/"*{dep-j2,copyparty} .
 | 
				
			||||||
	mv {x.,}jinja2 2>/dev/null || true
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
[ $repack ] || {
 | 
					[ $repack ] || {
 | 
				
			||||||
	echo collecting jinja2
 | 
						echo collecting jinja2
 | 
				
			||||||
	f="../build/Jinja2-2.6.tar.gz"
 | 
						f="../build/Jinja2-2.11.3.tar.gz"
 | 
				
			||||||
	[ -e "$f" ] ||
 | 
						[ -e "$f" ] ||
 | 
				
			||||||
		(url=https://files.pythonhosted.org/packages/25/c8/212b1c2fd6df9eaf536384b6c6619c4e70a3afd2dffdd00e5296ffbae940/Jinja2-2.6.tar.gz;
 | 
							(url=https://files.pythonhosted.org/packages/4f/e7/65300e6b32e69768ded990494809106f87da1d436418d5f1367ed3966fd7/Jinja2-2.11.3.tar.gz;
 | 
				
			||||||
		wget -O$f "$url" || curl -L "$url" >$f)
 | 
							wget -O$f "$url" || curl -L "$url" >$f)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	tar -zxf $f
 | 
						tar -zxf $f
 | 
				
			||||||
	mv Jinja2-*/jinja2 .
 | 
						mv Jinja2-*/src/jinja2 .
 | 
				
			||||||
	rm -rf Jinja2-* jinja2/testsuite jinja2/_markupsafe/tests.py jinja2/_stringdefs.py
 | 
						rm -rf Jinja2-*
 | 
				
			||||||
	
 | 
						
 | 
				
			||||||
	f=jinja2/lexer.py
 | 
						echo collecting markupsafe
 | 
				
			||||||
	sed -r '/.*föö.*/    raise SyntaxError/' <$f >t
 | 
						f="../build/MarkupSafe-1.1.1.tar.gz"
 | 
				
			||||||
	tmv $f
 | 
						[ -e "$f" ] ||
 | 
				
			||||||
 | 
							(url=https://files.pythonhosted.org/packages/b9/2e/64db92e53b86efccfaea71321f597fa2e1b2bd3853d8ce658568f7a13094/MarkupSafe-1.1.1.tar.gz;
 | 
				
			||||||
 | 
							wget -O$f "$url" || curl -L "$url" >$f)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	f=jinja2/_markupsafe/_constants.py
 | 
						tar -zxf $f
 | 
				
			||||||
	awk '!/: [0-9]+,?$/ || /(amp|gt|lt|quot|apos|nbsp).:/' <$f >t
 | 
						mv MarkupSafe-*/src/markupsafe .
 | 
				
			||||||
	tmv $f
 | 
						rm -rf MarkupSafe-* markupsafe/_speedups.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						mkdir dep-j2/
 | 
				
			||||||
 | 
						mv {markupsafe,jinja2} dep-j2/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	# msys2 tar is bad, make the best of it
 | 
						# msys2 tar is bad, make the best of it
 | 
				
			||||||
	echo collecting source
 | 
						echo collecting source
 | 
				
			||||||
@@ -118,7 +133,7 @@ git describe --tags >/dev/null 2>/dev/null && {
 | 
				
			|||||||
		exit 1
 | 
							exit 1
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	dt="$(git log -1 --format=%cd --date=format:'%Y,%m,%d' | sed -E 's/,0?/, /g')"
 | 
						dt="$(git log -1 --format=%cd --date=short | sed -E 's/-0?/, /g')"
 | 
				
			||||||
	printf 'git %3s: \033[36m%s\033[0m\n' ver "$ver" dt "$dt"
 | 
						printf 'git %3s: \033[36m%s\033[0m\n' ver "$ver" dt "$dt"
 | 
				
			||||||
	sed -ri '
 | 
						sed -ri '
 | 
				
			||||||
		s/^(VERSION =)(.*)/#\1\2\n\1 ('"$t_ver"')/;
 | 
							s/^(VERSION =)(.*)/#\1\2\n\1 ('"$t_ver"')/;
 | 
				
			||||||
@@ -165,9 +180,19 @@ done
 | 
				
			|||||||
	sed -r '/edit2">edit \(fancy/d' <$f >t && tmv "$f"
 | 
						sed -r '/edit2">edit \(fancy/d' <$f >t && tmv "$f"
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					[ $repack ] ||
 | 
				
			||||||
 | 
					find | grep -E '\.py$' |
 | 
				
			||||||
 | 
					  grep -vE '__version__' |
 | 
				
			||||||
 | 
					  tr '\n' '\0' |
 | 
				
			||||||
 | 
					  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
 | 
				
			||||||
 | 
					tmv "$f"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# up2k goes from 28k to 22k laff
 | 
					# up2k goes from 28k to 22k laff
 | 
				
			||||||
echo entabbening
 | 
					echo entabbening
 | 
				
			||||||
find | grep -E '\.(js|css|html|py)$' | while IFS= read -r f; do
 | 
					find | grep -E '\.(js|css|html)$' | while IFS= read -r f; do
 | 
				
			||||||
	unexpand -t 4 --first-only <"$f" >t
 | 
						unexpand -t 4 --first-only <"$f" >t
 | 
				
			||||||
	tmv "$f"
 | 
						tmv "$f"
 | 
				
			||||||
done
 | 
					done
 | 
				
			||||||
@@ -177,29 +202,40 @@ args=(--owner=1000 --group=1000)
 | 
				
			|||||||
[ "$OSTYPE" = msys ] &&
 | 
					[ "$OSTYPE" = msys ] &&
 | 
				
			||||||
	args=()
 | 
						args=()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
tar -cf tar "${args[@]}" --numeric-owner copyparty jinja2
 | 
					tar -cf tar "${args[@]}" --numeric-owner copyparty dep-j2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
echo compressing tar
 | 
					echo compressing tar
 | 
				
			||||||
# detect best level; bzip2 -7 is usually better than -9
 | 
					# detect best level; bzip2 -7 is usually better than -9
 | 
				
			||||||
for n in {2..9}; do cp tar t.$n; bzip2 -$n t.$n & done; wait; mv -v $(ls -1S t.*.bz2 | tail -n 1) tar.bz2
 | 
					[ $do_py ] && { for n in {2..9}; do cp tar t.$n; bzip2 -$n t.$n & done; wait; mv -v $(ls -1S t.*.bz2 | tail -n 1) tar.bz2; }
 | 
				
			||||||
for n in {2..9}; do cp tar t.$n;  xz -ze$n t.$n & done; wait; mv -v $(ls -1S t.*.xz  | tail -n 1) tar.xz
 | 
					[ $do_sh ] && { for n in {2..9}; do cp tar t.$n;  xz -ze$n t.$n & done; wait; mv -v $(ls -1S t.*.xz  | tail -n 1) tar.xz; }
 | 
				
			||||||
rm t.*
 | 
					rm t.* || true
 | 
				
			||||||
 | 
					exts=()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					[ $do_sh ] && {
 | 
				
			||||||
 | 
					exts+=(sh)
 | 
				
			||||||
echo creating unix sfx
 | 
					echo creating unix sfx
 | 
				
			||||||
(
 | 
					(
 | 
				
			||||||
	sed "s/PACK_TS/$ts/; s/PACK_HTS/$hts/; s/CPP_VER/$ver/" <../scripts/sfx.sh |
 | 
						sed "s/PACK_TS/$ts/; s/PACK_HTS/$hts/; s/CPP_VER/$ver/" <../scripts/sfx.sh |
 | 
				
			||||||
	grep -E '^sfx_eof$' -B 9001;
 | 
						grep -E '^sfx_eof$' -B 9001;
 | 
				
			||||||
	cat tar.xz
 | 
						cat tar.xz
 | 
				
			||||||
) >$sfx_out.sh
 | 
					) >$sfx_out.sh
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					[ $do_py ] && {
 | 
				
			||||||
 | 
					exts+=(py)
 | 
				
			||||||
echo creating generic sfx
 | 
					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
 | 
					mv sfx.out $sfx_out.py
 | 
				
			||||||
chmod 755 $sfx_out.*
 | 
					chmod 755 $sfx_out.*
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
printf "done:\n"
 | 
					printf "done:\n"
 | 
				
			||||||
printf "  %s\n" "$(realpath $sfx_out)."{sh,py}
 | 
					for ext in ${exts[@]}; do
 | 
				
			||||||
# rm -rf *
 | 
						printf "  %s\n" "$(realpath $sfx_out)."$ext
 | 
				
			||||||
 | 
					done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# tar -tvf ../sfx/tar | sed -r 's/(.* ....-..-.. ..:.. )(.*)/\2 `` \1/' | sort | sed -r 's/(.*) `` (.*)/\2 \1/'| less
 | 
					# apk add bash python3 tar xz bzip2
 | 
				
			||||||
# for n in {1..9}; do tar -tf tar | grep -vE '/$' | sed -r 's/(.*)\.(.*)/\2.\1/' | sort | sed -r 's/([^\.]+)\.(.*)/\2.\1/' | tar -cT- | bzip2 -c$n | wc -c; done 
 | 
					# while true; do ./make-sfx.sh; for f in ..//dist/copyparty-sfx.{sh,py}; do mv $f $f.$(wc -c <$f | awk '{print$1}'); done; done
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -35,6 +35,8 @@ ver="$1"
 | 
				
			|||||||
	exit 1
 | 
						exit 1
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					mv copyparty/web/deps/marked.full.js.gz srv/ || true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
mkdir -p dist
 | 
					mkdir -p dist
 | 
				
			||||||
zip_path="$(pwd)/dist/copyparty-$ver.zip"
 | 
					zip_path="$(pwd)/dist/copyparty-$ver.zip"
 | 
				
			||||||
tgz_path="$(pwd)/dist/copyparty-$ver.tar.gz"
 | 
					tgz_path="$(pwd)/dist/copyparty-$ver.tar.gz"
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										176
									
								
								scripts/sfx.py
									
									
									
									
									
								
							
							
						
						
									
										176
									
								
								scripts/sfx.py
									
									
									
									
									
								
							@@ -1,9 +1,8 @@
 | 
				
			|||||||
#!/usr/bin/env python
 | 
					#!/usr/bin/env python
 | 
				
			||||||
# coding: utf-8
 | 
					# coding: latin-1
 | 
				
			||||||
from __future__ import print_function, unicode_literals
 | 
					from __future__ import print_function, unicode_literals
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import re, os, sys, time, shutil, signal, tarfile, hashlib, platform, tempfile
 | 
					import os, sys, time, shutil, runpy, tarfile, hashlib, platform, tempfile, traceback
 | 
				
			||||||
import subprocess as sp
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
"""
 | 
					"""
 | 
				
			||||||
run me with any version of python, i will unpack and run copyparty
 | 
					run me with any version of python, i will unpack and run copyparty
 | 
				
			||||||
@@ -202,93 +201,6 @@ def u8(gen):
 | 
				
			|||||||
            yield s
 | 
					            yield s
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def get_py_win(ret):
 | 
					 | 
				
			||||||
    tops = []
 | 
					 | 
				
			||||||
    p = str(os.getenv("LocalAppdata"))
 | 
					 | 
				
			||||||
    if p:
 | 
					 | 
				
			||||||
        tops.append(os.path.join(p, "Programs", "Python"))
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    progfiles = {}
 | 
					 | 
				
			||||||
    for p in ["ProgramFiles", "ProgramFiles(x86)"]:
 | 
					 | 
				
			||||||
        p = str(os.getenv(p))
 | 
					 | 
				
			||||||
        if p:
 | 
					 | 
				
			||||||
            progfiles[p] = 1
 | 
					 | 
				
			||||||
            # 32bit apps get x86 for both
 | 
					 | 
				
			||||||
            if p.endswith(" (x86)"):
 | 
					 | 
				
			||||||
                progfiles[p[:-6]] = 1
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    tops += list(progfiles.keys())
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    for sysroot in [me, sys.executable]:
 | 
					 | 
				
			||||||
        sysroot = sysroot[:3].upper()
 | 
					 | 
				
			||||||
        if sysroot[1] == ":" and sysroot not in tops:
 | 
					 | 
				
			||||||
            tops.append(sysroot)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    # $WIRESHARK_SLOGAN
 | 
					 | 
				
			||||||
    for top in tops:
 | 
					 | 
				
			||||||
        try:
 | 
					 | 
				
			||||||
            for name1 in u8(sorted(os.listdir(top), reverse=True)):
 | 
					 | 
				
			||||||
                if name1.lower().startswith("python"):
 | 
					 | 
				
			||||||
                    path1 = os.path.join(top, name1)
 | 
					 | 
				
			||||||
                    try:
 | 
					 | 
				
			||||||
                        for name2 in u8(os.listdir(path1)):
 | 
					 | 
				
			||||||
                            if name2.lower() == "python.exe":
 | 
					 | 
				
			||||||
                                path2 = os.path.join(path1, name2)
 | 
					 | 
				
			||||||
                                ret[path2.lower()] = path2
 | 
					 | 
				
			||||||
                    except:
 | 
					 | 
				
			||||||
                        pass
 | 
					 | 
				
			||||||
        except:
 | 
					 | 
				
			||||||
            pass
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
def get_py_nix(ret):
 | 
					 | 
				
			||||||
    ptn = re.compile(r"^(python|pypy)[0-9\.-]*$")
 | 
					 | 
				
			||||||
    for bindir in os.getenv("PATH").split(":"):
 | 
					 | 
				
			||||||
        if not bindir:
 | 
					 | 
				
			||||||
            next
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        try:
 | 
					 | 
				
			||||||
            for fn in u8(os.listdir(bindir)):
 | 
					 | 
				
			||||||
                if ptn.match(fn):
 | 
					 | 
				
			||||||
                    fn = os.path.join(bindir, fn)
 | 
					 | 
				
			||||||
                    ret[fn.lower()] = fn
 | 
					 | 
				
			||||||
        except:
 | 
					 | 
				
			||||||
            pass
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
def read_py(binp):
 | 
					 | 
				
			||||||
    cmd = [
 | 
					 | 
				
			||||||
        binp,
 | 
					 | 
				
			||||||
        "-c",
 | 
					 | 
				
			||||||
        "import sys; sys.stdout.write(' '.join(str(x) for x in sys.version_info)); import jinja2",
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
    p = sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE)
 | 
					 | 
				
			||||||
    ver, _ = p.communicate()
 | 
					 | 
				
			||||||
    ver = ver.decode("utf-8").split(" ")[:3]
 | 
					 | 
				
			||||||
    ver = [int(x) if x.isdigit() else 0 for x in ver]
 | 
					 | 
				
			||||||
    return ver, p.returncode == 0
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
def get_pys():
 | 
					 | 
				
			||||||
    ver, chk = read_py(sys.executable)
 | 
					 | 
				
			||||||
    if chk or PY2:
 | 
					 | 
				
			||||||
        return [[chk, ver, sys.executable]]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    hits = {sys.executable.lower(): sys.executable}
 | 
					 | 
				
			||||||
    if platform.system() == "Windows":
 | 
					 | 
				
			||||||
        get_py_win(hits)
 | 
					 | 
				
			||||||
    else:
 | 
					 | 
				
			||||||
        get_py_nix(hits)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    ret = []
 | 
					 | 
				
			||||||
    for binp in hits.values():
 | 
					 | 
				
			||||||
        ver, chk = read_py(binp)
 | 
					 | 
				
			||||||
        ret.append([chk, ver, binp])
 | 
					 | 
				
			||||||
        msg("\t".join(str(x) for x in ret[-1]))
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    return ret
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
def yieldfile(fn):
 | 
					def yieldfile(fn):
 | 
				
			||||||
    with open(fn, "rb") as f:
 | 
					    with open(fn, "rb") as f:
 | 
				
			||||||
        for block in iter(lambda: f.read(64 * 1024), b""):
 | 
					        for block in iter(lambda: f.read(64 * 1024), b""):
 | 
				
			||||||
@@ -431,21 +343,24 @@ def get_payload():
 | 
				
			|||||||
                break
 | 
					                break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def confirm():
 | 
					def confirm(rv):
 | 
				
			||||||
    msg()
 | 
					    msg()
 | 
				
			||||||
 | 
					    msg(traceback.format_exc())
 | 
				
			||||||
    msg("*** hit enter to exit ***")
 | 
					    msg("*** hit enter to exit ***")
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        raw_input() if PY2 else input()
 | 
					        raw_input() if PY2 else input()
 | 
				
			||||||
    except:
 | 
					    except:
 | 
				
			||||||
        pass
 | 
					        pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    sys.exit(rv)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def run(tmp, py):
 | 
					
 | 
				
			||||||
 | 
					def run(tmp, j2ver):
 | 
				
			||||||
    global cpp
 | 
					    global cpp
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    msg("OK")
 | 
					    msg("jinja2:", j2ver or "bundled")
 | 
				
			||||||
    msg("will use:", py)
 | 
					    msg("sfxdir:", tmp)
 | 
				
			||||||
    msg("bound to:", tmp)
 | 
					    msg()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # "systemd-tmpfiles-clean.timer"?? HOW do you even come up with this shit
 | 
					    # "systemd-tmpfiles-clean.timer"?? HOW do you even come up with this shit
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
@@ -457,44 +372,25 @@ def run(tmp, py):
 | 
				
			|||||||
    except:
 | 
					    except:
 | 
				
			||||||
        pass
 | 
					        pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    fp_py = os.path.join(tmp, "py")
 | 
					    ld = [tmp, os.path.join(tmp, "dep-j2")]
 | 
				
			||||||
 | 
					    if j2ver:
 | 
				
			||||||
 | 
					        del ld[-1]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for x in ld:
 | 
				
			||||||
 | 
					        sys.path.insert(0, x)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        with open(fp_py, "wb") as f:
 | 
					        runpy.run_module(str("copyparty"), run_name=str("__main__"))
 | 
				
			||||||
            f.write(py.encode("utf-8") + b"\n")
 | 
					    except SystemExit as ex:
 | 
				
			||||||
 | 
					        if ex.code:
 | 
				
			||||||
 | 
					            confirm(ex.code)
 | 
				
			||||||
    except:
 | 
					    except:
 | 
				
			||||||
        pass
 | 
					        confirm(1)
 | 
				
			||||||
 | 
					 | 
				
			||||||
    # avoid loading ./copyparty.py
 | 
					 | 
				
			||||||
    cmd = [
 | 
					 | 
				
			||||||
        py,
 | 
					 | 
				
			||||||
        "-c",
 | 
					 | 
				
			||||||
        'import sys, runpy; sys.path.insert(0, r"'
 | 
					 | 
				
			||||||
        + tmp
 | 
					 | 
				
			||||||
        + '"); runpy.run_module("copyparty", run_name="__main__")',
 | 
					 | 
				
			||||||
    ] + list(sys.argv[1:])
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    msg("\n", cmd, "\n")
 | 
					 | 
				
			||||||
    cpp = sp.Popen(str(x) for x in cmd)
 | 
					 | 
				
			||||||
    try:
 | 
					 | 
				
			||||||
        cpp.wait()
 | 
					 | 
				
			||||||
    except:
 | 
					 | 
				
			||||||
        cpp.wait()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    if cpp.returncode != 0:
 | 
					 | 
				
			||||||
        confirm()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    sys.exit(cpp.returncode)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
def bye(sig, frame):
 | 
					 | 
				
			||||||
    if cpp is not None:
 | 
					 | 
				
			||||||
        cpp.terminate()
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def main():
 | 
					def main():
 | 
				
			||||||
    sysver = str(sys.version).replace("\n", "\n" + " " * 18)
 | 
					    sysver = str(sys.version).replace("\n", "\n" + " " * 18)
 | 
				
			||||||
    pktime = time.strftime("%Y-%m-%d, %H:%M:%S", time.gmtime(STAMP))
 | 
					    pktime = time.strftime("%Y-%m-%d, %H:%M:%S", time.gmtime(STAMP))
 | 
				
			||||||
    os.system("")
 | 
					 | 
				
			||||||
    msg()
 | 
					    msg()
 | 
				
			||||||
    msg("   this is: copyparty", VER)
 | 
					    msg("   this is: copyparty", VER)
 | 
				
			||||||
    msg(" packed at:", pktime, "UTC,", STAMP)
 | 
					    msg(" packed at:", pktime, "UTC,", STAMP)
 | 
				
			||||||
@@ -523,36 +419,14 @@ def main():
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    # skip 0
 | 
					    # skip 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    signal.signal(signal.SIGTERM, bye)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    tmp = unpack()
 | 
					    tmp = unpack()
 | 
				
			||||||
    fp_py = os.path.join(tmp, "py")
 | 
					 | 
				
			||||||
    if os.path.exists(fp_py):
 | 
					 | 
				
			||||||
        with open(fp_py, "rb") as f:
 | 
					 | 
				
			||||||
            py = f.read().decode("utf-8").rstrip()
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return run(tmp, py)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    pys = get_pys()
 | 
					 | 
				
			||||||
    pys.sort(reverse=True)
 | 
					 | 
				
			||||||
    j2, ver, py = pys[0]
 | 
					 | 
				
			||||||
    if j2:
 | 
					 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
            os.rename(os.path.join(tmp, "jinja2"), os.path.join(tmp, "x.jinja2"))
 | 
					        from jinja2 import __version__ as j2ver
 | 
				
			||||||
    except:
 | 
					    except:
 | 
				
			||||||
            pass
 | 
					        j2ver = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return run(tmp, py)
 | 
					    run(tmp, j2ver)
 | 
				
			||||||
 | 
					 | 
				
			||||||
    msg("\n  could not find jinja2; will use py2 + the bundled version\n")
 | 
					 | 
				
			||||||
    for _, ver, py in pys:
 | 
					 | 
				
			||||||
        if ver > [2, 7] and ver < [3, 0]:
 | 
					 | 
				
			||||||
            return run(tmp, py)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    m = "\033[1;31m\n\n\ncould not find a python with jinja2 installed; please do one of these:\n\n  pip install --user jinja2\n\n  install python2\n\n\033[0m"
 | 
					 | 
				
			||||||
    msg(m)
 | 
					 | 
				
			||||||
    confirm()
 | 
					 | 
				
			||||||
    sys.exit(1)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if __name__ == "__main__":
 | 
					if __name__ == "__main__":
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										77
									
								
								scripts/uncomment.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								scripts/uncomment.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,77 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/env python
 | 
				
			||||||
 | 
					# coding: utf-8
 | 
				
			||||||
 | 
					from __future__ import print_function, unicode_literals
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import io
 | 
				
			||||||
 | 
					import sys
 | 
				
			||||||
 | 
					import tokenize
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def uncomment(fpath):
 | 
				
			||||||
 | 
					    """ modified https://stackoverflow.com/a/62074206 """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    with open(fpath, "rb") as f:
 | 
				
			||||||
 | 
					        orig = f.read().decode("utf-8")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    out = ""
 | 
				
			||||||
 | 
					    for ln in orig.split("\n"):
 | 
				
			||||||
 | 
					        if not ln.startswith("#"):
 | 
				
			||||||
 | 
					            break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        out += ln + "\n"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    io_obj = io.StringIO(orig)
 | 
				
			||||||
 | 
					    prev_toktype = tokenize.INDENT
 | 
				
			||||||
 | 
					    last_lineno = -1
 | 
				
			||||||
 | 
					    last_col = 0
 | 
				
			||||||
 | 
					    for tok in tokenize.generate_tokens(io_obj.readline):
 | 
				
			||||||
 | 
					        # print(repr(tok))
 | 
				
			||||||
 | 
					        token_type = tok[0]
 | 
				
			||||||
 | 
					        token_string = tok[1]
 | 
				
			||||||
 | 
					        start_line, start_col = tok[2]
 | 
				
			||||||
 | 
					        end_line, end_col = tok[3]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if start_line > last_lineno:
 | 
				
			||||||
 | 
					            last_col = 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if start_col > last_col:
 | 
				
			||||||
 | 
					            out += " " * (start_col - last_col)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        is_legalese = (
 | 
				
			||||||
 | 
					            "copyright" in token_string.lower() or "license" in token_string.lower()
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if token_type == tokenize.STRING:
 | 
				
			||||||
 | 
					            if (
 | 
				
			||||||
 | 
					                prev_toktype != tokenize.INDENT
 | 
				
			||||||
 | 
					                and prev_toktype != tokenize.NEWLINE
 | 
				
			||||||
 | 
					                and start_col > 0
 | 
				
			||||||
 | 
					                or is_legalese
 | 
				
			||||||
 | 
					            ):
 | 
				
			||||||
 | 
					                out += token_string
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                out += '"a"'
 | 
				
			||||||
 | 
					        elif token_type != tokenize.COMMENT or is_legalese:
 | 
				
			||||||
 | 
					            out += token_string
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        prev_toktype = token_type
 | 
				
			||||||
 | 
					        last_lineno = end_line
 | 
				
			||||||
 | 
					        last_col = end_col
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # out = "\n".join(x for x in out.splitlines() if x.strip())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    with open(fpath, "wb") as f:
 | 
				
			||||||
 | 
					        f.write(out.encode("utf-8"))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def main():
 | 
				
			||||||
 | 
					    print("uncommenting", end="")
 | 
				
			||||||
 | 
					    for f in sys.argv[1:]:
 | 
				
			||||||
 | 
					        print(".", end="")
 | 
				
			||||||
 | 
					        uncomment(f)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    print("k")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if __name__ == "__main__":
 | 
				
			||||||
 | 
					    main()
 | 
				
			||||||
							
								
								
									
										4
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								setup.py
									
									
									
									
									
								
							@@ -2,10 +2,8 @@
 | 
				
			|||||||
# coding: utf-8
 | 
					# coding: utf-8
 | 
				
			||||||
from __future__ import print_function
 | 
					from __future__ import print_function
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import io
 | 
					 | 
				
			||||||
import os
 | 
					import os
 | 
				
			||||||
import sys
 | 
					import sys
 | 
				
			||||||
from glob import glob
 | 
					 | 
				
			||||||
from shutil import rmtree
 | 
					from shutil import rmtree
 | 
				
			||||||
 | 
					
 | 
				
			||||||
setuptools_available = True
 | 
					setuptools_available = True
 | 
				
			||||||
@@ -110,13 +108,13 @@ args = {
 | 
				
			|||||||
        "Programming Language :: Python :: 2",
 | 
					        "Programming Language :: Python :: 2",
 | 
				
			||||||
        "Programming Language :: Python :: 2.7",
 | 
					        "Programming Language :: Python :: 2.7",
 | 
				
			||||||
        "Programming Language :: Python :: 3",
 | 
					        "Programming Language :: Python :: 3",
 | 
				
			||||||
        "Programming Language :: Python :: 3.2",
 | 
					 | 
				
			||||||
        "Programming Language :: Python :: 3.3",
 | 
					        "Programming Language :: Python :: 3.3",
 | 
				
			||||||
        "Programming Language :: Python :: 3.4",
 | 
					        "Programming Language :: Python :: 3.4",
 | 
				
			||||||
        "Programming Language :: Python :: 3.5",
 | 
					        "Programming Language :: Python :: 3.5",
 | 
				
			||||||
        "Programming Language :: Python :: 3.6",
 | 
					        "Programming Language :: Python :: 3.6",
 | 
				
			||||||
        "Programming Language :: Python :: 3.7",
 | 
					        "Programming Language :: Python :: 3.7",
 | 
				
			||||||
        "Programming Language :: Python :: 3.8",
 | 
					        "Programming Language :: Python :: 3.8",
 | 
				
			||||||
 | 
					        "Programming Language :: Python :: 3.9",
 | 
				
			||||||
        "Programming Language :: Python :: Implementation :: CPython",
 | 
					        "Programming Language :: Python :: Implementation :: CPython",
 | 
				
			||||||
        "Programming Language :: Python :: Implementation :: PyPy",
 | 
					        "Programming Language :: Python :: Implementation :: PyPy",
 | 
				
			||||||
        "Environment :: Console",
 | 
					        "Environment :: Console",
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -16,6 +16,14 @@ from copyparty.authsrv import AuthSrv
 | 
				
			|||||||
from copyparty import util
 | 
					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".split()}
 | 
				
			||||||
 | 
					        ex["mtp"] = []
 | 
				
			||||||
 | 
					        ex["mte"] = "a"
 | 
				
			||||||
 | 
					        super(Cfg, self).__init__(a=a, v=v, c=c, **ex)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class TestVFS(unittest.TestCase):
 | 
					class TestVFS(unittest.TestCase):
 | 
				
			||||||
    def dump(self, vfs):
 | 
					    def dump(self, vfs):
 | 
				
			||||||
        print(json.dumps(vfs, indent=4, sort_keys=True, default=lambda o: o.__dict__))
 | 
					        print(json.dumps(vfs, indent=4, sort_keys=True, default=lambda o: o.__dict__))
 | 
				
			||||||
@@ -35,7 +43,13 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
    def ls(self, vfs, vpath, uname):
 | 
					    def ls(self, vfs, vpath, uname):
 | 
				
			||||||
        """helper for resolving and listing a folder"""
 | 
					        """helper for resolving and listing a folder"""
 | 
				
			||||||
        vn, rem = vfs.get(vpath, uname, True, False)
 | 
					        vn, rem = vfs.get(vpath, uname, True, False)
 | 
				
			||||||
        return vn.ls(rem, uname)
 | 
					        r1 = vn.ls(rem, uname, False)
 | 
				
			||||||
 | 
					        r2 = vn.ls(rem, uname, False)
 | 
				
			||||||
 | 
					        self.assertEqual(r1, r2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        fsdir, real, virt = r1
 | 
				
			||||||
 | 
					        real = [x[0] for x in real]
 | 
				
			||||||
 | 
					        return fsdir, real, virt
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def runcmd(self, *argv):
 | 
					    def runcmd(self, *argv):
 | 
				
			||||||
        p = sp.Popen(argv, stdout=sp.PIPE, stderr=sp.PIPE)
 | 
					        p = sp.Popen(argv, stdout=sp.PIPE, stderr=sp.PIPE)
 | 
				
			||||||
@@ -78,7 +92,7 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
        finally:
 | 
					        finally:
 | 
				
			||||||
            return ret
 | 
					            return ret
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def log(self, src, msg):
 | 
					    def log(self, src, msg, c=0):
 | 
				
			||||||
        pass
 | 
					        pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def test(self):
 | 
					    def test(self):
 | 
				
			||||||
@@ -102,7 +116,7 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
                            f.write(fn)
 | 
					                            f.write(fn)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # defaults
 | 
					        # defaults
 | 
				
			||||||
        vfs = AuthSrv(Namespace(c=None, a=[], v=[]), self.log).vfs
 | 
					        vfs = AuthSrv(Cfg(), self.log).vfs
 | 
				
			||||||
        self.assertEqual(vfs.nodes, {})
 | 
					        self.assertEqual(vfs.nodes, {})
 | 
				
			||||||
        self.assertEqual(vfs.vpath, "")
 | 
					        self.assertEqual(vfs.vpath, "")
 | 
				
			||||||
        self.assertEqual(vfs.realpath, td)
 | 
					        self.assertEqual(vfs.realpath, td)
 | 
				
			||||||
@@ -110,7 +124,7 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
        self.assertEqual(vfs.uwrite, ["*"])
 | 
					        self.assertEqual(vfs.uwrite, ["*"])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # single read-only rootfs (relative path)
 | 
					        # single read-only rootfs (relative path)
 | 
				
			||||||
        vfs = AuthSrv(Namespace(c=None, a=[], v=["a/ab/::r"]), self.log).vfs
 | 
					        vfs = AuthSrv(Cfg(v=["a/ab/::r"]), self.log).vfs
 | 
				
			||||||
        self.assertEqual(vfs.nodes, {})
 | 
					        self.assertEqual(vfs.nodes, {})
 | 
				
			||||||
        self.assertEqual(vfs.vpath, "")
 | 
					        self.assertEqual(vfs.vpath, "")
 | 
				
			||||||
        self.assertEqual(vfs.realpath, os.path.join(td, "a", "ab"))
 | 
					        self.assertEqual(vfs.realpath, os.path.join(td, "a", "ab"))
 | 
				
			||||||
@@ -118,9 +132,7 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
        self.assertEqual(vfs.uwrite, [])
 | 
					        self.assertEqual(vfs.uwrite, [])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # single read-only rootfs (absolute path)
 | 
					        # single read-only rootfs (absolute path)
 | 
				
			||||||
        vfs = AuthSrv(
 | 
					        vfs = AuthSrv(Cfg(v=[td + "//a/ac/../aa//::r"]), self.log).vfs
 | 
				
			||||||
            Namespace(c=None, a=[], v=[td + "//a/ac/../aa//::r"]), self.log
 | 
					 | 
				
			||||||
        ).vfs
 | 
					 | 
				
			||||||
        self.assertEqual(vfs.nodes, {})
 | 
					        self.assertEqual(vfs.nodes, {})
 | 
				
			||||||
        self.assertEqual(vfs.vpath, "")
 | 
					        self.assertEqual(vfs.vpath, "")
 | 
				
			||||||
        self.assertEqual(vfs.realpath, os.path.join(td, "a", "aa"))
 | 
					        self.assertEqual(vfs.realpath, os.path.join(td, "a", "aa"))
 | 
				
			||||||
@@ -129,7 +141,7 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        # read-only rootfs with write-only subdirectory (read-write for k)
 | 
					        # read-only rootfs with write-only subdirectory (read-write for k)
 | 
				
			||||||
        vfs = AuthSrv(
 | 
					        vfs = AuthSrv(
 | 
				
			||||||
            Namespace(c=None, a=["k:k"], v=[".::r:ak", "a/ac/acb:a/ac/acb:w:ak"]),
 | 
					            Cfg(a=["k:k"], v=[".::r:ak", "a/ac/acb:a/ac/acb:w:ak"]),
 | 
				
			||||||
            self.log,
 | 
					            self.log,
 | 
				
			||||||
        ).vfs
 | 
					        ).vfs
 | 
				
			||||||
        self.assertEqual(len(vfs.nodes), 1)
 | 
					        self.assertEqual(len(vfs.nodes), 1)
 | 
				
			||||||
@@ -192,7 +204,10 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
        self.assertEqual(list(virt), [])
 | 
					        self.assertEqual(list(virt), [])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # admin-only rootfs with all-read-only subfolder
 | 
					        # admin-only rootfs with all-read-only subfolder
 | 
				
			||||||
        vfs = AuthSrv(Namespace(c=None, a=["k:k"], v=[".::ak", "a:a:r"]), self.log,).vfs
 | 
					        vfs = AuthSrv(
 | 
				
			||||||
 | 
					            Cfg(a=["k:k"], v=[".::ak", "a:a:r"]),
 | 
				
			||||||
 | 
					            self.log,
 | 
				
			||||||
 | 
					        ).vfs
 | 
				
			||||||
        self.assertEqual(len(vfs.nodes), 1)
 | 
					        self.assertEqual(len(vfs.nodes), 1)
 | 
				
			||||||
        self.assertEqual(vfs.vpath, "")
 | 
					        self.assertEqual(vfs.vpath, "")
 | 
				
			||||||
        self.assertEqual(vfs.realpath, td)
 | 
					        self.assertEqual(vfs.realpath, td)
 | 
				
			||||||
@@ -211,9 +226,7 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        # breadth-first construction
 | 
					        # breadth-first construction
 | 
				
			||||||
        vfs = AuthSrv(
 | 
					        vfs = AuthSrv(
 | 
				
			||||||
            Namespace(
 | 
					            Cfg(
 | 
				
			||||||
                c=None,
 | 
					 | 
				
			||||||
                a=[],
 | 
					 | 
				
			||||||
                v=[
 | 
					                v=[
 | 
				
			||||||
                    "a/ac/acb:a/ac/acb:w",
 | 
					                    "a/ac/acb:a/ac/acb:w",
 | 
				
			||||||
                    "a:a:w",
 | 
					                    "a:a:w",
 | 
				
			||||||
@@ -234,7 +247,7 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
        self.undot(vfs, "./.././foo/..", "")
 | 
					        self.undot(vfs, "./.././foo/..", "")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # shadowing
 | 
					        # shadowing
 | 
				
			||||||
        vfs = AuthSrv(Namespace(c=None, a=[], v=[".::r", "b:a/ac:r"]), self.log).vfs
 | 
					        vfs = AuthSrv(Cfg(v=[".::r", "b:a/ac:r"]), self.log).vfs
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        fsp, r1, v1 = self.ls(vfs, "", "*")
 | 
					        fsp, r1, v1 = self.ls(vfs, "", "*")
 | 
				
			||||||
        self.assertEqual(fsp, td)
 | 
					        self.assertEqual(fsp, td)
 | 
				
			||||||
@@ -271,7 +284,7 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
                ).encode("utf-8")
 | 
					                ).encode("utf-8")
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        au = AuthSrv(Namespace(c=[cfg_path], a=[], v=[]), self.log)
 | 
					        au = AuthSrv(Cfg(c=[cfg_path]), self.log)
 | 
				
			||||||
        self.assertEqual(au.user["a"], "123")
 | 
					        self.assertEqual(au.user["a"], "123")
 | 
				
			||||||
        self.assertEqual(au.user["asd"], "fgh:jkl")
 | 
					        self.assertEqual(au.user["asd"], "fgh:jkl")
 | 
				
			||||||
        n = au.vfs
 | 
					        n = au.vfs
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user