mirror of
				https://github.com/9001/copyparty.git
				synced 2025-11-04 13:53:18 +00:00 
			
		
		
		
	Compare commits
	
		
			42 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					0c786b0766 | ||
| 
						 | 
					68c7528911 | ||
| 
						 | 
					26e18ae800 | ||
| 
						 | 
					c30dc0b546 | ||
| 
						 | 
					f94aa46a11 | ||
| 
						 | 
					403261a293 | ||
| 
						 | 
					c7d9cbb11f | ||
| 
						 | 
					57e1c53cbb | ||
| 
						 | 
					0754b553dd | ||
| 
						 | 
					50661d941b | ||
| 
						 | 
					c5db7c1a0c | ||
| 
						 | 
					2cef5365f7 | ||
| 
						 | 
					fbc4e94007 | ||
| 
						 | 
					037ed5a2ad | ||
| 
						 | 
					69dfa55705 | ||
| 
						 | 
					a79a5c4e3e | ||
| 
						 | 
					7e80eabfe6 | ||
| 
						 | 
					375b72770d | ||
| 
						 | 
					e2dd683def | ||
| 
						 | 
					9eba50c6e4 | ||
| 
						 | 
					5a579dba52 | ||
| 
						 | 
					e86c719575 | ||
| 
						 | 
					0e87f35547 | ||
| 
						 | 
					b6d3d791a5 | ||
| 
						 | 
					c9c3302664 | ||
| 
						 | 
					c3e4d65b80 | ||
| 
						 | 
					27a03510c5 | ||
| 
						 | 
					ed7727f7cb | ||
| 
						 | 
					127ec10c0d | ||
| 
						 | 
					5a9c0ad225 | ||
| 
						 | 
					7e8daf650e | ||
| 
						 | 
					0cf737b4ce | ||
| 
						 | 
					74635e0113 | ||
| 
						 | 
					e5c4f49901 | ||
| 
						 | 
					e4654ee7f1 | ||
| 
						 | 
					e5d05c05ed | ||
| 
						 | 
					73c4f99687 | ||
| 
						 | 
					28c12ef3bf | ||
| 
						 | 
					eed82dbb54 | ||
| 
						 | 
					2c4b4ab928 | ||
| 
						 | 
					505a8fc6f6 | ||
| 
						 | 
					e4801d9b06 | 
							
								
								
									
										150
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										150
									
								
								README.md
									
									
									
									
									
								
							@@ -12,6 +12,8 @@ turn your phone or raspi into a portable file server with resumable uploads/down
 | 
				
			|||||||
* *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`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					📷 screenshots: [browser](#the-browser) // [upload](#uploading) // [md-viewer](#markdown-viewer) // [search](#searching) // [fsearch](#file-search) // [zip-DL](#zip-downloads) // [ie4](#browser-support)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
## readme toc
 | 
					## readme toc
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -20,8 +22,16 @@ turn your phone or raspi into a portable file server with resumable uploads/down
 | 
				
			|||||||
    * [notes](#notes)
 | 
					    * [notes](#notes)
 | 
				
			||||||
    * [status](#status)
 | 
					    * [status](#status)
 | 
				
			||||||
* [bugs](#bugs)
 | 
					* [bugs](#bugs)
 | 
				
			||||||
* [usage](#usage)
 | 
					    * [not my bugs](#not-my-bugs)
 | 
				
			||||||
 | 
					* [the browser](#the-browser)
 | 
				
			||||||
 | 
					    * [tabs](#tabs)
 | 
				
			||||||
 | 
					    * [hotkeys](#hotkeys)
 | 
				
			||||||
 | 
					    * [tree-mode](#tree-mode)
 | 
				
			||||||
    * [zip downloads](#zip-downloads)
 | 
					    * [zip downloads](#zip-downloads)
 | 
				
			||||||
 | 
					    * [uploading](#uploading)
 | 
				
			||||||
 | 
					        * [file-search](#file-search)
 | 
				
			||||||
 | 
					    * [markdown viewer](#markdown-viewer)
 | 
				
			||||||
 | 
					    * [other tricks](#other-tricks)
 | 
				
			||||||
* [searching](#searching)
 | 
					* [searching](#searching)
 | 
				
			||||||
    * [search configuration](#search-configuration)
 | 
					    * [search configuration](#search-configuration)
 | 
				
			||||||
    * [metadata from audio files](#metadata-from-audio-files)
 | 
					    * [metadata from audio files](#metadata-from-audio-files)
 | 
				
			||||||
@@ -29,6 +39,7 @@ turn your phone or raspi into a portable file server with resumable uploads/down
 | 
				
			|||||||
    * [complete examples](#complete-examples)
 | 
					    * [complete examples](#complete-examples)
 | 
				
			||||||
* [browser support](#browser-support)
 | 
					* [browser support](#browser-support)
 | 
				
			||||||
* [client examples](#client-examples)
 | 
					* [client examples](#client-examples)
 | 
				
			||||||
 | 
					* [up2k](#up2k)
 | 
				
			||||||
* [dependencies](#dependencies)
 | 
					* [dependencies](#dependencies)
 | 
				
			||||||
    * [optional gpl stuff](#optional-gpl-stuff)
 | 
					    * [optional gpl stuff](#optional-gpl-stuff)
 | 
				
			||||||
* [sfx](#sfx)
 | 
					* [sfx](#sfx)
 | 
				
			||||||
@@ -53,9 +64,9 @@ you may also want these, especially on servers:
 | 
				
			|||||||
## notes
 | 
					## notes
 | 
				
			||||||
 | 
					
 | 
				
			||||||
* iPhone/iPad: use Firefox to download files
 | 
					* iPhone/iPad: use Firefox to download files
 | 
				
			||||||
* Android-Chrome: set max "parallel uploads" for 200% upload speed (android bug)
 | 
					* Android-Chrome: increase "parallel uploads" for higher speed (android bug)
 | 
				
			||||||
* Android-Firefox: takes a while to select files (in order to avoid the above android-chrome issue)
 | 
					* Android-Firefox: takes a while to select files (their fix for ☝️)
 | 
				
			||||||
* Desktop-Firefox: may use gigabytes of RAM if your connection is great and your files are massive
 | 
					* Desktop-Firefox: ~~may use gigabytes of RAM if your files are massive~~ *seems to be OK now*
 | 
				
			||||||
* paper-printing is affected by dark/light-mode! use lightmode for color, darkmode for grayscale
 | 
					* paper-printing is affected by dark/light-mode! use lightmode for color, darkmode for grayscale
 | 
				
			||||||
  * because no browsers currently implement the media-query to do this properly orz
 | 
					  * because no browsers currently implement the media-query to do this properly orz
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -101,17 +112,44 @@ summary: it works! you can use it! (but technically not even close to beta)
 | 
				
			|||||||
* 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
 | 
					* 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
 | 
					* probably more, pls let me know
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## not my bugs
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# usage
 | 
					* Windows: msys2-python 3.8.6 occasionally throws "RuntimeError: release unlocked lock" when leaving a scoped mutex in up2k
 | 
				
			||||||
 | 
					  * this is an msys2 bug, the regular windows edition of python is fine
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# the browser
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## tabs
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* `[🔎]` search by size, date, path/name, mp3-tags ... see [searching](#searching)
 | 
				
			||||||
 | 
					* `[🚀]` and `[🎈]` are the uploaders, see [uploading](#uploading)
 | 
				
			||||||
 | 
					* `[📂]` mkdir, create directories
 | 
				
			||||||
 | 
					* `[📝]` new-md, create a new markdown document
 | 
				
			||||||
 | 
					* `[📟]` send-msg, either to server-log or into textfiles if `--urlform save`
 | 
				
			||||||
 | 
					* `[⚙️]` client configuration options
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## hotkeys
 | 
				
			||||||
 | 
					
 | 
				
			||||||
the browser has the following hotkeys
 | 
					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
 | 
					* `I/K` prev/next folder
 | 
				
			||||||
* `P` parent folder
 | 
					* `P` parent folder
 | 
				
			||||||
 | 
					* when playing audio:
 | 
				
			||||||
 | 
					  * `0..9` jump to 10%..90%
 | 
				
			||||||
 | 
					  * `U/O` skip 10sec back/forward
 | 
				
			||||||
 | 
					  * `J/L` prev/next song
 | 
				
			||||||
 | 
					    * `J` also starts playing the folder
 | 
				
			||||||
 | 
					
 | 
				
			||||||
you can link a particular timestamp in an audio file by adding it to the URL, such as `&20` / `&20s` / `&1m20` / `&1:20` after the `.../#af-c8960dab`
 | 
					
 | 
				
			||||||
 | 
					## tree-mode
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					by default there's a breadcrumbs path; you can replace this with a tree-browser sidebar thing by clicking the 🌲
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					click `[-]` and `[+]` to adjust the size, and the `[a]` toggles if the tree should widen dynamically as you go deeper or stay fixed-size
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
## zip downloads
 | 
					## zip downloads
 | 
				
			||||||
@@ -130,12 +168,80 @@ the `zip` link next to folders can produce various types of zip/tar files using
 | 
				
			|||||||
* `zip_crc` will take longer to download since the server has to read each file twice
 | 
					* `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
 | 
					  * please let me know if you find a program old enough to actually need this
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					you can also zip a selection of files or folders by clicking them in the browser, that brings up a selection editor and zip button in the bottom right
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## uploading
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					two upload methods are available in the html client:
 | 
				
			||||||
 | 
					* 🎈 bup, the basic uploader, supports almost every browser since netscape 4.0
 | 
				
			||||||
 | 
					* 🚀 up2k, the fancy one
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					up2k has several advantages:
 | 
				
			||||||
 | 
					* you can drop folders into the browser (files are added recursively)
 | 
				
			||||||
 | 
					* files are processed in chunks, and each chunk is checksummed
 | 
				
			||||||
 | 
					  * uploads resume if they are interrupted (for example by a reboot)
 | 
				
			||||||
 | 
					  * server detects any corruption; the client reuploads affected chunks
 | 
				
			||||||
 | 
					  * the client doesn't upload anything that already exists on the server
 | 
				
			||||||
 | 
					* the last-modified timestamp of the file is preserved
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					see [up2k](#up2k) for details on how it works
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					the up2k UI is the epitome of polished inutitive experiences:
 | 
				
			||||||
 | 
					* "parallel uploads" specifies how many chunks to upload at the same time
 | 
				
			||||||
 | 
					* `[🏃]` analysis of other files should continue while one is uploading
 | 
				
			||||||
 | 
					* `[💭]` ask for confirmation before files are added to the list
 | 
				
			||||||
 | 
					* `[💤]` sync uploading between other copyparty tabs so only one is active
 | 
				
			||||||
 | 
					* `[🔎]` switch between upload and file-search mode
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					and then theres the tabs below it,
 | 
				
			||||||
 | 
					* `[ok]` is uploads which completed successfully
 | 
				
			||||||
 | 
					* `[ng]` is the uploads which failed / got rejected (already exists, ...)
 | 
				
			||||||
 | 
					* `[done]` shows a combined list of `[ok]` and `[ng]`, chronological order
 | 
				
			||||||
 | 
					* `[busy]` files which are currently hashing, pending-upload, or uploading
 | 
				
			||||||
 | 
					  * plus up to 3 entries each from `[done]` and `[que]` for context
 | 
				
			||||||
 | 
					* `[que]` is all the files that are still queued
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					protip: you can avoid scaring away users by hiding some of the UI with hacks like [docs/minimal-up2k.html](docs/minimal-up2k.html)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### file-search
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					in the 🚀 up2k tab, after toggling the `[🔎]` switch green, any files/folders you drop onto the dropzone will be hashed on the client-side. Each hash is sent to the server which checks if that file exists somewhere already
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					files go into `[ok]` if they exist (and you get a link to where it is), otherwise they land in `[ng]`
 | 
				
			||||||
 | 
					* the main reason filesearch is combined with the uploader is cause the code was too spaghetti to separate it out somewhere else
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					adding the same file multiple times is blocked, so if you first search for a file and then decide to upload it, you have to click the `[cleanup]` button to discard `[done]` files
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					note that since up2k has to read the file twice, 🎈 bup can be up to 2x faster if your internet connection is faster than the read-speed of your HDD
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					up2k has saved a few uploads from becoming corrupted in-transfer already; caught an android phone on wifi redhanded in wireshark with a bitflip, however bup with https would *probably* have noticed as well thanks to tls also functioning as an integrity check
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## markdown viewer
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* the document preview has a max-width which is the same as an A4 paper when printed
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## other tricks
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* you can link a particular timestamp in an audio file by adding it to the URL, such as `&20` / `&20s` / `&1m20` / `&t=1:20` after the `.../#af-c8960dab`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# searching
 | 
					# 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:
 | 
					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...
 | 
					* make search queries by `size`/`date`/`directory-path`/`filename`, or...
 | 
				
			||||||
* drag/drop a local file to see if the same contents exist somewhere on the server (you get the URL if it does)
 | 
					* drag/drop a local file to see if the same contents exist somewhere on the server, see [file-search](#file-search)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
path/name queries are space-separated, AND'ed together, and words are negated with a `-` prefix, so for example:
 | 
					path/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
 | 
					* 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
 | 
				
			||||||
@@ -201,6 +307,8 @@ copyparty can invoke external programs to collect additional metadata for files
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
# browser support
 | 
					# browser support
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
`ie` = internet-explorer, `ff` = firefox, `c` = chrome, `iOS` = iPhone/iPad, `Andr` = Android
 | 
					`ie` = internet-explorer, `ff` = firefox, `c` = chrome, `iOS` = iPhone/iPad, `Andr` = Android
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| feature         | ie6 | ie9 | ie10 | ie11 | ff 52 | c 49 | iOS | Andr |
 | 
					| feature         | ie6 | ie9 | ie10 | ie11 | ff 52 | c 49 | iOS | Andr |
 | 
				
			||||||
@@ -231,7 +339,7 @@ quick summary of more eccentric web-browsers trying to view a directory index:
 | 
				
			|||||||
* lynx (2.8.9/macports) can browse, login, upload/mkdir/msg
 | 
					* lynx (2.8.9/macports) can browse, login, upload/mkdir/msg
 | 
				
			||||||
* w3m (0.5.3/macports) can browse, login, upload at 100kB/s, mkdir/msg
 | 
					* w3m (0.5.3/macports) can browse, login, upload at 100kB/s, mkdir/msg
 | 
				
			||||||
* netsurf (3.10/arch) is basically ie6 with much better css (javascript has almost no effect)
 | 
					* netsurf (3.10/arch) is basically ie6 with much better css (javascript has almost no effect)
 | 
				
			||||||
* netscape 4.0 and 4.5 can browse (text is yellow on white), upload with `?b=u`
 | 
					* ie4 and netscape 4.0 can browse (text is yellow on white), upload with `?b=u`
 | 
				
			||||||
* SerenityOS (22d13d8) hits a page fault, works with `?b=u`, file input not-impl, url params are multiplying
 | 
					* SerenityOS (22d13d8) hits a page fault, works with `?b=u`, file input not-impl, url params are multiplying
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# client examples
 | 
					# client examples
 | 
				
			||||||
@@ -258,6 +366,22 @@ copyparty returns a truncated sha512sum of your PUT/POST as base64; you can gene
 | 
				
			|||||||
    b512 <movie.mkv
 | 
					    b512 <movie.mkv
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# up2k
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					quick outline of the up2k protocol, see [uploading](#uploading) for the web-client
 | 
				
			||||||
 | 
					* the up2k client splits a file into an "optimal" number of chunks
 | 
				
			||||||
 | 
					  * 1 MiB each, unless that becomes more than 256 chunks
 | 
				
			||||||
 | 
					  * tries 1.5M, 2M, 3, 4, 6, ... until <= 256# or chunksize >= 32M
 | 
				
			||||||
 | 
					* client posts the list of hashes, filename, size, last-modified
 | 
				
			||||||
 | 
					* server creates the `wark`, an identifier for this upload
 | 
				
			||||||
 | 
					  * `sha512( salt + filesize + chunk_hashes )`
 | 
				
			||||||
 | 
					  * and a sparse file is created for the chunks to drop into
 | 
				
			||||||
 | 
					* client uploads each chunk
 | 
				
			||||||
 | 
					  * header entries for the chunk-hash and wark
 | 
				
			||||||
 | 
					  * server writes chunks into place based on the hash
 | 
				
			||||||
 | 
					* client does another handshake with the hashlist; server replies with OK or a list of chunks to reupload
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# dependencies
 | 
					# dependencies
 | 
				
			||||||
 | 
					
 | 
				
			||||||
* `jinja2` (is built into the SFX)
 | 
					* `jinja2` (is built into the SFX)
 | 
				
			||||||
@@ -274,12 +398,12 @@ copyparty returns a truncated sha512sum of your PUT/POST as base64; you can gene
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
some bundled tools have copyleft dependencies, see [./bin/#mtag](bin/#mtag)
 | 
					some bundled tools have copyleft dependencies, see [./bin/#mtag](bin/#mtag)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
these are standalone and will never be imported / evaluated by copyparty
 | 
					these are standalone programs and will never be imported / evaluated by copyparty
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# sfx
 | 
					# sfx
 | 
				
			||||||
 | 
					
 | 
				
			||||||
currently there are two self-contained binaries:
 | 
					currently there are two self-contained "binaries":
 | 
				
			||||||
* [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) -- pure python, works everywhere
 | 
					* [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) -- pure python, works everywhere
 | 
				
			||||||
* [copyparty-sfx.sh](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.sh) -- smaller, but only for linux and macos
 | 
					* [copyparty-sfx.sh](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.sh) -- smaller, but only for linux and macos
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -16,12 +16,17 @@ if platform.system() == "Windows":
 | 
				
			|||||||
VT100 = not WINDOWS or WINDOWS >= [10, 0, 14393]
 | 
					VT100 = not WINDOWS or WINDOWS >= [10, 0, 14393]
 | 
				
			||||||
# introduced in anniversary update
 | 
					# introduced in anniversary update
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ANYWIN = WINDOWS or sys.platform in ["msys"]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
MACOS = platform.system() == "Darwin"
 | 
					MACOS = platform.system() == "Darwin"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class EnvParams(object):
 | 
					class EnvParams(object):
 | 
				
			||||||
    def __init__(self):
 | 
					    def __init__(self):
 | 
				
			||||||
        self.mod = os.path.dirname(os.path.realpath(__file__))
 | 
					        self.mod = os.path.dirname(os.path.realpath(__file__))
 | 
				
			||||||
 | 
					        if self.mod.endswith("__init__"):
 | 
				
			||||||
 | 
					            self.mod = os.path.dirname(self.mod)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if sys.platform == "win32":
 | 
					        if sys.platform == "win32":
 | 
				
			||||||
            self.cfg = os.path.normpath(os.environ["APPDATA"] + "/copyparty")
 | 
					            self.cfg = os.path.normpath(os.environ["APPDATA"] + "/copyparty")
 | 
				
			||||||
        elif sys.platform == "darwin":
 | 
					        elif sys.platform == "darwin":
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -247,6 +247,7 @@ def run_argparse(argv, formatter):
 | 
				
			|||||||
    ap.add_argument("--no-zip", action="store_true", help="disable download as zip/tar")
 | 
					    ap.add_argument("--no-zip", action="store_true", help="disable download as zip/tar")
 | 
				
			||||||
    ap.add_argument("--no-sendfile", action="store_true", help="disable sendfile (for debugging)")
 | 
					    ap.add_argument("--no-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("--no-scandir", action="store_true", help="disable scandir (for debugging)")
 | 
				
			||||||
 | 
					    ap.add_argument("--sparse", metavar="MiB", type=int, default=4, help="up2k min.size threshold (mswin-only)")
 | 
				
			||||||
    ap.add_argument("--urlform", metavar="MODE", type=str, default="print,get", help="how to handle url-forms")
 | 
					    ap.add_argument("--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")
 | 
					    ap.add_argument("--salt", type=str, default="hunter2", help="up2k file-hash salt")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,8 +1,8 @@
 | 
				
			|||||||
# coding: utf-8
 | 
					# coding: utf-8
 | 
				
			||||||
 | 
					
 | 
				
			||||||
VERSION = (0, 10, 13)
 | 
					VERSION = (0, 10, 17)
 | 
				
			||||||
CODENAME = "zip it"
 | 
					CODENAME = "zip it"
 | 
				
			||||||
BUILD_DT = (2021, 4, 20)
 | 
					BUILD_DT = (2021, 5, 12)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
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)
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -141,7 +141,12 @@ class VFS(object):
 | 
				
			|||||||
        real.sort()
 | 
					        real.sort()
 | 
				
			||||||
        if not rem:
 | 
					        if not rem:
 | 
				
			||||||
            for name, vn2 in sorted(self.nodes.items()):
 | 
					            for name, vn2 in sorted(self.nodes.items()):
 | 
				
			||||||
                if uname in vn2.uread or "*" in vn2.uread:
 | 
					                if (
 | 
				
			||||||
 | 
					                    uname in vn2.uread
 | 
				
			||||||
 | 
					                    or "*" in vn2.uread
 | 
				
			||||||
 | 
					                    or uname in vn2.uwrite
 | 
				
			||||||
 | 
					                    or "*" in vn2.uwrite
 | 
				
			||||||
 | 
					                ):
 | 
				
			||||||
                    virt_vis[name] = vn2
 | 
					                    virt_vis[name] = vn2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            # no vfs nodes in the list of real inodes
 | 
					            # no vfs nodes in the list of real inodes
 | 
				
			||||||
@@ -241,6 +246,7 @@ class AuthSrv(object):
 | 
				
			|||||||
        self.args = args
 | 
					        self.args = args
 | 
				
			||||||
        self.log_func = log_func
 | 
					        self.log_func = log_func
 | 
				
			||||||
        self.warn_anonwrite = warn_anonwrite
 | 
					        self.warn_anonwrite = warn_anonwrite
 | 
				
			||||||
 | 
					        self.line_ctr = 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if WINDOWS:
 | 
					        if WINDOWS:
 | 
				
			||||||
            self.re_vol = re.compile(r"^([a-zA-Z]:[\\/][^:]*|[^:]*):([^:]*):(.*)$")
 | 
					            self.re_vol = re.compile(r"^([a-zA-Z]:[\\/][^:]*|[^:]*):([^:]*):(.*)$")
 | 
				
			||||||
@@ -266,7 +272,9 @@ class AuthSrv(object):
 | 
				
			|||||||
    def _parse_config_file(self, fd, user, mread, mwrite, mflags, mount):
 | 
					    def _parse_config_file(self, fd, user, mread, mwrite, mflags, mount):
 | 
				
			||||||
        vol_src = None
 | 
					        vol_src = None
 | 
				
			||||||
        vol_dst = None
 | 
					        vol_dst = None
 | 
				
			||||||
 | 
					        self.line_ctr = 0
 | 
				
			||||||
        for ln in [x.decode("utf-8").strip() for x in fd]:
 | 
					        for ln in [x.decode("utf-8").strip() for x in fd]:
 | 
				
			||||||
 | 
					            self.line_ctr += 1
 | 
				
			||||||
            if not ln and vol_src is not None:
 | 
					            if not ln and vol_src is not None:
 | 
				
			||||||
                vol_src = None
 | 
					                vol_src = None
 | 
				
			||||||
                vol_dst = None
 | 
					                vol_dst = None
 | 
				
			||||||
@@ -296,7 +304,12 @@ class AuthSrv(object):
 | 
				
			|||||||
                mflags[vol_dst] = {}
 | 
					                mflags[vol_dst] = {}
 | 
				
			||||||
                continue
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            lvl, uname = ln.split(" ")
 | 
					            if len(ln) > 1:
 | 
				
			||||||
 | 
					                lvl, uname = ln.split(" ")
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                lvl = ln
 | 
				
			||||||
 | 
					                uname = "*"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            self._read_vol_str(
 | 
					            self._read_vol_str(
 | 
				
			||||||
                lvl, uname, mread[vol_dst], mwrite[vol_dst], mflags[vol_dst]
 | 
					                lvl, uname, mread[vol_dst], mwrite[vol_dst], mflags[vol_dst]
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
@@ -374,7 +387,12 @@ class AuthSrv(object):
 | 
				
			|||||||
        if self.args.c:
 | 
					        if self.args.c:
 | 
				
			||||||
            for cfg_fn in self.args.c:
 | 
					            for cfg_fn in self.args.c:
 | 
				
			||||||
                with open(cfg_fn, "rb") as f:
 | 
					                with open(cfg_fn, "rb") as f:
 | 
				
			||||||
                    self._parse_config_file(f, user, mread, mwrite, mflags, mount)
 | 
					                    try:
 | 
				
			||||||
 | 
					                        self._parse_config_file(f, user, mread, mwrite, mflags, mount)
 | 
				
			||||||
 | 
					                    except:
 | 
				
			||||||
 | 
					                        m = "\n\033[1;31m\nerror in config file {} on line {}:\n\033[0m"
 | 
				
			||||||
 | 
					                        print(m.format(cfg_fn, self.line_ctr))
 | 
				
			||||||
 | 
					                        raise
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        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
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -13,7 +13,7 @@ import ctypes
 | 
				
			|||||||
from datetime import datetime
 | 
					from datetime import datetime
 | 
				
			||||||
import calendar
 | 
					import calendar
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from .__init__ import E, PY2, WINDOWS
 | 
					from .__init__ import E, PY2, WINDOWS, ANYWIN
 | 
				
			||||||
from .util import *  # noqa  # pylint: disable=unused-wildcard-import
 | 
					from .util import *  # noqa  # pylint: disable=unused-wildcard-import
 | 
				
			||||||
from .szip import StreamZip
 | 
					from .szip import StreamZip
 | 
				
			||||||
from .star import StreamTar
 | 
					from .star import StreamTar
 | 
				
			||||||
@@ -182,10 +182,8 @@ class HttpCli(object):
 | 
				
			|||||||
        self.out_headers.update(headers)
 | 
					        self.out_headers.update(headers)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # default to utf8 html if no content-type is set
 | 
					        # default to utf8 html if no content-type is set
 | 
				
			||||||
        try:
 | 
					        if not mime:
 | 
				
			||||||
            mime = mime or self.out_headers["Content-Type"]
 | 
					            mime = self.out_headers.get("Content-Type", "text/html; charset=UTF-8")
 | 
				
			||||||
        except KeyError:
 | 
					 | 
				
			||||||
            mime = "text/html; charset=UTF-8"
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        self.out_headers["Content-Type"] = mime
 | 
					        self.out_headers["Content-Type"] = mime
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -261,12 +259,14 @@ class HttpCli(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
                self.absolute_urls = True
 | 
					                self.absolute_urls = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # go home if verboten
 | 
					 | 
				
			||||||
        self.readable, self.writable = self.conn.auth.vfs.can_access(
 | 
					        self.readable, self.writable = self.conn.auth.vfs.can_access(
 | 
				
			||||||
            self.vpath, self.uname
 | 
					            self.vpath, self.uname
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        if not self.readable and not self.writable:
 | 
					        if not self.readable and not self.writable:
 | 
				
			||||||
            self.log("inaccessible: [{}]".format(self.vpath))
 | 
					            if self.vpath:
 | 
				
			||||||
 | 
					                self.log("inaccessible: [{}]".format(self.vpath))
 | 
				
			||||||
 | 
					                raise Pebkac(404)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            self.uparam = {"h": False}
 | 
					            self.uparam = {"h": False}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if "h" in self.uparam:
 | 
					        if "h" in self.uparam:
 | 
				
			||||||
@@ -534,7 +534,7 @@ class HttpCli(object):
 | 
				
			|||||||
            self.log("qj: " + repr(vbody))
 | 
					            self.log("qj: " + repr(vbody))
 | 
				
			||||||
            hits = idx.fsearch(vols, body)
 | 
					            hits = idx.fsearch(vols, body)
 | 
				
			||||||
            msg = repr(hits)
 | 
					            msg = repr(hits)
 | 
				
			||||||
            taglist = []
 | 
					            taglist = {}
 | 
				
			||||||
        else:
 | 
					        else:
 | 
				
			||||||
            # search by query params
 | 
					            # search by query params
 | 
				
			||||||
            self.log("qj: " + repr(body))
 | 
					            self.log("qj: " + repr(body))
 | 
				
			||||||
@@ -626,7 +626,7 @@ class HttpCli(object):
 | 
				
			|||||||
            self.loud_reply(x, status=500)
 | 
					            self.loud_reply(x, status=500)
 | 
				
			||||||
            return False
 | 
					            return False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if not WINDOWS and num_left == 0:
 | 
					        if not ANYWIN and num_left == 0:
 | 
				
			||||||
            times = (int(time.time()), int(lastmod))
 | 
					            times = (int(time.time()), int(lastmod))
 | 
				
			||||||
            self.log("no more chunks, setting times {}".format(times))
 | 
					            self.log("no more chunks, setting times {}".format(times))
 | 
				
			||||||
            try:
 | 
					            try:
 | 
				
			||||||
@@ -680,7 +680,7 @@ 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("/")
 | 
				
			||||||
        esc_paths = [quotep(vpath), html_escape(vpath)]
 | 
					        esc_paths = [quotep(vpath), html_escape(vpath, crlf=True)]
 | 
				
			||||||
        html = self.j2(
 | 
					        html = self.j2(
 | 
				
			||||||
            "msg",
 | 
					            "msg",
 | 
				
			||||||
            h2='<a href="/{}">go to /{}</a>'.format(*esc_paths),
 | 
					            h2='<a href="/{}">go to /{}</a>'.format(*esc_paths),
 | 
				
			||||||
@@ -741,7 +741,9 @@ class HttpCli(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
                if p_file and not nullwrite:
 | 
					                if p_file and not nullwrite:
 | 
				
			||||||
                    fdir = os.path.join(vfs.realpath, rem)
 | 
					                    fdir = os.path.join(vfs.realpath, rem)
 | 
				
			||||||
                    fname = sanitize_fn(p_file)
 | 
					                    fname = sanitize_fn(
 | 
				
			||||||
 | 
					                        p_file, bad=[".prologue.html", ".epilogue.html"]
 | 
				
			||||||
 | 
					                    )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                    if not os.path.isdir(fsenc(fdir)):
 | 
					                    if not os.path.isdir(fsenc(fdir)):
 | 
				
			||||||
                        raise Pebkac(404, "that folder does not exist")
 | 
					                        raise Pebkac(404, "that folder does not exist")
 | 
				
			||||||
@@ -1181,17 +1183,16 @@ class HttpCli(object):
 | 
				
			|||||||
        template = self.j2(tpl)
 | 
					        template = self.j2(tpl)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        st = os.stat(fsenc(fs_path))
 | 
					        st = os.stat(fsenc(fs_path))
 | 
				
			||||||
        # sz_md = st.st_size
 | 
					 | 
				
			||||||
        ts_md = st.st_mtime
 | 
					        ts_md = st.st_mtime
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        st = os.stat(fsenc(html_path))
 | 
					        st = os.stat(fsenc(html_path))
 | 
				
			||||||
        ts_html = st.st_mtime
 | 
					        ts_html = st.st_mtime
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # TODO dont load into memory ;_;
 | 
					        sz_md = 0
 | 
				
			||||||
        #   (trivial fix, count the &'s)
 | 
					        for buf in yieldfile(fs_path):
 | 
				
			||||||
        with open(fsenc(fs_path), "rb") as f:
 | 
					            sz_md += len(buf)
 | 
				
			||||||
            md = f.read().replace(b"&", b"&")
 | 
					            for c, v in [[b"&", 4], [b"<", 3], [b">", 3]]:
 | 
				
			||||||
            sz_md = len(md)
 | 
					                sz_md += (len(buf) - len(buf.replace(c, b""))) * v
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        file_ts = max(ts_md, ts_html)
 | 
					        file_ts = max(ts_md, ts_html)
 | 
				
			||||||
        file_lastmod, do_send = self._chk_lastmod(file_ts)
 | 
					        file_lastmod, do_send = self._chk_lastmod(file_ts)
 | 
				
			||||||
@@ -1199,27 +1200,34 @@ class HttpCli(object):
 | 
				
			|||||||
        self.out_headers["Cache-Control"] = "no-cache"
 | 
					        self.out_headers["Cache-Control"] = "no-cache"
 | 
				
			||||||
        status = 200 if do_send else 304
 | 
					        status = 200 if do_send else 304
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        boundary = "\roll\tide"
 | 
				
			||||||
        targs = {
 | 
					        targs = {
 | 
				
			||||||
            "edit": "edit" in self.uparam,
 | 
					            "edit": "edit" in self.uparam,
 | 
				
			||||||
            "title": html_escape(self.vpath),
 | 
					            "title": html_escape(self.vpath, crlf=True),
 | 
				
			||||||
            "lastmod": int(ts_md * 1000),
 | 
					            "lastmod": int(ts_md * 1000),
 | 
				
			||||||
            "md_plug": "true" if self.args.emp else "false",
 | 
					            "md_plug": "true" if self.args.emp else "false",
 | 
				
			||||||
            "md_chk_rate": self.args.mcr,
 | 
					            "md_chk_rate": self.args.mcr,
 | 
				
			||||||
            "md": "",
 | 
					            "md": boundary,
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        sz_html = len(template.render(**targs).encode("utf-8"))
 | 
					        html = template.render(**targs).encode("utf-8")
 | 
				
			||||||
        self.send_headers(sz_html + sz_md, status)
 | 
					        html = html.split(boundary.encode("utf-8"))
 | 
				
			||||||
 | 
					        if len(html) != 2:
 | 
				
			||||||
 | 
					            raise Exception("boundary appears in " + html_path)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.send_headers(sz_md + len(html[0]) + len(html[1]), status)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        logmsg += unicode(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
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # TODO jinja2 can stream this right?
 | 
					 | 
				
			||||||
        targs["md"] = md.decode("utf-8", "replace")
 | 
					 | 
				
			||||||
        html = template.render(**targs).encode("utf-8")
 | 
					 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            self.s.sendall(html)
 | 
					            self.s.sendall(html[0])
 | 
				
			||||||
 | 
					            for buf in yieldfile(fs_path):
 | 
				
			||||||
 | 
					                self.s.sendall(html_bescape(buf))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            self.s.sendall(html[1])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        except:
 | 
					        except:
 | 
				
			||||||
            self.log(logmsg + " \033[31md/c\033[0m")
 | 
					            self.log(logmsg + " \033[31md/c\033[0m")
 | 
				
			||||||
            return False
 | 
					            return False
 | 
				
			||||||
@@ -1300,7 +1308,7 @@ class HttpCli(object):
 | 
				
			|||||||
                else:
 | 
					                else:
 | 
				
			||||||
                    vpath += "/" + node
 | 
					                    vpath += "/" + node
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                vpnodes.append([quotep(vpath) + "/", html_escape(node)])
 | 
					                vpnodes.append([quotep(vpath) + "/", html_escape(node, crlf=True)])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        vn, rem = self.auth.vfs.get(
 | 
					        vn, rem = self.auth.vfs.get(
 | 
				
			||||||
            self.vpath, self.uname, self.readable, self.writable
 | 
					            self.vpath, self.uname, self.readable, self.writable
 | 
				
			||||||
@@ -1311,6 +1319,94 @@ class HttpCli(object):
 | 
				
			|||||||
            # print(abspath)
 | 
					            # print(abspath)
 | 
				
			||||||
            raise Pebkac(404)
 | 
					            raise Pebkac(404)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        srv_info = []
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            if not self.args.nih:
 | 
				
			||||||
 | 
					                srv_info.append(unicode(socket.gethostname()).split(".")[0])
 | 
				
			||||||
 | 
					        except:
 | 
				
			||||||
 | 
					            self.log("#wow #whoa")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            # some fuses misbehave
 | 
				
			||||||
 | 
					            if not self.args.nid:
 | 
				
			||||||
 | 
					                if WINDOWS:
 | 
				
			||||||
 | 
					                    bfree = ctypes.c_ulonglong(0)
 | 
				
			||||||
 | 
					                    ctypes.windll.kernel32.GetDiskFreeSpaceExW(
 | 
				
			||||||
 | 
					                        ctypes.c_wchar_p(abspath), None, None, ctypes.pointer(bfree)
 | 
				
			||||||
 | 
					                    )
 | 
				
			||||||
 | 
					                    srv_info.append(humansize(bfree.value) + " free")
 | 
				
			||||||
 | 
					                else:
 | 
				
			||||||
 | 
					                    sv = os.statvfs(abspath)
 | 
				
			||||||
 | 
					                    free = humansize(sv.f_frsize * sv.f_bfree, True)
 | 
				
			||||||
 | 
					                    total = humansize(sv.f_frsize * sv.f_blocks, True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    srv_info.append(free + " free")
 | 
				
			||||||
 | 
					                    srv_info.append(total)
 | 
				
			||||||
 | 
					        except:
 | 
				
			||||||
 | 
					            pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        srv_info = "</span> /// <span>".join(srv_info)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        perms = []
 | 
				
			||||||
 | 
					        if self.readable:
 | 
				
			||||||
 | 
					            perms.append("read")
 | 
				
			||||||
 | 
					        if self.writable:
 | 
				
			||||||
 | 
					            perms.append("write")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        url_suf = self.urlq()
 | 
				
			||||||
 | 
					        is_ls = "ls" in self.uparam
 | 
				
			||||||
 | 
					        ts = ""  # "?{}".format(time.time())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        tpl = "browser"
 | 
				
			||||||
 | 
					        if "b" in self.uparam:
 | 
				
			||||||
 | 
					            tpl = "browser2"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        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")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ls_ret = {
 | 
				
			||||||
 | 
					            "dirs": [],
 | 
				
			||||||
 | 
					            "files": [],
 | 
				
			||||||
 | 
					            "taglist": [],
 | 
				
			||||||
 | 
					            "srvinf": srv_info,
 | 
				
			||||||
 | 
					            "perms": perms,
 | 
				
			||||||
 | 
					            "logues": logues,
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        j2a = {
 | 
				
			||||||
 | 
					            "vdir": quotep(self.vpath),
 | 
				
			||||||
 | 
					            "vpnodes": vpnodes,
 | 
				
			||||||
 | 
					            "files": [],
 | 
				
			||||||
 | 
					            "ts": ts,
 | 
				
			||||||
 | 
					            "perms": json.dumps(perms),
 | 
				
			||||||
 | 
					            "taglist": [],
 | 
				
			||||||
 | 
					            "tag_order": [],
 | 
				
			||||||
 | 
					            "have_up2k_idx": ("e2d" in vn.flags),
 | 
				
			||||||
 | 
					            "have_tags_idx": ("e2t" in vn.flags),
 | 
				
			||||||
 | 
					            "have_zip": (not self.args.no_zip),
 | 
				
			||||||
 | 
					            "have_b_u": (self.writable and self.uparam.get("b") == "u"),
 | 
				
			||||||
 | 
					            "url_suf": url_suf,
 | 
				
			||||||
 | 
					            "logues": logues,
 | 
				
			||||||
 | 
					            "title": html_escape(self.vpath, crlf=True),
 | 
				
			||||||
 | 
					            "srv_info": srv_info,
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if not self.readable:
 | 
				
			||||||
 | 
					            if is_ls:
 | 
				
			||||||
 | 
					                ret = json.dumps(ls_ret)
 | 
				
			||||||
 | 
					                self.reply(ret.encode("utf-8", "replace"), mime="application/json")
 | 
				
			||||||
 | 
					                return True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if not os.path.isdir(fsenc(abspath)):
 | 
				
			||||||
 | 
					                raise Pebkac(404)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            html = self.j2(tpl, **j2a)
 | 
				
			||||||
 | 
					            self.reply(html.encode("utf-8", "replace"))
 | 
				
			||||||
 | 
					            return True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if not os.path.isdir(fsenc(abspath)):
 | 
					        if not os.path.isdir(fsenc(abspath)):
 | 
				
			||||||
            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)
 | 
				
			||||||
@@ -1354,15 +1450,11 @@ class HttpCli(object):
 | 
				
			|||||||
        if rem == ".hist":
 | 
					        if rem == ".hist":
 | 
				
			||||||
            hidden = ["up2k."]
 | 
					            hidden = ["up2k."]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        is_ls = "ls" in self.uparam
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        icur = None
 | 
					        icur = None
 | 
				
			||||||
        if "e2t" in vn.flags:
 | 
					        if "e2t" in vn.flags:
 | 
				
			||||||
            idx = self.conn.get_u2idx()
 | 
					            idx = self.conn.get_u2idx()
 | 
				
			||||||
            icur = idx.get_cur(vn.realpath)
 | 
					            icur = idx.get_cur(vn.realpath)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        url_suf = self.urlq()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        dirs = []
 | 
					        dirs = []
 | 
				
			||||||
        files = []
 | 
					        files = []
 | 
				
			||||||
        for fn in vfs_ls:
 | 
					        for fn in vfs_ls:
 | 
				
			||||||
@@ -1394,7 +1486,7 @@ class HttpCli(object):
 | 
				
			|||||||
                    margin = '<a href="{}?zip">zip</a>'.format(quotep(href))
 | 
					                    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, crlf=True), hist[fn][0]
 | 
				
			||||||
                )
 | 
					                )
 | 
				
			||||||
            else:
 | 
					            else:
 | 
				
			||||||
                margin = "-"
 | 
					                margin = "-"
 | 
				
			||||||
@@ -1453,91 +1545,21 @@ class HttpCli(object):
 | 
				
			|||||||
            for f in dirs:
 | 
					            for f in dirs:
 | 
				
			||||||
                f["tags"] = {}
 | 
					                f["tags"] = {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        srv_info = []
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        try:
 | 
					 | 
				
			||||||
            if not self.args.nih:
 | 
					 | 
				
			||||||
                srv_info.append(unicode(socket.gethostname()).split(".")[0])
 | 
					 | 
				
			||||||
        except:
 | 
					 | 
				
			||||||
            self.log("#wow #whoa")
 | 
					 | 
				
			||||||
            pass
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        try:
 | 
					 | 
				
			||||||
            # some fuses misbehave
 | 
					 | 
				
			||||||
            if not self.args.nid:
 | 
					 | 
				
			||||||
                if WINDOWS:
 | 
					 | 
				
			||||||
                    bfree = ctypes.c_ulonglong(0)
 | 
					 | 
				
			||||||
                    ctypes.windll.kernel32.GetDiskFreeSpaceExW(
 | 
					 | 
				
			||||||
                        ctypes.c_wchar_p(abspath), None, None, ctypes.pointer(bfree)
 | 
					 | 
				
			||||||
                    )
 | 
					 | 
				
			||||||
                    srv_info.append(humansize(bfree.value) + " free")
 | 
					 | 
				
			||||||
                else:
 | 
					 | 
				
			||||||
                    sv = os.statvfs(abspath)
 | 
					 | 
				
			||||||
                    free = humansize(sv.f_frsize * sv.f_bfree, True)
 | 
					 | 
				
			||||||
                    total = humansize(sv.f_frsize * sv.f_blocks, True)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
                    srv_info.append(free + " free")
 | 
					 | 
				
			||||||
                    srv_info.append(total)
 | 
					 | 
				
			||||||
        except:
 | 
					 | 
				
			||||||
            pass
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        srv_info = "</span> /// <span>".join(srv_info)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        perms = []
 | 
					 | 
				
			||||||
        if self.readable:
 | 
					 | 
				
			||||||
            perms.append("read")
 | 
					 | 
				
			||||||
        if self.writable:
 | 
					 | 
				
			||||||
            perms.append("write")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        logues = ["", ""]
 | 
					 | 
				
			||||||
        for n, fn in enumerate([".prologue.html", ".epilogue.html"]):
 | 
					 | 
				
			||||||
            fn = os.path.join(abspath, fn)
 | 
					 | 
				
			||||||
            if os.path.exists(fsenc(fn)):
 | 
					 | 
				
			||||||
                with open(fsenc(fn), "rb") as f:
 | 
					 | 
				
			||||||
                    logues[n] = f.read().decode("utf-8")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if is_ls:
 | 
					        if is_ls:
 | 
				
			||||||
            [x.pop(k) for k in ["name", "dt"] for y in [dirs, files] for x in y]
 | 
					            [x.pop(k) for k in ["name", "dt"] for y in [dirs, files] for x in y]
 | 
				
			||||||
            ret = {
 | 
					            ls_ret["dirs"] = dirs
 | 
				
			||||||
                "dirs": dirs,
 | 
					            ls_ret["files"] = files
 | 
				
			||||||
                "files": files,
 | 
					            ls_ret["taglist"] = taglist
 | 
				
			||||||
                "srvinf": srv_info,
 | 
					            ret = json.dumps(ls_ret)
 | 
				
			||||||
                "perms": perms,
 | 
					 | 
				
			||||||
                "logues": logues,
 | 
					 | 
				
			||||||
                "taglist": taglist,
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
            ret = json.dumps(ret)
 | 
					 | 
				
			||||||
            self.reply(ret.encode("utf-8", "replace"), mime="application/json")
 | 
					            self.reply(ret.encode("utf-8", "replace"), mime="application/json")
 | 
				
			||||||
            return True
 | 
					            return True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        ts = ""
 | 
					        j2a["files"] = dirs + files
 | 
				
			||||||
        # ts = "?{}".format(time.time())
 | 
					        j2a["logues"] = logues
 | 
				
			||||||
 | 
					        j2a["taglist"] = taglist
 | 
				
			||||||
 | 
					        if "mte" in vn.flags:
 | 
				
			||||||
 | 
					            j2a["tag_order"] = json.dumps(vn.flags["mte"].split(","))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        dirs.extend(files)
 | 
					        html = self.j2(tpl, **j2a)
 | 
				
			||||||
 | 
					 | 
				
			||||||
        tpl = "browser"
 | 
					 | 
				
			||||||
        if "b" in self.uparam:
 | 
					 | 
				
			||||||
            tpl = "browser2"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        html = self.j2(
 | 
					 | 
				
			||||||
            tpl,
 | 
					 | 
				
			||||||
            vdir=quotep(self.vpath),
 | 
					 | 
				
			||||||
            vpnodes=vpnodes,
 | 
					 | 
				
			||||||
            files=dirs,
 | 
					 | 
				
			||||||
            ts=ts,
 | 
					 | 
				
			||||||
            perms=json.dumps(perms),
 | 
					 | 
				
			||||||
            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),
 | 
					 | 
				
			||||||
            have_b_u=(self.writable and self.uparam.get("b") == "u"),
 | 
					 | 
				
			||||||
            url_suf=url_suf,
 | 
					 | 
				
			||||||
            logues=logues,
 | 
					 | 
				
			||||||
            title=html_escape(self.vpath),
 | 
					 | 
				
			||||||
            srv_info=srv_info,
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.reply(html.encode("utf-8", "replace"))
 | 
					        self.reply(html.encode("utf-8", "replace"))
 | 
				
			||||||
        return True
 | 
					        return True
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -87,7 +87,7 @@ def gen_hdr(h_pos, fn, sz, lastmod, utf8, crc32, pre_crc):
 | 
				
			|||||||
    ret += struct.pack("<LL", vsz, vsz)
 | 
					    ret += struct.pack("<LL", vsz, vsz)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # windows support (the "?" replace below too)
 | 
					    # windows support (the "?" replace below too)
 | 
				
			||||||
    fn = sanitize_fn(fn, "/")
 | 
					    fn = sanitize_fn(fn, ok="/")
 | 
				
			||||||
    bfn = fn.encode("utf-8" if utf8 else "cp437", "replace").replace(b"?", b"_")
 | 
					    bfn = fn.encode("utf-8" if utf8 else "cp437", "replace").replace(b"?", b"_")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    z64_len = len(z64v) * 8 + 4 if z64v else 0
 | 
					    z64_len = len(z64v) * 8 + 4 if z64v else 0
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -16,7 +16,7 @@ import traceback
 | 
				
			|||||||
import subprocess as sp
 | 
					import subprocess as sp
 | 
				
			||||||
from copy import deepcopy
 | 
					from copy import deepcopy
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from .__init__ import WINDOWS
 | 
					from .__init__ import WINDOWS, ANYWIN
 | 
				
			||||||
from .util import (
 | 
					from .util import (
 | 
				
			||||||
    Pebkac,
 | 
					    Pebkac,
 | 
				
			||||||
    Queue,
 | 
					    Queue,
 | 
				
			||||||
@@ -79,7 +79,7 @@ class Up2k(object):
 | 
				
			|||||||
            if self.sqlite_ver < (3, 9):
 | 
					            if self.sqlite_ver < (3, 9):
 | 
				
			||||||
                self.no_expr_idx = True
 | 
					                self.no_expr_idx = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if WINDOWS:
 | 
					        if ANYWIN:
 | 
				
			||||||
            # usually fails to set lastmod too quickly
 | 
					            # usually fails to set lastmod too quickly
 | 
				
			||||||
            self.lastmod_q = Queue()
 | 
					            self.lastmod_q = Queue()
 | 
				
			||||||
            thr = threading.Thread(target=self._lastmodder)
 | 
					            thr = threading.Thread(target=self._lastmodder)
 | 
				
			||||||
@@ -668,12 +668,6 @@ class Up2k(object):
 | 
				
			|||||||
            cur.close()
 | 
					            cur.close()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def _start_mpool(self):
 | 
					    def _start_mpool(self):
 | 
				
			||||||
        if WINDOWS and False:
 | 
					 | 
				
			||||||
            nah = open(os.devnull, "wb")
 | 
					 | 
				
			||||||
            wmic = "processid={}".format(os.getpid())
 | 
					 | 
				
			||||||
            wmic = ["wmic", "process", "where", wmic, "call", "setpriority"]
 | 
					 | 
				
			||||||
            sp.call(wmic + ["below normal"], stdout=nah, stderr=nah)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        # mp.pool.ThreadPool and concurrent.futures.ThreadPoolExecutor
 | 
					        # mp.pool.ThreadPool and concurrent.futures.ThreadPoolExecutor
 | 
				
			||||||
        # both do crazy runahead so lets reinvent another wheel
 | 
					        # both do crazy runahead so lets reinvent another wheel
 | 
				
			||||||
        nw = os.cpu_count() if hasattr(os, "cpu_count") else 4
 | 
					        nw = os.cpu_count() if hasattr(os, "cpu_count") else 4
 | 
				
			||||||
@@ -698,12 +692,6 @@ class Up2k(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        mpool.join()
 | 
					        mpool.join()
 | 
				
			||||||
        done = self._flush_mpool(wcur)
 | 
					        done = self._flush_mpool(wcur)
 | 
				
			||||||
        if WINDOWS and False:
 | 
					 | 
				
			||||||
            nah = open(os.devnull, "wb")
 | 
					 | 
				
			||||||
            wmic = "processid={}".format(os.getpid())
 | 
					 | 
				
			||||||
            wmic = ["wmic", "process", "where", wmic, "call", "setpriority"]
 | 
					 | 
				
			||||||
            sp.call(wmic + ["below normal"], stdout=nah, stderr=nah)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        return done
 | 
					        return done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def _tag_thr(self, q):
 | 
					    def _tag_thr(self, q):
 | 
				
			||||||
@@ -903,7 +891,7 @@ class Up2k(object):
 | 
				
			|||||||
            if cj["ptop"] not in self.registry:
 | 
					            if cj["ptop"] not in self.registry:
 | 
				
			||||||
                raise Pebkac(410, "location unavailable")
 | 
					                raise Pebkac(410, "location unavailable")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        cj["name"] = sanitize_fn(cj["name"])
 | 
					        cj["name"] = sanitize_fn(cj["name"], bad=[".prologue.html", ".epilogue.html"])
 | 
				
			||||||
        cj["poke"] = time.time()
 | 
					        cj["poke"] = time.time()
 | 
				
			||||||
        wark = self._get_wark(cj)
 | 
					        wark = self._get_wark(cj)
 | 
				
			||||||
        now = time.time()
 | 
					        now = time.time()
 | 
				
			||||||
@@ -1110,8 +1098,9 @@ class Up2k(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            atomic_move(src, dst)
 | 
					            atomic_move(src, dst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            if WINDOWS:
 | 
					            if ANYWIN:
 | 
				
			||||||
                self.lastmod_q.put([dst, (int(time.time()), int(job["lmod"]))])
 | 
					                a = [dst, job["size"], (int(time.time()), int(job["lmod"]))]
 | 
				
			||||||
 | 
					                self.lastmod_q.put(a)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            # legit api sware 2 me mum
 | 
					            # legit api sware 2 me mum
 | 
				
			||||||
            if self.idx_wark(
 | 
					            if self.idx_wark(
 | 
				
			||||||
@@ -1212,6 +1201,17 @@ class Up2k(object):
 | 
				
			|||||||
        suffix = ".{:.6f}-{}".format(job["t0"], job["addr"])
 | 
					        suffix = ".{:.6f}-{}".format(job["t0"], job["addr"])
 | 
				
			||||||
        with ren_open(tnam, "wb", fdir=pdir, suffix=suffix) as f:
 | 
					        with ren_open(tnam, "wb", fdir=pdir, suffix=suffix) as f:
 | 
				
			||||||
            f, job["tnam"] = f["orz"]
 | 
					            f, job["tnam"] = f["orz"]
 | 
				
			||||||
 | 
					            if (
 | 
				
			||||||
 | 
					                ANYWIN
 | 
				
			||||||
 | 
					                and self.args.sparse
 | 
				
			||||||
 | 
					                and self.args.sparse * 1024 * 1024 <= job["size"]
 | 
				
			||||||
 | 
					            ):
 | 
				
			||||||
 | 
					                fp = os.path.join(pdir, job["tnam"])
 | 
				
			||||||
 | 
					                try:
 | 
				
			||||||
 | 
					                    sp.check_call(["fsutil", "sparse", "setflag", fp])
 | 
				
			||||||
 | 
					                except:
 | 
				
			||||||
 | 
					                    self.log("could not sparse [{}]".format(fp), 3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            f.seek(job["size"] - 1)
 | 
					            f.seek(job["size"] - 1)
 | 
				
			||||||
            f.write(b"e")
 | 
					            f.write(b"e")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1223,13 +1223,19 @@ class Up2k(object):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            # self.log("lmod: got {}".format(len(ready)))
 | 
					            # self.log("lmod: got {}".format(len(ready)))
 | 
				
			||||||
            time.sleep(5)
 | 
					            time.sleep(5)
 | 
				
			||||||
            for path, times in ready:
 | 
					            for path, sz, times in ready:
 | 
				
			||||||
                self.log("lmod: setting times {} on {}".format(times, path))
 | 
					                self.log("lmod: setting times {} on {}".format(times, path))
 | 
				
			||||||
                try:
 | 
					                try:
 | 
				
			||||||
                    os.utime(fsenc(path), times)
 | 
					                    os.utime(fsenc(path), times)
 | 
				
			||||||
                except:
 | 
					                except:
 | 
				
			||||||
                    self.log("lmod: failed to utime ({}, {})".format(path, times))
 | 
					                    self.log("lmod: failed to utime ({}, {})".format(path, times))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if self.args.sparse and self.args.sparse * 1024 * 1024 <= sz:
 | 
				
			||||||
 | 
					                    try:
 | 
				
			||||||
 | 
					                        sp.check_call(["fsutil", "sparse", "setflag", path, "0"])
 | 
				
			||||||
 | 
					                    except:
 | 
				
			||||||
 | 
					                        self.log("could not unsparse [{}]".format(path), 3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def _snapshot(self):
 | 
					    def _snapshot(self):
 | 
				
			||||||
        persist_interval = 30  # persist unfinished uploads index every 30 sec
 | 
					        persist_interval = 30  # persist unfinished uploads index every 30 sec
 | 
				
			||||||
        discard_interval = 21600  # drop unfinished uploads after 6 hours inactivity
 | 
					        discard_interval = 21600  # drop unfinished uploads after 6 hours inactivity
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -16,7 +16,7 @@ import mimetypes
 | 
				
			|||||||
import contextlib
 | 
					import contextlib
 | 
				
			||||||
import subprocess as sp  # nosec
 | 
					import subprocess as sp  # nosec
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from .__init__ import PY2, WINDOWS
 | 
					from .__init__ import PY2, WINDOWS, ANYWIN
 | 
				
			||||||
from .stolen import surrogateescape
 | 
					from .stolen import surrogateescape
 | 
				
			||||||
 | 
					
 | 
				
			||||||
FAKE_MP = False
 | 
					FAKE_MP = False
 | 
				
			||||||
@@ -576,12 +576,12 @@ def undot(path):
 | 
				
			|||||||
    return "/".join(ret)
 | 
					    return "/".join(ret)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def sanitize_fn(fn, ok=""):
 | 
					def sanitize_fn(fn, ok="", bad=[]):
 | 
				
			||||||
    if "/" not in ok:
 | 
					    if "/" not in ok:
 | 
				
			||||||
        fn = fn.replace("\\", "/").split("/")[-1]
 | 
					        fn = fn.replace("\\", "/").split("/")[-1]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if WINDOWS:
 | 
					    if ANYWIN:
 | 
				
			||||||
        for bad, good in [x for x in [
 | 
					        remap = [
 | 
				
			||||||
            ["<", "<"],
 | 
					            ["<", "<"],
 | 
				
			||||||
            [">", ">"],
 | 
					            [">", ">"],
 | 
				
			||||||
            [":", ":"],
 | 
					            [":", ":"],
 | 
				
			||||||
@@ -591,15 +591,16 @@ def sanitize_fn(fn, ok=""):
 | 
				
			|||||||
            ["|", "|"],
 | 
					            ["|", "|"],
 | 
				
			||||||
            ["?", "?"],
 | 
					            ["?", "?"],
 | 
				
			||||||
            ["*", "*"],
 | 
					            ["*", "*"],
 | 
				
			||||||
        ] if x[0] not in ok]:
 | 
					        ]
 | 
				
			||||||
 | 
					        for bad, good in [x for x in remap if x[0] not in ok]:
 | 
				
			||||||
            fn = fn.replace(bad, good)
 | 
					            fn = fn.replace(bad, good)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        bad = ["con", "prn", "aux", "nul"]
 | 
					        bad.extend(["con", "prn", "aux", "nul"])
 | 
				
			||||||
        for n in range(1, 10):
 | 
					        for n in range(1, 10):
 | 
				
			||||||
            bad += "com{0} lpt{0}".format(n).split(" ")
 | 
					            bad += "com{0} lpt{0}".format(n).split(" ")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if fn.lower() in bad:
 | 
					    if fn.lower() in bad:
 | 
				
			||||||
            fn = "_" + fn
 | 
					        fn = "_" + fn
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return fn.strip()
 | 
					    return fn.strip()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -615,17 +616,24 @@ def exclude_dotfiles(filepaths):
 | 
				
			|||||||
    return [x for x in filepaths if not x.split("/")[-1].startswith(".")]
 | 
					    return [x for x in filepaths if not x.split("/")[-1].startswith(".")]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def html_escape(s, quote=False):
 | 
					def html_escape(s, quote=False, crlf=False):
 | 
				
			||||||
    """html.escape but also newlines"""
 | 
					    """html.escape but also newlines"""
 | 
				
			||||||
    s = (
 | 
					    s = s.replace("&", "&").replace("<", "<").replace(">", ">")
 | 
				
			||||||
        s.replace("&", "&")
 | 
					 | 
				
			||||||
        .replace("<", "<")
 | 
					 | 
				
			||||||
        .replace(">", ">")
 | 
					 | 
				
			||||||
        .replace("\r", "
")
 | 
					 | 
				
			||||||
        .replace("\n", "
")
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    if quote:
 | 
					    if quote:
 | 
				
			||||||
        s = s.replace('"', """).replace("'", "'")
 | 
					        s = s.replace('"', """).replace("'", "'")
 | 
				
			||||||
 | 
					    if crlf:
 | 
				
			||||||
 | 
					        s = s.replace("\r", "
").replace("\n", "
")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return s
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def html_bescape(s, quote=False, crlf=False):
 | 
				
			||||||
 | 
					    """html.escape but bytestrings"""
 | 
				
			||||||
 | 
					    s = s.replace(b"&", b"&").replace(b"<", b"<").replace(b">", b">")
 | 
				
			||||||
 | 
					    if quote:
 | 
				
			||||||
 | 
					        s = s.replace(b'"', b""").replace(b"'", b"'")
 | 
				
			||||||
 | 
					    if crlf:
 | 
				
			||||||
 | 
					        s = s.replace(b"\r", b"
").replace(b"\n", b"
")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return s
 | 
					    return s
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,7 +6,7 @@ html,body,tr,th,td,#files,a {
 | 
				
			|||||||
	background: none;
 | 
						background: none;
 | 
				
			||||||
	font-weight: inherit;
 | 
						font-weight: inherit;
 | 
				
			||||||
	font-size: inherit;
 | 
						font-size: inherit;
 | 
				
			||||||
	padding: none;
 | 
						padding: 0;
 | 
				
			||||||
	border: none;
 | 
						border: none;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html {
 | 
					html {
 | 
				
			||||||
@@ -68,7 +68,7 @@ a, #files tbody div a:last-child {
 | 
				
			|||||||
	color: #999;
 | 
						color: #999;
 | 
				
			||||||
	font-weight: normal;
 | 
						font-weight: normal;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#files tr+tr:hover {
 | 
					#files tr:hover {
 | 
				
			||||||
	background: #1c1c1c;
 | 
						background: #1c1c1c;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#files thead th {
 | 
					#files thead th {
 | 
				
			||||||
@@ -90,8 +90,6 @@ a, #files tbody div a:last-child {
 | 
				
			|||||||
#files td {
 | 
					#files td {
 | 
				
			||||||
	margin: 0;
 | 
						margin: 0;
 | 
				
			||||||
	padding: 0 .5em;
 | 
						padding: 0 .5em;
 | 
				
			||||||
}
 | 
					 | 
				
			||||||
#files td {
 | 
					 | 
				
			||||||
	border-bottom: 1px solid #111;
 | 
						border-bottom: 1px solid #111;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#files td+td+td {
 | 
					#files td+td+td {
 | 
				
			||||||
@@ -187,6 +185,16 @@ a, #files tbody div a:last-child {
 | 
				
			|||||||
	background: #925;
 | 
						background: #925;
 | 
				
			||||||
	border-color: #c37;
 | 
						border-color: #c37;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#files tr.sel a {
 | 
				
			||||||
 | 
						color: #fff;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#files tr.sel a.play {
 | 
				
			||||||
 | 
						color: #fc5;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#files tr.sel a.play.act {
 | 
				
			||||||
 | 
						color: #fff;
 | 
				
			||||||
 | 
						text-shadow: 0 0 1px #fff;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#blocked {
 | 
					#blocked {
 | 
				
			||||||
	position: fixed;
 | 
						position: fixed;
 | 
				
			||||||
	top: 0;
 | 
						top: 0;
 | 
				
			||||||
@@ -343,10 +351,10 @@ a, #files tbody div a:last-child {
 | 
				
			|||||||
	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) {
 | 
					@media (min-width: 80em) {
 | 
				
			||||||
	#barpos,
 | 
						#barpos,
 | 
				
			||||||
	#barbuf {
 | 
						#barbuf {
 | 
				
			||||||
		width: calc(100% - 24em);
 | 
							width: calc(100% - 21em);
 | 
				
			||||||
		left: 9.8em;
 | 
							left: 9.8em;
 | 
				
			||||||
		top: .7em;
 | 
							top: .7em;
 | 
				
			||||||
		height: 1.6em;
 | 
							height: 1.6em;
 | 
				
			||||||
@@ -356,6 +364,9 @@ a, #files tbody div a:last-child {
 | 
				
			|||||||
		bottom: -3.2em;
 | 
							bottom: -3.2em;
 | 
				
			||||||
		height: 3.2em;
 | 
							height: 3.2em;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						#pvol {
 | 
				
			||||||
 | 
							max-width: 9em;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -487,9 +498,6 @@ input[type="checkbox"]:checked+label {
 | 
				
			|||||||
	border-collapse: collapse;
 | 
						border-collapse: collapse;
 | 
				
			||||||
	width: 100%;
 | 
						width: 100%;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#files td div a:last-child {
 | 
					 | 
				
			||||||
	width: 100%;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
#wrap {
 | 
					#wrap {
 | 
				
			||||||
	margin-top: 2em;
 | 
						margin-top: 2em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -508,9 +516,7 @@ input[type="checkbox"]:checked+label {
 | 
				
			|||||||
#thx_ff {
 | 
					#thx_ff {
 | 
				
			||||||
	padding: 5em 0;
 | 
						padding: 5em 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#tree::-webkit-scrollbar-track {
 | 
					#tree::-webkit-scrollbar-track,
 | 
				
			||||||
	background: #333;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
#tree::-webkit-scrollbar {
 | 
					#tree::-webkit-scrollbar {
 | 
				
			||||||
	background: #333;
 | 
						background: #333;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -685,6 +691,15 @@ input[type="checkbox"]:checked+label {
 | 
				
			|||||||
	font-family: monospace, monospace;
 | 
						font-family: monospace, monospace;
 | 
				
			||||||
	line-height: 2em;
 | 
						line-height: 2em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#pvol,
 | 
				
			||||||
 | 
					#barbuf,
 | 
				
			||||||
 | 
					#barpos,
 | 
				
			||||||
 | 
					#u2conf label {
 | 
				
			||||||
 | 
						-webkit-user-select: none;
 | 
				
			||||||
 | 
						-moz-user-select: none;
 | 
				
			||||||
 | 
						-ms-user-select: none;
 | 
				
			||||||
 | 
						user-select: none;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -704,7 +719,7 @@ html.light #srch_form {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
html.light #ops a.act {
 | 
					html.light #ops a.act {
 | 
				
			||||||
	box-shadow: 0 .2em .2em #ccc;
 | 
						box-shadow: 0 .2em .2em #ccc;
 | 
				
			||||||
	background: #f7f7f7;
 | 
						background: #fff;
 | 
				
			||||||
	border-color: #07a;
 | 
						border-color: #07a;
 | 
				
			||||||
	padding-top: .4em;
 | 
						padding-top: .4em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -761,7 +776,7 @@ html.light #files {
 | 
				
			|||||||
html.light #files thead th {
 | 
					html.light #files thead th {
 | 
				
			||||||
	background: #eee;
 | 
						background: #eee;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.light #files tr+tr td {
 | 
					html.light #files tr td {
 | 
				
			||||||
	border-top: 1px solid #ddd;
 | 
						border-top: 1px solid #ddd;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.light #files td {
 | 
					html.light #files td {
 | 
				
			||||||
@@ -785,8 +800,12 @@ html.light tr.play td {
 | 
				
			|||||||
html.light tr.play a {
 | 
					html.light tr.play a {
 | 
				
			||||||
	color: #406;
 | 
						color: #406;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					html.light #files th:hover .cfg,
 | 
				
			||||||
 | 
					html.light #files th.min .cfg {
 | 
				
			||||||
 | 
						background: #ccc;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
html.light #files > thead > tr > th.min span {
 | 
					html.light #files > thead > tr > th.min span {
 | 
				
			||||||
	background: linear-gradient(90deg, rgba(68,68,68,0), rgba(68,68,68,0.2) 70%, rgba(68,68,68,0.5));
 | 
						background: linear-gradient(90deg, rgba(204,204,204,0), rgba(204,204,204,0.5) 70%, #ccc);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.light #blocked {
 | 
					html.light #blocked {
 | 
				
			||||||
	background: #eee;
 | 
						background: #eee;
 | 
				
			||||||
@@ -808,6 +827,9 @@ html.light #files tr.sel td {
 | 
				
			|||||||
html.light #files tr.sel a {
 | 
					html.light #files tr.sel a {
 | 
				
			||||||
	color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					html.light #files tr.sel a.play.act {
 | 
				
			||||||
 | 
						color: #fb0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
html.light input[type="checkbox"] + label {
 | 
					html.light input[type="checkbox"] + label {
 | 
				
			||||||
	color: #333;
 | 
						color: #333;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -854,4 +876,14 @@ html.light #files a:hover,
 | 
				
			|||||||
html.light #files tr.sel a:hover {
 | 
					html.light #files tr.sel a:hover {
 | 
				
			||||||
	color: #000;
 | 
						color: #000;
 | 
				
			||||||
	background: #fff;
 | 
						background: #fff;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					html.light #tree {
 | 
				
			||||||
 | 
						scrollbar-color: #a70 #ddd;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					html.light #tree::-webkit-scrollbar-track,
 | 
				
			||||||
 | 
					html.light #tree::-webkit-scrollbar {
 | 
				
			||||||
 | 
						background: #ddd;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#tree::-webkit-scrollbar-thumb {
 | 
				
			||||||
 | 
						background: #da0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -9,7 +9,7 @@ function dbg(msg) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// extract songs + add play column
 | 
					// extract songs + add play column
 | 
				
			||||||
function MPlayer() {
 | 
					function MPlayer() {
 | 
				
			||||||
	this.id = new Date().getTime();
 | 
						this.id = Date.now();
 | 
				
			||||||
	this.au = null;
 | 
						this.au = null;
 | 
				
			||||||
	this.au_native = null;
 | 
						this.au_native = null;
 | 
				
			||||||
	this.au_ogvjs = null;
 | 
						this.au_ogvjs = null;
 | 
				
			||||||
@@ -17,15 +17,17 @@ function MPlayer() {
 | 
				
			|||||||
	this.tracks = {};
 | 
						this.tracks = {};
 | 
				
			||||||
	this.order = [];
 | 
						this.order = [];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var re_audio = /\.(opus|ogg|m4a|aac|mp3|wav|flac)$/i;
 | 
						var re_audio = /\.(opus|ogg|m4a|aac|mp3|wav|flac)$/i,
 | 
				
			||||||
	var trs = document.querySelectorAll('#files tbody tr');
 | 
							trs = QSA('#files tbody tr');
 | 
				
			||||||
	for (var a = 0, aa = trs.length; a < aa; a++) {
 | 
					
 | 
				
			||||||
		var tds = trs[a].getElementsByTagName('td');
 | 
						for (var a = 0, aa = trs.length; a < aa; a++) {
 | 
				
			||||||
		var link = tds[1].getElementsByTagName('a');
 | 
							var tds = trs[a].getElementsByTagName('td'),
 | 
				
			||||||
		link = link[link.length - 1];
 | 
								link = tds[1].getElementsByTagName('a');
 | 
				
			||||||
		var url = link.getAttribute('href');
 | 
					
 | 
				
			||||||
 | 
							link = link[link.length - 1];
 | 
				
			||||||
 | 
							var url = link.getAttribute('href'),
 | 
				
			||||||
 | 
								m = re_audio.exec(url);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		var m = re_audio.exec(url);
 | 
					 | 
				
			||||||
		if (m) {
 | 
							if (m) {
 | 
				
			||||||
			var tid = link.getAttribute('id');
 | 
								var tid = link.getAttribute('id');
 | 
				
			||||||
			this.order.push(tid);
 | 
								this.order.push(tid);
 | 
				
			||||||
@@ -54,8 +56,9 @@ function MPlayer() {
 | 
				
			|||||||
	};
 | 
						};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	this.read_order = function () {
 | 
						this.read_order = function () {
 | 
				
			||||||
		var order = [];
 | 
							var order = [],
 | 
				
			||||||
		var links = document.querySelectorAll('#files>tbody>tr>td:nth-child(1)>a');
 | 
								links = QSA('#files>tbody>tr>td:nth-child(1)>a');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		for (var a = 0, aa = links.length; a < aa; a++) {
 | 
							for (var a = 0, aa = links.length; a < aa; a++) {
 | 
				
			||||||
			var tid = links[a].getAttribute('id');
 | 
								var tid = links[a].getAttribute('id');
 | 
				
			||||||
			if (!tid || tid.indexOf('af-') !== 0)
 | 
								if (!tid || tid.indexOf('af-') !== 0)
 | 
				
			||||||
@@ -73,12 +76,12 @@ makeSortable(ebi('files'), mp.read_order.bind(mp));
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// toggle player widget
 | 
					// toggle player widget
 | 
				
			||||||
var widget = (function () {
 | 
					var widget = (function () {
 | 
				
			||||||
	var ret = {};
 | 
						var ret = {},
 | 
				
			||||||
	var widget = ebi('widget');
 | 
							widget = ebi('widget'),
 | 
				
			||||||
	var wtico = ebi('wtico');
 | 
							wtico = ebi('wtico'),
 | 
				
			||||||
	var touchmode = false;
 | 
							touchmode = false,
 | 
				
			||||||
	var side_open = false;
 | 
							side_open = false,
 | 
				
			||||||
	var was_paused = true;
 | 
							was_paused = true;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ret.open = function () {
 | 
						ret.open = function () {
 | 
				
			||||||
		if (side_open)
 | 
							if (side_open)
 | 
				
			||||||
@@ -107,159 +110,170 @@ var widget = (function () {
 | 
				
			|||||||
			ebi('bplay').innerHTML = paused ? '▶' : '⏸';
 | 
								ebi('bplay').innerHTML = paused ? '▶' : '⏸';
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
	var click_handler = function (e) {
 | 
						wtico.onclick = function (e) {
 | 
				
			||||||
		if (!touchmode)
 | 
							if (!touchmode)
 | 
				
			||||||
			ret.toggle(e);
 | 
								ret.toggle(e);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		return false;
 | 
							return false;
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
	wtico.onclick = click_handler;
 | 
					 | 
				
			||||||
	return ret;
 | 
						return ret;
 | 
				
			||||||
})();
 | 
					})();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function canvas_cfg(can) {
 | 
				
			||||||
 | 
						var r = {},
 | 
				
			||||||
 | 
							b = can.getBoundingClientRect(),
 | 
				
			||||||
 | 
							mul = window.devicePixelRatio || 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						r.w = b.width;
 | 
				
			||||||
 | 
						r.h = b.height;
 | 
				
			||||||
 | 
						can.width = r.w * mul;
 | 
				
			||||||
 | 
						can.height = r.h * mul;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						r.can = can;
 | 
				
			||||||
 | 
						r.ctx = can.getContext('2d');
 | 
				
			||||||
 | 
						r.ctx.scale(mul, mul);
 | 
				
			||||||
 | 
						return r;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function glossy_grad(can, hsl) {
 | 
				
			||||||
 | 
						var g = can.ctx.createLinearGradient(0, 0, 0, can.h),
 | 
				
			||||||
 | 
							s = [0, 0.49, 0.50, 1];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (var a = 0; a < hsl.length; a++)
 | 
				
			||||||
 | 
							g.addColorStop(s[a], 'hsl(' + hsl[a] + ')');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return g;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// buffer/position bar
 | 
					// buffer/position bar
 | 
				
			||||||
var pbar = (function () {
 | 
					var pbar = (function () {
 | 
				
			||||||
	var r = {};
 | 
						var r = {},
 | 
				
			||||||
	r.bcan = ebi('barbuf');
 | 
							gradh = -1,
 | 
				
			||||||
	r.pcan = ebi('barpos');
 | 
							grad;
 | 
				
			||||||
	r.bctx = r.bcan.getContext('2d');
 | 
					 | 
				
			||||||
	r.pctx = r.pcan.getContext('2d');
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var bctx = r.bctx;
 | 
						function onresize() {
 | 
				
			||||||
	var pctx = r.pctx;
 | 
							r.buf = canvas_cfg(ebi('barbuf'));
 | 
				
			||||||
	var scale = (window.devicePixelRatio || 1) / (
 | 
							r.pos = canvas_cfg(ebi('barpos'));
 | 
				
			||||||
		bctx.webkitBackingStorePixelRatio ||
 | 
							r.drawbuf();
 | 
				
			||||||
		bctx.mozBackingStorePixelRatio ||
 | 
							r.drawpos();
 | 
				
			||||||
		bctx.msBackingStorePixelRatio ||
 | 
						}
 | 
				
			||||||
		bctx.oBackingStorePixelRatio ||
 | 
					 | 
				
			||||||
		bctx.BackingStorePixelRatio || 1);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	var gradh = 0;
 | 
					 | 
				
			||||||
	var grad = null;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	r.drawbuf = function () {
 | 
						r.drawbuf = function () {
 | 
				
			||||||
		if (!mp.au)
 | 
							if (!mp.au)
 | 
				
			||||||
			return;
 | 
								return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		var cs = getComputedStyle(r.bcan);
 | 
							var bc = r.buf,
 | 
				
			||||||
		var sw = parseInt(cs['width']);
 | 
								bctx = bc.ctx,
 | 
				
			||||||
		var sh = parseInt(cs['height']);
 | 
								sm = bc.w * 1.0 / mp.au.duration;
 | 
				
			||||||
		var sm = sw * 1.0 / mp.au.duration;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
		r.bcan.width = (sw * scale);
 | 
							if (gradh != bc.h) {
 | 
				
			||||||
		r.bcan.height = (sh * scale);
 | 
								gradh = bc.h;
 | 
				
			||||||
		bctx.setTransform(scale, 0, 0, scale, 0, 0);
 | 
								grad = glossy_grad(bc, [
 | 
				
			||||||
 | 
									'85,35%,42%',
 | 
				
			||||||
		if (!grad || gradh != sh) {
 | 
									'85,40%,49%',
 | 
				
			||||||
			grad = bctx.createLinearGradient(0, 0, 0, sh);
 | 
									'85,37%,47%',
 | 
				
			||||||
			grad.addColorStop(0, 'hsl(85,35%,42%)');
 | 
									'85,35%,42%'
 | 
				
			||||||
			grad.addColorStop(0.49, 'hsl(85,40%,49%)');
 | 
								]);
 | 
				
			||||||
			grad.addColorStop(0.50, 'hsl(85,37%,47%)');
 | 
					 | 
				
			||||||
			grad.addColorStop(1, 'hsl(85,35%,42%)');
 | 
					 | 
				
			||||||
			gradh = sh;
 | 
					 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		bctx.fillStyle = grad;
 | 
							bctx.fillStyle = grad;
 | 
				
			||||||
		bctx.clearRect(0, 0, sw, sh);
 | 
							bctx.clearRect(0, 0, bc.w, bc.h);
 | 
				
			||||||
		for (var a = 0; a < mp.au.buffered.length; a++) {
 | 
							for (var a = 0; a < mp.au.buffered.length; a++) {
 | 
				
			||||||
			var x1 = sm * mp.au.buffered.start(a);
 | 
								var x1 = sm * mp.au.buffered.start(a),
 | 
				
			||||||
			var x2 = sm * mp.au.buffered.end(a);
 | 
									x2 = sm * mp.au.buffered.end(a);
 | 
				
			||||||
			bctx.fillRect(x1, 0, x2 - x1, sh);
 | 
					
 | 
				
			||||||
 | 
								bctx.fillRect(x1, 0, x2 - x1, bc.h);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	r.drawpos = function () {
 | 
						r.drawpos = function () {
 | 
				
			||||||
		if (!mp.au)
 | 
							if (!mp.au)
 | 
				
			||||||
			return;
 | 
								return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		var cs = getComputedStyle(r.bcan);
 | 
							var bc = r.buf,
 | 
				
			||||||
		var sw = parseInt(cs['width']);
 | 
								pc = r.pos,
 | 
				
			||||||
		var sh = parseInt(cs['height']);
 | 
								pctx = pc.ctx,
 | 
				
			||||||
		var sm = sw * 1.0 / mp.au.duration;
 | 
								sm = bc.w * 1.0 / mp.au.duration;
 | 
				
			||||||
 | 
					 | 
				
			||||||
		r.pcan.width = (sw * scale);
 | 
					 | 
				
			||||||
		r.pcan.height = (sh * scale);
 | 
					 | 
				
			||||||
		pctx.setTransform(scale, 0, 0, scale, 0, 0);
 | 
					 | 
				
			||||||
		pctx.clearRect(0, 0, sw, sh);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							pctx.clearRect(0, 0, pc.w, pc.h);
 | 
				
			||||||
		pctx.fillStyle = 'rgba(204,255,128,0.15)';
 | 
							pctx.fillStyle = 'rgba(204,255,128,0.15)';
 | 
				
			||||||
		for (var p = 1, mins = mp.au.duration / 10; p <= mins; p++)
 | 
							for (var p = 1, mins = mp.au.duration / 10; p <= mins; p++)
 | 
				
			||||||
			pctx.fillRect(Math.floor(sm * p * 10), 0, 2, sh);
 | 
								pctx.fillRect(Math.floor(sm * p * 10), 0, 2, pc.h);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		pctx.fillStyle = '#9b7';
 | 
							pctx.fillStyle = '#9b7';
 | 
				
			||||||
		pctx.fillStyle = 'rgba(192,255,96,0.5)';
 | 
							pctx.fillStyle = 'rgba(192,255,96,0.5)';
 | 
				
			||||||
		for (var p = 1, mins = mp.au.duration / 60; p <= mins; p++)
 | 
							for (var p = 1, mins = mp.au.duration / 60; p <= mins; p++)
 | 
				
			||||||
			pctx.fillRect(Math.floor(sm * p * 60), 0, 2, sh);
 | 
								pctx.fillRect(Math.floor(sm * p * 60), 0, 2, pc.h);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		var w = 8;
 | 
							var w = 8,
 | 
				
			||||||
		var x = sm * mp.au.currentTime;
 | 
								x = sm * mp.au.currentTime;
 | 
				
			||||||
		pctx.fillStyle = '#573'; pctx.fillRect((x - w / 2) - 1, 0, w + 2, sh);
 | 
					
 | 
				
			||||||
		pctx.fillStyle = '#dfc'; pctx.fillRect((x - w / 2), 0, 8, sh);
 | 
							pctx.fillStyle = '#573'; pctx.fillRect((x - w / 2) - 1, 0, w + 2, pc.h);
 | 
				
			||||||
 | 
							pctx.fillStyle = '#dfc'; pctx.fillRect((x - w / 2), 0, 8, pc.h);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		pctx.fillStyle = '#fff';
 | 
							pctx.fillStyle = '#fff';
 | 
				
			||||||
		pctx.font = '1em sans-serif';
 | 
							pctx.font = '1em sans-serif';
 | 
				
			||||||
		var txt = s2ms(mp.au.duration);
 | 
							var txt = s2ms(mp.au.duration),
 | 
				
			||||||
		var tw = pctx.measureText(txt).width;
 | 
								tw = pctx.measureText(txt).width;
 | 
				
			||||||
		pctx.fillText(txt, sw - (tw + 8), sh / 3 * 2);
 | 
					
 | 
				
			||||||
 | 
							pctx.fillText(txt, pc.w - (tw + 8), pc.h / 3 * 2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		txt = s2ms(mp.au.currentTime);
 | 
							txt = s2ms(mp.au.currentTime);
 | 
				
			||||||
		tw = pctx.measureText(txt).width;
 | 
							tw = pctx.measureText(txt).width;
 | 
				
			||||||
		var gw = pctx.measureText("88:88::").width;
 | 
							var gw = pctx.measureText("88:88::").width,
 | 
				
			||||||
		var xt = x < sw / 2 ? (x + 8) : (Math.min(sw - gw, x - 8) - tw);
 | 
								xt = x < pc.w / 2 ? (x + 8) : (Math.min(pc.w - gw, x - 8) - tw);
 | 
				
			||||||
		pctx.fillText(txt, xt, sh / 3 * 2);
 | 
					
 | 
				
			||||||
 | 
							pctx.fillText(txt, xt, pc.h / 3 * 2);
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						window.addEventListener('resize', onresize);
 | 
				
			||||||
 | 
						onresize();
 | 
				
			||||||
	return r;
 | 
						return r;
 | 
				
			||||||
})();
 | 
					})();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// volume bar
 | 
					// volume bar
 | 
				
			||||||
var vbar = (function () {
 | 
					var vbar = (function () {
 | 
				
			||||||
	var r = {};
 | 
						var r = {},
 | 
				
			||||||
	r.can = ebi('pvol');
 | 
							gradh = -1,
 | 
				
			||||||
	r.ctx = r.can.getContext('2d');
 | 
							can, ctx, w, h, grad1, grad2;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var bctx = r.ctx;
 | 
						function onresize() {
 | 
				
			||||||
	var scale = (window.devicePixelRatio || 1) / (
 | 
							r.can = canvas_cfg(ebi('pvol'));
 | 
				
			||||||
		bctx.webkitBackingStorePixelRatio ||
 | 
							can = r.can.can;
 | 
				
			||||||
		bctx.mozBackingStorePixelRatio ||
 | 
							ctx = r.can.ctx;
 | 
				
			||||||
		bctx.msBackingStorePixelRatio ||
 | 
							w = r.can.w;
 | 
				
			||||||
		bctx.oBackingStorePixelRatio ||
 | 
							h = r.can.h;
 | 
				
			||||||
		bctx.BackingStorePixelRatio || 1);
 | 
							r.draw();
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	var gradh = 0;
 | 
					 | 
				
			||||||
	var grad1 = null;
 | 
					 | 
				
			||||||
	var grad2 = null;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	r.draw = function () {
 | 
						r.draw = function () {
 | 
				
			||||||
		var cs = getComputedStyle(r.can);
 | 
							if (gradh != h) {
 | 
				
			||||||
		var sw = parseInt(cs['width']);
 | 
								gradh = h;
 | 
				
			||||||
		var sh = parseInt(cs['height']);
 | 
								grad1 = glossy_grad(r.can, [
 | 
				
			||||||
 | 
									'50,45%,42%',
 | 
				
			||||||
		r.can.width = (sw * scale);
 | 
									'50,50%,49%',
 | 
				
			||||||
		r.can.height = (sh * scale);
 | 
									'50,47%,47%',
 | 
				
			||||||
		bctx.setTransform(scale, 0, 0, scale, 0, 0);
 | 
									'50,45%,42%'
 | 
				
			||||||
 | 
								]);
 | 
				
			||||||
		if (!grad1 || gradh != sh) {
 | 
								grad2 = glossy_grad(r.can, [
 | 
				
			||||||
			gradh = sh;
 | 
									'205,10%,16%',
 | 
				
			||||||
 | 
									'205,15%,20%',
 | 
				
			||||||
			grad1 = bctx.createLinearGradient(0, 0, 0, sh);
 | 
									'205,13%,18%',
 | 
				
			||||||
			grad1.addColorStop(0, 'hsl(50,45%,42%)');
 | 
									'205,10%,16%'
 | 
				
			||||||
			grad1.addColorStop(0.49, 'hsl(50,50%,49%)');
 | 
								]);
 | 
				
			||||||
			grad1.addColorStop(0.50, 'hsl(50,47%,47%)');
 | 
					 | 
				
			||||||
			grad1.addColorStop(1, 'hsl(50,45%,42%)');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			grad2 = bctx.createLinearGradient(0, 0, 0, sh);
 | 
					 | 
				
			||||||
			grad2.addColorStop(0, 'hsl(205,10%,16%)');
 | 
					 | 
				
			||||||
			grad2.addColorStop(0.49, 'hsl(205,15%,20%)');
 | 
					 | 
				
			||||||
			grad2.addColorStop(0.50, 'hsl(205,13%,18%)');
 | 
					 | 
				
			||||||
			grad2.addColorStop(1, 'hsl(205,10%,16%)');
 | 
					 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		bctx.fillStyle = grad2; bctx.fillRect(0, 0, sw, sh);
 | 
							ctx.fillStyle = grad2; ctx.fillRect(0, 0, w, h);
 | 
				
			||||||
		bctx.fillStyle = grad1; bctx.fillRect(0, 0, sw * mp.vol, sh);
 | 
							ctx.fillStyle = grad1; ctx.fillRect(0, 0, w * mp.vol, h);
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
 | 
						window.addEventListener('resize', onresize);
 | 
				
			||||||
 | 
						onresize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var rect;
 | 
						var rect;
 | 
				
			||||||
	function mousedown(e) {
 | 
						function mousedown(e) {
 | 
				
			||||||
		rect = r.can.getBoundingClientRect();
 | 
							rect = can.getBoundingClientRect();
 | 
				
			||||||
		mousemove(e);
 | 
							mousemove(e);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	function mousemove(e) {
 | 
						function mousemove(e) {
 | 
				
			||||||
@@ -267,34 +281,34 @@ var vbar = (function () {
 | 
				
			|||||||
			e = e.changedTouches[0];
 | 
								e = e.changedTouches[0];
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		else if (e.buttons === 0) {
 | 
							else if (e.buttons === 0) {
 | 
				
			||||||
			r.can.onmousemove = null;
 | 
								can.onmousemove = null;
 | 
				
			||||||
			return;
 | 
								return;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		var x = e.clientX - rect.left;
 | 
							var x = e.clientX - rect.left,
 | 
				
			||||||
		var mul = x * 1.0 / rect.width;
 | 
								mul = x * 1.0 / rect.width;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if (mul > 0.98)
 | 
							if (mul > 0.98)
 | 
				
			||||||
			mul = 1;
 | 
								mul = 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		mp.setvol(mul);
 | 
							mp.setvol(mul);
 | 
				
			||||||
		r.draw();
 | 
							r.draw();
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	r.can.onmousedown = function (e) {
 | 
						can.onmousedown = function (e) {
 | 
				
			||||||
		if (e.button !== 0)
 | 
							if (e.button !== 0)
 | 
				
			||||||
			return;
 | 
								return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		r.can.onmousemove = mousemove;
 | 
							can.onmousemove = mousemove;
 | 
				
			||||||
		mousedown(e);
 | 
							mousedown(e);
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
	r.can.onmouseup = function (e) {
 | 
						can.onmouseup = function (e) {
 | 
				
			||||||
		if (e.button === 0)
 | 
							if (e.button === 0)
 | 
				
			||||||
			r.can.onmousemove = null;
 | 
								can.onmousemove = null;
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
	if (window.Touch) {
 | 
						if (window.Touch) {
 | 
				
			||||||
		r.can.ontouchstart = mousedown;
 | 
							can.ontouchstart = mousedown;
 | 
				
			||||||
		r.can.ontouchmove = mousemove;
 | 
							can.ontouchmove = mousemove;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	r.draw();
 | 
					 | 
				
			||||||
	return r;
 | 
						return r;
 | 
				
			||||||
})();
 | 
					})();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -358,8 +372,9 @@ function song_skip(n) {
 | 
				
			|||||||
			return play(0);
 | 
								return play(0);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		var rect = pbar.pcan.getBoundingClientRect();
 | 
							var rect = pbar.buf.can.getBoundingClientRect(),
 | 
				
			||||||
		var x = e.clientX - rect.left;
 | 
								x = e.clientX - rect.left;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		seek_au_mul(x * 1.0 / rect.width);
 | 
							seek_au_mul(x * 1.0 / rect.width);
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
})();
 | 
					})();
 | 
				
			||||||
@@ -367,8 +382,9 @@ function song_skip(n) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// periodic tasks
 | 
					// periodic tasks
 | 
				
			||||||
(function () {
 | 
					(function () {
 | 
				
			||||||
	var nth = 0;
 | 
						var nth = 0,
 | 
				
			||||||
	var last_skip_url = '';
 | 
							last_skip_url = '';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var progress_updater = function () {
 | 
						var progress_updater = function () {
 | 
				
			||||||
		if (!mp.au) {
 | 
							if (!mp.au) {
 | 
				
			||||||
			widget.paused(true);
 | 
								widget.paused(true);
 | 
				
			||||||
@@ -389,8 +405,9 @@ function song_skip(n) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
			// switch to next track if approaching the end
 | 
								// switch to next track if approaching the end
 | 
				
			||||||
			if (last_skip_url != mp.au.src) {
 | 
								if (last_skip_url != mp.au.src) {
 | 
				
			||||||
				var pos = mp.au.currentTime;
 | 
									var pos = mp.au.currentTime,
 | 
				
			||||||
				var len = mp.au.duration;
 | 
										len = mp.au.duration;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
				if (pos > 0 && pos > len - 0.1) {
 | 
									if (pos > 0 && pos > len - 0.1) {
 | 
				
			||||||
					last_skip_url = mp.au.src;
 | 
										last_skip_url = mp.au.src;
 | 
				
			||||||
					song_skip(1);
 | 
										song_skip(1);
 | 
				
			||||||
@@ -532,8 +549,8 @@ function play(tid, seek, call_depth) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// event from the audio object if something breaks
 | 
					// event from the audio object if something breaks
 | 
				
			||||||
function evau_error(e) {
 | 
					function evau_error(e) {
 | 
				
			||||||
	var err = '';
 | 
						var err = '',
 | 
				
			||||||
	var eplaya = (e && e.target) || (window.event && window.event.srcElement);
 | 
							eplaya = (e && e.target) || (window.event && window.event.srcElement);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	switch (eplaya.error.code) {
 | 
						switch (eplaya.error.code) {
 | 
				
			||||||
		case eplaya.error.MEDIA_ERR_ABORTED:
 | 
							case eplaya.error.MEDIA_ERR_ABORTED:
 | 
				
			||||||
@@ -563,8 +580,9 @@ function evau_error(e) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// show a fullscreen message
 | 
					// show a fullscreen message
 | 
				
			||||||
function show_modal(html) {
 | 
					function show_modal(html) {
 | 
				
			||||||
	var body = document.body || document.getElementsByTagName('body')[0];
 | 
						var body = document.body || document.getElementsByTagName('body')[0],
 | 
				
			||||||
	var div = document.createElement('div');
 | 
							div = mknod('div');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	div.setAttribute('id', 'blocked');
 | 
						div.setAttribute('id', 'blocked');
 | 
				
			||||||
	div.innerHTML = html;
 | 
						div.innerHTML = html;
 | 
				
			||||||
	unblocked();
 | 
						unblocked();
 | 
				
			||||||
@@ -586,10 +604,10 @@ function autoplay_blocked(seek) {
 | 
				
			|||||||
		'<div id="blk_play"><a href="#" id="blk_go"></a></div>' +
 | 
							'<div id="blk_play"><a href="#" id="blk_go"></a></div>' +
 | 
				
			||||||
		'<div id="blk_abrt"><a href="#" id="blk_na">Cancel<br />(show file list)</a></div>');
 | 
							'<div id="blk_abrt"><a href="#" id="blk_na">Cancel<br />(show file list)</a></div>');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var go = ebi('blk_go');
 | 
						var go = ebi('blk_go'),
 | 
				
			||||||
	var na = ebi('blk_na');
 | 
							na = ebi('blk_na'),
 | 
				
			||||||
 | 
							fn = mp.tracks[mp.au.tid].split(/\//).pop();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var fn = mp.tracks[mp.au.tid].split(/\//).pop();
 | 
					 | 
				
			||||||
	fn = uricom_dec(fn.replace(/\+/g, ' '))[0];
 | 
						fn = uricom_dec(fn.replace(/\+/g, ' '))[0];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	go.textContent = 'Play "' + fn + '"';
 | 
						go.textContent = 'Play "' + fn + '"';
 | 
				
			||||||
@@ -625,7 +643,7 @@ function autoplay_blocked(seek) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function tree_neigh(n) {
 | 
					function tree_neigh(n) {
 | 
				
			||||||
	var links = document.querySelectorAll('#treeul li>a+a');
 | 
						var links = QSA('#treeul li>a+a');
 | 
				
			||||||
	if (!links.length) {
 | 
						if (!links.length) {
 | 
				
			||||||
		alert('switch to the tree for that');
 | 
							alert('switch to the tree for that');
 | 
				
			||||||
		return;
 | 
							return;
 | 
				
			||||||
@@ -651,7 +669,7 @@ function tree_neigh(n) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function tree_up() {
 | 
					function tree_up() {
 | 
				
			||||||
	var act = document.querySelector('#treeul a.hl');
 | 
						var act = QS('#treeul a.hl');
 | 
				
			||||||
	if (!act) {
 | 
						if (!act) {
 | 
				
			||||||
		alert('switch to the tree for that');
 | 
							alert('switch to the tree for that');
 | 
				
			||||||
		return;
 | 
							return;
 | 
				
			||||||
@@ -714,7 +732,7 @@ document.onkeydown = function (e) {
 | 
				
			|||||||
	];
 | 
						];
 | 
				
			||||||
	var oldcfg = [];
 | 
						var oldcfg = [];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (document.querySelector('#srch_form.tags')) {
 | 
						if (QS('#srch_form.tags')) {
 | 
				
			||||||
		sconf.push(["tags",
 | 
							sconf.push(["tags",
 | 
				
			||||||
			["tags", "tags", "tags contains   (^=start, end=$)", "46"]
 | 
								["tags", "tags", "tags contains   (^=start, end=$)", "46"]
 | 
				
			||||||
		]);
 | 
							]);
 | 
				
			||||||
@@ -723,13 +741,15 @@ document.onkeydown = function (e) {
 | 
				
			|||||||
		]);
 | 
							]);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var trs = [];
 | 
						var trs = [],
 | 
				
			||||||
	var orig_html = null;
 | 
							orig_html = null;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (var a = 0; a < sconf.length; a++) {
 | 
						for (var a = 0; a < sconf.length; a++) {
 | 
				
			||||||
		var html = ['<tr><td><br />' + sconf[a][0] + '</td>'];
 | 
							var html = ['<tr><td><br />' + sconf[a][0] + '</td>'];
 | 
				
			||||||
		for (var b = 1; b < 3; b++) {
 | 
							for (var b = 1; b < 3; b++) {
 | 
				
			||||||
			var hn = "srch_" + sconf[a][b][0];
 | 
								var hn = "srch_" + sconf[a][b][0],
 | 
				
			||||||
			var csp = (sconf[a].length == 2) ? 2 : 1;
 | 
									csp = (sconf[a].length == 2) ? 2 : 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			html.push(
 | 
								html.push(
 | 
				
			||||||
				'<td colspan="' + csp + '"><input id="' + hn + 'c" type="checkbox">\n' +
 | 
									'<td colspan="' + csp + '"><input id="' + hn + 'c" type="checkbox">\n' +
 | 
				
			||||||
				'<label for="' + hn + 'c">' + sconf[a][b][2] + '</label>\n' +
 | 
									'<label for="' + hn + 'c">' + sconf[a][b][2] + '</label>\n' +
 | 
				
			||||||
@@ -747,7 +767,7 @@ document.onkeydown = function (e) {
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
	ebi('srch_form').innerHTML = html.join('\n');
 | 
						ebi('srch_form').innerHTML = html.join('\n');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var o = document.querySelectorAll('#op_search input');
 | 
						var o = QSA('#op_search input');
 | 
				
			||||||
	for (var a = 0; a < o.length; a++) {
 | 
						for (var a = 0; a < o.length; a++) {
 | 
				
			||||||
		o[a].oninput = ev_search_input;
 | 
							o[a].oninput = ev_search_input;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
@@ -758,28 +778,30 @@ document.onkeydown = function (e) {
 | 
				
			|||||||
		o.style.color = err ? '#f09' : '#c90';
 | 
							o.style.color = err ? '#f09' : '#c90';
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var search_timeout;
 | 
						var search_timeout,
 | 
				
			||||||
	var search_in_progress = 0;
 | 
							search_in_progress = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	function ev_search_input() {
 | 
						function ev_search_input() {
 | 
				
			||||||
		var v = this.value;
 | 
							var v = this.value,
 | 
				
			||||||
		var id = this.getAttribute('id');
 | 
								id = this.getAttribute('id');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if (id.slice(-1) == 'v') {
 | 
							if (id.slice(-1) == 'v') {
 | 
				
			||||||
			var chk = ebi(id.slice(0, -1) + 'c');
 | 
								var chk = ebi(id.slice(0, -1) + 'c');
 | 
				
			||||||
			chk.checked = ((v + '').length > 0);
 | 
								chk.checked = ((v + '').length > 0);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		clearTimeout(search_timeout);
 | 
							clearTimeout(search_timeout);
 | 
				
			||||||
		var now = new Date().getTime();
 | 
							if (Date.now() - search_in_progress > 30 * 1000)
 | 
				
			||||||
		if (now - search_in_progress > 30 * 1000)
 | 
					 | 
				
			||||||
			search_timeout = setTimeout(do_search, 200);
 | 
								search_timeout = setTimeout(do_search, 200);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	function do_search() {
 | 
						function do_search() {
 | 
				
			||||||
		search_in_progress = new Date().getTime();
 | 
							search_in_progress = Date.now();
 | 
				
			||||||
		srch_msg(false, "searching...");
 | 
							srch_msg(false, "searching...");
 | 
				
			||||||
		clearTimeout(search_timeout);
 | 
							clearTimeout(search_timeout);
 | 
				
			||||||
		var params = {};
 | 
					
 | 
				
			||||||
		var o = document.querySelectorAll('#op_search input[type="text"]');
 | 
							var params = {},
 | 
				
			||||||
 | 
								o = QSA('#op_search input[type="text"]');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		for (var a = 0; a < o.length; a++) {
 | 
							for (var a = 0; a < o.length; a++) {
 | 
				
			||||||
			var chk = ebi(o[a].getAttribute('id').slice(0, -1) + 'c');
 | 
								var chk = ebi(o[a].getAttribute('id').slice(0, -1) + 'c');
 | 
				
			||||||
			if (!chk.checked)
 | 
								if (!chk.checked)
 | 
				
			||||||
@@ -792,7 +814,7 @@ document.onkeydown = function (e) {
 | 
				
			|||||||
		xhr.open('POST', '/?srch', true);
 | 
							xhr.open('POST', '/?srch', true);
 | 
				
			||||||
		xhr.setRequestHeader('Content-Type', 'text/plain');
 | 
							xhr.setRequestHeader('Content-Type', 'text/plain');
 | 
				
			||||||
		xhr.onreadystatechange = xhr_search_results;
 | 
							xhr.onreadystatechange = xhr_search_results;
 | 
				
			||||||
		xhr.ts = new Date().getTime();
 | 
							xhr.ts = Date.now();
 | 
				
			||||||
		xhr.send(JSON.stringify(params));
 | 
							xhr.send(JSON.stringify(params));
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -982,12 +1004,13 @@ var treectl = (function () {
 | 
				
			|||||||
		if (!entreed || treectl.hidden)
 | 
							if (!entreed || treectl.hidden)
 | 
				
			||||||
			return;
 | 
								return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		var q = '#tree';
 | 
							var q = '#tree',
 | 
				
			||||||
		var nq = 0;
 | 
								nq = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		while (dyn) {
 | 
							while (dyn) {
 | 
				
			||||||
			nq++;
 | 
								nq++;
 | 
				
			||||||
			q += '>ul>li';
 | 
								q += '>ul>li';
 | 
				
			||||||
			if (!document.querySelector(q))
 | 
								if (!QS(q))
 | 
				
			||||||
				break;
 | 
									break;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		var w = treesz + nq;
 | 
							var w = treesz + nq;
 | 
				
			||||||
@@ -1001,7 +1024,7 @@ var treectl = (function () {
 | 
				
			|||||||
		xhr.top = top;
 | 
							xhr.top = top;
 | 
				
			||||||
		xhr.dst = dst;
 | 
							xhr.dst = dst;
 | 
				
			||||||
		xhr.rst = rst;
 | 
							xhr.rst = rst;
 | 
				
			||||||
		xhr.ts = new Date().getTime();
 | 
							xhr.ts = Date.now();
 | 
				
			||||||
		xhr.open('GET', dst + '?tree=' + top, true);
 | 
							xhr.open('GET', dst + '?tree=' + top, true);
 | 
				
			||||||
		xhr.onreadystatechange = recvtree;
 | 
							xhr.onreadystatechange = recvtree;
 | 
				
			||||||
		xhr.send();
 | 
							xhr.send();
 | 
				
			||||||
@@ -1026,10 +1049,11 @@ var treectl = (function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
		var top = this.top == '.' ? this.dst : this.top,
 | 
							var top = this.top == '.' ? this.dst : this.top,
 | 
				
			||||||
			name = uricom_dec(top.split('/').slice(-2)[0])[0],
 | 
								name = uricom_dec(top.split('/').slice(-2)[0])[0],
 | 
				
			||||||
			rtop = top.replace(/^\/+/, "");
 | 
								rtop = top.replace(/^\/+/, ""),
 | 
				
			||||||
 | 
								res;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		try {
 | 
							try {
 | 
				
			||||||
			var res = JSON.parse(this.responseText);
 | 
								res = JSON.parse(this.responseText);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		catch (ex) {
 | 
							catch (ex) {
 | 
				
			||||||
			return;
 | 
								return;
 | 
				
			||||||
@@ -1045,7 +1069,7 @@ var treectl = (function () {
 | 
				
			|||||||
				esc(top) + '">' + esc(name) +
 | 
									esc(top) + '">' + esc(name) +
 | 
				
			||||||
				"</a>\n<ul>\n" + html + "</ul>";
 | 
									"</a>\n<ul>\n" + html + "</ul>";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			var links = document.querySelectorAll('#treeul a+a');
 | 
								var links = QSA('#treeul a+a');
 | 
				
			||||||
			for (var a = 0, aa = links.length; a < aa; a++) {
 | 
								for (var a = 0, aa = links.length; a < aa; a++) {
 | 
				
			||||||
				if (links[a].getAttribute('href') == top) {
 | 
									if (links[a].getAttribute('href') == top) {
 | 
				
			||||||
					var o = links[a].parentNode;
 | 
										var o = links[a].parentNode;
 | 
				
			||||||
@@ -1054,21 +1078,22 @@ var treectl = (function () {
 | 
				
			|||||||
				}
 | 
									}
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		document.querySelector('#treeul>li>a+a').textContent = '[root]';
 | 
							QS('#treeul>li>a+a').textContent = '[root]';
 | 
				
			||||||
		despin('#tree');
 | 
							despin('#tree');
 | 
				
			||||||
		reload_tree();
 | 
							reload_tree();
 | 
				
			||||||
		onresize();
 | 
							onresize();
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	function reload_tree() {
 | 
						function reload_tree() {
 | 
				
			||||||
		var cdir = get_evpath();
 | 
							var cdir = get_evpath(),
 | 
				
			||||||
		var links = document.querySelectorAll('#treeul a+a');
 | 
								links = QSA('#treeul a+a');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		for (var a = 0, aa = links.length; a < aa; a++) {
 | 
							for (var a = 0, aa = links.length; a < aa; a++) {
 | 
				
			||||||
			var href = links[a].getAttribute('href');
 | 
								var href = links[a].getAttribute('href');
 | 
				
			||||||
			links[a].setAttribute('class', href == cdir ? 'hl' : '');
 | 
								links[a].setAttribute('class', href == cdir ? 'hl' : '');
 | 
				
			||||||
			links[a].onclick = treego;
 | 
								links[a].onclick = treego;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		links = document.querySelectorAll('#treeul li>a:first-child');
 | 
							links = QSA('#treeul li>a:first-child');
 | 
				
			||||||
		for (var a = 0, aa = links.length; a < aa; a++) {
 | 
							for (var a = 0, aa = links.length; a < aa; a++) {
 | 
				
			||||||
			links[a].setAttribute('dst', links[a].nextSibling.getAttribute('href'));
 | 
								links[a].setAttribute('dst', links[a].nextSibling.getAttribute('href'));
 | 
				
			||||||
			links[a].onclick = treegrow;
 | 
								links[a].onclick = treegrow;
 | 
				
			||||||
@@ -1089,7 +1114,7 @@ var treectl = (function () {
 | 
				
			|||||||
		var xhr = new XMLHttpRequest();
 | 
							var xhr = new XMLHttpRequest();
 | 
				
			||||||
		xhr.top = url;
 | 
							xhr.top = url;
 | 
				
			||||||
		xhr.hpush = hpush;
 | 
							xhr.hpush = hpush;
 | 
				
			||||||
		xhr.ts = new Date().getTime();
 | 
							xhr.ts = Date.now();
 | 
				
			||||||
		xhr.open('GET', xhr.top + '?ls', true);
 | 
							xhr.open('GET', xhr.top + '?ls', true);
 | 
				
			||||||
		xhr.onreadystatechange = recvls;
 | 
							xhr.onreadystatechange = recvls;
 | 
				
			||||||
		xhr.send();
 | 
							xhr.send();
 | 
				
			||||||
@@ -1139,12 +1164,13 @@ var treectl = (function () {
 | 
				
			|||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		ebi('srv_info').innerHTML = '<span>' + res.srvinf + '</span>';
 | 
							ebi('srv_info').innerHTML = '<span>' + res.srvinf + '</span>';
 | 
				
			||||||
		var nodes = res.dirs.concat(res.files);
 | 
					 | 
				
			||||||
		nodes = sortfiles(nodes);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
		var top = this.top;
 | 
							var top = this.top,
 | 
				
			||||||
		var html = mk_files_header(res.taglist);
 | 
								nodes = res.dirs.concat(res.files),
 | 
				
			||||||
 | 
								html = mk_files_header(res.taglist);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		html.push('<tbody>');
 | 
							html.push('<tbody>');
 | 
				
			||||||
 | 
							nodes = sortfiles(nodes);
 | 
				
			||||||
		for (var a = 0; a < nodes.length; a++) {
 | 
							for (var a = 0; a < nodes.length; a++) {
 | 
				
			||||||
			var r = nodes[a],
 | 
								var r = nodes[a],
 | 
				
			||||||
				ln = ['<tr><td>' + r.lead + '</td><td><a href="' +
 | 
									ln = ['<tr><td>' + r.lead + '</td><td><a href="' +
 | 
				
			||||||
@@ -1262,16 +1288,16 @@ var treectl = (function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
function enspin(sel) {
 | 
					function enspin(sel) {
 | 
				
			||||||
	despin(sel);
 | 
						despin(sel);
 | 
				
			||||||
	var d = document.createElement('div');
 | 
						var d = mknod('div');
 | 
				
			||||||
	d.setAttribute('class', 'dumb_loader_thing');
 | 
						d.setAttribute('class', 'dumb_loader_thing');
 | 
				
			||||||
	d.innerHTML = '🌲';
 | 
						d.innerHTML = '🌲';
 | 
				
			||||||
	var tgt = document.querySelector(sel);
 | 
						var tgt = QS(sel);
 | 
				
			||||||
	tgt.insertBefore(d, tgt.childNodes[0]);
 | 
						tgt.insertBefore(d, tgt.childNodes[0]);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function despin(sel) {
 | 
					function despin(sel) {
 | 
				
			||||||
	var o = document.querySelectorAll(sel + '>.dumb_loader_thing');
 | 
						var o = QSA(sel + '>.dumb_loader_thing');
 | 
				
			||||||
	for (var a = o.length - 1; a >= 0; a--)
 | 
						for (var a = o.length - 1; a >= 0; a--)
 | 
				
			||||||
		o[a].parentNode.removeChild(o[a]);
 | 
							o[a].parentNode.removeChild(o[a]);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -1280,17 +1306,17 @@ function despin(sel) {
 | 
				
			|||||||
function apply_perms(perms) {
 | 
					function apply_perms(perms) {
 | 
				
			||||||
	perms = perms || [];
 | 
						perms = perms || [];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var o = document.querySelectorAll('#ops>a[data-perm]');
 | 
						var o = QSA('#ops>a[data-perm]');
 | 
				
			||||||
	for (var a = 0; a < o.length; a++)
 | 
						for (var a = 0; a < o.length; a++)
 | 
				
			||||||
		o[a].style.display = 'none';
 | 
							o[a].style.display = 'none';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (var a = 0; a < perms.length; a++) {
 | 
						for (var a = 0; a < perms.length; a++) {
 | 
				
			||||||
		o = document.querySelectorAll('#ops>a[data-perm="' + perms[a] + '"]');
 | 
							o = QSA('#ops>a[data-perm="' + perms[a] + '"]');
 | 
				
			||||||
		for (var b = 0; b < o.length; b++)
 | 
							for (var b = 0; b < o.length; b++)
 | 
				
			||||||
			o[b].style.display = 'inline';
 | 
								o[b].style.display = 'inline';
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var act = document.querySelector('#ops>a.act');
 | 
						var act = QS('#ops>a.act');
 | 
				
			||||||
	if (act) {
 | 
						if (act) {
 | 
				
			||||||
		var areq = act.getAttribute('data-perm');
 | 
							var areq = act.getAttribute('data-perm');
 | 
				
			||||||
		if (areq && !has(perms, areq))
 | 
							if (areq && !has(perms, areq))
 | 
				
			||||||
@@ -1299,8 +1325,10 @@ function apply_perms(perms) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	document.body.setAttribute('perms', perms.join(' '));
 | 
						document.body.setAttribute('perms', perms.join(' '));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var have_write = has(perms, "write");
 | 
						var have_write = has(perms, "write"),
 | 
				
			||||||
	var tds = document.querySelectorAll('#u2conf td');
 | 
							have_read = has(perms, "read"),
 | 
				
			||||||
 | 
							tds = QSA('#u2conf td');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (var a = 0; a < tds.length; a++) {
 | 
						for (var a = 0; a < tds.length; a++) {
 | 
				
			||||||
		tds[a].style.display =
 | 
							tds[a].style.display =
 | 
				
			||||||
			(have_write || tds[a].getAttribute('data-perm') == 'read') ?
 | 
								(have_write || tds[a].getAttribute('data-perm') == 'read') ?
 | 
				
			||||||
@@ -1309,13 +1337,19 @@ function apply_perms(perms) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	if (window['up2k'])
 | 
						if (window['up2k'])
 | 
				
			||||||
		up2k.set_fsearch();
 | 
							up2k.set_fsearch();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ebi('widget').style.display = have_read ? '' : 'none';
 | 
				
			||||||
 | 
						ebi('files').style.display = have_read ? '' : 'none';
 | 
				
			||||||
 | 
						if (!have_read)
 | 
				
			||||||
 | 
							goto('up2k');
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function find_file_col(txt) {
 | 
					function find_file_col(txt) {
 | 
				
			||||||
	var tds = ebi('files').tHead.getElementsByTagName('th');
 | 
						var i = -1,
 | 
				
			||||||
	var i = -1;
 | 
							min = false,
 | 
				
			||||||
	var min = false;
 | 
							tds = ebi('files').tHead.getElementsByTagName('th');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (var a = 0; a < tds.length; a++) {
 | 
						for (var a = 0; a < tds.length; a++) {
 | 
				
			||||||
		var spans = tds[a].getElementsByTagName('span');
 | 
							var spans = tds[a].getElementsByTagName('span');
 | 
				
			||||||
		if (spans.length && spans[0].textContent == txt) {
 | 
							if (spans.length && spans[0].textContent == txt) {
 | 
				
			||||||
@@ -1340,8 +1374,9 @@ function mk_files_header(taglist) {
 | 
				
			|||||||
		'<th name="sz" sort="int"><span>Size</span></th>'
 | 
							'<th name="sz" sort="int"><span>Size</span></th>'
 | 
				
			||||||
	];
 | 
						];
 | 
				
			||||||
	for (var a = 0; a < taglist.length; a++) {
 | 
						for (var a = 0; a < taglist.length; a++) {
 | 
				
			||||||
		var tag = taglist[a];
 | 
							var tag = taglist[a],
 | 
				
			||||||
		var c1 = tag.slice(0, 1).toUpperCase();
 | 
								c1 = tag.slice(0, 1).toUpperCase();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		tag = c1 + tag.slice(1);
 | 
							tag = c1 + tag.slice(1);
 | 
				
			||||||
		if (c1 == '.')
 | 
							if (c1 == '.')
 | 
				
			||||||
			tag = '<th name="tags/' + tag + '" sort="int"><span>' + tag.slice(1);
 | 
								tag = '<th name="tags/' + tag + '" sort="int"><span>' + tag.slice(1);
 | 
				
			||||||
@@ -1363,10 +1398,11 @@ var filecols = (function () {
 | 
				
			|||||||
	var hidden = jread('filecols', []);
 | 
						var hidden = jread('filecols', []);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var add_btns = function () {
 | 
						var add_btns = function () {
 | 
				
			||||||
		var ths = document.querySelectorAll('#files th>span');
 | 
							var ths = QSA('#files th>span');
 | 
				
			||||||
		for (var a = 0, aa = ths.length; a < aa; a++) {
 | 
							for (var a = 0, aa = ths.length; a < aa; a++) {
 | 
				
			||||||
			var th = ths[a].parentElement;
 | 
								var th = ths[a].parentElement,
 | 
				
			||||||
			var is_hidden = has(hidden, ths[a].textContent);
 | 
									is_hidden = has(hidden, ths[a].textContent);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			th.innerHTML = '<div class="cfg"><a href="#">' +
 | 
								th.innerHTML = '<div class="cfg"><a href="#">' +
 | 
				
			||||||
				(is_hidden ? '+' : '-') + '</a></div>' + ths[a].outerHTML;
 | 
									(is_hidden ? '+' : '-') + '</a></div>' + ths[a].outerHTML;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1378,7 +1414,7 @@ var filecols = (function () {
 | 
				
			|||||||
		add_btns();
 | 
							add_btns();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		var ohidden = [],
 | 
							var ohidden = [],
 | 
				
			||||||
			ths = document.querySelectorAll('#files th'),
 | 
								ths = QSA('#files th'),
 | 
				
			||||||
			ncols = ths.length;
 | 
								ncols = ths.length;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		for (var a = 0; a < ncols; a++) {
 | 
							for (var a = 0; a < ncols; a++) {
 | 
				
			||||||
@@ -1396,8 +1432,9 @@ var filecols = (function () {
 | 
				
			|||||||
			clmod(ths[a], 'min', cls)
 | 
								clmod(ths[a], 'min', cls)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		for (var a = 0; a < ncols; a++) {
 | 
							for (var a = 0; a < ncols; a++) {
 | 
				
			||||||
			var cls = has(ohidden, a) ? 'min' : '';
 | 
								var cls = has(ohidden, a) ? 'min' : '',
 | 
				
			||||||
			var tds = document.querySelectorAll('#files>tbody>tr>td:nth-child(' + (a + 1) + ')');
 | 
									tds = QSA('#files>tbody>tr>td:nth-child(' + (a + 1) + ')');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			for (var b = 0, bb = tds.length; b < bb; b++) {
 | 
								for (var b = 0, bb = tds.length; b < bb; b++) {
 | 
				
			||||||
				tds[b].setAttribute('class', cls);
 | 
									tds[b].setAttribute('class', cls);
 | 
				
			||||||
				if (a < 2)
 | 
									if (a < 2)
 | 
				
			||||||
@@ -1475,9 +1512,9 @@ var mukey = (function () {
 | 
				
			|||||||
			"6m ", "7m ", "8m ", "9m ", "10m", "11m", "12m", "1m ", "2m ", "3m ", "4m ", "5m "
 | 
								"6m ", "7m ", "8m ", "9m ", "10m", "11m", "12m", "1m ", "2m ", "3m ", "4m ", "5m "
 | 
				
			||||||
		]
 | 
							]
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
	var map = {};
 | 
						var map = {},
 | 
				
			||||||
 | 
							html = [];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var html = [];
 | 
					 | 
				
			||||||
	for (var k in maps) {
 | 
						for (var k in maps) {
 | 
				
			||||||
		if (!maps.hasOwnProperty(k))
 | 
							if (!maps.hasOwnProperty(k))
 | 
				
			||||||
			continue;
 | 
								continue;
 | 
				
			||||||
@@ -1551,7 +1588,7 @@ var mukey = (function () {
 | 
				
			|||||||
	ebi('key_' + notation).checked = true;
 | 
						ebi('key_' + notation).checked = true;
 | 
				
			||||||
	load_notation(notation);
 | 
						load_notation(notation);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var o = document.querySelectorAll('#key_notation input');
 | 
						var o = QSA('#key_notation input');
 | 
				
			||||||
	for (var a = 0; a < o.length; a++) {
 | 
						for (var a = 0; a < o.length; a++) {
 | 
				
			||||||
		o[a].onchange = set_key_notation;
 | 
							o[a].onchange = set_key_notation;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
@@ -1563,7 +1600,7 @@ var mukey = (function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function addcrc() {
 | 
					function addcrc() {
 | 
				
			||||||
	var links = document.querySelectorAll(
 | 
						var links = QSA(
 | 
				
			||||||
		'#files>tbody>tr>td:first-child+td>' + (
 | 
							'#files>tbody>tr>td:first-child+td>' + (
 | 
				
			||||||
			ebi('unsearch') ? 'div>a:last-child' : 'a'));
 | 
								ebi('unsearch') ? 'div>a:last-child' : 'a'));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1586,7 +1623,7 @@ function addcrc() {
 | 
				
			|||||||
		o.setAttribute('class', tt ? '' : 'off');
 | 
							o.setAttribute('class', tt ? '' : 'off');
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var btns = document.querySelectorAll('#ops, #ops>a');
 | 
						var btns = QSA('#ops, #ops>a');
 | 
				
			||||||
	for (var a = 0; a < btns.length; a++) {
 | 
						for (var a = 0; a < btns.length; a++) {
 | 
				
			||||||
		btns[a].onmouseenter = set_tooltip;
 | 
							btns[a].onmouseenter = set_tooltip;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
@@ -1638,7 +1675,7 @@ var arcfmt = (function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	function render() {
 | 
						function render() {
 | 
				
			||||||
		var arg = arcv[arcfmts.indexOf(fmt)],
 | 
							var arg = arcv[arcfmts.indexOf(fmt)],
 | 
				
			||||||
			tds = document.querySelectorAll('#files tbody td:first-child a');
 | 
								tds = QSA('#files tbody td:first-child a');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		for (var a = 0, aa = tds.length; a < aa; a++) {
 | 
							for (var a = 0, aa = tds.length; a < aa; a++) {
 | 
				
			||||||
			var o = tds[a], txt = o.textContent, href = o.getAttribute('href');
 | 
								var o = tds[a], txt = o.textContent, href = o.getAttribute('href');
 | 
				
			||||||
@@ -1672,7 +1709,7 @@ var arcfmt = (function () {
 | 
				
			|||||||
		try_render();
 | 
							try_render();
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var o = document.querySelectorAll('#arc_fmt input');
 | 
						var o = QSA('#arc_fmt input');
 | 
				
			||||||
	for (var a = 0; a < o.length; a++) {
 | 
						for (var a = 0; a < o.length; a++) {
 | 
				
			||||||
		o[a].onchange = change_fmt;
 | 
							o[a].onchange = change_fmt;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
@@ -1685,8 +1722,9 @@ var arcfmt = (function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
var msel = (function () {
 | 
					var msel = (function () {
 | 
				
			||||||
	function getsel() {
 | 
						function getsel() {
 | 
				
			||||||
		var names = [];
 | 
							var names = [],
 | 
				
			||||||
		var links = document.querySelectorAll('#files tbody tr.sel td:nth-child(2) a');
 | 
								links = QSA('#files tbody tr.sel td:nth-child(2) a');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		for (var a = 0, aa = links.length; a < aa; a++)
 | 
							for (var a = 0, aa = links.length; a < aa; a++)
 | 
				
			||||||
			names.push(links[a].getAttribute('href').replace(/\/$/, "").split('/').slice(-1));
 | 
								names.push(links[a].getAttribute('href').replace(/\/$/, "").split('/').slice(-1));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1703,7 +1741,7 @@ var msel = (function () {
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
	function evsel(e, fun) {
 | 
						function evsel(e, fun) {
 | 
				
			||||||
		ev(e);
 | 
							ev(e);
 | 
				
			||||||
		var trs = document.querySelectorAll('#files tbody tr');
 | 
							var trs = QSA('#files tbody tr');
 | 
				
			||||||
		for (var a = 0, aa = trs.length; a < aa; a++)
 | 
							for (var a = 0, aa = trs.length; a < aa; a++)
 | 
				
			||||||
			clmod(trs[a], 'sel', fun);
 | 
								clmod(trs[a], 'sel', fun);
 | 
				
			||||||
		selui();
 | 
							selui();
 | 
				
			||||||
@@ -1716,10 +1754,11 @@ var msel = (function () {
 | 
				
			|||||||
	};
 | 
						};
 | 
				
			||||||
	ebi('selzip').onclick = function (e) {
 | 
						ebi('selzip').onclick = function (e) {
 | 
				
			||||||
		ev(e);
 | 
							ev(e);
 | 
				
			||||||
		var names = getsel();
 | 
							var names = getsel(),
 | 
				
			||||||
		var arg = ebi('selzip').getAttribute('fmt');
 | 
								arg = ebi('selzip').getAttribute('fmt'),
 | 
				
			||||||
		var txt = names.join('\n');
 | 
								txt = names.join('\n'),
 | 
				
			||||||
		var frm = document.createElement('form');
 | 
								frm = mknod('form');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		frm.setAttribute('action', '?' + arg);
 | 
							frm.setAttribute('action', '?' + arg);
 | 
				
			||||||
		frm.setAttribute('method', 'post');
 | 
							frm.setAttribute('method', 'post');
 | 
				
			||||||
		frm.setAttribute('target', '_blank');
 | 
							frm.setAttribute('target', '_blank');
 | 
				
			||||||
@@ -1728,7 +1767,7 @@ var msel = (function () {
 | 
				
			|||||||
			'<textarea name="files" id="ziptxt"></textarea>';
 | 
								'<textarea name="files" id="ziptxt"></textarea>';
 | 
				
			||||||
		frm.style.display = 'none';
 | 
							frm.style.display = 'none';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		var oldform = document.querySelector('#widgeti>form');
 | 
							var oldform = QS('#widgeti>form');
 | 
				
			||||||
		if (oldform)
 | 
							if (oldform)
 | 
				
			||||||
			oldform.parentNode.removeChild(oldform);
 | 
								oldform.parentNode.removeChild(oldform);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1739,7 +1778,7 @@ var msel = (function () {
 | 
				
			|||||||
		frm.submit();
 | 
							frm.submit();
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
	function render() {
 | 
						function render() {
 | 
				
			||||||
		var tds = document.querySelectorAll('#files tbody td+td+td');
 | 
							var tds = QSA('#files tbody td+td+td');
 | 
				
			||||||
		for (var a = 0, aa = tds.length; a < aa; a++) {
 | 
							for (var a = 0, aa = tds.length; a < aa; a++) {
 | 
				
			||||||
			tds[a].onclick = seltgl;
 | 
								tds[a].onclick = seltgl;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
@@ -1770,21 +1809,22 @@ function reload_mp() {
 | 
				
			|||||||
function reload_browser(not_mp) {
 | 
					function reload_browser(not_mp) {
 | 
				
			||||||
	filecols.set_style();
 | 
						filecols.set_style();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var parts = get_evpath().split('/');
 | 
						var parts = get_evpath().split('/'),
 | 
				
			||||||
	var rm = document.querySelectorAll('#path>a+a+a');
 | 
							rm = QSA('#path>a+a+a');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (a = rm.length - 1; a >= 0; a--)
 | 
						for (a = rm.length - 1; a >= 0; a--)
 | 
				
			||||||
		rm[a].parentNode.removeChild(rm[a]);
 | 
							rm[a].parentNode.removeChild(rm[a]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var link = '/';
 | 
						var link = '/';
 | 
				
			||||||
	for (var a = 1; a < parts.length - 1; a++) {
 | 
						for (var a = 1; a < parts.length - 1; a++) {
 | 
				
			||||||
		link += parts[a] + '/';
 | 
							link += parts[a] + '/';
 | 
				
			||||||
		var o = document.createElement('a');
 | 
							var o = mknod('a');
 | 
				
			||||||
		o.setAttribute('href', link);
 | 
							o.setAttribute('href', link);
 | 
				
			||||||
		o.textContent = uricom_dec(parts[a])[0];
 | 
							o.textContent = uricom_dec(parts[a])[0];
 | 
				
			||||||
		ebi('path').appendChild(o);
 | 
							ebi('path').appendChild(o);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var oo = document.querySelectorAll('#files>tbody>tr>td:nth-child(3)');
 | 
						var oo = QSA('#files>tbody>tr>td:nth-child(3)');
 | 
				
			||||||
	for (var a = 0, aa = oo.length; a < aa; a++) {
 | 
						for (var a = 0, aa = oo.length; a < aa; a++) {
 | 
				
			||||||
		var sz = oo[a].textContent.replace(/ /g, ""),
 | 
							var sz = oo[a].textContent.replace(/ /g, ""),
 | 
				
			||||||
			hsz = sz.replace(/\B(?=(\d{3})+(?!\d))/g, " ");
 | 
								hsz = sz.replace(/\B(?=(\d{3})+(?!\d))/g, " ");
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -50,6 +50,9 @@ pre code:last-child {
 | 
				
			|||||||
pre code::before {
 | 
					pre code::before {
 | 
				
			||||||
	content: counter(precode);
 | 
						content: counter(precode);
 | 
				
			||||||
	-webkit-user-select: none;
 | 
						-webkit-user-select: none;
 | 
				
			||||||
 | 
						-moz-user-select: none;
 | 
				
			||||||
 | 
						-ms-user-select: none;
 | 
				
			||||||
 | 
						user-select: none;
 | 
				
			||||||
	display: inline-block;
 | 
						display: inline-block;
 | 
				
			||||||
	text-align: right;
 | 
						text-align: right;
 | 
				
			||||||
	font-size: .75em;
 | 
						font-size: .75em;
 | 
				
			||||||
@@ -591,12 +594,3 @@ blink {
 | 
				
			|||||||
		color: #940;
 | 
							color: #940;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					 | 
				
			||||||
/*
 | 
					 | 
				
			||||||
*[data-ln]:before {
 | 
					 | 
				
			||||||
	content: attr(data-ln);
 | 
					 | 
				
			||||||
	font-size: .8em;
 | 
					 | 
				
			||||||
	margin: 0 .4em;
 | 
					 | 
				
			||||||
	color: #f0c;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
*/
 | 
					 | 
				
			||||||
@@ -46,7 +46,7 @@ function statify(obj) {
 | 
				
			|||||||
    var ua = navigator.userAgent;
 | 
					    var ua = navigator.userAgent;
 | 
				
			||||||
    if (ua.indexOf(') Gecko/') !== -1 && /Linux| Mac /.exec(ua)) {
 | 
					    if (ua.indexOf(') Gecko/') !== -1 && /Linux| Mac /.exec(ua)) {
 | 
				
			||||||
        // necessary on ff-68.7 at least
 | 
					        // necessary on ff-68.7 at least
 | 
				
			||||||
        var s = document.createElement('style');
 | 
					        var s = mknod('style');
 | 
				
			||||||
        s.innerHTML = '@page { margin: .5in .6in .8in .6in; }';
 | 
					        s.innerHTML = '@page { margin: .5in .6in .8in .6in; }';
 | 
				
			||||||
        console.log(s.innerHTML);
 | 
					        console.log(s.innerHTML);
 | 
				
			||||||
        document.head.appendChild(s);
 | 
					        document.head.appendChild(s);
 | 
				
			||||||
@@ -175,12 +175,12 @@ function md_plug_err(ex, js) {
 | 
				
			|||||||
        msg = "Line " + ln + ", " + msg;
 | 
					        msg = "Line " + ln + ", " + msg;
 | 
				
			||||||
        var lns = js.split('\n');
 | 
					        var lns = js.split('\n');
 | 
				
			||||||
        if (ln < lns.length) {
 | 
					        if (ln < lns.length) {
 | 
				
			||||||
            o = document.createElement('span');
 | 
					            o = mknod('span');
 | 
				
			||||||
            o.style.cssText = 'color:#ac2;font-size:.9em;font-family:scp;display:block';
 | 
					            o.style.cssText = 'color:#ac2;font-size:.9em;font-family:scp;display:block';
 | 
				
			||||||
            o.textContent = lns[ln - 1];
 | 
					            o.textContent = lns[ln - 1];
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    errbox = document.createElement('div');
 | 
					    errbox = mknod('div');
 | 
				
			||||||
    errbox.setAttribute('id', 'md_errbox');
 | 
					    errbox.setAttribute('id', 'md_errbox');
 | 
				
			||||||
    errbox.style.cssText = 'position:absolute;top:0;left:0;padding:1em .5em;background:#2b2b2b;color:#fc5'
 | 
					    errbox.style.cssText = 'position:absolute;top:0;left:0;padding:1em .5em;background:#2b2b2b;color:#fc5'
 | 
				
			||||||
    errbox.textContent = msg;
 | 
					    errbox.textContent = msg;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,126 +1,125 @@
 | 
				
			|||||||
#toc {
 | 
					#toc {
 | 
				
			||||||
    display: none;
 | 
						display: none;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mtw {
 | 
					#mtw {
 | 
				
			||||||
    display: block;
 | 
						display: block;
 | 
				
			||||||
    position: fixed;
 | 
						position: fixed;
 | 
				
			||||||
    left: .5em;
 | 
						left: .5em;
 | 
				
			||||||
    bottom: 0;
 | 
						bottom: 0;
 | 
				
			||||||
    width: calc(100% - 56em);
 | 
						width: calc(100% - 56em);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mw {
 | 
					#mw {
 | 
				
			||||||
    left: calc(100% - 55em);
 | 
						left: calc(100% - 55em);
 | 
				
			||||||
    overflow-y: auto;
 | 
						overflow-y: auto;
 | 
				
			||||||
    position: fixed;
 | 
						position: fixed;
 | 
				
			||||||
    bottom: 0;
 | 
						bottom: 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* single-screen */
 | 
					/* single-screen */
 | 
				
			||||||
#mtw.preview,
 | 
					#mtw.preview,
 | 
				
			||||||
#mw.editor {
 | 
					#mw.editor {
 | 
				
			||||||
    opacity: 0;
 | 
						opacity: 0;
 | 
				
			||||||
    z-index: 1;
 | 
						z-index: 1;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mw.preview,
 | 
					#mw.preview,
 | 
				
			||||||
#mtw.editor {
 | 
					#mtw.editor {
 | 
				
			||||||
    z-index: 5;
 | 
						z-index: 5;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mtw.single,
 | 
					#mtw.single,
 | 
				
			||||||
#mw.single {
 | 
					#mw.single {
 | 
				
			||||||
    margin: 0;
 | 
						margin: 0;
 | 
				
			||||||
    left: 1em;
 | 
						left: 1em;
 | 
				
			||||||
    left: max(1em, calc((100% - 56em) / 2));
 | 
						left: max(1em, calc((100% - 56em) / 2));
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mtw.single {
 | 
					#mtw.single {
 | 
				
			||||||
    width: 55em;
 | 
						width: 55em;
 | 
				
			||||||
    width: min(55em, calc(100% - 2em));
 | 
						width: min(55em, calc(100% - 2em));
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#mp {
 | 
					#mp {
 | 
				
			||||||
    position: relative;
 | 
						position: relative;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mt, #mtr {
 | 
					#mt, #mtr {
 | 
				
			||||||
    width: 100%;
 | 
						width: 100%;
 | 
				
			||||||
    height: calc(100% - 1px);
 | 
						height: calc(100% - 1px);
 | 
				
			||||||
    color: #444;
 | 
						color: #444;
 | 
				
			||||||
    background: #f7f7f7;
 | 
						background: #f7f7f7;
 | 
				
			||||||
    border: 1px solid #999;
 | 
						border: 1px solid #999;
 | 
				
			||||||
    outline: none;
 | 
						outline: none;
 | 
				
			||||||
    padding: 0;
 | 
						padding: 0;
 | 
				
			||||||
    margin: 0;
 | 
						margin: 0;
 | 
				
			||||||
    font-family: 'consolas', monospace, monospace;
 | 
						font-family: 'consolas', monospace, monospace;
 | 
				
			||||||
    white-space: pre-wrap;
 | 
						white-space: pre-wrap;
 | 
				
			||||||
    word-break: break-word;
 | 
						word-break: break-word;
 | 
				
			||||||
    overflow-wrap: break-word;
 | 
						overflow-wrap: break-word;
 | 
				
			||||||
    word-wrap: break-word; /*ie*/
 | 
						word-wrap: break-word; /*ie*/
 | 
				
			||||||
    overflow-y: scroll;
 | 
						overflow-y: scroll;
 | 
				
			||||||
    line-height: 1.3em;
 | 
						line-height: 1.3em;
 | 
				
			||||||
    font-size: .9em;
 | 
						font-size: .9em;
 | 
				
			||||||
    position: relative;
 | 
						position: relative;
 | 
				
			||||||
    scrollbar-color: #eb0 #f7f7f7;
 | 
						scrollbar-color: #eb0 #f7f7f7;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark #mt {
 | 
					html.dark #mt {
 | 
				
			||||||
    color: #eee;
 | 
						color: #eee;
 | 
				
			||||||
    background: #222;
 | 
						background: #222;
 | 
				
			||||||
    border: 1px solid #777;
 | 
						border: 1px solid #777;
 | 
				
			||||||
    scrollbar-color: #b80 #282828;
 | 
						scrollbar-color: #b80 #282828;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mtr {
 | 
					#mtr {
 | 
				
			||||||
    position: absolute;
 | 
						position: absolute;
 | 
				
			||||||
    top: 0;
 | 
						top: 0;
 | 
				
			||||||
    left: 0;
 | 
						left: 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#save.force-save {
 | 
					#save.force-save {
 | 
				
			||||||
    color: #400;
 | 
						color: #400;
 | 
				
			||||||
    background: #f97;
 | 
						background: #f97;
 | 
				
			||||||
    border-radius: .15em;
 | 
						border-radius: .15em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark #save.force-save {
 | 
					html.dark #save.force-save {
 | 
				
			||||||
    color: #fca;
 | 
						color: #fca;
 | 
				
			||||||
    background: #720;
 | 
						background: #720;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#save.disabled {
 | 
					#save.disabled {
 | 
				
			||||||
    opacity: .4;
 | 
						opacity: .4;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#helpbox,
 | 
					#helpbox,
 | 
				
			||||||
#toast {
 | 
					#toast {
 | 
				
			||||||
    background: #f7f7f7;
 | 
						background: #f7f7f7;
 | 
				
			||||||
    border-radius: .4em;
 | 
						border-radius: .4em;
 | 
				
			||||||
    z-index: 9001;
 | 
						z-index: 9001;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#helpbox {
 | 
					#helpbox {
 | 
				
			||||||
    display: none;
 | 
						display: none;
 | 
				
			||||||
    position: fixed;
 | 
						position: fixed;
 | 
				
			||||||
    padding: 2em;
 | 
						padding: 2em;
 | 
				
			||||||
    top: 4em;
 | 
						top: 4em;
 | 
				
			||||||
    overflow-y: auto;
 | 
						overflow-y: auto;
 | 
				
			||||||
    box-shadow: 0 .5em 2em #777;
 | 
						box-shadow: 0 .5em 2em #777;
 | 
				
			||||||
    height: calc(100% - 12em);
 | 
						height: calc(100% - 12em);
 | 
				
			||||||
    left: calc(50% - 15em);
 | 
						left: calc(50% - 15em);
 | 
				
			||||||
    right: 0;
 | 
						right: 0;
 | 
				
			||||||
    width: 30em;
 | 
						width: 30em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#helpclose {
 | 
					#helpclose {
 | 
				
			||||||
    display: block;
 | 
						display: block;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark #helpbox {
 | 
					html.dark #helpbox {
 | 
				
			||||||
    box-shadow: 0 .5em 2em #444;
 | 
						box-shadow: 0 .5em 2em #444;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark #helpbox,
 | 
					html.dark #helpbox,
 | 
				
			||||||
html.dark #toast {
 | 
					html.dark #toast {
 | 
				
			||||||
    background: #222;
 | 
						background: #222;
 | 
				
			||||||
    border: 1px solid #079;
 | 
						border: 1px solid #079;
 | 
				
			||||||
    border-width: 1px 0;
 | 
						border-width: 1px 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#toast {
 | 
					#toast {
 | 
				
			||||||
    font-weight: bold;
 | 
						font-weight: bold;
 | 
				
			||||||
    text-align: center;
 | 
						text-align: center;
 | 
				
			||||||
    padding: .6em 0;
 | 
						padding: .6em 0;
 | 
				
			||||||
    position: fixed;
 | 
						position: fixed;
 | 
				
			||||||
    z-index: 9001;
 | 
						top: 30%;
 | 
				
			||||||
    top: 30%;
 | 
						transition: opacity 0.2s ease-in-out;
 | 
				
			||||||
    transition: opacity 0.2s ease-in-out;
 | 
						opacity: 1;
 | 
				
			||||||
    opacity: 1;
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -16,7 +16,7 @@ var dom_sbs = ebi('sbs');
 | 
				
			|||||||
var dom_nsbs = ebi('nsbs');
 | 
					var dom_nsbs = ebi('nsbs');
 | 
				
			||||||
var dom_tbox = ebi('toolsbox');
 | 
					var dom_tbox = ebi('toolsbox');
 | 
				
			||||||
var dom_ref = (function () {
 | 
					var dom_ref = (function () {
 | 
				
			||||||
    var d = document.createElement('div');
 | 
					    var d = mknod('div');
 | 
				
			||||||
    d.setAttribute('id', 'mtr');
 | 
					    d.setAttribute('id', 'mtr');
 | 
				
			||||||
    dom_swrap.appendChild(d);
 | 
					    dom_swrap.appendChild(d);
 | 
				
			||||||
    d = ebi('mtr');
 | 
					    d = ebi('mtr');
 | 
				
			||||||
@@ -71,7 +71,7 @@ var map_src = [];
 | 
				
			|||||||
var map_pre = [];
 | 
					var map_pre = [];
 | 
				
			||||||
function genmap(dom, oldmap) {
 | 
					function genmap(dom, oldmap) {
 | 
				
			||||||
    var find = nlines;
 | 
					    var find = nlines;
 | 
				
			||||||
    while (oldmap && find --> 0) {
 | 
					    while (oldmap && find-- > 0) {
 | 
				
			||||||
        var tmap = genmapq(dom, '*[data-ln="' + find + '"]');
 | 
					        var tmap = genmapq(dom, '*[data-ln="' + find + '"]');
 | 
				
			||||||
        if (!tmap || !tmap.length)
 | 
					        if (!tmap || !tmap.length)
 | 
				
			||||||
            continue;
 | 
					            continue;
 | 
				
			||||||
@@ -94,7 +94,7 @@ var nlines = 0;
 | 
				
			|||||||
var draw_md = (function () {
 | 
					var draw_md = (function () {
 | 
				
			||||||
    var delay = 1;
 | 
					    var delay = 1;
 | 
				
			||||||
    function draw_md() {
 | 
					    function draw_md() {
 | 
				
			||||||
        var t0 = new Date().getTime();
 | 
					        var t0 = Date.now();
 | 
				
			||||||
        var src = dom_src.value;
 | 
					        var src = dom_src.value;
 | 
				
			||||||
        convert_markdown(src, dom_pre);
 | 
					        convert_markdown(src, dom_pre);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -110,7 +110,7 @@ var draw_md = (function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        cls(ebi('save'), 'disabled', src == server_md);
 | 
					        cls(ebi('save'), 'disabled', src == server_md);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var t1 = new Date().getTime();
 | 
					        var t1 = Date.now();
 | 
				
			||||||
        delay = t1 - t0 > 100 ? 25 : 1;
 | 
					        delay = t1 - t0 > 100 ? 25 : 1;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -252,7 +252,7 @@ function Modpoll() {
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        console.log('modpoll...');
 | 
					        console.log('modpoll...');
 | 
				
			||||||
        var url = (document.location + '').split('?')[0] + '?raw&_=' + new Date().getTime();
 | 
					        var url = (document.location + '').split('?')[0] + '?raw&_=' + Date.now();
 | 
				
			||||||
        var xhr = new XMLHttpRequest();
 | 
					        var xhr = new XMLHttpRequest();
 | 
				
			||||||
        xhr.modpoll = this;
 | 
					        xhr.modpoll = this;
 | 
				
			||||||
        xhr.open('GET', url, true);
 | 
					        xhr.open('GET', url, true);
 | 
				
			||||||
@@ -399,7 +399,7 @@ function save_cb() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
function run_savechk(lastmod, txt, btn, ntry) {
 | 
					function run_savechk(lastmod, txt, btn, ntry) {
 | 
				
			||||||
    // download the saved doc from the server and compare
 | 
					    // download the saved doc from the server and compare
 | 
				
			||||||
    var url = (document.location + '').split('?')[0] + '?raw&_=' + new Date().getTime();
 | 
					    var url = (document.location + '').split('?')[0] + '?raw&_=' + Date.now();
 | 
				
			||||||
    var xhr = new XMLHttpRequest();
 | 
					    var xhr = new XMLHttpRequest();
 | 
				
			||||||
    xhr.open('GET', url, true);
 | 
					    xhr.open('GET', url, true);
 | 
				
			||||||
    xhr.responseType = 'text';
 | 
					    xhr.responseType = 'text';
 | 
				
			||||||
@@ -455,7 +455,7 @@ function toast(autoclose, style, width, msg) {
 | 
				
			|||||||
        ok.parentNode.removeChild(ok);
 | 
					        ok.parentNode.removeChild(ok);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    style = "width:" + width + "em;left:calc(50% - " + (width / 2) + "em);" + style;
 | 
					    style = "width:" + width + "em;left:calc(50% - " + (width / 2) + "em);" + style;
 | 
				
			||||||
    ok = document.createElement('div');
 | 
					    ok = mknod('div');
 | 
				
			||||||
    ok.setAttribute('id', 'toast');
 | 
					    ok.setAttribute('id', 'toast');
 | 
				
			||||||
    ok.setAttribute('style', style);
 | 
					    ok.setAttribute('style', style);
 | 
				
			||||||
    ok.innerHTML = msg;
 | 
					    ok.innerHTML = msg;
 | 
				
			||||||
@@ -1049,7 +1049,7 @@ action_stack = (function () {
 | 
				
			|||||||
        var p1 = from.length,
 | 
					        var p1 = from.length,
 | 
				
			||||||
            p2 = to.length;
 | 
					            p2 = to.length;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        while (p1 --> 0 && p2 --> 0)
 | 
					        while (p1-- > 0 && p2-- > 0)
 | 
				
			||||||
            if (from[p1] != to[p2])
 | 
					            if (from[p1] != to[p2])
 | 
				
			||||||
                break;
 | 
					                break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1142,14 +1142,3 @@ action_stack = (function () {
 | 
				
			|||||||
        _ref: ref
 | 
					        _ref: ref
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
})();
 | 
					})();
 | 
				
			||||||
 | 
					 | 
				
			||||||
/*
 | 
					 | 
				
			||||||
ebi('help').onclick = function () {
 | 
					 | 
				
			||||||
    var c1 = getComputedStyle(dom_src).cssText.split(';');
 | 
					 | 
				
			||||||
    var c2 = getComputedStyle(dom_ref).cssText.split(';');
 | 
					 | 
				
			||||||
    var max = Math.min(c1.length, c2.length);
 | 
					 | 
				
			||||||
    for (var a = 0; a < max; a++)
 | 
					 | 
				
			||||||
        if (c1[a] !== c2[a])
 | 
					 | 
				
			||||||
            console.log(c1[a] + '\n' + c2[a]);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
*/
 | 
					 | 
				
			||||||
 
 | 
				
			|||||||
@@ -8,68 +8,58 @@ html .editor-toolbar>i.separator { border-left: 1px solid #ccc; }
 | 
				
			|||||||
html .editor-toolbar.disabled-for-preview>button:not(.no-disable) { opacity: .35 }
 | 
					html .editor-toolbar.disabled-for-preview>button:not(.no-disable) { opacity: .35 }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
html {
 | 
					html {
 | 
				
			||||||
    line-height: 1.5em;
 | 
						line-height: 1.5em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html, body {
 | 
					html, body {
 | 
				
			||||||
    margin: 0;
 | 
						margin: 0;
 | 
				
			||||||
    padding: 0;
 | 
						padding: 0;
 | 
				
			||||||
    min-height: 100%;
 | 
						min-height: 100%;
 | 
				
			||||||
    font-family: sans-serif;
 | 
						font-family: sans-serif;
 | 
				
			||||||
    background: #f7f7f7;
 | 
						background: #f7f7f7;
 | 
				
			||||||
    color: #333;
 | 
						color: #333;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mn {
 | 
					#mn {
 | 
				
			||||||
    font-weight: normal;
 | 
						font-weight: normal;
 | 
				
			||||||
    margin: 1.3em 0 .7em 1em;
 | 
						margin: 1.3em 0 .7em 1em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mn a {
 | 
					#mn a {
 | 
				
			||||||
    color: #444;
 | 
						color: #444;
 | 
				
			||||||
    margin: 0 0 0 -.2em;
 | 
						margin: 0 0 0 -.2em;
 | 
				
			||||||
    padding: 0 0 0 .4em;
 | 
						padding: 0 0 0 .4em;
 | 
				
			||||||
    text-decoration: none;
 | 
						text-decoration: none;
 | 
				
			||||||
    /* ie: */
 | 
						/* ie: */
 | 
				
			||||||
    border-bottom: .1em solid #777\9;
 | 
						border-bottom: .1em solid #777\9;
 | 
				
			||||||
    margin-right: 1em\9;
 | 
						margin-right: 1em\9;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mn a:first-child {
 | 
					#mn a:first-child {
 | 
				
			||||||
    padding-left: .5em;
 | 
						padding-left: .5em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mn a:last-child {
 | 
					#mn a:last-child {
 | 
				
			||||||
    padding-right: .5em;
 | 
						padding-right: .5em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mn a:not(:last-child):after {
 | 
					#mn a:not(:last-child):after {
 | 
				
			||||||
    content: '';
 | 
						content: '';
 | 
				
			||||||
    width: 1.05em;
 | 
						width: 1.05em;
 | 
				
			||||||
    height: 1.05em;
 | 
						height: 1.05em;
 | 
				
			||||||
    margin: -.2em .3em -.2em -.4em;
 | 
						margin: -.2em .3em -.2em -.4em;
 | 
				
			||||||
    display: inline-block;
 | 
						display: inline-block;
 | 
				
			||||||
    border: 1px solid rgba(0,0,0,0.2);
 | 
						border: 1px solid rgba(0,0,0,0.2);
 | 
				
			||||||
    border-width: .2em .2em 0 0;
 | 
						border-width: .2em .2em 0 0;
 | 
				
			||||||
    transform: rotate(45deg);
 | 
						transform: rotate(45deg);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#mn a:hover {
 | 
					#mn a:hover {
 | 
				
			||||||
    color: #000;
 | 
						color: #000;
 | 
				
			||||||
    text-decoration: underline;
 | 
						text-decoration: underline;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
html .editor-toolbar>button.disabled {
 | 
					html .editor-toolbar>button.disabled {
 | 
				
			||||||
    opacity: .35;
 | 
						opacity: .35;
 | 
				
			||||||
    pointer-events: none;
 | 
						pointer-events: none;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html .editor-toolbar>button.save.force-save {
 | 
					html .editor-toolbar>button.save.force-save {
 | 
				
			||||||
    background: #f97;
 | 
						background: #f97;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/*
 | 
					 | 
				
			||||||
*[data-ln]:before {
 | 
					 | 
				
			||||||
	content: attr(data-ln);
 | 
					 | 
				
			||||||
	font-size: .8em;
 | 
					 | 
				
			||||||
	margin: 0 .4em;
 | 
					 | 
				
			||||||
	color: #f0c;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
.cm-header { font-size: .4em !important }
 | 
					 | 
				
			||||||
*/
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -101,29 +91,29 @@ html .editor-toolbar>button.save.force-save {
 | 
				
			|||||||
	line-height: 1.1em;
 | 
						line-height: 1.1em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
.mdo a {
 | 
					.mdo a {
 | 
				
			||||||
    color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
    background: #39b;
 | 
						background: #39b;
 | 
				
			||||||
    text-decoration: none;
 | 
						text-decoration: none;
 | 
				
			||||||
    padding: 0 .3em;
 | 
						padding: 0 .3em;
 | 
				
			||||||
    border: none;
 | 
						border: none;
 | 
				
			||||||
    border-bottom: .07em solid #079;
 | 
						border-bottom: .07em solid #079;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
.mdo h2 {
 | 
					.mdo h2 {
 | 
				
			||||||
    color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
    background: #555;
 | 
						background: #555;
 | 
				
			||||||
    margin-top: 2em;
 | 
						margin-top: 2em;
 | 
				
			||||||
    border-bottom: .22em solid #999;
 | 
						border-bottom: .22em solid #999;
 | 
				
			||||||
    border-top: none;
 | 
						border-top: none;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
.mdo h1 {
 | 
					.mdo h1 {
 | 
				
			||||||
    color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
    background: #444;
 | 
						background: #444;
 | 
				
			||||||
    font-weight: normal;
 | 
						font-weight: normal;
 | 
				
			||||||
    border-top: .4em solid #fb0;
 | 
						border-top: .4em solid #fb0;
 | 
				
			||||||
    border-bottom: .4em solid #777;
 | 
						border-bottom: .4em solid #777;
 | 
				
			||||||
    border-radius: 0 1em 0 1em;
 | 
						border-radius: 0 1em 0 1em;
 | 
				
			||||||
    margin: 3em 0 1em 0;
 | 
						margin: 3em 0 1em 0;
 | 
				
			||||||
    padding: .5em 0;
 | 
						padding: .5em 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
h1, h2 {
 | 
					h1, h2 {
 | 
				
			||||||
	line-height: 1.5em;
 | 
						line-height: 1.5em;
 | 
				
			||||||
@@ -197,14 +187,14 @@ th {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
/* mde support */
 | 
					/* mde support */
 | 
				
			||||||
.mdo {
 | 
					.mdo {
 | 
				
			||||||
    padding: 1em;
 | 
						padding: 1em;
 | 
				
			||||||
    background: #f7f7f7;
 | 
						background: #f7f7f7;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .mdo {
 | 
					html.dark .mdo {
 | 
				
			||||||
    background: #1c1c1c;
 | 
						background: #1c1c1c;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
.CodeMirror {
 | 
					.CodeMirror {
 | 
				
			||||||
    background: #f7f7f7;
 | 
						background: #f7f7f7;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -214,108 +204,108 @@ html.dark .mdo {
 | 
				
			|||||||
/* darkmode */
 | 
					/* darkmode */
 | 
				
			||||||
html.dark .mdo,
 | 
					html.dark .mdo,
 | 
				
			||||||
html.dark .CodeMirror {
 | 
					html.dark .CodeMirror {
 | 
				
			||||||
    border-color: #222;
 | 
						border-color: #222;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark,
 | 
					html.dark,
 | 
				
			||||||
html.dark body,
 | 
					html.dark body,
 | 
				
			||||||
html.dark .CodeMirror {
 | 
					html.dark .CodeMirror {
 | 
				
			||||||
    background: #222;
 | 
						background: #222;
 | 
				
			||||||
    color: #ccc;
 | 
						color: #ccc;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .CodeMirror-cursor {
 | 
					html.dark .CodeMirror-cursor {
 | 
				
			||||||
    border-color: #fff;
 | 
						border-color: #fff;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .CodeMirror-selected {
 | 
					html.dark .CodeMirror-selected {
 | 
				
			||||||
    box-shadow: 0 0 1px #0cf inset;
 | 
						box-shadow: 0 0 1px #0cf inset;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .CodeMirror-selected,
 | 
					html.dark .CodeMirror-selected,
 | 
				
			||||||
html.dark .CodeMirror-selectedtext {
 | 
					html.dark .CodeMirror-selectedtext {
 | 
				
			||||||
    border-radius: .1em;
 | 
						border-radius: .1em;
 | 
				
			||||||
    background: #246;
 | 
						background: #246;
 | 
				
			||||||
    color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .mdo a {
 | 
					html.dark .mdo a {
 | 
				
			||||||
    background: #057;
 | 
						background: #057;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .mdo h1 a, html.dark .mdo h4 a,
 | 
					html.dark .mdo h1 a, html.dark .mdo h4 a,
 | 
				
			||||||
html.dark .mdo h2 a, html.dark .mdo h5 a,
 | 
					html.dark .mdo h2 a, html.dark .mdo h5 a,
 | 
				
			||||||
html.dark .mdo h3 a, html.dark .mdo h6 a {
 | 
					html.dark .mdo h3 a, html.dark .mdo h6 a {
 | 
				
			||||||
    color: inherit;
 | 
						color: inherit;
 | 
				
			||||||
    background: none;
 | 
						background: none;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark pre,
 | 
					html.dark pre,
 | 
				
			||||||
html.dark code {
 | 
					html.dark code {
 | 
				
			||||||
    color: #8c0;
 | 
						color: #8c0;
 | 
				
			||||||
    background: #1a1a1a;
 | 
						background: #1a1a1a;
 | 
				
			||||||
    border: .07em solid #333;
 | 
						border: .07em solid #333;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .mdo ul,
 | 
					html.dark .mdo ul,
 | 
				
			||||||
html.dark .mdo ol {
 | 
					html.dark .mdo ol {
 | 
				
			||||||
    border-color: #444;
 | 
						border-color: #444;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .mdo>ul,
 | 
					html.dark .mdo>ul,
 | 
				
			||||||
html.dark .mdo>ol {
 | 
					html.dark .mdo>ol {
 | 
				
			||||||
    border-color: #555;
 | 
						border-color: #555;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark strong {
 | 
					html.dark strong {
 | 
				
			||||||
    color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark p>em,
 | 
					html.dark p>em,
 | 
				
			||||||
html.dark li>em,
 | 
					html.dark li>em,
 | 
				
			||||||
html.dark td>em {
 | 
					html.dark td>em {
 | 
				
			||||||
    color: #f94;
 | 
						color: #f94;
 | 
				
			||||||
    border-color: #666;
 | 
						border-color: #666;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark h1 {
 | 
					html.dark h1 {
 | 
				
			||||||
    background: #383838;
 | 
						background: #383838;
 | 
				
			||||||
    border-top: .4em solid #b80;
 | 
						border-top: .4em solid #b80;
 | 
				
			||||||
    border-bottom: .4em solid #4c4c4c;
 | 
						border-bottom: .4em solid #4c4c4c;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark h2 {
 | 
					html.dark h2 {
 | 
				
			||||||
    background: #444;
 | 
						background: #444;
 | 
				
			||||||
    border-bottom: .22em solid #555;
 | 
						border-bottom: .22em solid #555;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark td,
 | 
					html.dark td,
 | 
				
			||||||
html.dark th {
 | 
					html.dark th {
 | 
				
			||||||
    border-color: #444;
 | 
						border-color: #444;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark blockquote {
 | 
					html.dark blockquote {
 | 
				
			||||||
    background: #282828;
 | 
						background: #282828;
 | 
				
			||||||
    border: .07em dashed #444;
 | 
						border: .07em dashed #444;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
html.dark #mn a {
 | 
					html.dark #mn a {
 | 
				
			||||||
    color: #ccc;
 | 
						color: #ccc;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark #mn a:not(:last-child):after {
 | 
					html.dark #mn a:not(:last-child):after {
 | 
				
			||||||
    border-color: rgba(255,255,255,0.3);
 | 
						border-color: rgba(255,255,255,0.3);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .editor-toolbar {
 | 
					html.dark .editor-toolbar {
 | 
				
			||||||
    border-color: #2c2c2c;
 | 
						border-color: #2c2c2c;
 | 
				
			||||||
    background: #1c1c1c;
 | 
						background: #1c1c1c;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .editor-toolbar>i.separator {
 | 
					html.dark .editor-toolbar>i.separator {
 | 
				
			||||||
    border-left: 1px solid #444;
 | 
						border-left: 1px solid #444;
 | 
				
			||||||
    border-right: 1px solid #111;
 | 
						border-right: 1px solid #111;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .editor-toolbar>button {
 | 
					html.dark .editor-toolbar>button {
 | 
				
			||||||
    margin-left: -1px; border: 1px solid rgba(255,255,255,0.1);
 | 
						margin-left: -1px; border: 1px solid rgba(255,255,255,0.1);
 | 
				
			||||||
    color: #aaa;
 | 
						color: #aaa;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
html.dark .editor-toolbar>button:hover {
 | 
					html.dark .editor-toolbar>button:hover {
 | 
				
			||||||
    color: #333;
 | 
						color: #333;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .editor-toolbar>button.active {
 | 
					html.dark .editor-toolbar>button.active {
 | 
				
			||||||
    color: #333;
 | 
						color: #333;
 | 
				
			||||||
    border-color: #ec1;
 | 
						border-color: #ec1;
 | 
				
			||||||
    background: #c90;
 | 
						background: #c90;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.dark .editor-toolbar::after,
 | 
					html.dark .editor-toolbar::after,
 | 
				
			||||||
html.dark .editor-toolbar::before {
 | 
					html.dark .editor-toolbar::before {
 | 
				
			||||||
    background: none;
 | 
						background: none;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -71,7 +71,7 @@ var mde = (function () {
 | 
				
			|||||||
})();
 | 
					})();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function set_jumpto() {
 | 
					function set_jumpto() {
 | 
				
			||||||
    document.querySelector('.editor-preview-side').onclick = jumpto;
 | 
					    QS('.editor-preview-side').onclick = jumpto;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function jumpto(ev) {
 | 
					function jumpto(ev) {
 | 
				
			||||||
@@ -94,7 +94,7 @@ function md_changed(mde, on_srv) {
 | 
				
			|||||||
        window.md_saved = mde.value();
 | 
					        window.md_saved = mde.value();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    var md_now = mde.value();
 | 
					    var md_now = mde.value();
 | 
				
			||||||
    var save_btn = document.querySelector('.editor-toolbar button.save');
 | 
					    var save_btn = QS('.editor-toolbar button.save');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (md_now == window.md_saved)
 | 
					    if (md_now == window.md_saved)
 | 
				
			||||||
        save_btn.classList.add('disabled');
 | 
					        save_btn.classList.add('disabled');
 | 
				
			||||||
@@ -105,7 +105,7 @@ function md_changed(mde, on_srv) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function save(mde) {
 | 
					function save(mde) {
 | 
				
			||||||
    var save_btn = document.querySelector('.editor-toolbar button.save');
 | 
					    var save_btn = QS('.editor-toolbar button.save');
 | 
				
			||||||
    if (save_btn.classList.contains('disabled')) {
 | 
					    if (save_btn.classList.contains('disabled')) {
 | 
				
			||||||
        alert('there is nothing to save');
 | 
					        alert('there is nothing to save');
 | 
				
			||||||
        return;
 | 
					        return;
 | 
				
			||||||
@@ -212,7 +212,7 @@ function save_chk() {
 | 
				
			|||||||
    last_modified = this.lastmod;
 | 
					    last_modified = this.lastmod;
 | 
				
			||||||
    md_changed(this.mde, true);
 | 
					    md_changed(this.mde, true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    var ok = document.createElement('div');
 | 
					    var ok = mknod('div');
 | 
				
			||||||
    ok.setAttribute('style', 'font-size:6em;font-family:serif;font-weight:bold;color:#cf6;background:#444;border-radius:.3em;padding:.6em 0;position:fixed;top:30%;left:calc(50% - 2em);width:4em;text-align:center;z-index:9001;transition:opacity 0.2s ease-in-out;opacity:1');
 | 
					    ok.setAttribute('style', 'font-size:6em;font-family:serif;font-weight:bold;color:#cf6;background:#444;border-radius:.3em;padding:.6em 0;position:fixed;top:30%;left:calc(50% - 2em);width:4em;text-align:center;z-index:9001;transition:opacity 0.2s ease-in-out;opacity:1');
 | 
				
			||||||
    ok.innerHTML = 'OK✔️';
 | 
					    ok.innerHTML = 'OK✔️';
 | 
				
			||||||
    var parent = ebi('m');
 | 
					    var parent = ebi('m');
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -3,7 +3,7 @@ html,body,tr,th,td,#files,a {
 | 
				
			|||||||
	background: none;
 | 
						background: none;
 | 
				
			||||||
	font-weight: inherit;
 | 
						font-weight: inherit;
 | 
				
			||||||
	font-size: inherit;
 | 
						font-size: inherit;
 | 
				
			||||||
	padding: none;
 | 
						padding: 0;
 | 
				
			||||||
	border: none;
 | 
						border: none;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html {
 | 
					html {
 | 
				
			||||||
@@ -20,8 +20,8 @@ body {
 | 
				
			|||||||
	padding-bottom: 5em;
 | 
						padding-bottom: 5em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#box {
 | 
					#box {
 | 
				
			||||||
    padding: .5em 1em;
 | 
						padding: .5em 1em;
 | 
				
			||||||
    background: #2c2c2c;
 | 
						background: #2c2c2c;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
pre {
 | 
					pre {
 | 
				
			||||||
	font-family: monospace, monospace;
 | 
						font-family: monospace, monospace;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -55,7 +55,7 @@ function up2k_flagbus() {
 | 
				
			|||||||
            dbg(who, 'hi me (??)');
 | 
					            dbg(who, 'hi me (??)');
 | 
				
			||||||
            return;
 | 
					            return;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        flag.act = new Date().getTime();
 | 
					        flag.act = Date.now();
 | 
				
			||||||
        if (what == "want") {
 | 
					        if (what == "want") {
 | 
				
			||||||
            // lowest id wins, don't care if that's us
 | 
					            // lowest id wins, don't care if that's us
 | 
				
			||||||
            if (who < flag.id) {
 | 
					            if (who < flag.id) {
 | 
				
			||||||
@@ -209,7 +209,7 @@ function U2pvis(act, btns) {
 | 
				
			|||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    this.perc = function (bd, bd0, sz, t0) {
 | 
					    this.perc = function (bd, bd0, sz, t0) {
 | 
				
			||||||
        var td = new Date().getTime() - t0,
 | 
					        var td = Date.now() - t0,
 | 
				
			||||||
            p = bd * 100.0 / sz,
 | 
					            p = bd * 100.0 / sz,
 | 
				
			||||||
            nb = bd - bd0,
 | 
					            nb = bd - bd0,
 | 
				
			||||||
            spd = nb / (td / 1000),
 | 
					            spd = nb / (td / 1000),
 | 
				
			||||||
@@ -219,25 +219,28 @@ function U2pvis(act, btns) {
 | 
				
			|||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    this.hashed = function (fobj) {
 | 
					    this.hashed = function (fobj) {
 | 
				
			||||||
        var fo = this.tab[fobj.n];
 | 
					        var fo = this.tab[fobj.n],
 | 
				
			||||||
        var nb = fo.bt * (++fo.nh / fo.cb.length);
 | 
					            nb = fo.bt * (++fo.nh / fo.cb.length),
 | 
				
			||||||
        var p = this.perc(nb, 0, fobj.size, fobj.t1);
 | 
					            p = this.perc(nb, 0, fobj.size, fobj.t1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        fo.hp = '{0}%, {1}, {2} MB/s'.format(
 | 
					        fo.hp = '{0}%, {1}, {2} MB/s'.format(
 | 
				
			||||||
            p[0].toFixed(2), p[1], p[2].toFixed(2)
 | 
					            p[0].toFixed(2), p[1], p[2].toFixed(2)
 | 
				
			||||||
        );
 | 
					        );
 | 
				
			||||||
        if (!this.is_act(fo.in))
 | 
					        if (!this.is_act(fo.in))
 | 
				
			||||||
            return;
 | 
					            return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var obj = ebi('f{0}p'.format(fobj.n));
 | 
					        var obj = ebi('f{0}p'.format(fobj.n)),
 | 
				
			||||||
 | 
					            o1 = p[0] - 2, o2 = p[0] - 0.1, o3 = p[0];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        obj.innerHTML = fo.hp;
 | 
					        obj.innerHTML = fo.hp;
 | 
				
			||||||
        obj.style.color = '#fff';
 | 
					        obj.style.color = '#fff';
 | 
				
			||||||
        var o1 = p[0] - 2, o2 = p[0] - 0.1, o3 = p[0];
 | 
					 | 
				
			||||||
        obj.style.background = 'linear-gradient(90deg, #025, #06a ' + o1 + '%, #09d ' + o2 + '%, #333 ' + o3 + '%, #333 99%, #777)';
 | 
					        obj.style.background = 'linear-gradient(90deg, #025, #06a ' + o1 + '%, #09d ' + o2 + '%, #333 ' + o3 + '%, #333 99%, #777)';
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    this.prog = function (fobj, nchunk, cbd) {
 | 
					    this.prog = function (fobj, nchunk, cbd) {
 | 
				
			||||||
        var fo = this.tab[fobj.n];
 | 
					        var fo = this.tab[fobj.n],
 | 
				
			||||||
        var delta = cbd - fo.cb[nchunk];
 | 
					            delta = cbd - fo.cb[nchunk];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        fo.cb[nchunk] = cbd;
 | 
					        fo.cb[nchunk] = cbd;
 | 
				
			||||||
        fo.bd += delta;
 | 
					        fo.bd += delta;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -249,10 +252,11 @@ function U2pvis(act, btns) {
 | 
				
			|||||||
        if (!this.is_act(fo.in))
 | 
					        if (!this.is_act(fo.in))
 | 
				
			||||||
            return;
 | 
					            return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var obj = ebi('f{0}p'.format(fobj.n));
 | 
					        var obj = ebi('f{0}p'.format(fobj.n)),
 | 
				
			||||||
 | 
					            o1 = p[0] - 2, o2 = p[0] - 0.1, o3 = p[0];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        obj.innerHTML = fo.hp;
 | 
					        obj.innerHTML = fo.hp;
 | 
				
			||||||
        obj.style.color = '#fff';
 | 
					        obj.style.color = '#fff';
 | 
				
			||||||
        var o1 = p[0] - 2, o2 = p[0] - 0.1, o3 = p[0];
 | 
					 | 
				
			||||||
        obj.style.background = 'linear-gradient(90deg, #050, #270 ' + o1 + '%, #4b0 ' + o2 + '%, #333 ' + o3 + '%, #333 99%, #777)';
 | 
					        obj.style.background = 'linear-gradient(90deg, #050, #270 ' + o1 + '%, #4b0 ' + o2 + '%, #333 ' + o3 + '%, #333 99%, #777)';
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -287,24 +291,14 @@ function U2pvis(act, btns) {
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    this.bzw_log = function (first, last) {
 | 
					 | 
				
			||||||
        console.log("first %d   head %d   tail %d   last %d", first, this.head, this.tail, last);
 | 
					 | 
				
			||||||
        var trs = document.querySelectorAll('#u2tab>tbody>tr'), msg = [];
 | 
					 | 
				
			||||||
        for (var a = 0; a < trs.length; a++)
 | 
					 | 
				
			||||||
            msg.push(trs[a].getAttribute('id'));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        console.log(msg.join(' '));
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    this.bzw = function () {
 | 
					    this.bzw = function () {
 | 
				
			||||||
        var first = document.querySelector('#u2tab>tbody>tr:first-child');
 | 
					        var first = QS('#u2tab>tbody>tr:first-child');
 | 
				
			||||||
        if (!first)
 | 
					        if (!first)
 | 
				
			||||||
            return;
 | 
					            return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var last = document.querySelector('#u2tab>tbody>tr:last-child');
 | 
					        var last = QS('#u2tab>tbody>tr:last-child');
 | 
				
			||||||
        first = parseInt(first.getAttribute('id').slice(1));
 | 
					        first = parseInt(first.getAttribute('id').slice(1));
 | 
				
			||||||
        last = parseInt(last.getAttribute('id').slice(1));
 | 
					        last = parseInt(last.getAttribute('id').slice(1));
 | 
				
			||||||
        //this.bzw_log(first, last);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        while (this.head - first > this.wsz) {
 | 
					        while (this.head - first > this.wsz) {
 | 
				
			||||||
            var obj = ebi('f' + (first++));
 | 
					            var obj = ebi('f' + (first++));
 | 
				
			||||||
@@ -315,12 +309,10 @@ function U2pvis(act, btns) {
 | 
				
			|||||||
            if (!obj)
 | 
					            if (!obj)
 | 
				
			||||||
                this.addrow(last);
 | 
					                this.addrow(last);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        //this.bzw_log(first, last);
 | 
					 | 
				
			||||||
        //console.log('--');
 | 
					 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    this.drawcard = function (cat) {
 | 
					    this.drawcard = function (cat) {
 | 
				
			||||||
        var cards = document.querySelectorAll('#u2cards>a>span');
 | 
					        var cards = QSA('#u2cards>a>span');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (cat == "q") {
 | 
					        if (cat == "q") {
 | 
				
			||||||
            cards[4].innerHTML = this.ctr[cat];
 | 
					            cards[4].innerHTML = this.ctr[cat];
 | 
				
			||||||
@@ -343,9 +335,9 @@ function U2pvis(act, btns) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    this.changecard = function (card) {
 | 
					    this.changecard = function (card) {
 | 
				
			||||||
        this.act = card;
 | 
					        this.act = card;
 | 
				
			||||||
        var html = [];
 | 
					 | 
				
			||||||
        this.head = -1;
 | 
					        this.head = -1;
 | 
				
			||||||
        this.tail = -1;
 | 
					        this.tail = -1;
 | 
				
			||||||
 | 
					        var html = [];
 | 
				
			||||||
        for (var a = 0; a < this.tab.length; a++) {
 | 
					        for (var a = 0; a < this.tab.length; a++) {
 | 
				
			||||||
            var rt = this.tab[a].in;
 | 
					            var rt = this.tab[a].in;
 | 
				
			||||||
            if (this.is_act(rt)) {
 | 
					            if (this.is_act(rt)) {
 | 
				
			||||||
@@ -382,7 +374,7 @@ function U2pvis(act, btns) {
 | 
				
			|||||||
        if (as_html)
 | 
					        if (as_html)
 | 
				
			||||||
            return '<tr id="f' + nfile + '">' + ret + '</tr>';
 | 
					            return '<tr id="f' + nfile + '">' + ret + '</tr>';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var obj = document.createElement('tr');
 | 
					        var obj = mknod('tr');
 | 
				
			||||||
        obj.setAttribute('id', 'f' + nfile);
 | 
					        obj.setAttribute('id', 'f' + nfile);
 | 
				
			||||||
        obj.innerHTML = ret;
 | 
					        obj.innerHTML = ret;
 | 
				
			||||||
        return obj;
 | 
					        return obj;
 | 
				
			||||||
@@ -394,7 +386,7 @@ function U2pvis(act, btns) {
 | 
				
			|||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    var that = this;
 | 
					    var that = this;
 | 
				
			||||||
    btns = document.querySelectorAll(btns + '>a[act]');
 | 
					    btns = QSA(btns + '>a[act]');
 | 
				
			||||||
    for (var a = 0; a < btns.length; a++) {
 | 
					    for (var a = 0; a < btns.length; a++) {
 | 
				
			||||||
        btns[a].onclick = function (e) {
 | 
					        btns[a].onclick = function (e) {
 | 
				
			||||||
            ev(e);
 | 
					            ev(e);
 | 
				
			||||||
@@ -428,8 +420,11 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        ebi('u2notbtn').innerHTML = '';
 | 
					        ebi('u2notbtn').innerHTML = '';
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    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 suggest_up2k = 'this is the basic uploader; <a href="#" id="u2yea">up2k</a> is better';
 | 
				
			||||||
    var is_https = (window.location + '').indexOf('https:') === 0;
 | 
					
 | 
				
			||||||
 | 
					    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>',
 | 
				
			||||||
 | 
					        is_https = (window.location + '').indexOf('https:') === 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (is_https)
 | 
					    if (is_https)
 | 
				
			||||||
        // chrome<37 firefox<34 edge<12 ie<11 opera<24 safari<10.1
 | 
					        // chrome<37 firefox<34 edge<12 ie<11 opera<24 safari<10.1
 | 
				
			||||||
        shame = 'your browser is impressively ancient';
 | 
					        shame = 'your browser is impressively ancient';
 | 
				
			||||||
@@ -448,34 +443,43 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // show uploader if the user only has write-access
 | 
					    // show uploader if the user only has write-access
 | 
				
			||||||
    if (!ebi('files'))
 | 
					    var perms = (document.body.getAttribute('perms') + '').split(' ');
 | 
				
			||||||
 | 
					    if (!has(perms, 'read'))
 | 
				
			||||||
        goto('up2k');
 | 
					        goto('up2k');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // shows or clears an error message in the basic uploader ui
 | 
					    // shows or clears a message in the basic uploader ui
 | 
				
			||||||
    function setmsg(msg) {
 | 
					    function setmsg(msg, type) {
 | 
				
			||||||
        if (msg !== undefined) {
 | 
					        if (msg !== undefined) {
 | 
				
			||||||
            ebi('u2err').setAttribute('class', 'err');
 | 
					            ebi('u2err').setAttribute('class', type);
 | 
				
			||||||
            ebi('u2err').innerHTML = msg;
 | 
					            ebi('u2err').innerHTML = msg;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        else {
 | 
					        else {
 | 
				
			||||||
            ebi('u2err').setAttribute('class', '');
 | 
					            ebi('u2err').setAttribute('class', '');
 | 
				
			||||||
            ebi('u2err').innerHTML = '';
 | 
					            ebi('u2err').innerHTML = '';
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        if (msg == suggest_up2k) {
 | 
				
			||||||
 | 
					            ebi('u2yea').onclick = function (e) {
 | 
				
			||||||
 | 
					                ev(e);
 | 
				
			||||||
 | 
					                goto('up2k');
 | 
				
			||||||
 | 
					            };
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // switches to the basic uploader with msg as error message
 | 
					    // switches to the basic uploader with msg as error message
 | 
				
			||||||
    function un2k(msg) {
 | 
					    function un2k(msg) {
 | 
				
			||||||
        setmsg(msg);
 | 
					        setmsg(msg, 'err');
 | 
				
			||||||
        return false;
 | 
					        return false;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // 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) {
 | 
				
			||||||
        ev(e);
 | 
					        ev(e);
 | 
				
			||||||
        setmsg();
 | 
					        setmsg(suggest_up2k, 'msg');
 | 
				
			||||||
        goto('bup');
 | 
					        goto('bup');
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    setmsg(suggest_up2k, 'msg');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!String.prototype.format) {
 | 
					    if (!String.prototype.format) {
 | 
				
			||||||
        String.prototype.format = function () {
 | 
					        String.prototype.format = function () {
 | 
				
			||||||
            var args = arguments;
 | 
					            var args = arguments;
 | 
				
			||||||
@@ -486,13 +490,14 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        };
 | 
					        };
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    var parallel_uploads = icfg_get('nthread');
 | 
					    var parallel_uploads = icfg_get('nthread'),
 | 
				
			||||||
    var multitask = bcfg_get('multitask', true);
 | 
					        multitask = bcfg_get('multitask', true),
 | 
				
			||||||
    var ask_up = bcfg_get('ask_up', true);
 | 
					        ask_up = bcfg_get('ask_up', true),
 | 
				
			||||||
    var flag_en = bcfg_get('flag_en', false);
 | 
					        flag_en = bcfg_get('flag_en', false),
 | 
				
			||||||
    var fsearch = bcfg_get('fsearch', false);
 | 
					        fsearch = bcfg_get('fsearch', false),
 | 
				
			||||||
 | 
					        fdom_ctr = 0,
 | 
				
			||||||
 | 
					        min_filebuf = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    var fdom_ctr = 0;
 | 
					 | 
				
			||||||
    var st = {
 | 
					    var st = {
 | 
				
			||||||
        "files": [],
 | 
					        "files": [],
 | 
				
			||||||
        "todo": {
 | 
					        "todo": {
 | 
				
			||||||
@@ -542,8 +547,9 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        e.stopPropagation();
 | 
					        e.stopPropagation();
 | 
				
			||||||
        e.preventDefault();
 | 
					        e.preventDefault();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var files;
 | 
					        var files,
 | 
				
			||||||
        var is_itemlist = false;
 | 
					            is_itemlist = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (e.dataTransfer) {
 | 
					        if (e.dataTransfer) {
 | 
				
			||||||
            if (e.dataTransfer.items) {
 | 
					            if (e.dataTransfer.items) {
 | 
				
			||||||
                files = e.dataTransfer.items; // DataTransferItemList
 | 
					                files = e.dataTransfer.items; // DataTransferItemList
 | 
				
			||||||
@@ -557,9 +563,10 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
            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 = [];
 | 
					            good_files = [],
 | 
				
			||||||
        var dirs = [];
 | 
					            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) {
 | 
				
			||||||
@@ -644,12 +651,13 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    function gotallfiles(good_files, bad_files) {
 | 
					    function gotallfiles(good_files, bad_files) {
 | 
				
			||||||
        if (bad_files.length > 0) {
 | 
					        if (bad_files.length > 0) {
 | 
				
			||||||
            var ntot = bad_files.length + good_files.length;
 | 
					            var ntot = bad_files.length + good_files.length,
 | 
				
			||||||
            var msg = 'These {0} files (of {1} total) were skipped because they are empty:\n'.format(bad_files.length, ntot);
 | 
					                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++)
 | 
					            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 (good_files.length - bad_files.length <= 1 && /(android)/i.test(navigator.userAgent))
 | 
					            if (good_files.length - bad_files.length <= 1 && ANDROID)
 | 
				
			||||||
                msg += '\nFirefox-Android has a bug which prevents selecting multiple files. Try selecting one file at a time. For more info, see firefox bug 1456557';
 | 
					                msg += '\nFirefox-Android has a bug which prevents selecting multiple files. Try selecting one file at a time. For more info, see firefox bug 1456557';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            alert(msg);
 | 
					            alert(msg);
 | 
				
			||||||
@@ -663,9 +671,10 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
            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][0];
 | 
					            var fobj = good_files[a][0],
 | 
				
			||||||
            var now = new Date().getTime();
 | 
					                now = Date.now(),
 | 
				
			||||||
            var lmod = fobj.lastModified || now;
 | 
					                lmod = fobj.lastModified || now;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            var entry = {
 | 
					            var entry = {
 | 
				
			||||||
                "n": parseInt(st.files.length.toString()),
 | 
					                "n": parseInt(st.files.length.toString()),
 | 
				
			||||||
                "t0": now,
 | 
					                "t0": now,
 | 
				
			||||||
@@ -701,7 +710,7 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    function more_one_file() {
 | 
					    function more_one_file() {
 | 
				
			||||||
        fdom_ctr++;
 | 
					        fdom_ctr++;
 | 
				
			||||||
        var elm = document.createElement('div')
 | 
					        var elm = mknod('div');
 | 
				
			||||||
        elm.innerHTML = '<input id="file{0}" type="file" name="file{0}[]" multiple="multiple" />'.format(fdom_ctr);
 | 
					        elm.innerHTML = '<input id="file{0}" type="file" name="file{0}[]" multiple="multiple" />'.format(fdom_ctr);
 | 
				
			||||||
        ebi('u2form').appendChild(elm);
 | 
					        ebi('u2form').appendChild(elm);
 | 
				
			||||||
        ebi('file' + fdom_ctr).addEventListener('change', gotfile, false);
 | 
					        ebi('file' + fdom_ctr).addEventListener('change', gotfile, false);
 | 
				
			||||||
@@ -748,26 +757,23 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    var tasker = (function () {
 | 
					    var tasker = (function () {
 | 
				
			||||||
        var mutex = false;
 | 
					        var tto = null,
 | 
				
			||||||
        var was_busy = false;
 | 
					            running = false,
 | 
				
			||||||
 | 
					            was_busy = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        function defer() {
 | 
				
			||||||
 | 
					            running = false;
 | 
				
			||||||
 | 
					            clearTimeout(tto);
 | 
				
			||||||
 | 
					            tto = setTimeout(taskerd, 100);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        function taskerd() {
 | 
					        function taskerd() {
 | 
				
			||||||
            if (mutex)
 | 
					            if (running)
 | 
				
			||||||
                return;
 | 
					                return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            mutex = true;
 | 
					            clearTimeout(tto);
 | 
				
			||||||
 | 
					            running = 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 !=
 | 
					                var is_busy = 0 !=
 | 
				
			||||||
                    st.todo.hash.length +
 | 
					                    st.todo.hash.length +
 | 
				
			||||||
                    st.todo.handshake.length +
 | 
					                    st.todo.handshake.length +
 | 
				
			||||||
@@ -779,21 +785,16 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                if (was_busy != is_busy) {
 | 
					                if (was_busy != is_busy) {
 | 
				
			||||||
                    was_busy = is_busy;
 | 
					                    was_busy = is_busy;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                    if (is_busy)
 | 
					                    window[(is_busy ? "add" : "remove") +
 | 
				
			||||||
                        window.addEventListener("beforeunload", warn_uploader_busy);
 | 
					                        "EventListener"]("beforeunload", warn_uploader_busy);
 | 
				
			||||||
                    else
 | 
					 | 
				
			||||||
                        window.removeEventListener("beforeunload", warn_uploader_busy);
 | 
					 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                if (flag) {
 | 
					                if (flag) {
 | 
				
			||||||
                    if (is_busy) {
 | 
					                    if (is_busy) {
 | 
				
			||||||
                        var now = new Date().getTime();
 | 
					                        var now = Date.now();
 | 
				
			||||||
                        flag.take(now);
 | 
					                        flag.take(now);
 | 
				
			||||||
                        if (!flag.ours) {
 | 
					                        if (!flag.ours)
 | 
				
			||||||
                            setTimeout(taskerd, 100);
 | 
					                            return defer();
 | 
				
			||||||
                            mutex = false;
 | 
					 | 
				
			||||||
                            return;
 | 
					 | 
				
			||||||
                        }
 | 
					 | 
				
			||||||
                    }
 | 
					                    }
 | 
				
			||||||
                    else if (flag.ours) {
 | 
					                    else if (flag.ours) {
 | 
				
			||||||
                        flag.give();
 | 
					                        flag.give();
 | 
				
			||||||
@@ -835,11 +836,8 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                    mou_ikkai = true;
 | 
					                    mou_ikkai = true;
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                if (!mou_ikkai) {
 | 
					                if (!mou_ikkai)
 | 
				
			||||||
                    setTimeout(taskerd, 100);
 | 
					                    return defer();
 | 
				
			||||||
                    mutex = false;
 | 
					 | 
				
			||||||
                    return;
 | 
					 | 
				
			||||||
                }
 | 
					 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        taskerd();
 | 
					        taskerd();
 | 
				
			||||||
@@ -853,47 +851,47 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    // https://gist.github.com/jonleighton/958841
 | 
					    // https://gist.github.com/jonleighton/958841
 | 
				
			||||||
    function buf2b64(arrayBuffer) {
 | 
					    function buf2b64(arrayBuffer) {
 | 
				
			||||||
        var base64 = '';
 | 
					        var base64 = '',
 | 
				
			||||||
        var encodings = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_';
 | 
					            cset = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_',
 | 
				
			||||||
        var bytes = new Uint8Array(arrayBuffer);
 | 
					            src = new Uint8Array(arrayBuffer),
 | 
				
			||||||
        var byteLength = bytes.byteLength;
 | 
					            nbytes = src.byteLength,
 | 
				
			||||||
        var byteRemainder = byteLength % 3;
 | 
					            byteRem = nbytes % 3,
 | 
				
			||||||
        var mainLength = byteLength - byteRemainder;
 | 
					            mainLen = nbytes - byteRem,
 | 
				
			||||||
        var a, b, c, d;
 | 
					            a, b, c, d, chunk;
 | 
				
			||||||
        var chunk;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        for (var i = 0; i < mainLength; i = i + 3) {
 | 
					        for (var i = 0; i < mainLen; i = i + 3) {
 | 
				
			||||||
            chunk = (bytes[i] << 16) | (bytes[i + 1] << 8) | bytes[i + 2];
 | 
					            chunk = (src[i] << 16) | (src[i + 1] << 8) | src[i + 2];
 | 
				
			||||||
            // create 8*3=24bit segment then split into 6bit segments
 | 
					            // create 8*3=24bit segment then split into 6bit segments
 | 
				
			||||||
            a = (chunk & 16515072) >> 18; // 16515072 = (2^6 - 1) << 18
 | 
					            a = (chunk & 16515072) >> 18; // (2^6 - 1) << 18
 | 
				
			||||||
            b = (chunk & 258048) >> 12; // 258048   = (2^6 - 1) << 12
 | 
					            b = (chunk & 258048) >> 12; // (2^6 - 1) << 12
 | 
				
			||||||
            c = (chunk & 4032) >> 6; // 4032     = (2^6 - 1) << 6
 | 
					            c = (chunk & 4032) >> 6; // (2^6 - 1) << 6
 | 
				
			||||||
            d = chunk & 63;               // 63       = 2^6 - 1
 | 
					            d = chunk & 63; // 2^6 - 1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            // Convert the raw binary segments to the appropriate ASCII encoding
 | 
					            // Convert the raw binary segments to the appropriate ASCII encoding
 | 
				
			||||||
            base64 += encodings[a] + encodings[b] + encodings[c] + encodings[d];
 | 
					            base64 += cset[a] + cset[b] + cset[c] + cset[d];
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (byteRemainder == 1) {
 | 
					        if (byteRem == 1) {
 | 
				
			||||||
            chunk = bytes[mainLength];
 | 
					            chunk = src[mainLen];
 | 
				
			||||||
            a = (chunk & 252) >> 2; // 252 = (2^6 - 1) << 2
 | 
					            a = (chunk & 252) >> 2; // (2^6 - 1) << 2
 | 
				
			||||||
            b = (chunk & 3) << 4; // 3   = 2^2 - 1  (zero 4 LSB)
 | 
					            b = (chunk & 3) << 4; // 2^2 - 1  (zero 4 LSB)
 | 
				
			||||||
            base64 += encodings[a] + encodings[b];//+ '==';
 | 
					            base64 += cset[a] + cset[b];//+ '==';
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        else if (byteRemainder == 2) {
 | 
					        else if (byteRem == 2) {
 | 
				
			||||||
            chunk = (bytes[mainLength] << 8) | bytes[mainLength + 1];
 | 
					            chunk = (src[mainLen] << 8) | src[mainLen + 1];
 | 
				
			||||||
            a = (chunk & 64512) >> 10; // 64512 = (2^6 - 1) << 10
 | 
					            a = (chunk & 64512) >> 10; // (2^6 - 1) << 10
 | 
				
			||||||
            b = (chunk & 1008) >> 4; // 1008  = (2^6 - 1) << 4
 | 
					            b = (chunk & 1008) >> 4; // (2^6 - 1) << 4
 | 
				
			||||||
            c = (chunk & 15) << 2; // 15    = 2^4 - 1  (zero 2 LSB)
 | 
					            c = (chunk & 15) << 2; // 2^4 - 1  (zero 2 LSB)
 | 
				
			||||||
            base64 += encodings[a] + encodings[b] + encodings[c];//+ '=';
 | 
					            base64 += cset[a] + cset[b] + cset[c];//+ '=';
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return base64;
 | 
					        return base64;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function get_chunksize(filesize) {
 | 
					    function get_chunksize(filesize) {
 | 
				
			||||||
        var chunksize = 1024 * 1024;
 | 
					        var chunksize = 1024 * 1024,
 | 
				
			||||||
        var stepsize = 512 * 1024;
 | 
					            stepsize = 512 * 1024;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        while (true) {
 | 
					        while (true) {
 | 
				
			||||||
            for (var mul = 1; mul <= 2; mul++) {
 | 
					            for (var mul = 1; mul <= 2; mul++) {
 | 
				
			||||||
                var nchunks = Math.ceil(filesize / chunksize);
 | 
					                var nchunks = Math.ceil(filesize / chunksize);
 | 
				
			||||||
@@ -906,25 +904,11 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function ensure_rendered(func) {
 | 
					 | 
				
			||||||
        var hidden = false;
 | 
					 | 
				
			||||||
        var keys = ['hidden', 'msHidden', 'webkitHidden'];
 | 
					 | 
				
			||||||
        for (var a = 0; a < keys.length; a++)
 | 
					 | 
				
			||||||
            if (typeof document[keys[a]] !== "undefined")
 | 
					 | 
				
			||||||
                hidden = document[keys[a]];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (hidden)
 | 
					 | 
				
			||||||
            return func();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        window.requestAnimationFrame(func);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    function exec_hash() {
 | 
					    function exec_hash() {
 | 
				
			||||||
        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;
 | 
					        st.bytes.hashed += t.size;
 | 
				
			||||||
        t.bytes_uploaded = 0;
 | 
					        t.bytes_uploaded = 0;
 | 
				
			||||||
        t.t1 = new Date().getTime();
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var bpend = 0,
 | 
					        var bpend = 0,
 | 
				
			||||||
            nchunk = 0,
 | 
					            nchunk = 0,
 | 
				
			||||||
@@ -936,13 +920,14 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        pvis.move(t.n, 'bz');
 | 
					        pvis.move(t.n, 'bz');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var segm_next = function () {
 | 
					        var segm_next = function () {
 | 
				
			||||||
            if (nchunk >= nchunks || (bpend > chunksize && bpend >= 32 * 1024 * 1024))
 | 
					            if (nchunk >= nchunks || (bpend > chunksize && bpend >= min_filebuf))
 | 
				
			||||||
                return false;
 | 
					                return false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            var reader = new FileReader(),
 | 
					            var reader = new FileReader(),
 | 
				
			||||||
                nch = nchunk++,
 | 
					                nch = nchunk++,
 | 
				
			||||||
                car = nch * chunksize,
 | 
					                car = nch * chunksize,
 | 
				
			||||||
                cdr = car + chunksize;
 | 
					                cdr = car + chunksize,
 | 
				
			||||||
 | 
					                t0 = Date.now();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            if (cdr >= t.size)
 | 
					            if (cdr >= t.size)
 | 
				
			||||||
                cdr = t.size;
 | 
					                cdr = t.size;
 | 
				
			||||||
@@ -950,9 +935,19 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
            bpend += cdr - car;
 | 
					            bpend += cdr - car;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            reader.onload = function (e) {
 | 
					            reader.onload = function (e) {
 | 
				
			||||||
 | 
					                if (!min_filebuf && nch == 1) {
 | 
				
			||||||
 | 
					                    min_filebuf = 1;
 | 
				
			||||||
 | 
					                    var td = Date.now() - t0;
 | 
				
			||||||
 | 
					                    if (td > 50) {
 | 
				
			||||||
 | 
					                        ebi('u2foot').innerHTML += "<p>excessive filereader latency (" + td + " ms), increasing readahead</p>";
 | 
				
			||||||
 | 
					                        min_filebuf = 32 * 1024 * 1024;
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
                hash_calc(nch, e.target.result);
 | 
					                hash_calc(nch, e.target.result);
 | 
				
			||||||
            };
 | 
					            };
 | 
				
			||||||
            reader.onerror = segm_err;
 | 
					            reader.onerror = function () {
 | 
				
			||||||
 | 
					                alert('y o u   b r o k e    i t\nerror: ' + reader.error);
 | 
				
			||||||
 | 
					            };
 | 
				
			||||||
            reader.readAsArrayBuffer(
 | 
					            reader.readAsArrayBuffer(
 | 
				
			||||||
                bobslice.call(t.fobj, car, cdr));
 | 
					                bobslice.call(t.fobj, car, cdr));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -963,8 +958,9 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
            while (segm_next());
 | 
					            while (segm_next());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            var hash_done = function (hashbuf) {
 | 
					            var hash_done = function (hashbuf) {
 | 
				
			||||||
                var hslice = new Uint8Array(hashbuf).subarray(0, 32);
 | 
					                var hslice = new Uint8Array(hashbuf).subarray(0, 32),
 | 
				
			||||||
                var b64str = buf2b64(hslice).replace(/=$/, '');
 | 
					                    b64str = buf2b64(hslice).replace(/=$/, '');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                hashtab[nch] = b64str;
 | 
					                hashtab[nch] = b64str;
 | 
				
			||||||
                t.hash.push(nch);
 | 
					                t.hash.push(nch);
 | 
				
			||||||
                pvis.hashed(t);
 | 
					                pvis.hashed(t);
 | 
				
			||||||
@@ -978,7 +974,7 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                    t.hash.push(hashtab[a]);
 | 
					                    t.hash.push(hashtab[a]);
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                t.t2 = new Date().getTime();
 | 
					                t.t2 = Date.now();
 | 
				
			||||||
                if (t.n == 0 && window.location.hash == '#dbg') {
 | 
					                if (t.n == 0 && window.location.hash == '#dbg') {
 | 
				
			||||||
                    var spd = (t.size / ((t.t2 - t.t1) / 1000.)) / (1024 * 1024.);
 | 
					                    var spd = (t.size / ((t.t2 - t.t1) / 1000.)) / (1024 * 1024.);
 | 
				
			||||||
                    alert('{0} ms, {1} MB/s\n'.format(t.t2 - t.t1, spd.toFixed(3)) + t.hash.join('\n'));
 | 
					                    alert('{0} ms, {1} MB/s\n'.format(t.t2 - t.t1, spd.toFixed(3)) + t.hash.join('\n'));
 | 
				
			||||||
@@ -1000,10 +996,7 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
            }
 | 
					            }
 | 
				
			||||||
        };
 | 
					        };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var segm_err = function () {
 | 
					        t.t1 = Date.now();
 | 
				
			||||||
            alert('y o u   b r o k e    i t\nerror: ' + reader.error);
 | 
					 | 
				
			||||||
        };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        segm_next();
 | 
					        segm_next();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1022,8 +1015,9 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                var response = JSON.parse(xhr.responseText);
 | 
					                var response = JSON.parse(xhr.responseText);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                if (!response.name) {
 | 
					                if (!response.name) {
 | 
				
			||||||
                    var msg = '';
 | 
					                    var msg = '',
 | 
				
			||||||
                    var smsg = '';
 | 
					                        smsg = '';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                    if (!response || !response.hits || !response.hits.length) {
 | 
					                    if (!response || !response.hits || !response.hits.length) {
 | 
				
			||||||
                        msg = 'not found on server';
 | 
					                        msg = 'not found on server';
 | 
				
			||||||
                        smsg = '404';
 | 
					                        smsg = '404';
 | 
				
			||||||
@@ -1056,10 +1050,11 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                    pvis.seth(t.n, 0, linksplit(esc(t.purl + t.name)).join(' '));
 | 
					                    pvis.seth(t.n, 0, linksplit(esc(t.purl + t.name)).join(' '));
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                var chunksize = get_chunksize(t.size);
 | 
					                var chunksize = get_chunksize(t.size),
 | 
				
			||||||
                var cdr_idx = Math.ceil(t.size / chunksize) - 1;
 | 
					                    cdr_idx = Math.ceil(t.size / chunksize) - 1,
 | 
				
			||||||
                var cdr_sz = (t.size % chunksize) || chunksize;
 | 
					                    cdr_sz = (t.size % chunksize) || chunksize,
 | 
				
			||||||
                var cbd = [];
 | 
					                    cbd = [];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                for (var a = 0; a <= cdr_idx; a++) {
 | 
					                for (var a = 0; a <= cdr_idx; a++) {
 | 
				
			||||||
                    cbd.push(a == cdr_idx ? cdr_sz : chunksize);
 | 
					                    cbd.push(a == cdr_idx ? cdr_sz : chunksize);
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
@@ -1080,8 +1075,9 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                pvis.setat(t.n, cbd);
 | 
					                pvis.setat(t.n, cbd);
 | 
				
			||||||
                pvis.prog(t, 0, cbd[0]);
 | 
					                pvis.prog(t, 0, cbd[0]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                var done = true;
 | 
					                var done = true,
 | 
				
			||||||
                var msg = '🎷🐛';
 | 
					                    msg = '🎷🐛';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                if (t.postlist.length > 0) {
 | 
					                if (t.postlist.length > 0) {
 | 
				
			||||||
                    for (var a = 0; a < t.postlist.length; a++)
 | 
					                    for (var a = 0; a < t.postlist.length; a++)
 | 
				
			||||||
                        st.todo.upload.push({
 | 
					                        st.todo.upload.push({
 | 
				
			||||||
@@ -1098,10 +1094,12 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                if (done) {
 | 
					                if (done) {
 | 
				
			||||||
                    t.done = true;
 | 
					                    t.done = true;
 | 
				
			||||||
                    st.bytes.uploaded += t.size - t.bytes_uploaded;
 | 
					                    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.t4 - t.t3) / 1000.)) / (1024 * 1024.);
 | 
					                        spd2 = (t.size / ((t.t4 - t.t3) / 1000.)) / (1024 * 1024.);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                    pvis.seth(t.n, 2, 'hash {0}, up {1} MB/s'.format(
 | 
					                    pvis.seth(t.n, 2, 'hash {0}, up {1} MB/s'.format(
 | 
				
			||||||
                        spd1.toFixed(2), spd2.toFixed(2)));
 | 
					                        spd1.toFixed(2), spd2.toFixed(2)));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                    pvis.move(t.n, 'ok');
 | 
					                    pvis.move(t.n, 'ok');
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
                else t.t4 = undefined;
 | 
					                else t.t4 = undefined;
 | 
				
			||||||
@@ -1168,16 +1166,18 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        var upt = st.todo.upload.shift();
 | 
					        var upt = st.todo.upload.shift();
 | 
				
			||||||
        st.busy.upload.push(upt);
 | 
					        st.busy.upload.push(upt);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var npart = upt.npart;
 | 
					        var npart = upt.npart,
 | 
				
			||||||
        var t = st.files[upt.nfile];
 | 
					            t = st.files[upt.nfile];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (!t.t3)
 | 
					        if (!t.t3)
 | 
				
			||||||
            t.t3 = new Date().getTime();
 | 
					            t.t3 = Date.now();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        pvis.seth(t.n, 1, "🚀 send");
 | 
					        pvis.seth(t.n, 1, "🚀 send");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var chunksize = get_chunksize(t.size);
 | 
					        var chunksize = get_chunksize(t.size),
 | 
				
			||||||
        var car = npart * chunksize;
 | 
					            car = npart * chunksize,
 | 
				
			||||||
        var cdr = car + chunksize;
 | 
					            cdr = car + chunksize;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (cdr >= t.size)
 | 
					        if (cdr >= t.size)
 | 
				
			||||||
            cdr = t.size;
 | 
					            cdr = t.size;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1193,7 +1193,7 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
                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.t4 = new Date().getTime();
 | 
					                    t.t4 = Date.now();
 | 
				
			||||||
                    pvis.seth(t.n, 1, 'verifying');
 | 
					                    pvis.seth(t.n, 1, 'verifying');
 | 
				
			||||||
                    st.todo.handshake.unshift(t);
 | 
					                    st.todo.handshake.unshift(t);
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
@@ -1242,20 +1242,20 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
    onresize();
 | 
					    onresize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function desc_show(e) {
 | 
					    function desc_show(e) {
 | 
				
			||||||
        var msg = this.getAttribute('alt');
 | 
					        var msg = this.getAttribute('alt'),
 | 
				
			||||||
        msg = msg.replace(/\$N/g, "<br />");
 | 
					            cdesc = ebi('u2cdesc');
 | 
				
			||||||
        var cdesc = ebi('u2cdesc');
 | 
					
 | 
				
			||||||
        cdesc.innerHTML = msg;
 | 
					        cdesc.innerHTML = msg.replace(/\$N/g, "<br />");
 | 
				
			||||||
        cdesc.setAttribute('class', 'show');
 | 
					        cdesc.setAttribute('class', 'show');
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    function desc_hide(e) {
 | 
					    function desc_hide(e) {
 | 
				
			||||||
        ebi('u2cdesc').setAttribute('class', '');
 | 
					        ebi('u2cdesc').setAttribute('class', '');
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    var o = document.querySelectorAll('#u2conf *[alt]');
 | 
					    var o = QSA('#u2conf *[alt]');
 | 
				
			||||||
    for (var a = o.length - 1; a >= 0; a--) {
 | 
					    for (var a = o.length - 1; a >= 0; a--) {
 | 
				
			||||||
        o[a].parentNode.getElementsByTagName('input')[0].setAttribute('alt', o[a].getAttribute('alt'));
 | 
					        o[a].parentNode.getElementsByTagName('input')[0].setAttribute('alt', o[a].getAttribute('alt'));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    var o = document.querySelectorAll('#u2conf *[alt]');
 | 
					    var o = QSA('#u2conf *[alt]');
 | 
				
			||||||
    for (var a = 0; a < o.length; a++) {
 | 
					    for (var a = 0; a < o.length; a++) {
 | 
				
			||||||
        o[a].onfocus = desc_show;
 | 
					        o[a].onfocus = desc_show;
 | 
				
			||||||
        o[a].onblur = desc_hide;
 | 
					        o[a].onblur = desc_hide;
 | 
				
			||||||
@@ -1309,8 +1309,8 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function set_fsearch(new_state) {
 | 
					    function set_fsearch(new_state) {
 | 
				
			||||||
        var perms = document.body.getAttribute('perms');
 | 
					        var perms = document.body.getAttribute('perms'),
 | 
				
			||||||
        var read_only = false;
 | 
					            read_only = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (!ebi('fsearch')) {
 | 
					        if (!ebi('fsearch')) {
 | 
				
			||||||
            new_state = false;
 | 
					            new_state = false;
 | 
				
			||||||
@@ -1326,16 +1326,16 @@ function up2k_init(have_crypto) {
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        try {
 | 
					        try {
 | 
				
			||||||
            document.querySelector('label[for="fsearch"]').style.opacity = read_only ? '0' : '1';
 | 
					            QS('label[for="fsearch"]').style.opacity = read_only ? '0' : '1';
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        catch (ex) { }
 | 
					        catch (ex) { }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        try {
 | 
					        try {
 | 
				
			||||||
            var fun = fsearch ? 'add' : 'remove';
 | 
					            var fun = fsearch ? 'add' : 'remove',
 | 
				
			||||||
            ebi('op_up2k').classList[fun]('srch');
 | 
					                ico = fsearch ? '🔎' : '🚀',
 | 
				
			||||||
 | 
					                desc = fsearch ? 'Search' : 'Upload';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            var ico = fsearch ? '🔎' : '🚀';
 | 
					            ebi('op_up2k').classList[fun]('srch');
 | 
				
			||||||
            var desc = fsearch ? 'Search' : 'Upload';
 | 
					 | 
				
			||||||
            ebi('u2bm').innerHTML = ico + ' <sup>' + desc + '</sup>';
 | 
					            ebi('u2bm').innerHTML = ico + ' <sup>' + desc + '</sup>';
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        catch (ex) { }
 | 
					        catch (ex) { }
 | 
				
			||||||
@@ -1402,5 +1402,5 @@ function warn_uploader_busy(e) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if (document.querySelector('#op_up2k.act'))
 | 
					if (QS('#op_up2k.act'))
 | 
				
			||||||
    goto_up2k();
 | 
					    goto_up2k();
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -19,6 +19,10 @@
 | 
				
			|||||||
	color: #f87;
 | 
						color: #f87;
 | 
				
			||||||
	padding: .5em;
 | 
						padding: .5em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#u2err.msg {
 | 
				
			||||||
 | 
						color: #999;
 | 
				
			||||||
 | 
						padding: .5em;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
#u2btn {
 | 
					#u2btn {
 | 
				
			||||||
	color: #eee;
 | 
						color: #eee;
 | 
				
			||||||
	background: #555;
 | 
						background: #555;
 | 
				
			||||||
@@ -86,12 +90,14 @@
 | 
				
			|||||||
	font-family: sans-serif;
 | 
						font-family: sans-serif;
 | 
				
			||||||
	width: auto;
 | 
						width: auto;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2tab tr+tr:hover td {
 | 
					#u2tab tbody tr:hover td {
 | 
				
			||||||
	background: #222;
 | 
						background: #222;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2cards {
 | 
					#u2cards {
 | 
				
			||||||
	margin: 2.5em auto -2.5em auto;
 | 
						padding: 1em 0 .3em 1em;
 | 
				
			||||||
 | 
						margin: 1.5em auto -2.5em auto;
 | 
				
			||||||
	text-align: center;
 | 
						text-align: center;
 | 
				
			||||||
 | 
						overflow: hidden;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2cards.w {
 | 
					#u2cards.w {
 | 
				
			||||||
	width: 45em;
 | 
						width: 45em;
 | 
				
			||||||
@@ -110,10 +116,15 @@
 | 
				
			|||||||
	border-radius: 0 .4em 0 0;
 | 
						border-radius: 0 .4em 0 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2cards a.act {
 | 
					#u2cards a.act {
 | 
				
			||||||
	border-width: 1px 1px 0 1px;
 | 
						padding-bottom: .5em;
 | 
				
			||||||
 | 
						border-width: 1px 1px .1em 1px;
 | 
				
			||||||
	border-radius: .3em .3em 0 0;
 | 
						border-radius: .3em .3em 0 0;
 | 
				
			||||||
	margin-left: -1px;
 | 
						margin-left: -1px;
 | 
				
			||||||
	background: transparent;
 | 
						background: linear-gradient(to bottom, #464, #333 80%);
 | 
				
			||||||
 | 
						box-shadow: 0 -.17em .67em #280;
 | 
				
			||||||
 | 
						border-color: #7c5 #583 #333 #583;
 | 
				
			||||||
 | 
						position: relative;
 | 
				
			||||||
 | 
						color: #fd7;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2cards span {
 | 
					#u2cards span {
 | 
				
			||||||
	color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
@@ -134,12 +145,13 @@
 | 
				
			|||||||
	outline: none;
 | 
						outline: none;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2conf .txtbox {
 | 
					#u2conf .txtbox {
 | 
				
			||||||
	width: 4em;
 | 
						width: 3em;
 | 
				
			||||||
	color: #fff;
 | 
						color: #fff;
 | 
				
			||||||
	background: #444;
 | 
						background: #444;
 | 
				
			||||||
	border: 1px solid #777;
 | 
						border: 1px solid #777;
 | 
				
			||||||
	font-size: 1.2em;
 | 
						font-size: 1.2em;
 | 
				
			||||||
	padding: .15em 0;
 | 
						padding: .15em 0;
 | 
				
			||||||
 | 
						height: 1.05em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2conf .txtbox.err {
 | 
					#u2conf .txtbox.err {
 | 
				
			||||||
	background: #922;
 | 
						background: #922;
 | 
				
			||||||
@@ -151,13 +163,12 @@
 | 
				
			|||||||
	border-radius: .1em;
 | 
						border-radius: .1em;
 | 
				
			||||||
	font-size: 1.5em;
 | 
						font-size: 1.5em;
 | 
				
			||||||
	padding: .1em 0;
 | 
						padding: .1em 0;
 | 
				
			||||||
	margin: 0 -.25em;
 | 
						margin: 0 -1px;
 | 
				
			||||||
	width: 1.5em;
 | 
						width: 1.5em;
 | 
				
			||||||
	height: 1em;
 | 
						height: 1em;
 | 
				
			||||||
	display: inline-block;
 | 
						display: inline-block;
 | 
				
			||||||
	position: relative;
 | 
						position: relative;
 | 
				
			||||||
	line-height: 1em;
 | 
						bottom: -0.08em;
 | 
				
			||||||
	bottom: -.08em;
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2conf input+a {
 | 
					#u2conf input+a {
 | 
				
			||||||
	background: #d80;
 | 
						background: #d80;
 | 
				
			||||||
@@ -168,7 +179,6 @@
 | 
				
			|||||||
	height: 1em;
 | 
						height: 1em;
 | 
				
			||||||
	padding: .4em 0;
 | 
						padding: .4em 0;
 | 
				
			||||||
	display: block;
 | 
						display: block;
 | 
				
			||||||
	user-select: none;
 | 
					 | 
				
			||||||
	border-radius: .25em;
 | 
						border-radius: .25em;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2conf input[type="checkbox"] {
 | 
					#u2conf input[type="checkbox"] {
 | 
				
			||||||
@@ -208,12 +218,13 @@
 | 
				
			|||||||
	text-align: center;
 | 
						text-align: center;
 | 
				
			||||||
	overflow: hidden;
 | 
						overflow: hidden;
 | 
				
			||||||
	margin: 0 -2em;
 | 
						margin: 0 -2em;
 | 
				
			||||||
	height: 0;
 | 
					 | 
				
			||||||
	padding: 0 1em;
 | 
						padding: 0 1em;
 | 
				
			||||||
 | 
						height: 0;
 | 
				
			||||||
	opacity: .1;
 | 
						opacity: .1;
 | 
				
			||||||
    transition: all 0.14s ease-in-out;
 | 
						transition: all 0.14s ease-in-out;
 | 
				
			||||||
	border-radius: .4em;
 | 
					 | 
				
			||||||
	box-shadow: 0 .2em .5em #222;
 | 
						box-shadow: 0 .2em .5em #222;
 | 
				
			||||||
 | 
						border-radius: .4em;
 | 
				
			||||||
 | 
						z-index: 1;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#u2cdesc.show {
 | 
					#u2cdesc.show {
 | 
				
			||||||
	padding: 1em;
 | 
						padding: 1em;
 | 
				
			||||||
@@ -256,7 +267,10 @@ html.light #u2cards a {
 | 
				
			|||||||
	background: linear-gradient(to bottom, #eee, #fff);
 | 
						background: linear-gradient(to bottom, #eee, #fff);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.light #u2cards a.act {
 | 
					html.light #u2cards a.act {
 | 
				
			||||||
 | 
						color: #037;
 | 
				
			||||||
	background: inherit;
 | 
						background: inherit;
 | 
				
			||||||
 | 
						box-shadow: 0 -.17em .67em #0ad;
 | 
				
			||||||
 | 
						border-color: #09c #05a #eee #05a;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
html.light #u2conf .txtbox {
 | 
					html.light #u2conf .txtbox {
 | 
				
			||||||
	background: #fff;
 | 
						background: #fff;
 | 
				
			||||||
@@ -272,4 +286,10 @@ html.light #u2cdesc {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
html.light #op_up2k.srch #u2btn {
 | 
					html.light #op_up2k.srch #u2btn {
 | 
				
			||||||
	border-color: #a80;
 | 
						border-color: #a80;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					html.light #u2foot {
 | 
				
			||||||
 | 
						color: #000;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					html.light #u2tab tbody tr:hover td {
 | 
				
			||||||
 | 
						background: #fff;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -36,7 +36,7 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            <table id="u2conf">
 | 
					            <table id="u2conf">
 | 
				
			||||||
                <tr>
 | 
					                <tr>
 | 
				
			||||||
                    <td>parallel uploads</td>
 | 
					                    <td>parallel uploads:</td>
 | 
				
			||||||
                    <td rowspan="2">
 | 
					                    <td rowspan="2">
 | 
				
			||||||
                        <input type="checkbox" id="multitask" />
 | 
					                        <input type="checkbox" id="multitask" />
 | 
				
			||||||
                        <label for="multitask" alt="continue hashing other files while uploading">🏃</label>
 | 
					                        <label for="multitask" alt="continue hashing other files while uploading">🏃</label>
 | 
				
			||||||
@@ -59,9 +59,9 @@
 | 
				
			|||||||
                </tr>
 | 
					                </tr>
 | 
				
			||||||
                <tr>
 | 
					                <tr>
 | 
				
			||||||
                    <td>
 | 
					                    <td>
 | 
				
			||||||
                        <a href="#" id="nthread_sub">–</a>
 | 
					                        <a href="#" id="nthread_sub">–</a><input
 | 
				
			||||||
                        <input class="txtbox" id="nthread" value="2" />
 | 
					                            class="txtbox" id="nthread" value="2"/><a
 | 
				
			||||||
                        <a href="#" id="nthread_add">+</a>
 | 
					                            href="#" id="nthread_add">+</a>
 | 
				
			||||||
                    </td>
 | 
					                    </td>
 | 
				
			||||||
                </tr>
 | 
					                </tr>
 | 
				
			||||||
            </table>
 | 
					            </table>
 | 
				
			||||||
@@ -99,5 +99,5 @@
 | 
				
			|||||||
            </table>
 | 
					            </table>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            <p id="u2foot"></p>
 | 
					            <p id="u2foot"></p>
 | 
				
			||||||
            <p id="u2footfoot">( if you don't need lastmod timestamps, resumable uploads or progress bars just use the <a href="#" id="u2nope">basic uploader</a>)</p>
 | 
					            <p id="u2footfoot">( you can use the <a href="#" id="u2nope">basic uploader</a> if you don't need lastmod timestamps, resumable uploads, or progress bars )</p>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,7 +6,8 @@ if (!window['console'])
 | 
				
			|||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
var clickev = window.Touch ? 'touchstart' : 'click';
 | 
					var clickev = window.Touch ? 'touchstart' : 'click',
 | 
				
			||||||
 | 
					    ANDROID = /(android)/i.test(navigator.userAgent);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// error handler for mobile devices
 | 
					// error handler for mobile devices
 | 
				
			||||||
@@ -49,9 +50,11 @@ function vis_exh(msg, url, lineNo, columnNo, error) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function ebi(id) {
 | 
					var ebi = document.getElementById.bind(document),
 | 
				
			||||||
    return document.getElementById(id);
 | 
					    QS = document.querySelector.bind(document),
 | 
				
			||||||
}
 | 
					    QSA = document.querySelectorAll.bind(document),
 | 
				
			||||||
 | 
					    mknod = document.createElement.bind(document);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function ev(e) {
 | 
					function ev(e) {
 | 
				
			||||||
    e = e || window.event;
 | 
					    e = e || window.event;
 | 
				
			||||||
@@ -89,7 +92,7 @@ if (!String.startsWith) {
 | 
				
			|||||||
// https://stackoverflow.com/a/950146
 | 
					// https://stackoverflow.com/a/950146
 | 
				
			||||||
function import_js(url, cb) {
 | 
					function import_js(url, cb) {
 | 
				
			||||||
    var head = document.head || document.getElementsByTagName('head')[0];
 | 
					    var head = document.head || document.getElementsByTagName('head')[0];
 | 
				
			||||||
    var script = document.createElement('script');
 | 
					    var script = mknod('script');
 | 
				
			||||||
    script.type = 'text/javascript';
 | 
					    script.type = 'text/javascript';
 | 
				
			||||||
    script.src = url;
 | 
					    script.src = url;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -274,7 +277,7 @@ function makeSortable(table, cb) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
(function () {
 | 
					(function () {
 | 
				
			||||||
    var ops = document.querySelectorAll('#ops>a');
 | 
					    var ops = QSA('#ops>a');
 | 
				
			||||||
    for (var a = 0; a < ops.length; a++) {
 | 
					    for (var a = 0; a < ops.length; a++) {
 | 
				
			||||||
        ops[a].onclick = opclick;
 | 
					        ops[a].onclick = opclick;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -289,25 +292,25 @@ function opclick(e) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    swrite('opmode', dest || null);
 | 
					    swrite('opmode', dest || null);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    var input = document.querySelector('.opview.act input:not([type="hidden"])')
 | 
					    var input = QS('.opview.act input:not([type="hidden"])')
 | 
				
			||||||
    if (input)
 | 
					    if (input)
 | 
				
			||||||
        input.focus();
 | 
					        input.focus();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function goto(dest) {
 | 
					function goto(dest) {
 | 
				
			||||||
    var obj = document.querySelectorAll('.opview.act');
 | 
					    var obj = QSA('.opview.act');
 | 
				
			||||||
    for (var a = obj.length - 1; a >= 0; a--)
 | 
					    for (var a = obj.length - 1; a >= 0; a--)
 | 
				
			||||||
        clmod(obj[a], 'act');
 | 
					        clmod(obj[a], 'act');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    obj = document.querySelectorAll('#ops>a');
 | 
					    obj = QSA('#ops>a');
 | 
				
			||||||
    for (var a = obj.length - 1; a >= 0; a--)
 | 
					    for (var a = obj.length - 1; a >= 0; a--)
 | 
				
			||||||
        clmod(obj[a], 'act');
 | 
					        clmod(obj[a], 'act');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (dest) {
 | 
					    if (dest) {
 | 
				
			||||||
        var ui = ebi('op_' + dest);
 | 
					        var ui = ebi('op_' + dest);
 | 
				
			||||||
        clmod(ui, 'act', true);
 | 
					        clmod(ui, 'act', true);
 | 
				
			||||||
        document.querySelector('#ops>a[data-dest=' + dest + ']').className += " act";
 | 
					        QS('#ops>a[data-dest=' + dest + ']').className += " act";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        var fn = window['goto_' + dest];
 | 
					        var fn = window['goto_' + dest];
 | 
				
			||||||
        if (fn)
 | 
					        if (fn)
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -32,9 +32,13 @@ r
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
# and a folder where anyone can upload
 | 
					# and a folder where anyone can upload
 | 
				
			||||||
# but nobody can see the contents
 | 
					# but nobody can see the contents
 | 
				
			||||||
 | 
					# and set the e2d flag to enable the uploads database
 | 
				
			||||||
 | 
					# and set the nodupe flag to reject duplicate uploads
 | 
				
			||||||
/home/ed/inc
 | 
					/home/ed/inc
 | 
				
			||||||
/dump
 | 
					/dump
 | 
				
			||||||
w
 | 
					w
 | 
				
			||||||
 | 
					c e2d
 | 
				
			||||||
 | 
					c nodupe
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# this entire config file can be replaced with these arguments:
 | 
					# this entire config file can be replaced with these arguments:
 | 
				
			||||||
# -u ed:123 -u k:k -v .::r:aed -v priv:priv:rk:aed -v /home/ed/Music:music:r -v /home/ed/inc:dump:w
 | 
					# -u ed:123 -u k:k -v .::r:aed -v priv:priv:rk:aed -v /home/ed/Music:music:r -v /home/ed/inc:dump:w
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										29
									
								
								docs/minimal-up2k.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								docs/minimal-up2k.html
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,29 @@
 | 
				
			|||||||
 | 
					<!--
 | 
				
			||||||
 | 
					    save this as .epilogue.html inside a
 | 
				
			||||||
 | 
					    write-only folder to declutter the UI
 | 
				
			||||||
 | 
					-->
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* make the up2k ui REALLY minimal by hiding a bunch of stuff: */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #ops, #tree, #path,  /* main tabs and navigators (tree/breadcrumbs) */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #u2cleanup, #u2conf tr:first-child>td[rowspan]:not(#u2btn_cw),  /* most of the config options */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #u2cards  /* and the upload progress tabs */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    {display:none!important}  /* do it! */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* add some margins because now it's weird */
 | 
				
			||||||
 | 
					    .opview {margin-top: 2.5em}
 | 
				
			||||||
 | 
					    #op_up2k {margin-top: 5em}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* and embiggen the upload button */
 | 
				
			||||||
 | 
					    #u2conf #u2btn, #u2btn {padding:1.5em 0}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<a href="#" onclick="this.parentNode.innerHTML='';">show advanced options</a>
 | 
				
			||||||
@@ -171,7 +171,7 @@ Range: bytes=26-         Content-Range: bytes */26
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
var tsh = [];
 | 
					var tsh = [];
 | 
				
			||||||
function convert_markdown(md_text, dest_dom) {
 | 
					function convert_markdown(md_text, dest_dom) {
 | 
				
			||||||
    tsh.push(new Date().getTime());
 | 
					    tsh.push(Date.now());
 | 
				
			||||||
    while (tsh.length > 10)
 | 
					    while (tsh.length > 10)
 | 
				
			||||||
        tsh.shift();
 | 
					        tsh.shift();
 | 
				
			||||||
    if (tsh.length > 1) {
 | 
					    if (tsh.length > 1) {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -117,7 +117,7 @@ cd sfx
 | 
				
			|||||||
ver=
 | 
					ver=
 | 
				
			||||||
git describe --tags >/dev/null 2>/dev/null && {
 | 
					git describe --tags >/dev/null 2>/dev/null && {
 | 
				
			||||||
	git_ver="$(git describe --tags)";  # v0.5.5-2-gb164aa0
 | 
						git_ver="$(git describe --tags)";  # v0.5.5-2-gb164aa0
 | 
				
			||||||
	ver="$(printf '%s\n' "$git_ver" | sed -r 's/^v//; s/-g?/./g')";
 | 
						ver="$(printf '%s\n' "$git_ver" | sed -r 's/^v//')";
 | 
				
			||||||
	t_ver=
 | 
						t_ver=
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	printf '%s\n' "$git_ver" | grep -qE '^v[0-9\.]+$' && {
 | 
						printf '%s\n' "$git_ver" | grep -qE '^v[0-9\.]+$' && {
 | 
				
			||||||
@@ -199,12 +199,19 @@ find | grep -E '\.(js|css|html)$' | while IFS= read -r f; do
 | 
				
			|||||||
	tmv "$f"
 | 
						tmv "$f"
 | 
				
			||||||
done
 | 
					done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo gen tarlist
 | 
				
			||||||
 | 
					for d in copyparty dep-j2; do find $d -type f; done |
 | 
				
			||||||
 | 
					sed -r 's/(.*)\.(.*)/\2 \1/' | LC_ALL=C sort |
 | 
				
			||||||
 | 
					sed -r 's/([^ ]*) (.*)/\2.\1/' | grep -vE '/list1?$' > list1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					(grep -vE 'gz$' list1; grep -E 'gz$' list1) >list
 | 
				
			||||||
 | 
					
 | 
				
			||||||
echo creating tar
 | 
					echo creating tar
 | 
				
			||||||
args=(--owner=1000 --group=1000)
 | 
					args=(--owner=1000 --group=1000)
 | 
				
			||||||
[ "$OSTYPE" = msys ] &&
 | 
					[ "$OSTYPE" = msys ] &&
 | 
				
			||||||
	args=()
 | 
						args=()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
tar -cf tar "${args[@]}" --numeric-owner copyparty dep-j2
 | 
					tar -cf tar "${args[@]}" --numeric-owner -T list
 | 
				
			||||||
 | 
					
 | 
				
			||||||
pc=bzip2
 | 
					pc=bzip2
 | 
				
			||||||
pe=bz2
 | 
					pe=bz2
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										161
									
								
								scripts/sfx.py
									
									
									
									
									
								
							
							
						
						
									
										161
									
								
								scripts/sfx.py
									
									
									
									
									
								
							@@ -2,7 +2,8 @@
 | 
				
			|||||||
# coding: latin-1
 | 
					# coding: latin-1
 | 
				
			||||||
from __future__ import print_function, unicode_literals
 | 
					from __future__ import print_function, unicode_literals
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import os, sys, time, shutil, threading, tarfile, hashlib, platform, tempfile, traceback
 | 
					import re, os, sys, time, shutil, signal, threading, tarfile, hashlib, platform, tempfile, traceback
 | 
				
			||||||
 | 
					import subprocess as sp
 | 
				
			||||||
 | 
					
 | 
				
			||||||
"""
 | 
					"""
 | 
				
			||||||
run me with any version of python, i will unpack and run copyparty
 | 
					run me with any version of python, i will unpack and run copyparty
 | 
				
			||||||
@@ -26,22 +27,21 @@ CKSUM = None
 | 
				
			|||||||
STAMP = None
 | 
					STAMP = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
PY2 = sys.version_info[0] == 2
 | 
					PY2 = sys.version_info[0] == 2
 | 
				
			||||||
WINDOWS = sys.platform == "win32"
 | 
					WINDOWS = sys.platform in ["win32", "msys"]
 | 
				
			||||||
sys.dont_write_bytecode = True
 | 
					sys.dont_write_bytecode = True
 | 
				
			||||||
me = os.path.abspath(os.path.realpath(__file__))
 | 
					me = os.path.abspath(os.path.realpath(__file__))
 | 
				
			||||||
cpp = None
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def eprint(*args, **kwargs):
 | 
					def eprint(*a, **ka):
 | 
				
			||||||
    kwargs["file"] = sys.stderr
 | 
					    ka["file"] = sys.stderr
 | 
				
			||||||
    print(*args, **kwargs)
 | 
					    print(*a, **ka)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def msg(*args, **kwargs):
 | 
					def msg(*a, **ka):
 | 
				
			||||||
    if args:
 | 
					    if a:
 | 
				
			||||||
        args = ["[SFX]", args[0]] + list(args[1:])
 | 
					        a = ["[SFX]", a[0]] + list(a[1:])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    eprint(*args, **kwargs)
 | 
					    eprint(*a, **ka)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# skip 1
 | 
					# skip 1
 | 
				
			||||||
@@ -156,6 +156,9 @@ def encode(data, size, cksum, ver, ts):
 | 
				
			|||||||
                skip = True
 | 
					                skip = True
 | 
				
			||||||
                continue
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if ln.strip().startswith("# fmt: "):
 | 
				
			||||||
 | 
					                continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            unpk += ln + "\n"
 | 
					            unpk += ln + "\n"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        for k, v in [
 | 
					        for k, v in [
 | 
				
			||||||
@@ -209,11 +212,11 @@ def yieldfile(fn):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def hashfile(fn):
 | 
					def hashfile(fn):
 | 
				
			||||||
    hasher = hashlib.md5()
 | 
					    h = hashlib.md5()
 | 
				
			||||||
    for block in yieldfile(fn):
 | 
					    for block in yieldfile(fn):
 | 
				
			||||||
        hasher.update(block)
 | 
					        h.update(block)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return hasher.hexdigest()
 | 
					    return h.hexdigest()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def unpack():
 | 
					def unpack():
 | 
				
			||||||
@@ -222,9 +225,10 @@ def unpack():
 | 
				
			|||||||
    tag = "v" + str(STAMP)
 | 
					    tag = "v" + str(STAMP)
 | 
				
			||||||
    withpid = "{}.{}".format(name, os.getpid())
 | 
					    withpid = "{}.{}".format(name, os.getpid())
 | 
				
			||||||
    top = tempfile.gettempdir()
 | 
					    top = tempfile.gettempdir()
 | 
				
			||||||
    final = os.path.join(top, name)
 | 
					    opj = os.path.join
 | 
				
			||||||
    mine = os.path.join(top, withpid)
 | 
					    final = opj(top, name)
 | 
				
			||||||
    tar = os.path.join(mine, "tar")
 | 
					    mine = opj(top, withpid)
 | 
				
			||||||
 | 
					    tar = opj(mine, "tar")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        if tag in os.listdir(final):
 | 
					        if tag in os.listdir(final):
 | 
				
			||||||
@@ -233,28 +237,24 @@ def unpack():
 | 
				
			|||||||
    except:
 | 
					    except:
 | 
				
			||||||
        pass
 | 
					        pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    nwrite = 0
 | 
					    sz = 0
 | 
				
			||||||
    os.mkdir(mine)
 | 
					    os.mkdir(mine)
 | 
				
			||||||
    with open(tar, "wb") as f:
 | 
					    with open(tar, "wb") as f:
 | 
				
			||||||
        for buf in get_payload():
 | 
					        for buf in get_payload():
 | 
				
			||||||
            nwrite += len(buf)
 | 
					            sz += len(buf)
 | 
				
			||||||
            f.write(buf)
 | 
					            f.write(buf)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if nwrite != SIZE:
 | 
					    ck = hashfile(tar)
 | 
				
			||||||
        t = "\n\n  bad file:\n    expected {} bytes, got {}\n".format(SIZE, nwrite)
 | 
					    if ck != CKSUM:
 | 
				
			||||||
        raise Exception(t)
 | 
					        t = "\n\nexpected {} ({} byte)\nobtained {} ({} byte)\nsfx corrupt"
 | 
				
			||||||
 | 
					        raise Exception(t.format(CKSUM, SIZE, ck, sz))
 | 
				
			||||||
    cksum = hashfile(tar)
 | 
					 | 
				
			||||||
    if cksum != CKSUM:
 | 
					 | 
				
			||||||
        t = "\n\n  bad file:\n    {} expected,\n    {} obtained\n".format(CKSUM, cksum)
 | 
					 | 
				
			||||||
        raise Exception(t)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    with tarfile.open(tar, "r:bz2") as tf:
 | 
					    with tarfile.open(tar, "r:bz2") as tf:
 | 
				
			||||||
        tf.extractall(mine)
 | 
					        tf.extractall(mine)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    os.remove(tar)
 | 
					    os.remove(tar)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    with open(os.path.join(mine, tag), "wb") as f:
 | 
					    with open(opj(mine, tag), "wb") as f:
 | 
				
			||||||
        f.write(b"h\n")
 | 
					        f.write(b"h\n")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
@@ -272,25 +272,25 @@ def unpack():
 | 
				
			|||||||
    except:
 | 
					    except:
 | 
				
			||||||
        pass
 | 
					        pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for fn in u8(os.listdir(top)):
 | 
				
			||||||
 | 
					        if fn.startswith(name) and fn != withpid:
 | 
				
			||||||
 | 
					            try:
 | 
				
			||||||
 | 
					                old = opj(top, fn)
 | 
				
			||||||
 | 
					                if time.time() - os.path.getmtime(old) > 86400:
 | 
				
			||||||
 | 
					                    shutil.rmtree(old)
 | 
				
			||||||
 | 
					            except:
 | 
				
			||||||
 | 
					                pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        os.symlink(mine, final)
 | 
					        os.symlink(mine, final)
 | 
				
			||||||
    except:
 | 
					    except:
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            os.rename(mine, final)
 | 
					            os.rename(mine, final)
 | 
				
			||||||
 | 
					            return final
 | 
				
			||||||
        except:
 | 
					        except:
 | 
				
			||||||
            msg("reloc fail,", mine)
 | 
					            msg("reloc fail,", mine)
 | 
				
			||||||
            return mine
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    for fn in u8(os.listdir(top)):
 | 
					    return mine
 | 
				
			||||||
        if fn.startswith(name) and fn not in [name, withpid]:
 | 
					 | 
				
			||||||
            try:
 | 
					 | 
				
			||||||
                old = os.path.join(top, fn)
 | 
					 | 
				
			||||||
                if time.time() - os.path.getmtime(old) > 10:
 | 
					 | 
				
			||||||
                    shutil.rmtree(old)
 | 
					 | 
				
			||||||
            except:
 | 
					 | 
				
			||||||
                pass
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    return final
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def get_payload():
 | 
					def get_payload():
 | 
				
			||||||
@@ -307,37 +307,33 @@ def get_payload():
 | 
				
			|||||||
        if ofs < 0:
 | 
					        if ofs < 0:
 | 
				
			||||||
            raise Exception("could not find archive marker")
 | 
					            raise Exception("could not find archive marker")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # start reading from the final b"\n"
 | 
					        # start at final b"\n"
 | 
				
			||||||
        fpos = ofs + len(ptn) - 3
 | 
					        fpos = ofs + len(ptn) - 3
 | 
				
			||||||
        # msg("tar found at", fpos)
 | 
					 | 
				
			||||||
        f.seek(fpos)
 | 
					        f.seek(fpos)
 | 
				
			||||||
        dpos = 0
 | 
					        dpos = 0
 | 
				
			||||||
        leftovers = b""
 | 
					        rem = b""
 | 
				
			||||||
        while True:
 | 
					        while True:
 | 
				
			||||||
            rbuf = f.read(1024 * 32)
 | 
					            rbuf = f.read(1024 * 32)
 | 
				
			||||||
            if rbuf:
 | 
					            if rbuf:
 | 
				
			||||||
                buf = leftovers + rbuf
 | 
					                buf = rem + rbuf
 | 
				
			||||||
                ofs = buf.rfind(b"\n")
 | 
					                ofs = buf.rfind(b"\n")
 | 
				
			||||||
                if len(buf) <= 4:
 | 
					                if len(buf) <= 4:
 | 
				
			||||||
                    leftovers = buf
 | 
					                    rem = buf
 | 
				
			||||||
                    continue
 | 
					                    continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
                if ofs >= len(buf) - 4:
 | 
					                if ofs >= len(buf) - 4:
 | 
				
			||||||
                    leftovers = buf[ofs:]
 | 
					                    rem = buf[ofs:]
 | 
				
			||||||
                    buf = buf[:ofs]
 | 
					                    buf = buf[:ofs]
 | 
				
			||||||
                else:
 | 
					                else:
 | 
				
			||||||
                    leftovers = b"\n# "
 | 
					                    rem = b"\n# "
 | 
				
			||||||
            else:
 | 
					            else:
 | 
				
			||||||
                buf = leftovers
 | 
					                buf = rem
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            fpos += len(buf) + 1
 | 
					            fpos += len(buf) + 1
 | 
				
			||||||
            buf = (
 | 
					            for a, b in [[b"\n# ", b""], [b"\n#r", b"\r"], [b"\n#n", b"\n"]]:
 | 
				
			||||||
                buf.replace(b"\n# ", b"")
 | 
					                buf = buf.replace(a, b)
 | 
				
			||||||
                .replace(b"\n#r", b"\r")
 | 
					 | 
				
			||||||
                .replace(b"\n#n", b"\n")
 | 
					 | 
				
			||||||
            )
 | 
					 | 
				
			||||||
            dpos += len(buf) - 1
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            dpos += len(buf) - 1
 | 
				
			||||||
            yield buf
 | 
					            yield buf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            if not rbuf:
 | 
					            if not rbuf:
 | 
				
			||||||
@@ -361,7 +357,7 @@ def utime(top):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
def confirm(rv):
 | 
					def confirm(rv):
 | 
				
			||||||
    msg()
 | 
					    msg()
 | 
				
			||||||
    msg(traceback.format_exc())
 | 
					    msg("retcode", rv if rv else traceback.format_exc())
 | 
				
			||||||
    msg("*** hit enter to exit ***")
 | 
					    msg("*** hit enter to exit ***")
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        raw_input() if PY2 else input()
 | 
					        raw_input() if PY2 else input()
 | 
				
			||||||
@@ -371,10 +367,8 @@ def confirm(rv):
 | 
				
			|||||||
    sys.exit(rv)
 | 
					    sys.exit(rv)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def run(tmp, j2ver):
 | 
					def run(tmp, j2):
 | 
				
			||||||
    global cpp
 | 
					    msg("jinja2:", j2 or "bundled")
 | 
				
			||||||
 | 
					 | 
				
			||||||
    msg("jinja2:", j2ver or "bundled")
 | 
					 | 
				
			||||||
    msg("sfxdir:", tmp)
 | 
					    msg("sfxdir:", tmp)
 | 
				
			||||||
    msg()
 | 
					    msg()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -384,7 +378,6 @@ def run(tmp, j2ver):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        fd = os.open(tmp, os.O_RDONLY)
 | 
					        fd = os.open(tmp, os.O_RDONLY)
 | 
				
			||||||
        fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB)
 | 
					        fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB)
 | 
				
			||||||
        tmp = os.readlink(tmp)  # can't flock a symlink, even with O_NOFOLLOW
 | 
					 | 
				
			||||||
    except Exception as ex:
 | 
					    except Exception as ex:
 | 
				
			||||||
        if not WINDOWS:
 | 
					        if not WINDOWS:
 | 
				
			||||||
            msg("\033[31mflock:", repr(ex))
 | 
					            msg("\033[31mflock:", repr(ex))
 | 
				
			||||||
@@ -394,22 +387,39 @@ def run(tmp, j2ver):
 | 
				
			|||||||
    t.start()
 | 
					    t.start()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    ld = [tmp, os.path.join(tmp, "dep-j2")]
 | 
					    ld = [tmp, os.path.join(tmp, "dep-j2")]
 | 
				
			||||||
    if j2ver:
 | 
					    if j2:
 | 
				
			||||||
        del ld[-1]
 | 
					        del ld[-1]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if any([re.match(r"^-.*j[0-9]", x) for x in sys.argv]):
 | 
				
			||||||
 | 
					        run_s(ld)
 | 
				
			||||||
 | 
					    else:
 | 
				
			||||||
 | 
					        run_i(ld)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def run_i(ld):
 | 
				
			||||||
    for x in ld:
 | 
					    for x in ld:
 | 
				
			||||||
        sys.path.insert(0, x)
 | 
					        sys.path.insert(0, x)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    try:
 | 
					    from copyparty.__main__ import main as p
 | 
				
			||||||
        from copyparty.__main__ import main as copyparty
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        copyparty()
 | 
					    p()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    except SystemExit as ex:
 | 
					
 | 
				
			||||||
        if ex.code:
 | 
					def run_s(ld):
 | 
				
			||||||
            confirm(ex.code)
 | 
					    # fmt: off
 | 
				
			||||||
    except:
 | 
					    c = "import sys,runpy;" + "".join(['sys.path.insert(0,r"' + x + '");' for x in ld]) + 'runpy.run_module("copyparty",run_name="__main__")'
 | 
				
			||||||
        confirm(1)
 | 
					    c = [str(x) for x in [sys.executable, "-c", c] + list(sys.argv[1:])]
 | 
				
			||||||
 | 
					    # fmt: on
 | 
				
			||||||
 | 
					    msg("\n", c, "\n")
 | 
				
			||||||
 | 
					    p = sp.Popen(c)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def bye(*a):
 | 
				
			||||||
 | 
					        p.send_signal(signal.SIGINT)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    signal.signal(signal.SIGTERM, bye)
 | 
				
			||||||
 | 
					    p.wait()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    raise SystemExit(p.returncode)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def main():
 | 
					def main():
 | 
				
			||||||
@@ -443,14 +453,23 @@ def main():
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    # skip 0
 | 
					    # skip 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    tmp = unpack()
 | 
					    tmp = os.path.realpath(unpack())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        from jinja2 import __version__ as j2ver
 | 
					        from jinja2 import __version__ as j2
 | 
				
			||||||
    except:
 | 
					    except:
 | 
				
			||||||
        j2ver = None
 | 
					        j2 = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    run(tmp, j2ver)
 | 
					    try:
 | 
				
			||||||
 | 
					        run(tmp, j2)
 | 
				
			||||||
 | 
					    except SystemExit as ex:
 | 
				
			||||||
 | 
					        c = ex.code
 | 
				
			||||||
 | 
					        if c not in [0, -15]:
 | 
				
			||||||
 | 
					            confirm(ex.code)
 | 
				
			||||||
 | 
					    except KeyboardInterrupt:
 | 
				
			||||||
 | 
					        pass
 | 
				
			||||||
 | 
					    except:
 | 
				
			||||||
 | 
					        confirm(0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if __name__ == "__main__":
 | 
					if __name__ == "__main__":
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										0
									
								
								tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										33
									
								
								tests/run.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										33
									
								
								tests/run.py
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1,33 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/env python3
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import sys
 | 
				
			||||||
 | 
					import runpy
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					host = sys.argv[1]
 | 
				
			||||||
 | 
					sys.argv = sys.argv[:1] + sys.argv[2:]
 | 
				
			||||||
 | 
					sys.path.insert(0, ".")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def rp():
 | 
				
			||||||
 | 
					    runpy.run_module("unittest", run_name="__main__")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if host == "vmprof":
 | 
				
			||||||
 | 
					    rp()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					elif host == "cprofile":
 | 
				
			||||||
 | 
					    import cProfile
 | 
				
			||||||
 | 
					    import pstats
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    log_fn = "cprofile.log"
 | 
				
			||||||
 | 
					    cProfile.run("rp()", log_fn)
 | 
				
			||||||
 | 
					    p = pstats.Stats(log_fn)
 | 
				
			||||||
 | 
					    p.sort_stats(pstats.SortKey.CUMULATIVE).print_stats(64)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					"""
 | 
				
			||||||
 | 
					python3.9 tests/run.py cprofile -v tests/test_httpcli.py
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					python3.9 -m pip install --user vmprof
 | 
				
			||||||
 | 
					python3.9 -m vmprof --lines -o vmprof.log tests/run.py vmprof -v tests/test_httpcli.py
 | 
				
			||||||
 | 
					"""
 | 
				
			||||||
							
								
								
									
										202
									
								
								tests/test_httpcli.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										202
									
								
								tests/test_httpcli.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,202 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/env python
 | 
				
			||||||
 | 
					# coding: utf-8
 | 
				
			||||||
 | 
					from __future__ import print_function, unicode_literals
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import io
 | 
				
			||||||
 | 
					import os
 | 
				
			||||||
 | 
					import time
 | 
				
			||||||
 | 
					import shutil
 | 
				
			||||||
 | 
					import pprint
 | 
				
			||||||
 | 
					import tarfile
 | 
				
			||||||
 | 
					import unittest
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from argparse import Namespace
 | 
				
			||||||
 | 
					from copyparty.authsrv import AuthSrv
 | 
				
			||||||
 | 
					from copyparty.httpcli import HttpCli
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from tests import util as tu
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def hdr(query):
 | 
				
			||||||
 | 
					    h = "GET /{} HTTP/1.1\r\nCookie: cppwd=o\r\nConnection: close\r\n\r\n"
 | 
				
			||||||
 | 
					    return h.format(query).encode("utf-8")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Cfg(Namespace):
 | 
				
			||||||
 | 
					    def __init__(self, a=[], v=[], c=None):
 | 
				
			||||||
 | 
					        super(Cfg, self).__init__(
 | 
				
			||||||
 | 
					            a=a,
 | 
				
			||||||
 | 
					            v=v,
 | 
				
			||||||
 | 
					            c=c,
 | 
				
			||||||
 | 
					            ed=False,
 | 
				
			||||||
 | 
					            no_zip=False,
 | 
				
			||||||
 | 
					            no_scandir=False,
 | 
				
			||||||
 | 
					            no_sendfile=True,
 | 
				
			||||||
 | 
					            nih=True,
 | 
				
			||||||
 | 
					            mtp=[],
 | 
				
			||||||
 | 
					            mte="a",
 | 
				
			||||||
 | 
					            **{k: False for k in "e2d e2ds e2dsa e2t e2ts e2tsr".split()}
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class TestHttpCli(unittest.TestCase):
 | 
				
			||||||
 | 
					    def test(self):
 | 
				
			||||||
 | 
					        td = os.path.join(tu.get_ramdisk(), "vfs")
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            shutil.rmtree(td)
 | 
				
			||||||
 | 
					        except OSError:
 | 
				
			||||||
 | 
					            pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        os.mkdir(td)
 | 
				
			||||||
 | 
					        os.chdir(td)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.dtypes = ["ra", "ro", "rx", "wa", "wo", "wx", "aa", "ao", "ax"]
 | 
				
			||||||
 | 
					        self.can_read = ["ra", "ro", "aa", "ao"]
 | 
				
			||||||
 | 
					        self.can_write = ["wa", "wo", "aa", "ao"]
 | 
				
			||||||
 | 
					        self.fn = "g{:x}g".format(int(time.time() * 3))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        allfiles = []
 | 
				
			||||||
 | 
					        allvols = []
 | 
				
			||||||
 | 
					        for top in self.dtypes:
 | 
				
			||||||
 | 
					            allvols.append(top)
 | 
				
			||||||
 | 
					            allfiles.append("/".join([top, self.fn]))
 | 
				
			||||||
 | 
					            for s1 in self.dtypes:
 | 
				
			||||||
 | 
					                p = "/".join([top, s1])
 | 
				
			||||||
 | 
					                allvols.append(p)
 | 
				
			||||||
 | 
					                allfiles.append(p + "/" + self.fn)
 | 
				
			||||||
 | 
					                allfiles.append(p + "/n/" + self.fn)
 | 
				
			||||||
 | 
					                for s2 in self.dtypes:
 | 
				
			||||||
 | 
					                    p = "/".join([top, s1, "n", s2])
 | 
				
			||||||
 | 
					                    os.makedirs(p)
 | 
				
			||||||
 | 
					                    allvols.append(p)
 | 
				
			||||||
 | 
					                    allfiles.append(p + "/" + self.fn)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for fp in allfiles:
 | 
				
			||||||
 | 
					            with open(fp, "w") as f:
 | 
				
			||||||
 | 
					                f.write("ok {}\n".format(fp))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for top in self.dtypes:
 | 
				
			||||||
 | 
					            vcfg = []
 | 
				
			||||||
 | 
					            for vol in allvols:
 | 
				
			||||||
 | 
					                if not vol.startswith(top):
 | 
				
			||||||
 | 
					                    continue
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                mode = vol[-2]
 | 
				
			||||||
 | 
					                usr = vol[-1]
 | 
				
			||||||
 | 
					                if usr == "a":
 | 
				
			||||||
 | 
					                    usr = ""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if "/" not in vol:
 | 
				
			||||||
 | 
					                    vol += "/"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                top, sub = vol.split("/", 1)
 | 
				
			||||||
 | 
					                vcfg.append("{0}/{1}:{1}:{2}{3}".format(top, sub, mode, usr))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            pprint.pprint(vcfg)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            self.args = Cfg(v=vcfg, a=["o:o", "x:x"])
 | 
				
			||||||
 | 
					            self.auth = AuthSrv(self.args, self.log)
 | 
				
			||||||
 | 
					            vfiles = [x for x in allfiles if x.startswith(top)]
 | 
				
			||||||
 | 
					            for fp in vfiles:
 | 
				
			||||||
 | 
					                rok, wok = self.can_rw(fp)
 | 
				
			||||||
 | 
					                furl = fp.split("/", 1)[1]
 | 
				
			||||||
 | 
					                durl = furl.rsplit("/", 1)[0] if "/" in furl else ""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                # file download
 | 
				
			||||||
 | 
					                h, ret = self.curl(furl)
 | 
				
			||||||
 | 
					                res = "ok " + fp in ret
 | 
				
			||||||
 | 
					                print("[{}] {} {} = {}".format(fp, rok, wok, res))
 | 
				
			||||||
 | 
					                if rok != res:
 | 
				
			||||||
 | 
					                    print("\033[33m{}\n# {}\033[0m".format(ret, furl))
 | 
				
			||||||
 | 
					                    self.fail()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                # file browser: html
 | 
				
			||||||
 | 
					                h, ret = self.curl(durl)
 | 
				
			||||||
 | 
					                res = "'{}'".format(self.fn) in ret
 | 
				
			||||||
 | 
					                print(res)
 | 
				
			||||||
 | 
					                if rok != res:
 | 
				
			||||||
 | 
					                    print("\033[33m{}\n# {}\033[0m".format(ret, durl))
 | 
				
			||||||
 | 
					                    self.fail()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                # file browser: json
 | 
				
			||||||
 | 
					                url = durl + "?ls"
 | 
				
			||||||
 | 
					                h, ret = self.curl(url)
 | 
				
			||||||
 | 
					                res = '"{}"'.format(self.fn) in ret
 | 
				
			||||||
 | 
					                print(res)
 | 
				
			||||||
 | 
					                if rok != res:
 | 
				
			||||||
 | 
					                    print("\033[33m{}\n# {}\033[0m".format(ret, url))
 | 
				
			||||||
 | 
					                    self.fail()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                # tar
 | 
				
			||||||
 | 
					                url = durl + "?tar"
 | 
				
			||||||
 | 
					                h, b = self.curl(url, True)
 | 
				
			||||||
 | 
					                # with open(os.path.join(td, "tar"), "wb") as f:
 | 
				
			||||||
 | 
					                #    f.write(b)
 | 
				
			||||||
 | 
					                try:
 | 
				
			||||||
 | 
					                    tar = tarfile.open(fileobj=io.BytesIO(b)).getnames()
 | 
				
			||||||
 | 
					                except:
 | 
				
			||||||
 | 
					                    tar = []
 | 
				
			||||||
 | 
					                tar = ["/".join([y for y in [top, durl, x] if y]) for x in tar]
 | 
				
			||||||
 | 
					                tar = [[x] + self.can_rw(x) for x in tar]
 | 
				
			||||||
 | 
					                tar_ok = [x[0] for x in tar if x[1]]
 | 
				
			||||||
 | 
					                tar_ng = [x[0] for x in tar if not x[1]]
 | 
				
			||||||
 | 
					                self.assertEqual([], tar_ng)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if durl.split("/")[-1] in self.can_read:
 | 
				
			||||||
 | 
					                    ref = [x for x in vfiles if self.in_dive(top + "/" + durl, x)]
 | 
				
			||||||
 | 
					                    for f in ref:
 | 
				
			||||||
 | 
					                        print("{}: {}".format("ok" if f in tar_ok else "NG", f))
 | 
				
			||||||
 | 
					                    ref.sort()
 | 
				
			||||||
 | 
					                    tar_ok.sort()
 | 
				
			||||||
 | 
					                    self.assertEqual(ref, tar_ok)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                # stash
 | 
				
			||||||
 | 
					                h, ret = self.put(url)
 | 
				
			||||||
 | 
					                res = h.startswith("HTTP/1.1 200 ")
 | 
				
			||||||
 | 
					                self.assertEqual(res, wok)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def can_rw(self, fp):
 | 
				
			||||||
 | 
					        # lowest non-neutral folder declares permissions
 | 
				
			||||||
 | 
					        expect = fp.split("/")[:-1]
 | 
				
			||||||
 | 
					        for x in reversed(expect):
 | 
				
			||||||
 | 
					            if x != "n":
 | 
				
			||||||
 | 
					                expect = x
 | 
				
			||||||
 | 
					                break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return [expect in self.can_read, expect in self.can_write]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def in_dive(self, top, fp):
 | 
				
			||||||
 | 
					        # archiver bails at first inaccessible subvolume
 | 
				
			||||||
 | 
					        top = top.strip("/").split("/")
 | 
				
			||||||
 | 
					        fp = fp.split("/")
 | 
				
			||||||
 | 
					        for f1, f2 in zip(top, fp):
 | 
				
			||||||
 | 
					            if f1 != f2:
 | 
				
			||||||
 | 
					                return False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for f in fp[len(top) :]:
 | 
				
			||||||
 | 
					            if f == self.fn:
 | 
				
			||||||
 | 
					                return True
 | 
				
			||||||
 | 
					            if f not in self.can_read and f != "n":
 | 
				
			||||||
 | 
					                return False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def put(self, url):
 | 
				
			||||||
 | 
					        buf = "PUT /{0} HTTP/1.1\r\nCookie: cppwd=o\r\nConnection: close\r\nContent-Length: {1}\r\n\r\nok {0}\n"
 | 
				
			||||||
 | 
					        buf = buf.format(url, len(url) + 4).encode("utf-8")
 | 
				
			||||||
 | 
					        conn = tu.VHttpConn(self.args, self.auth, self.log, buf)
 | 
				
			||||||
 | 
					        HttpCli(conn).run()
 | 
				
			||||||
 | 
					        return conn.s._reply.decode("utf-8").split("\r\n\r\n", 1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def curl(self, url, binary=False):
 | 
				
			||||||
 | 
					        conn = tu.VHttpConn(self.args, self.auth, self.log, hdr(url))
 | 
				
			||||||
 | 
					        HttpCli(conn).run()
 | 
				
			||||||
 | 
					        if binary:
 | 
				
			||||||
 | 
					            h, b = conn.s._reply.split(b"\r\n\r\n", 1)
 | 
				
			||||||
 | 
					            return [h.decode("utf-8"), b]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return conn.s._reply.decode("utf-8").split("\r\n\r\n", 1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def log(self, src, msg, c=0):
 | 
				
			||||||
 | 
					        # print(repr(msg))
 | 
				
			||||||
 | 
					        pass
 | 
				
			||||||
@@ -3,18 +3,18 @@
 | 
				
			|||||||
from __future__ import print_function, unicode_literals
 | 
					from __future__ import print_function, unicode_literals
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import os
 | 
					import os
 | 
				
			||||||
import time
 | 
					 | 
				
			||||||
import json
 | 
					import json
 | 
				
			||||||
import shutil
 | 
					import shutil
 | 
				
			||||||
import tempfile
 | 
					import tempfile
 | 
				
			||||||
import unittest
 | 
					import unittest
 | 
				
			||||||
import subprocess as sp  # nosec
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
from textwrap import dedent
 | 
					from textwrap import dedent
 | 
				
			||||||
from argparse import Namespace
 | 
					from argparse import Namespace
 | 
				
			||||||
from copyparty.authsrv import AuthSrv
 | 
					from copyparty.authsrv import AuthSrv
 | 
				
			||||||
from copyparty import util
 | 
					from copyparty import util
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from tests import util as tu
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class Cfg(Namespace):
 | 
					class Cfg(Namespace):
 | 
				
			||||||
    def __init__(self, a=[], v=[], c=None):
 | 
					    def __init__(self, a=[], v=[], c=None):
 | 
				
			||||||
@@ -51,52 +51,11 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
        real = [x[0] for x in real]
 | 
					        real = [x[0] for x in real]
 | 
				
			||||||
        return fsdir, real, virt
 | 
					        return fsdir, real, virt
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def runcmd(self, *argv):
 | 
					 | 
				
			||||||
        p = sp.Popen(argv, stdout=sp.PIPE, stderr=sp.PIPE)
 | 
					 | 
				
			||||||
        stdout, stderr = p.communicate()
 | 
					 | 
				
			||||||
        stdout = stdout.decode("utf-8")
 | 
					 | 
				
			||||||
        stderr = stderr.decode("utf-8")
 | 
					 | 
				
			||||||
        return [p.returncode, stdout, stderr]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def chkcmd(self, *argv):
 | 
					 | 
				
			||||||
        ok, sout, serr = self.runcmd(*argv)
 | 
					 | 
				
			||||||
        if ok != 0:
 | 
					 | 
				
			||||||
            raise Exception(serr)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        return sout, serr
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def get_ramdisk(self):
 | 
					 | 
				
			||||||
        for vol in ["/dev/shm", "/Volumes/cptd"]:  # nosec (singleton test)
 | 
					 | 
				
			||||||
            if os.path.exists(vol):
 | 
					 | 
				
			||||||
                return vol
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if os.path.exists("/Volumes"):
 | 
					 | 
				
			||||||
            devname, _ = self.chkcmd("hdiutil", "attach", "-nomount", "ram://8192")
 | 
					 | 
				
			||||||
            devname = devname.strip()
 | 
					 | 
				
			||||||
            print("devname: [{}]".format(devname))
 | 
					 | 
				
			||||||
            for _ in range(10):
 | 
					 | 
				
			||||||
                try:
 | 
					 | 
				
			||||||
                    _, _ = self.chkcmd(
 | 
					 | 
				
			||||||
                        "diskutil", "eraseVolume", "HFS+", "cptd", devname
 | 
					 | 
				
			||||||
                    )
 | 
					 | 
				
			||||||
                    return "/Volumes/cptd"
 | 
					 | 
				
			||||||
                except Exception as ex:
 | 
					 | 
				
			||||||
                    print(repr(ex))
 | 
					 | 
				
			||||||
                    time.sleep(0.25)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            raise Exception("ramdisk creation failed")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        ret = os.path.join(tempfile.gettempdir(), "copyparty-test")
 | 
					 | 
				
			||||||
        try:
 | 
					 | 
				
			||||||
            os.mkdir(ret)
 | 
					 | 
				
			||||||
        finally:
 | 
					 | 
				
			||||||
            return ret
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def log(self, src, msg, c=0):
 | 
					    def log(self, src, msg, c=0):
 | 
				
			||||||
        pass
 | 
					        pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def test(self):
 | 
					    def test(self):
 | 
				
			||||||
        td = os.path.join(self.get_ramdisk(), "vfs")
 | 
					        td = os.path.join(tu.get_ramdisk(), "vfs")
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            shutil.rmtree(td)
 | 
					            shutil.rmtree(td)
 | 
				
			||||||
        except OSError:
 | 
					        except OSError:
 | 
				
			||||||
@@ -268,7 +227,7 @@ class TestVFS(unittest.TestCase):
 | 
				
			|||||||
        self.assertEqual(list(v1), list(v2))
 | 
					        self.assertEqual(list(v1), list(v2))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # config file parser
 | 
					        # config file parser
 | 
				
			||||||
        cfg_path = os.path.join(self.get_ramdisk(), "test.cfg")
 | 
					        cfg_path = os.path.join(tu.get_ramdisk(), "test.cfg")
 | 
				
			||||||
        with open(cfg_path, "wb") as f:
 | 
					        with open(cfg_path, "wb") as f:
 | 
				
			||||||
            f.write(
 | 
					            f.write(
 | 
				
			||||||
                dedent(
 | 
					                dedent(
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										97
									
								
								tests/util.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								tests/util.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,97 @@
 | 
				
			|||||||
 | 
					import os
 | 
				
			||||||
 | 
					import time
 | 
				
			||||||
 | 
					import jinja2
 | 
				
			||||||
 | 
					import tempfile
 | 
				
			||||||
 | 
					import subprocess as sp
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from copyparty.util import Unrecv
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					J2_ENV = jinja2.Environment(loader=jinja2.BaseLoader)
 | 
				
			||||||
 | 
					J2_FILES = J2_ENV.from_string("{{ files|join('\n') }}")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def runcmd(*argv):
 | 
				
			||||||
 | 
					    p = sp.Popen(argv, stdout=sp.PIPE, stderr=sp.PIPE)
 | 
				
			||||||
 | 
					    stdout, stderr = p.communicate()
 | 
				
			||||||
 | 
					    stdout = stdout.decode("utf-8")
 | 
				
			||||||
 | 
					    stderr = stderr.decode("utf-8")
 | 
				
			||||||
 | 
					    return [p.returncode, stdout, stderr]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def chkcmd(*argv):
 | 
				
			||||||
 | 
					    ok, sout, serr = runcmd(*argv)
 | 
				
			||||||
 | 
					    if ok != 0:
 | 
				
			||||||
 | 
					        raise Exception(serr)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return sout, serr
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def get_ramdisk():
 | 
				
			||||||
 | 
					    for vol in ["/dev/shm", "/Volumes/cptd"]:  # nosec (singleton test)
 | 
				
			||||||
 | 
					        if os.path.exists(vol):
 | 
				
			||||||
 | 
					            return vol
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if os.path.exists("/Volumes"):
 | 
				
			||||||
 | 
					        devname, _ = chkcmd("hdiutil", "attach", "-nomount", "ram://32768")
 | 
				
			||||||
 | 
					        devname = devname.strip()
 | 
				
			||||||
 | 
					        print("devname: [{}]".format(devname))
 | 
				
			||||||
 | 
					        for _ in range(10):
 | 
				
			||||||
 | 
					            try:
 | 
				
			||||||
 | 
					                _, _ = chkcmd("diskutil", "eraseVolume", "HFS+", "cptd", devname)
 | 
				
			||||||
 | 
					                return "/Volumes/cptd"
 | 
				
			||||||
 | 
					            except Exception as ex:
 | 
				
			||||||
 | 
					                print(repr(ex))
 | 
				
			||||||
 | 
					                time.sleep(0.25)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        raise Exception("ramdisk creation failed")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ret = os.path.join(tempfile.gettempdir(), "copyparty-test")
 | 
				
			||||||
 | 
					    try:
 | 
				
			||||||
 | 
					        os.mkdir(ret)
 | 
				
			||||||
 | 
					    finally:
 | 
				
			||||||
 | 
					        return ret
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class NullBroker(object):
 | 
				
			||||||
 | 
					    def put(*args):
 | 
				
			||||||
 | 
					        pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class VSock(object):
 | 
				
			||||||
 | 
					    def __init__(self, buf):
 | 
				
			||||||
 | 
					        self._query = buf
 | 
				
			||||||
 | 
					        self._reply = b""
 | 
				
			||||||
 | 
					        self.sendall = self.send
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def recv(self, sz):
 | 
				
			||||||
 | 
					        ret = self._query[:sz]
 | 
				
			||||||
 | 
					        self._query = self._query[sz:]
 | 
				
			||||||
 | 
					        return ret
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def send(self, buf):
 | 
				
			||||||
 | 
					        self._reply += buf
 | 
				
			||||||
 | 
					        return len(buf)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class VHttpSrv(object):
 | 
				
			||||||
 | 
					    def __init__(self):
 | 
				
			||||||
 | 
					        self.broker = NullBroker()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        aliases = ["splash", "browser", "browser2", "msg", "md", "mde"]
 | 
				
			||||||
 | 
					        self.j2 = {x: J2_FILES for x in aliases}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class VHttpConn(object):
 | 
				
			||||||
 | 
					    def __init__(self, args, auth, log, buf):
 | 
				
			||||||
 | 
					        self.s = VSock(buf)
 | 
				
			||||||
 | 
					        self.sr = Unrecv(self.s)
 | 
				
			||||||
 | 
					        self.addr = ("127.0.0.1", "42069")
 | 
				
			||||||
 | 
					        self.args = args
 | 
				
			||||||
 | 
					        self.auth = auth
 | 
				
			||||||
 | 
					        self.log_func = log
 | 
				
			||||||
 | 
					        self.log_src = "a"
 | 
				
			||||||
 | 
					        self.hsrv = VHttpSrv()
 | 
				
			||||||
 | 
					        self.nbyte = 0
 | 
				
			||||||
 | 
					        self.workload = 0
 | 
				
			||||||
 | 
					        self.t0 = time.time()
 | 
				
			||||||
		Reference in New Issue
	
	Block a user