mirror of
https://github.com/9001/copyparty.git
synced 2025-10-28 10:33:33 +00:00
Compare commits
148 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
bbb1e165d6 | ||
|
|
fed8d94885 | ||
|
|
58040cc0ed | ||
|
|
03d692db66 | ||
|
|
903f8e8453 | ||
|
|
405ae1308e | ||
|
|
8a0f583d71 | ||
|
|
b6d7017491 | ||
|
|
0f0217d203 | ||
|
|
a203e33347 | ||
|
|
3b8f697dd4 | ||
|
|
78ba16f722 | ||
|
|
0fcfe79994 | ||
|
|
c0e6df4b63 | ||
|
|
322abdcb43 | ||
|
|
31100787ce | ||
|
|
c57d721be4 | ||
|
|
3b5a03e977 | ||
|
|
ed807ee43e | ||
|
|
073c130ae6 | ||
|
|
8810e0be13 | ||
|
|
f93016ab85 | ||
|
|
b19cf260c2 | ||
|
|
db03e1e7eb | ||
|
|
e0d975e36a | ||
|
|
cfeb15259f | ||
|
|
3b3f8fc8fb | ||
|
|
88bd2c084c | ||
|
|
bd367389b0 | ||
|
|
58ba71a76f | ||
|
|
d03e34d55d | ||
|
|
24f239a46c | ||
|
|
2c0826f85a | ||
|
|
c061461d01 | ||
|
|
e7982a04fe | ||
|
|
33b91a7513 | ||
|
|
9bb1323e44 | ||
|
|
e62bb807a5 | ||
|
|
3fc0d2cc4a | ||
|
|
0c786b0766 | ||
|
|
68c7528911 | ||
|
|
26e18ae800 | ||
|
|
c30dc0b546 | ||
|
|
f94aa46a11 | ||
|
|
403261a293 | ||
|
|
c7d9cbb11f | ||
|
|
57e1c53cbb | ||
|
|
0754b553dd | ||
|
|
50661d941b | ||
|
|
c5db7c1a0c | ||
|
|
2cef5365f7 | ||
|
|
fbc4e94007 | ||
|
|
037ed5a2ad | ||
|
|
69dfa55705 | ||
|
|
a79a5c4e3e | ||
|
|
7e80eabfe6 | ||
|
|
375b72770d | ||
|
|
e2dd683def | ||
|
|
9eba50c6e4 | ||
|
|
5a579dba52 | ||
|
|
e86c719575 | ||
|
|
0e87f35547 | ||
|
|
b6d3d791a5 | ||
|
|
c9c3302664 | ||
|
|
c3e4d65b80 | ||
|
|
27a03510c5 | ||
|
|
ed7727f7cb | ||
|
|
127ec10c0d | ||
|
|
5a9c0ad225 | ||
|
|
7e8daf650e | ||
|
|
0cf737b4ce | ||
|
|
74635e0113 | ||
|
|
e5c4f49901 | ||
|
|
e4654ee7f1 | ||
|
|
e5d05c05ed | ||
|
|
73c4f99687 | ||
|
|
28c12ef3bf | ||
|
|
eed82dbb54 | ||
|
|
2c4b4ab928 | ||
|
|
505a8fc6f6 | ||
|
|
e4801d9b06 | ||
|
|
04f1b2cf3a | ||
|
|
c06d928bb5 | ||
|
|
ab09927e7b | ||
|
|
779437db67 | ||
|
|
28cbdb652e | ||
|
|
2b2415a7d8 | ||
|
|
746a8208aa | ||
|
|
a2a041a98a | ||
|
|
10b436e449 | ||
|
|
4d62b34786 | ||
|
|
0546210687 | ||
|
|
f8c11faada | ||
|
|
16d6e9be1f | ||
|
|
aff8185f2e | ||
|
|
217d15fe81 | ||
|
|
171e93c201 | ||
|
|
acc1d2e9e3 | ||
|
|
49c2f37154 | ||
|
|
69e54497aa | ||
|
|
9aa1885669 | ||
|
|
4418508513 | ||
|
|
e897df3b34 | ||
|
|
8cd97ab0e7 | ||
|
|
bf4949353d | ||
|
|
98a944f7cc | ||
|
|
7c10f81c92 | ||
|
|
126ecc55c3 | ||
|
|
1034a51bd2 | ||
|
|
a2657887cc | ||
|
|
c14b17bfaf | ||
|
|
59ebc795e7 | ||
|
|
8e128d917e | ||
|
|
ea762b05e0 | ||
|
|
db374b19f1 | ||
|
|
ab3839ef36 | ||
|
|
9886c442f2 | ||
|
|
c8d1926d52 | ||
|
|
a6bd699e52 | ||
|
|
12143f2702 | ||
|
|
480705dee9 | ||
|
|
781d5094f4 | ||
|
|
5615cb94cd | ||
|
|
302302a2ac | ||
|
|
9761b4e3e9 | ||
|
|
0cf6924dca | ||
|
|
5fd81e9f90 | ||
|
|
52bf6f892b | ||
|
|
f3cce232a4 | ||
|
|
53d3c8b28e | ||
|
|
83fec3cca7 | ||
|
|
3cefc99b7d | ||
|
|
3a38dcbc05 | ||
|
|
7ff08bce57 | ||
|
|
fd490af434 | ||
|
|
1195b8f17e | ||
|
|
28dce13776 | ||
|
|
431f20177a | ||
|
|
87aff54d9d | ||
|
|
f50462de82 | ||
|
|
9bda8c7eb6 | ||
|
|
e83c63d239 | ||
|
|
b38533b0cc | ||
|
|
5ccca3fbd5 | ||
|
|
9e850fc3ab | ||
|
|
ffbfcd7e00 | ||
|
|
5ea7590748 | ||
|
|
290c3bc2bb |
2
.vscode/launch.json
vendored
2
.vscode/launch.json
vendored
@@ -14,6 +14,8 @@
|
|||||||
"-emp",
|
"-emp",
|
||||||
"-e2dsa",
|
"-e2dsa",
|
||||||
"-e2ts",
|
"-e2ts",
|
||||||
|
"-mtp",
|
||||||
|
".bpm=f,bin/mtag/audio-bpm.py",
|
||||||
"-a",
|
"-a",
|
||||||
"ed:wark",
|
"ed:wark",
|
||||||
"-v",
|
"-v",
|
||||||
|
|||||||
35
.vscode/launch.py
vendored
Normal file
35
.vscode/launch.py
vendored
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
# takes arguments from launch.json
|
||||||
|
# is used by no_dbg in tasks.json
|
||||||
|
# launches 10x faster than mspython debugpy
|
||||||
|
# and is stoppable with ^C
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import shlex
|
||||||
|
|
||||||
|
sys.path.insert(0, os.getcwd())
|
||||||
|
|
||||||
|
import jstyleson
|
||||||
|
from copyparty.__main__ import main as copyparty
|
||||||
|
|
||||||
|
with open(".vscode/launch.json", "r", encoding="utf-8") as f:
|
||||||
|
tj = f.read()
|
||||||
|
|
||||||
|
oj = jstyleson.loads(tj)
|
||||||
|
argv = oj["configurations"][0]["args"]
|
||||||
|
|
||||||
|
try:
|
||||||
|
sargv = " ".join([shlex.quote(x) for x in argv])
|
||||||
|
print(sys.executable + " -m copyparty " + sargv + "\n")
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
argv = [os.path.expanduser(x) if x.startswith("~") else x for x in argv]
|
||||||
|
try:
|
||||||
|
copyparty(["a"] + argv)
|
||||||
|
except SystemExit as ex:
|
||||||
|
if ex.code:
|
||||||
|
raise
|
||||||
|
|
||||||
|
print("\n\033[32mokke\033[0m")
|
||||||
|
sys.exit(1)
|
||||||
4
.vscode/tasks.json
vendored
4
.vscode/tasks.json
vendored
@@ -9,9 +9,7 @@
|
|||||||
{
|
{
|
||||||
"label": "no_dbg",
|
"label": "no_dbg",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "${config:python.pythonPath} -m copyparty -ed -emp -e2dsa -e2ts -a ed:wark -v srv::r:aed:cnodupe -v dist:dist:r ;exit 1"
|
"command": "${config:python.pythonPath} .vscode/launch.py"
|
||||||
// -v ~/Music/mt:mt:r:cmtp=.bpm=~/dev/copyparty/bin/mtag/audio-bpm.py:cmtp=key=~/dev/copyparty/bin/mtag/audio-key.py:ce2tsr
|
|
||||||
// -v ~/Music/mt:mt:r:cmtp=.bpm=~/dev/copyparty/bin/mtag/audio-bpm.py:ce2tsr
|
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
249
README.md
249
README.md
@@ -9,9 +9,12 @@
|
|||||||
turn your phone or raspi into a portable file server with resumable uploads/downloads using IE6 or any other browser
|
turn your phone or raspi into a portable file server with resumable uploads/downloads using IE6 or any other browser
|
||||||
|
|
||||||
* server runs on anything with `py2.7` or `py3.3+`
|
* server runs on anything with `py2.7` or `py3.3+`
|
||||||
* *resumable* uploads need `firefox 12+` / `chrome 6+` / `safari 6+` / `IE 10+`
|
* browse/upload with IE4 / netscape4.0 on win3.11 (heh)
|
||||||
|
* *resumable* uploads need `firefox 34+` / `chrome 41+` / `safari 7+` for full speed
|
||||||
* 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,13 +23,25 @@ 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)
|
* [general bugs](#general-bugs)
|
||||||
|
* [not my bugs](#not-my-bugs)
|
||||||
|
* [the browser](#the-browser)
|
||||||
|
* [tabs](#tabs)
|
||||||
|
* [hotkeys](#hotkeys)
|
||||||
|
* [tree-mode](#tree-mode)
|
||||||
|
* [zip downloads](#zip-downloads)
|
||||||
|
* [uploading](#uploading)
|
||||||
|
* [file-search](#file-search)
|
||||||
|
* [markdown viewer](#markdown-viewer)
|
||||||
|
* [other tricks](#other-tricks)
|
||||||
* [searching](#searching)
|
* [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)
|
||||||
* [file parser plugins](#file-parser-plugins)
|
* [file parser plugins](#file-parser-plugins)
|
||||||
* [complete examples](#complete-examples)
|
* [complete examples](#complete-examples)
|
||||||
|
* [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)
|
||||||
@@ -41,19 +56,19 @@ turn your phone or raspi into a portable file server with resumable uploads/down
|
|||||||
|
|
||||||
download [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) and you're all set!
|
download [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) and you're all set!
|
||||||
|
|
||||||
running the sfx without arguments (for example doubleclicking it on Windows) will let anyone access the current folder; see `-h` for help if you want accounts and volumes etc
|
running the sfx without arguments (for example doubleclicking it on Windows) will give everyone full access to the current folder; see `-h` for help if you want accounts and volumes etc
|
||||||
|
|
||||||
you may also want these, especially on servers:
|
you may also want these, especially on servers:
|
||||||
* [contrib/systemd/copyparty.service](contrib/systemd/copyparty.service) to run copyparty as a systemd service
|
* [contrib/systemd/copyparty.service](contrib/systemd/copyparty.service) to run copyparty as a systemd service
|
||||||
* [contrib/nginx/copyparty.conf](contrib/nginx/copyparty.conf) to reverse-proxy behind nginx (for legit https)
|
* [contrib/nginx/copyparty.conf](contrib/nginx/copyparty.conf) to reverse-proxy behind nginx (for better https)
|
||||||
|
|
||||||
|
|
||||||
## notes
|
## 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
|
||||||
|
|
||||||
@@ -78,8 +93,8 @@ you may also want these, especially on servers:
|
|||||||
* ☑ tree-view
|
* ☑ tree-view
|
||||||
* ☑ media player
|
* ☑ media player
|
||||||
* ✖ thumbnails
|
* ✖ thumbnails
|
||||||
* ✖ SPA (browse while uploading)
|
* ☑ SPA (browse while uploading)
|
||||||
* currently safe using the file-tree on the left only, not folders in the file list
|
* if you use the file-tree on the left only, not folders in the file list
|
||||||
* server indexing
|
* server indexing
|
||||||
* ☑ locate files by contents
|
* ☑ locate files by contents
|
||||||
* ☑ search by name/path/date/size
|
* ☑ search by name/path/date/size
|
||||||
@@ -95,24 +110,145 @@ summary: it works! you can use it! (but technically not even close to beta)
|
|||||||
|
|
||||||
* Windows: python 3.7 and older cannot read tags with ffprobe, so use mutagen or upgrade
|
* Windows: python 3.7 and older cannot read tags with ffprobe, so use mutagen or upgrade
|
||||||
* Windows: python 2.7 cannot index non-ascii filenames with `-e2d`
|
* Windows: python 2.7 cannot index non-ascii filenames with `-e2d`
|
||||||
|
* Windows: python 2.7 cannot handle filenames with mojibake
|
||||||
|
|
||||||
|
## general bugs
|
||||||
|
|
||||||
|
* all volumes must exist / be available on startup; up2k (mtp especially) gets funky otherwise
|
||||||
|
* cannot mount something at `/d1/d2/d3` unless `d2` exists inside `d1`
|
||||||
|
* 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
|
||||||
|
|
||||||
|
|
||||||
|
## tree-mode
|
||||||
|
|
||||||
|
by default there's a breadcrumbs path; you can replace this with a tree-browser sidebar thing by clicking the 🌲
|
||||||
|
|
||||||
|
click `[-]` and `[+]` to adjust the size, and the `[a]` toggles if the tree should widen dynamically as you go deeper or stay fixed-size
|
||||||
|
|
||||||
|
|
||||||
|
## zip downloads
|
||||||
|
|
||||||
|
the `zip` link next to folders can produce various types of zip/tar files using these alternatives in the browser settings tab:
|
||||||
|
|
||||||
|
| name | url-suffix | description |
|
||||||
|
|--|--|--|
|
||||||
|
| `tar` | `?tar` | plain gnutar, works great with `curl \| tar -xv` |
|
||||||
|
| `zip` | `?zip=utf8` | works everywhere, glitchy filenames on win7 and older |
|
||||||
|
| `zip_dos` | `?zip` | traditional cp437 (no unicode) to fix glitchy filenames |
|
||||||
|
| `zip_crc` | `?zip=crc` | cp437 with crc32 computed early for truly ancient software |
|
||||||
|
|
||||||
|
* hidden files (dotfiles) are excluded unless `-ed`
|
||||||
|
* the up2k.db is always excluded
|
||||||
|
* `zip_crc` will take longer to download since the server has to read each file twice
|
||||||
|
* please let me know if you find a program old enough to actually need this
|
||||||
|
|
||||||
|
you can also zip a selection of files or folders by clicking them in the browser, that brings up a selection editor and zip button in the bottom right
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## uploading
|
||||||
|
|
||||||
|
two upload methods are available in the html client:
|
||||||
|
* `🎈 bup`, the basic uploader, supports almost every browser since netscape 4.0
|
||||||
|
* `🚀 up2k`, the fancy one
|
||||||
|
|
||||||
|
up2k has several advantages:
|
||||||
|
* you can drop folders into the browser (files are added recursively)
|
||||||
|
* files are processed in chunks, and each chunk is checksummed
|
||||||
|
* uploads resume if they are interrupted (for example by a reboot)
|
||||||
|
* server detects any corruption; the client reuploads affected chunks
|
||||||
|
* the client doesn't upload anything that already exists on the server
|
||||||
|
* the last-modified timestamp of the file is preserved
|
||||||
|
|
||||||
|
see [up2k](#up2k) for details on how it works
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
**protip:** you can avoid scaring away users with [docs/minimal-up2k.html](docs/minimal-up2k.html) which makes it look [much simpler](https://user-images.githubusercontent.com/241032/118311195-dd6ca380-b4ef-11eb-86f3-75a3ff2e1332.png)
|
||||||
|
|
||||||
|
the up2k UI is the epitome of polished inutitive experiences:
|
||||||
|
* "parallel uploads" specifies how many chunks to upload at the same time
|
||||||
|
* `[🏃]` analysis of other files should continue while one is uploading
|
||||||
|
* `[💭]` ask for confirmation before files are added to the list
|
||||||
|
* `[💤]` sync uploading between other copyparty tabs so only one is active
|
||||||
|
* `[🔎]` switch between upload and file-search mode
|
||||||
|
|
||||||
|
and then theres the tabs below it,
|
||||||
|
* `[ok]` is uploads which completed successfully
|
||||||
|
* `[ng]` is the uploads which failed / got rejected (already exists, ...)
|
||||||
|
* `[done]` shows a combined list of `[ok]` and `[ng]`, chronological order
|
||||||
|
* `[busy]` files which are currently hashing, pending-upload, or uploading
|
||||||
|
* plus up to 3 entries each from `[done]` and `[que]` for context
|
||||||
|
* `[que]` is all the files that are still queued
|
||||||
|
|
||||||
|
### file-search
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
in the `[🚀 up2k]` tab, after toggling the `[🔎]` switch green, any files/folders you drop onto the dropzone will be hashed on the client-side. Each hash is sent to the server which checks if that file exists somewhere already
|
||||||
|
|
||||||
|
files go into `[ok]` if they exist (and you get a link to where it is), otherwise they land in `[ng]`
|
||||||
|
* the main reason filesearch is combined with the uploader is cause the code was too spaghetti to separate it out somewhere else, this is no longer the case but now i've warmed up to the idea too much
|
||||||
|
|
||||||
|
adding the same file multiple times is blocked, so if you first search for a file and then decide to upload it, you have to click the `[cleanup]` button to discard `[done]` files
|
||||||
|
|
||||||
|
note that since up2k has to read the file twice, `[🎈 bup]` can be up to 2x faster in extreme cases (if your internet connection is faster than the read-speed of your HDD)
|
||||||
|
|
||||||
|
up2k has saved a few uploads from becoming corrupted in-transfer already; caught an android phone on wifi redhanded in wireshark with a bitflip, however bup with https would *probably* have noticed as well thanks to tls also functioning as an integrity check
|
||||||
|
|
||||||
|
|
||||||
|
## markdown viewer
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
* the document preview has a max-width which is the same as an A4 paper when printed
|
||||||
|
|
||||||
|
|
||||||
|
## other tricks
|
||||||
|
|
||||||
|
* you can link a particular timestamp in an audio file by adding it to the URL, such as `&20` / `&20s` / `&1m20` / `&t=1:20` after the `.../#af-c8960dab`
|
||||||
|
|
||||||
|
|
||||||
# searching
|
# 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
|
||||||
@@ -169,6 +305,10 @@ copyparty can invoke external programs to collect additional metadata for files
|
|||||||
* `-mtp key=f,t5,~/bin/audio-key.py` uses `~/bin/audio-key.py` to get the `key` tag, replacing any existing metadata tag (`f,`), aborting if it takes longer than 5sec (`t5,`)
|
* `-mtp key=f,t5,~/bin/audio-key.py` uses `~/bin/audio-key.py` to get the `key` tag, replacing any existing metadata tag (`f,`), aborting if it takes longer than 5sec (`t5,`)
|
||||||
* `-v ~/music::r:cmtp=.bpm=~/bin/audio-bpm.py:cmtp=key=f,t5,~/bin/audio-key.py` both as a per-volume config wow this is getting ugly
|
* `-v ~/music::r:cmtp=.bpm=~/bin/audio-bpm.py:cmtp=key=f,t5,~/bin/audio-key.py` both as a per-volume config wow this is getting ugly
|
||||||
|
|
||||||
|
*but wait, there's more!* `-mtp` can be used for non-audio files as well using the `a` flag: `ay` only do audio files, `an` only do non-audio files, or `ad` do all files (d as in dontcare)
|
||||||
|
|
||||||
|
* `-mtp ext=an,~/bin/file-ext.py` runs `~/bin/file-ext.py` to get the `ext` tag only if file is not audio (`an`)
|
||||||
|
|
||||||
|
|
||||||
## complete examples
|
## complete examples
|
||||||
|
|
||||||
@@ -176,6 +316,47 @@ copyparty can invoke external programs to collect additional metadata for files
|
|||||||
`python copyparty-sfx.py -v /mnt/nas/music:/music:r -e2dsa -e2ts -mtp .bpm=f,audio-bpm.py -mtp key=f,audio-key.py`
|
`python copyparty-sfx.py -v /mnt/nas/music:/music:r -e2dsa -e2ts -mtp .bpm=f,audio-bpm.py -mtp key=f,audio-key.py`
|
||||||
|
|
||||||
|
|
||||||
|
# browser support
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
`ie` = internet-explorer, `ff` = firefox, `c` = chrome, `iOS` = iPhone/iPad, `Andr` = Android
|
||||||
|
|
||||||
|
| feature | ie6 | ie9 | ie10 | ie11 | ff 52 | c 49 | iOS | Andr |
|
||||||
|
| --------------- | --- | --- | ---- | ---- | ----- | ---- | --- | ---- |
|
||||||
|
| browse files | yep | yep | yep | yep | yep | yep | yep | yep |
|
||||||
|
| basic uploader | yep | yep | yep | yep | yep | yep | yep | yep |
|
||||||
|
| make directory | yep | yep | yep | yep | yep | yep | yep | yep |
|
||||||
|
| send message | yep | yep | yep | yep | yep | yep | yep | yep |
|
||||||
|
| set sort order | - | yep | yep | yep | yep | yep | yep | yep |
|
||||||
|
| zip selection | - | yep | yep | yep | yep | yep | yep | yep |
|
||||||
|
| directory tree | - | - | `*1` | yep | yep | yep | yep | yep |
|
||||||
|
| up2k | - | - | yep | yep | yep | yep | yep | yep |
|
||||||
|
| icons work | - | - | yep | yep | yep | yep | yep | yep |
|
||||||
|
| markdown editor | - | - | yep | yep | yep | yep | yep | yep |
|
||||||
|
| markdown viewer | - | - | yep | yep | yep | yep | yep | yep |
|
||||||
|
| play mp3/m4a | - | yep | yep | yep | yep | yep | yep | yep |
|
||||||
|
| play ogg/opus | - | - | - | - | yep | yep | `*2` | yep |
|
||||||
|
|
||||||
|
* internet explorer 6 to 8 behave the same
|
||||||
|
* firefox 52 and chrome 49 are the last winxp versions
|
||||||
|
* `*1` only public folders (login session is dropped) and no history / back-button
|
||||||
|
* `*2` using a wasm decoder which can sometimes get stuck and consumes a bit more power
|
||||||
|
|
||||||
|
quick summary of more eccentric web-browsers trying to view a directory index:
|
||||||
|
|
||||||
|
| browser | will it blend |
|
||||||
|
| ------- | ------------- |
|
||||||
|
| **safari** (14.0.3/macos) | is chrome with janky wasm, so playing opus can deadlock the javascript engine |
|
||||||
|
| **safari** (14.0.1/iOS) | same as macos, except it recovers from the deadlocks if you poke it a bit |
|
||||||
|
| **links** (2.21/macports) | can browse, login, upload/mkdir/msg |
|
||||||
|
| **lynx** (2.8.9/macports) | can browse, login, upload/mkdir/msg |
|
||||||
|
| **w3m** (0.5.3/macports) | can browse, login, upload at 100kB/s, mkdir/msg |
|
||||||
|
| **netsurf** (3.10/arch) | is basically ie6 with much better css (javascript has almost no effect) |
|
||||||
|
| **ie4** and **netscape** 4.0 | can browse (text is yellow on white), upload with `?b=u` |
|
||||||
|
| **SerenityOS** (22d13d8) | hits a page fault, works with `?b=u`, file input not-impl, url params are multiplying |
|
||||||
|
|
||||||
|
|
||||||
# client examples
|
# client examples
|
||||||
|
|
||||||
* javascript: dump some state into a file (two separate examples)
|
* javascript: dump some state into a file (two separate examples)
|
||||||
@@ -200,6 +381,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 chunks or size >= 32M
|
||||||
|
* client posts the list of hashes, filename, size, last-modified
|
||||||
|
* server creates the `wark`, an identifier for this upload
|
||||||
|
* `sha512( salt + filesize + chunk_hashes )`
|
||||||
|
* and a sparse file is created for the chunks to drop into
|
||||||
|
* client uploads each chunk
|
||||||
|
* header entries for the chunk-hash and wark
|
||||||
|
* server writes chunks into place based on the hash
|
||||||
|
* client does another handshake with the hashlist; server replies with OK or a list of chunks to reupload
|
||||||
|
|
||||||
|
|
||||||
# dependencies
|
# dependencies
|
||||||
|
|
||||||
* `jinja2` (is built into the SFX)
|
* `jinja2` (is built into the SFX)
|
||||||
@@ -216,14 +413,14 @@ 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, **recommended**
|
||||||
* [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, kinda deprecated
|
||||||
|
|
||||||
launch either of them (**use sfx.py on systemd**) and it'll unpack and run copyparty, assuming you have python installed of course
|
launch either of them (**use sfx.py on systemd**) and it'll unpack and run copyparty, assuming you have python installed of course
|
||||||
|
|
||||||
@@ -283,15 +480,21 @@ in the `scripts` folder:
|
|||||||
|
|
||||||
roughly sorted by priority
|
roughly sorted by priority
|
||||||
|
|
||||||
|
* separate sqlite table per tag
|
||||||
|
* audio fingerprinting
|
||||||
|
* readme.md as epilogue
|
||||||
* reduce up2k roundtrips
|
* reduce up2k roundtrips
|
||||||
* start from a chunk index and just go
|
* start from a chunk index and just go
|
||||||
* terminate client on bad data
|
* terminate client on bad data
|
||||||
* drop onto folders
|
|
||||||
* `os.copy_file_range` for up2k cloning
|
* `os.copy_file_range` for up2k cloning
|
||||||
* up2k partials ui
|
|
||||||
* support pillow-simd
|
* support pillow-simd
|
||||||
* cache sha512 chunks on client
|
* single sha512 across all up2k chunks? maybe
|
||||||
* comment field
|
|
||||||
* ~~look into android thumbnail cache file format~~ bad idea
|
|
||||||
* figure out the deal with pixel3a not being connectable as hotspot
|
* figure out the deal with pixel3a not being connectable as hotspot
|
||||||
* pixel3a having unpredictable 3sec latency in general :||||
|
* pixel3a having unpredictable 3sec latency in general :||||
|
||||||
|
|
||||||
|
discarded ideas
|
||||||
|
|
||||||
|
* up2k partials ui
|
||||||
|
* cache sha512 chunks on client
|
||||||
|
* comment field
|
||||||
|
* look into android thumbnail cache file format
|
||||||
|
|||||||
9
bin/mtag/file-ext.py
Normal file
9
bin/mtag/file-ext.py
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
|
"""
|
||||||
|
example that just prints the file extension
|
||||||
|
"""
|
||||||
|
|
||||||
|
print(sys.argv[1].split(".")[-1])
|
||||||
@@ -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":
|
||||||
|
|||||||
@@ -12,7 +12,6 @@ import re
|
|||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
import time
|
import time
|
||||||
import signal
|
|
||||||
import shutil
|
import shutil
|
||||||
import filecmp
|
import filecmp
|
||||||
import locale
|
import locale
|
||||||
@@ -56,6 +55,12 @@ class RiceFormatter(argparse.HelpFormatter):
|
|||||||
return "".join(indent + line + "\n" for line in text.splitlines())
|
return "".join(indent + line + "\n" for line in text.splitlines())
|
||||||
|
|
||||||
|
|
||||||
|
class Dodge11874(RiceFormatter):
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
kwargs["width"] = 9003
|
||||||
|
super(Dodge11874, self).__init__(*args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
def warn(msg):
|
def warn(msg):
|
||||||
print("\033[1mwarning:\033[0;33m {}\033[0m\n".format(msg))
|
print("\033[1mwarning:\033[0;33m {}\033[0m\n".format(msg))
|
||||||
|
|
||||||
@@ -167,7 +172,7 @@ def configure_ssl_ciphers(al):
|
|||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
|
|
||||||
|
|
||||||
def sighandler(signal=None, frame=None):
|
def sighandler(sig=None, frame=None):
|
||||||
msg = [""] * 5
|
msg = [""] * 5
|
||||||
for th in threading.enumerate():
|
for th in threading.enumerate():
|
||||||
msg.append(str(th))
|
msg.append(str(th))
|
||||||
@@ -177,34 +182,9 @@ def sighandler(signal=None, frame=None):
|
|||||||
print("\n".join(msg))
|
print("\n".join(msg))
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def run_argparse(argv, formatter):
|
||||||
time.strptime("19970815", "%Y%m%d") # python#7980
|
|
||||||
if WINDOWS:
|
|
||||||
os.system("rem") # enables colors
|
|
||||||
|
|
||||||
desc = py_desc().replace("[", "\033[1;30m[")
|
|
||||||
|
|
||||||
f = '\033[36mcopyparty v{} "\033[35m{}\033[36m" ({})\n{}\033[0m\n'
|
|
||||||
print(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc))
|
|
||||||
|
|
||||||
ensure_locale()
|
|
||||||
if HAVE_SSL:
|
|
||||||
ensure_cert()
|
|
||||||
|
|
||||||
deprecated = [["-e2s", "-e2ds"]]
|
|
||||||
for dk, nk in deprecated:
|
|
||||||
try:
|
|
||||||
idx = sys.argv.index(dk)
|
|
||||||
except:
|
|
||||||
continue
|
|
||||||
|
|
||||||
msg = "\033[1;31mWARNING:\033[0;1m\n {} \033[0;33mwas replaced with\033[0;1m {} \033[0;33mand will be removed\n\033[0m"
|
|
||||||
print(msg.format(dk, nk))
|
|
||||||
sys.argv[idx] = nk
|
|
||||||
time.sleep(2)
|
|
||||||
|
|
||||||
ap = argparse.ArgumentParser(
|
ap = argparse.ArgumentParser(
|
||||||
formatter_class=RiceFormatter,
|
formatter_class=formatter,
|
||||||
prog="copyparty",
|
prog="copyparty",
|
||||||
description="http file sharing hub v{} ({})".format(S_VERSION, S_BUILD_DT),
|
description="http file sharing hub v{} ({})".format(S_VERSION, S_BUILD_DT),
|
||||||
epilog=dedent(
|
epilog=dedent(
|
||||||
@@ -216,6 +196,9 @@ def main():
|
|||||||
|
|
||||||
list of cflags:
|
list of cflags:
|
||||||
"cnodupe" rejects existing files (instead of symlinking them)
|
"cnodupe" rejects existing files (instead of symlinking them)
|
||||||
|
"ce2d" sets -e2d (all -e2* args can be set using ce2* cflags)
|
||||||
|
"cd2t" disables metadata collection, overrides -e2t*
|
||||||
|
"cd2d" disables all database stuff, overrides -e2*
|
||||||
|
|
||||||
example:\033[35m
|
example:\033[35m
|
||||||
-a ed:hunter2 -v .::r:aed -v ../inc:dump:w:aed:cnodupe \033[36m
|
-a ed:hunter2 -v .::r:aed -v ../inc:dump:w:aed:cnodupe \033[36m
|
||||||
@@ -254,16 +237,15 @@ def main():
|
|||||||
ap.add_argument("-a", metavar="ACCT", type=str, action="append", help="add account")
|
ap.add_argument("-a", metavar="ACCT", type=str, action="append", help="add account")
|
||||||
ap.add_argument("-v", metavar="VOL", type=str, action="append", help="add volume")
|
ap.add_argument("-v", metavar="VOL", type=str, action="append", help="add volume")
|
||||||
ap.add_argument("-q", action="store_true", help="quiet")
|
ap.add_argument("-q", action="store_true", help="quiet")
|
||||||
ap.add_argument("--log-conn", action="store_true", help="print tcp-server msgs")
|
|
||||||
ap.add_argument("-ed", action="store_true", help="enable ?dots")
|
ap.add_argument("-ed", action="store_true", help="enable ?dots")
|
||||||
ap.add_argument("-emp", action="store_true", help="enable markdown plugins")
|
ap.add_argument("-emp", action="store_true", help="enable markdown plugins")
|
||||||
ap.add_argument("-mcr", metavar="SEC", type=int, default=60, help="md-editor mod-chk rate")
|
ap.add_argument("-mcr", metavar="SEC", type=int, default=60, help="md-editor mod-chk rate")
|
||||||
ap.add_argument("-nw", action="store_true", help="disable writes (benchmark)")
|
ap.add_argument("-nw", action="store_true", help="disable writes (benchmark)")
|
||||||
ap.add_argument("-nih", action="store_true", help="no info hostname")
|
ap.add_argument("-nih", action="store_true", help="no info hostname")
|
||||||
ap.add_argument("-nid", action="store_true", help="no info disk-usage")
|
ap.add_argument("-nid", action="store_true", help="no info disk-usage")
|
||||||
|
ap.add_argument("--dotpart", action="store_true", help="dotfile incomplete uploads")
|
||||||
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("--sparse", metavar="MiB", type=int, default=4, help="up2k min.size threshold (mswin-only)")
|
||||||
ap.add_argument("--no-scandir", action="store_true", help="disable scandir (for debugging)")
|
|
||||||
ap.add_argument("--urlform", metavar="MODE", type=str, default="print,get", help="how to handle url-forms")
|
ap.add_argument("--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")
|
||||||
|
|
||||||
@@ -289,10 +271,52 @@ def main():
|
|||||||
ap2.add_argument("--ciphers", metavar="LIST", help="set allowed ciphers")
|
ap2.add_argument("--ciphers", metavar="LIST", help="set allowed ciphers")
|
||||||
ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info")
|
ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info")
|
||||||
ap2.add_argument("--ssl-log", metavar="PATH", help="log master secrets")
|
ap2.add_argument("--ssl-log", metavar="PATH", help="log master secrets")
|
||||||
|
|
||||||
|
ap2 = ap.add_argument_group('debug options')
|
||||||
|
ap2.add_argument("--log-conn", action="store_true", help="print tcp-server msgs")
|
||||||
|
ap2.add_argument("--no-sendfile", action="store_true", help="disable sendfile")
|
||||||
|
ap2.add_argument("--no-scandir", action="store_true", help="disable scandir")
|
||||||
|
ap2.add_argument("--ihead", metavar="HEADER", action='append', help="dump incoming header")
|
||||||
|
ap2.add_argument("--lf-url", metavar="RE", type=str, default=r"^/\.cpr/", help="dont log URLs matching")
|
||||||
|
|
||||||
al = ap.parse_args()
|
return ap.parse_args(args=argv[1:])
|
||||||
# fmt: on
|
# fmt: on
|
||||||
|
|
||||||
|
|
||||||
|
def main(argv=None):
|
||||||
|
time.strptime("19970815", "%Y%m%d") # python#7980
|
||||||
|
if WINDOWS:
|
||||||
|
os.system("rem") # enables colors
|
||||||
|
|
||||||
|
if argv is None:
|
||||||
|
argv = sys.argv
|
||||||
|
|
||||||
|
desc = py_desc().replace("[", "\033[1;30m[")
|
||||||
|
|
||||||
|
f = '\033[36mcopyparty v{} "\033[35m{}\033[36m" ({})\n{}\033[0m\n'
|
||||||
|
print(f.format(S_VERSION, CODENAME, S_BUILD_DT, desc))
|
||||||
|
|
||||||
|
ensure_locale()
|
||||||
|
if HAVE_SSL:
|
||||||
|
ensure_cert()
|
||||||
|
|
||||||
|
deprecated = [["-e2s", "-e2ds"]]
|
||||||
|
for dk, nk in deprecated:
|
||||||
|
try:
|
||||||
|
idx = argv.index(dk)
|
||||||
|
except:
|
||||||
|
continue
|
||||||
|
|
||||||
|
msg = "\033[1;31mWARNING:\033[0;1m\n {} \033[0;33mwas replaced with\033[0;1m {} \033[0;33mand will be removed\n\033[0m"
|
||||||
|
print(msg.format(dk, nk))
|
||||||
|
argv[idx] = nk
|
||||||
|
time.sleep(2)
|
||||||
|
|
||||||
|
try:
|
||||||
|
al = run_argparse(argv, RiceFormatter)
|
||||||
|
except AssertionError:
|
||||||
|
al = run_argparse(argv, Dodge11874)
|
||||||
|
|
||||||
# propagate implications
|
# propagate implications
|
||||||
for k1, k2 in IMPLICATIONS:
|
for k1, k2 in IMPLICATIONS:
|
||||||
if getattr(al, k1):
|
if getattr(al, k1):
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
|
|
||||||
VERSION = (0, 10, 1)
|
VERSION = (0, 10, 22)
|
||||||
CODENAME = "zip it"
|
CODENAME = "zip it"
|
||||||
BUILD_DT = (2021, 3, 27)
|
BUILD_DT = (2021, 5, 18)
|
||||||
|
|
||||||
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)
|
||||||
|
|||||||
@@ -111,7 +111,27 @@ class VFS(object):
|
|||||||
if rem:
|
if rem:
|
||||||
rp += "/" + rem
|
rp += "/" + rem
|
||||||
|
|
||||||
return fsdec(os.path.realpath(fsenc(rp)))
|
try:
|
||||||
|
return fsdec(os.path.realpath(fsenc(rp)))
|
||||||
|
except:
|
||||||
|
if not WINDOWS:
|
||||||
|
raise
|
||||||
|
|
||||||
|
# cpython bug introduced in 3.8, still exists in 3.9.1;
|
||||||
|
# some win7sp1 and win10:20H2 boxes cannot realpath a
|
||||||
|
# networked drive letter such as b"n:" or b"n:\\"
|
||||||
|
#
|
||||||
|
# requirements to trigger:
|
||||||
|
# * bytestring (not unicode str)
|
||||||
|
# * just the drive letter (subfolders are ok)
|
||||||
|
# * networked drive (regular disks and vmhgfs are ok)
|
||||||
|
# * on an enterprise network (idk, cannot repro with samba)
|
||||||
|
#
|
||||||
|
# hits the following exceptions in succession:
|
||||||
|
# * access denied at L601: "path = _getfinalpathname(path)"
|
||||||
|
# * "cant concat str to bytes" at L621: "return path + tail"
|
||||||
|
#
|
||||||
|
return os.path.realpath(rp)
|
||||||
|
|
||||||
def ls(self, rem, uname, scandir, lstat=False):
|
def ls(self, rem, uname, scandir, lstat=False):
|
||||||
"""return user-readable [fsdir,real,virt] items at vpath"""
|
"""return user-readable [fsdir,real,virt] items at vpath"""
|
||||||
@@ -121,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
|
||||||
@@ -161,47 +186,45 @@ class VFS(object):
|
|||||||
for x in vfs.walk(wrel, "", uname, scandir, lstat):
|
for x in vfs.walk(wrel, "", uname, scandir, lstat):
|
||||||
yield x
|
yield x
|
||||||
|
|
||||||
def zipgen(self, vrem, rems, uname, dots, scandir):
|
def zipgen(self, vrem, flt, uname, dots, scandir):
|
||||||
vtops = [["", [self, vrem]]]
|
if flt:
|
||||||
if rems:
|
flt = {k: True for k in flt}
|
||||||
# list of subfolders to zip was provided,
|
|
||||||
# add all the ones uname is allowed to access
|
|
||||||
vtops = []
|
|
||||||
for rem in rems:
|
|
||||||
try:
|
|
||||||
d = rem if not vrem else vrem + "/" + rem
|
|
||||||
vn = self.get(d, uname, True, False)
|
|
||||||
vtops.append([rem, vn])
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
|
|
||||||
for rel, (vn, rem) in vtops:
|
for vpath, apath, files, rd, vd in self.walk("", vrem, uname, dots, scandir):
|
||||||
for vpath, apath, files, rd, vd in vn.walk(rel, rem, uname, dots, scandir):
|
if flt:
|
||||||
# print(repr([vpath, apath, [x[0] for x in files]]))
|
files = [x for x in files if x[0] in flt]
|
||||||
fnames = [n[0] for n in files]
|
|
||||||
vpaths = [vpath + "/" + n for n in fnames] if vpath else fnames
|
|
||||||
apaths = [os.path.join(apath, n) for n in fnames]
|
|
||||||
files = list(zip(vpaths, apaths, files))
|
|
||||||
|
|
||||||
if not dots:
|
rm = [x for x in rd if x[0] not in flt]
|
||||||
# dotfile filtering based on vpath (intended visibility)
|
[rd.remove(x) for x in rm]
|
||||||
files = [x for x in files if "/." not in "/" + x[0]]
|
|
||||||
|
|
||||||
rm = [x for x in rd if x[0].startswith(".")]
|
rm = [x for x in vd.keys() if x not in flt]
|
||||||
for x in rm:
|
[vd.pop(x) for x in rm]
|
||||||
rd.remove(x)
|
|
||||||
|
|
||||||
rm = [k for k in vd.keys() if k.startswith(".")]
|
flt = None
|
||||||
for x in rm:
|
|
||||||
del vd[x]
|
|
||||||
|
|
||||||
# up2k filetring based on actual abspath
|
# print(repr([vpath, apath, [x[0] for x in files]]))
|
||||||
files = [
|
fnames = [n[0] for n in files]
|
||||||
x for x in files if "{0}.hist{0}up2k.".format(os.sep) not in x[1]
|
vpaths = [vpath + "/" + n for n in fnames] if vpath else fnames
|
||||||
]
|
apaths = [os.path.join(apath, n) for n in fnames]
|
||||||
|
files = list(zip(vpaths, apaths, files))
|
||||||
|
|
||||||
for f in [{"vp": v, "ap": a, "st": n[1]} for v, a, n in files]:
|
if not dots:
|
||||||
yield f
|
# dotfile filtering based on vpath (intended visibility)
|
||||||
|
files = [x for x in files if "/." not in "/" + x[0]]
|
||||||
|
|
||||||
|
rm = [x for x in rd if x[0].startswith(".")]
|
||||||
|
for x in rm:
|
||||||
|
rd.remove(x)
|
||||||
|
|
||||||
|
rm = [k for k in vd.keys() if k.startswith(".")]
|
||||||
|
for x in rm:
|
||||||
|
del vd[x]
|
||||||
|
|
||||||
|
# up2k filetring based on actual abspath
|
||||||
|
files = [x for x in files if "{0}.hist{0}up2k.".format(os.sep) not in x[1]]
|
||||||
|
|
||||||
|
for f in [{"vp": v, "ap": a, "st": n[1]} for v, a, n in files]:
|
||||||
|
yield f
|
||||||
|
|
||||||
def user_tree(self, uname, readable=False, writable=False):
|
def user_tree(self, uname, readable=False, writable=False):
|
||||||
ret = []
|
ret = []
|
||||||
@@ -223,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]:[\\/][^:]*|[^:]*):([^:]*):(.*)$")
|
||||||
@@ -235,12 +259,6 @@ class AuthSrv(object):
|
|||||||
def log(self, msg, c=0):
|
def log(self, msg, c=0):
|
||||||
self.log_func("auth", msg, c)
|
self.log_func("auth", msg, c)
|
||||||
|
|
||||||
def invert(self, orig):
|
|
||||||
if PY2:
|
|
||||||
return {v: k for k, v in orig.iteritems()}
|
|
||||||
else:
|
|
||||||
return {v: k for k, v in orig.items()}
|
|
||||||
|
|
||||||
def laggy_iter(self, iterable):
|
def laggy_iter(self, iterable):
|
||||||
"""returns [value,isFinalValue]"""
|
"""returns [value,isFinalValue]"""
|
||||||
it = iter(iterable)
|
it = iter(iterable)
|
||||||
@@ -254,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
|
||||||
@@ -284,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]
|
||||||
)
|
)
|
||||||
@@ -362,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
|
||||||
@@ -497,7 +527,7 @@ class AuthSrv(object):
|
|||||||
with self.mutex:
|
with self.mutex:
|
||||||
self.vfs = vfs
|
self.vfs = vfs
|
||||||
self.user = user
|
self.user = user
|
||||||
self.iuser = self.invert(user)
|
self.iuser = {v: k for k, v in user.items()}
|
||||||
|
|
||||||
# import pprint
|
# import pprint
|
||||||
# pprint.pprint({"usr": user, "rd": mread, "wr": mwrite, "mnt": mount})
|
# pprint.pprint({"usr": user, "rd": mread, "wr": mwrite, "mnt": mount})
|
||||||
|
|||||||
@@ -51,7 +51,7 @@ class BrokerMp(object):
|
|||||||
self.procs.append(proc)
|
self.procs.append(proc)
|
||||||
proc.start()
|
proc.start()
|
||||||
|
|
||||||
if True:
|
if not self.args.q:
|
||||||
thr = threading.Thread(target=self.debug_load_balancer)
|
thr = threading.Thread(target=self.debug_load_balancer)
|
||||||
thr.daemon = True
|
thr.daemon = True
|
||||||
thr.start()
|
thr.start()
|
||||||
|
|||||||
@@ -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
|
||||||
@@ -74,7 +74,7 @@ class HttpCli(object):
|
|||||||
headerlines.pop(0)
|
headerlines.pop(0)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.mode, self.req, _ = headerlines[0].split(" ")
|
self.mode, self.req, self.http_ver = headerlines[0].split(" ")
|
||||||
except:
|
except:
|
||||||
raise Pebkac(400, "bad headers:\n" + "\n".join(headerlines))
|
raise Pebkac(400, "bad headers:\n" + "\n".join(headerlines))
|
||||||
|
|
||||||
@@ -93,29 +93,22 @@ class HttpCli(object):
|
|||||||
self.headers[k.lower()] = v.strip()
|
self.headers[k.lower()] = v.strip()
|
||||||
|
|
||||||
v = self.headers.get("connection", "").lower()
|
v = self.headers.get("connection", "").lower()
|
||||||
self.keepalive = not v.startswith("close")
|
self.keepalive = not v.startswith("close") and self.http_ver != "HTTP/1.0"
|
||||||
|
|
||||||
v = self.headers.get("x-forwarded-for", None)
|
v = self.headers.get("x-forwarded-for", None)
|
||||||
if v is not None and self.conn.addr[0] in ["127.0.0.1", "::1"]:
|
if v is not None and self.conn.addr[0] in ["127.0.0.1", "::1"]:
|
||||||
self.ip = v.split(",")[0]
|
self.ip = v.split(",")[0]
|
||||||
self.log_src = self.conn.set_rproxy(self.ip)
|
self.log_src = self.conn.set_rproxy(self.ip)
|
||||||
|
|
||||||
self.uname = "*"
|
if self.args.ihead:
|
||||||
if "cookie" in self.headers:
|
keys = self.args.ihead
|
||||||
cookies = self.headers["cookie"].split(";")
|
if "*" in keys:
|
||||||
for k, v in [x.split("=", 1) for x in cookies]:
|
keys = list(sorted(self.headers.keys()))
|
||||||
if k.strip() != "cppwd":
|
|
||||||
continue
|
|
||||||
|
|
||||||
v = unescape_cookie(v)
|
for k in keys:
|
||||||
if v in self.auth.iuser:
|
v = self.headers.get(k)
|
||||||
self.uname = self.auth.iuser[v]
|
if v is not None:
|
||||||
|
self.log("[H] {}: \033[33m[{}]".format(k, v), 6)
|
||||||
break
|
|
||||||
|
|
||||||
if self.uname:
|
|
||||||
self.rvol = self.auth.vfs.user_tree(self.uname, readable=True)
|
|
||||||
self.wvol = self.auth.vfs.user_tree(self.uname, writable=True)
|
|
||||||
|
|
||||||
# split req into vpath + uparam
|
# split req into vpath + uparam
|
||||||
uparam = {}
|
uparam = {}
|
||||||
@@ -137,13 +130,35 @@ class HttpCli(object):
|
|||||||
else:
|
else:
|
||||||
uparam[k.lower()] = False
|
uparam[k.lower()] = False
|
||||||
|
|
||||||
|
self.ouparam = {k: v for k, v in uparam.items()}
|
||||||
|
|
||||||
|
cookies = self.headers.get("cookie") or {}
|
||||||
|
if cookies:
|
||||||
|
cookies = [x.split("=", 1) for x in cookies.split(";") if "=" in x]
|
||||||
|
cookies = {k.strip(): unescape_cookie(v) for k, v in cookies}
|
||||||
|
for kc, ku in [["cppwd", "pw"], ["b", "b"]]:
|
||||||
|
if kc in cookies and ku not in uparam:
|
||||||
|
uparam[ku] = cookies[kc]
|
||||||
|
|
||||||
self.uparam = uparam
|
self.uparam = uparam
|
||||||
|
self.cookies = cookies
|
||||||
self.vpath = unquotep(vpath)
|
self.vpath = unquotep(vpath)
|
||||||
|
|
||||||
|
pwd = uparam.get("pw")
|
||||||
|
self.uname = self.auth.iuser.get(pwd, "*")
|
||||||
|
if self.uname:
|
||||||
|
self.rvol = self.auth.vfs.user_tree(self.uname, readable=True)
|
||||||
|
self.wvol = self.auth.vfs.user_tree(self.uname, writable=True)
|
||||||
|
|
||||||
ua = self.headers.get("user-agent", "")
|
ua = self.headers.get("user-agent", "")
|
||||||
if ua.startswith("rclone/"):
|
self.is_rclone = ua.startswith("rclone/")
|
||||||
|
if self.is_rclone:
|
||||||
uparam["raw"] = False
|
uparam["raw"] = False
|
||||||
uparam["dots"] = False
|
uparam["dots"] = False
|
||||||
|
uparam["b"] = False
|
||||||
|
cookies["b"] = False
|
||||||
|
|
||||||
|
self.do_log = not self.conn.lf_url or not self.conn.lf_url.match(self.req)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if self.mode in ["GET", "HEAD"]:
|
if self.mode in ["GET", "HEAD"]:
|
||||||
@@ -160,16 +175,18 @@ class HttpCli(object):
|
|||||||
except Pebkac as ex:
|
except Pebkac as ex:
|
||||||
try:
|
try:
|
||||||
# self.log("pebkac at httpcli.run #2: " + repr(ex))
|
# self.log("pebkac at httpcli.run #2: " + repr(ex))
|
||||||
self.keepalive = self._check_nonfatal(ex)
|
if not self._check_nonfatal(ex):
|
||||||
self.log("{}\033[0m: {}".format(str(ex), self.vpath), 3)
|
self.keepalive = False
|
||||||
msg = "<pre>{}: {}\r\n".format(str(ex), self.vpath)
|
|
||||||
|
self.log("{}\033[0m, {}".format(str(ex), self.vpath), 3)
|
||||||
|
msg = "<pre>{}\r\nURL: {}\r\n".format(str(ex), self.vpath)
|
||||||
self.reply(msg.encode("utf-8", "replace"), status=ex.code)
|
self.reply(msg.encode("utf-8", "replace"), status=ex.code)
|
||||||
return self.keepalive
|
return self.keepalive
|
||||||
except Pebkac:
|
except Pebkac:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def send_headers(self, length, status=200, mime=None, headers={}):
|
def send_headers(self, length, status=200, mime=None, headers={}):
|
||||||
response = ["HTTP/1.1 {} {}".format(status, HTTPCODE[status])]
|
response = ["{} {} {}".format(self.http_ver, status, HTTPCODE[status])]
|
||||||
|
|
||||||
if length is not None:
|
if length is not None:
|
||||||
response.append("Content-Length: " + unicode(length))
|
response.append("Content-Length: " + unicode(length))
|
||||||
@@ -181,10 +198,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
|
||||||
|
|
||||||
@@ -213,18 +228,58 @@ class HttpCli(object):
|
|||||||
self.log(body.rstrip())
|
self.log(body.rstrip())
|
||||||
self.reply(b"<pre>" + body.encode("utf-8") + b"\r\n", *list(args), **kwargs)
|
self.reply(b"<pre>" + body.encode("utf-8") + b"\r\n", *list(args), **kwargs)
|
||||||
|
|
||||||
|
def urlq(self, add={}, rm=[]):
|
||||||
|
"""
|
||||||
|
generates url query based on uparam (b, pw, all others)
|
||||||
|
removing anything in rm, adding pairs in add
|
||||||
|
"""
|
||||||
|
|
||||||
|
if self.is_rclone:
|
||||||
|
return ""
|
||||||
|
|
||||||
|
kv = {
|
||||||
|
k: v
|
||||||
|
for k, v in self.uparam.items()
|
||||||
|
if k not in rm and self.cookies.get(k) != v
|
||||||
|
}
|
||||||
|
kv.update(add)
|
||||||
|
if not kv:
|
||||||
|
return ""
|
||||||
|
|
||||||
|
r = ["{}={}".format(k, quotep(v)) if v else k for k, v in kv.items()]
|
||||||
|
return "?" + "&".join(r)
|
||||||
|
|
||||||
|
def redirect(
|
||||||
|
self, vpath, suf="", msg="aight", flavor="go to", click=True, use302=False
|
||||||
|
):
|
||||||
|
html = self.j2(
|
||||||
|
"msg",
|
||||||
|
h2='<a href="/{}">{} /{}</a>'.format(
|
||||||
|
quotep(vpath) + suf, flavor, html_escape(vpath, crlf=True) + suf
|
||||||
|
),
|
||||||
|
pre=msg,
|
||||||
|
click=click,
|
||||||
|
).encode("utf-8", "replace")
|
||||||
|
|
||||||
|
if use302:
|
||||||
|
h = {"Location": "/" + vpath, "Cache-Control": "no-cache"}
|
||||||
|
self.reply(html, status=302, headers=h)
|
||||||
|
else:
|
||||||
|
self.reply(html)
|
||||||
|
|
||||||
def handle_get(self):
|
def handle_get(self):
|
||||||
logmsg = "{:4} {}".format(self.mode, self.req)
|
if self.do_log:
|
||||||
|
logmsg = "{:4} {}".format(self.mode, self.req)
|
||||||
|
|
||||||
if "range" in self.headers:
|
if "range" in self.headers:
|
||||||
try:
|
try:
|
||||||
rval = self.headers["range"].split("=", 1)[1]
|
rval = self.headers["range"].split("=", 1)[1]
|
||||||
except:
|
except:
|
||||||
rval = self.headers["range"]
|
rval = self.headers["range"]
|
||||||
|
|
||||||
logmsg += " [\033[36m" + rval + "\033[0m]"
|
logmsg += " [\033[36m" + rval + "\033[0m]"
|
||||||
|
|
||||||
self.log(logmsg)
|
self.log(logmsg)
|
||||||
|
|
||||||
# "embedded" resources
|
# "embedded" resources
|
||||||
if self.vpath.startswith(".cpr"):
|
if self.vpath.startswith(".cpr"):
|
||||||
@@ -235,23 +290,27 @@ class HttpCli(object):
|
|||||||
return self.tx_tree()
|
return self.tx_tree()
|
||||||
|
|
||||||
# conditional redirect to single volumes
|
# conditional redirect to single volumes
|
||||||
if self.vpath == "" and not self.uparam:
|
if self.vpath == "" and not self.ouparam:
|
||||||
nread = len(self.rvol)
|
nread = len(self.rvol)
|
||||||
nwrite = len(self.wvol)
|
nwrite = len(self.wvol)
|
||||||
if nread + nwrite == 1 or (self.rvol == self.wvol and nread == 1):
|
if nread + nwrite == 1 or (self.rvol == self.wvol and nread == 1):
|
||||||
if nread == 1:
|
if nread == 1:
|
||||||
self.vpath = self.rvol[0]
|
vpath = self.rvol[0]
|
||||||
else:
|
else:
|
||||||
self.vpath = self.wvol[0]
|
vpath = self.wvol[0]
|
||||||
|
|
||||||
self.absolute_urls = True
|
if self.vpath != vpath:
|
||||||
|
self.redirect(vpath, flavor="redirecting to", use302=True)
|
||||||
|
return 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:
|
||||||
@@ -261,7 +320,9 @@ class HttpCli(object):
|
|||||||
return self.tx_browser()
|
return self.tx_browser()
|
||||||
|
|
||||||
def handle_options(self):
|
def handle_options(self):
|
||||||
self.log("OPTIONS " + self.req)
|
if self.do_log:
|
||||||
|
self.log("OPTIONS " + self.req)
|
||||||
|
|
||||||
self.send_headers(
|
self.send_headers(
|
||||||
None,
|
None,
|
||||||
204,
|
204,
|
||||||
@@ -321,8 +382,19 @@ class HttpCli(object):
|
|||||||
elif "print" in opt:
|
elif "print" in opt:
|
||||||
reader, _ = self.get_body_reader()
|
reader, _ = self.get_body_reader()
|
||||||
for buf in reader:
|
for buf in reader:
|
||||||
buf = buf.decode("utf-8", "replace")
|
orig = buf.decode("utf-8", "replace")
|
||||||
self.log("urlform @ {}\n {}\n".format(self.vpath, buf))
|
m = "urlform_raw {} @ {}\n {}\n"
|
||||||
|
self.log(m.format(len(orig), self.vpath, orig))
|
||||||
|
try:
|
||||||
|
plain = unquote(buf.replace(b"+", b" "))
|
||||||
|
plain = plain.decode("utf-8", "replace")
|
||||||
|
if buf.startswith(b"msg="):
|
||||||
|
plain = plain[4:]
|
||||||
|
|
||||||
|
m = "urlform_dec {} @ {}\n {}\n"
|
||||||
|
self.log(m.format(len(plain), self.vpath, plain))
|
||||||
|
except Exception as ex:
|
||||||
|
self.log(repr(ex))
|
||||||
|
|
||||||
if "get" in opt:
|
if "get" in opt:
|
||||||
return self.handle_get()
|
return self.handle_get()
|
||||||
@@ -397,8 +469,30 @@ class HttpCli(object):
|
|||||||
if act == "tput":
|
if act == "tput":
|
||||||
return self.handle_text_upload()
|
return self.handle_text_upload()
|
||||||
|
|
||||||
|
if act == "zip":
|
||||||
|
return self.handle_zip_post()
|
||||||
|
|
||||||
raise Pebkac(422, 'invalid action "{}"'.format(act))
|
raise Pebkac(422, 'invalid action "{}"'.format(act))
|
||||||
|
|
||||||
|
def handle_zip_post(self):
|
||||||
|
for k in ["zip", "tar"]:
|
||||||
|
v = self.uparam.get(k)
|
||||||
|
if v is not None:
|
||||||
|
break
|
||||||
|
|
||||||
|
if v is None:
|
||||||
|
raise Pebkac(422, "need zip or tar keyword")
|
||||||
|
|
||||||
|
vn, rem = self.auth.vfs.get(self.vpath, self.uname, True, False)
|
||||||
|
items = self.parser.require("files", 1024 * 1024)
|
||||||
|
if not items:
|
||||||
|
raise Pebkac(422, "need files list")
|
||||||
|
|
||||||
|
items = items.replace("\r", "").split("\n")
|
||||||
|
items = [unquotep(x) for x in items if items]
|
||||||
|
|
||||||
|
return self.tx_zip(k, v, vn, rem, items, self.args.ed)
|
||||||
|
|
||||||
def handle_post_json(self):
|
def handle_post_json(self):
|
||||||
try:
|
try:
|
||||||
remains = int(self.headers["content-length"])
|
remains = int(self.headers["content-length"])
|
||||||
@@ -486,7 +580,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))
|
||||||
@@ -578,7 +672,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:
|
||||||
@@ -597,13 +691,16 @@ class HttpCli(object):
|
|||||||
|
|
||||||
if pwd in self.auth.iuser:
|
if pwd in self.auth.iuser:
|
||||||
msg = "login ok"
|
msg = "login ok"
|
||||||
|
dt = datetime.utcfromtimestamp(time.time() + 60 * 60 * 24 * 365)
|
||||||
|
exp = dt.strftime("%a, %d %b %Y %H:%M:%S GMT")
|
||||||
else:
|
else:
|
||||||
msg = "naw dude"
|
msg = "naw dude"
|
||||||
pwd = "x" # nosec
|
pwd = "x" # nosec
|
||||||
|
exp = "Fri, 15 Aug 1997 01:00:00 GMT"
|
||||||
|
|
||||||
h = {"Set-Cookie": "cppwd={}; Path=/; SameSite=Lax".format(pwd)}
|
ck = "cppwd={}; Path=/; Expires={}; SameSite=Lax".format(pwd, exp)
|
||||||
html = self.j2("msg", h1=msg, h2='<a href="/">ack</a>', redir="/")
|
html = self.j2("msg", h1=msg, h2='<a href="/">ack</a>', redir="/")
|
||||||
self.reply(html.encode("utf-8"), headers=h)
|
self.reply(html.encode("utf-8"), headers={"Set-Cookie": ck})
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def handle_mkdir(self):
|
def handle_mkdir(self):
|
||||||
@@ -632,14 +729,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)]
|
self.redirect(vpath)
|
||||||
html = self.j2(
|
|
||||||
"msg",
|
|
||||||
h2='<a href="/{}">go to /{}</a>'.format(*esc_paths),
|
|
||||||
pre="aight",
|
|
||||||
click=True,
|
|
||||||
)
|
|
||||||
self.reply(html.encode("utf-8", "replace"))
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def handle_new_md(self):
|
def handle_new_md(self):
|
||||||
@@ -666,15 +756,7 @@ class HttpCli(object):
|
|||||||
f.write(b"`GRUNNUR`\n")
|
f.write(b"`GRUNNUR`\n")
|
||||||
|
|
||||||
vpath = "{}/{}".format(self.vpath, sanitized).lstrip("/")
|
vpath = "{}/{}".format(self.vpath, sanitized).lstrip("/")
|
||||||
html = self.j2(
|
self.redirect(vpath, "?edit")
|
||||||
"msg",
|
|
||||||
h2='<a href="/{}?edit">go to /{}?edit</a>'.format(
|
|
||||||
quotep(vpath), html_escape(vpath)
|
|
||||||
),
|
|
||||||
pre="aight",
|
|
||||||
click=True,
|
|
||||||
)
|
|
||||||
self.reply(html.encode("utf-8", "replace"))
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def handle_plain_upload(self):
|
def handle_plain_upload(self):
|
||||||
@@ -693,7 +775,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")
|
||||||
@@ -713,7 +797,7 @@ class HttpCli(object):
|
|||||||
if sz == 0:
|
if sz == 0:
|
||||||
raise Pebkac(400, "empty files in post")
|
raise Pebkac(400, "empty files in post")
|
||||||
|
|
||||||
files.append([sz, sha512_hex])
|
files.append([sz, sha512_hex, p_file, fname])
|
||||||
self.conn.hsrv.broker.put(
|
self.conn.hsrv.broker.put(
|
||||||
False, "up2k.hash_file", vfs.realpath, vfs.flags, rem, fname
|
False, "up2k.hash_file", vfs.realpath, vfs.flags, rem, fname
|
||||||
)
|
)
|
||||||
@@ -722,12 +806,16 @@ class HttpCli(object):
|
|||||||
except Pebkac:
|
except Pebkac:
|
||||||
if fname != os.devnull:
|
if fname != os.devnull:
|
||||||
fp = os.path.join(fdir, fname)
|
fp = os.path.join(fdir, fname)
|
||||||
|
fp2 = fp
|
||||||
|
if self.args.dotpart:
|
||||||
|
fp2 = os.path.join(fdir, "." + fname)
|
||||||
|
|
||||||
suffix = ".PARTIAL"
|
suffix = ".PARTIAL"
|
||||||
try:
|
try:
|
||||||
os.rename(fsenc(fp), fsenc(fp + suffix))
|
os.rename(fsenc(fp), fsenc(fp2 + suffix))
|
||||||
except:
|
except:
|
||||||
fp = fp[: -len(suffix)]
|
fp2 = fp2[: -len(suffix) - 1]
|
||||||
os.rename(fsenc(fp), fsenc(fp + suffix))
|
os.rename(fsenc(fp), fsenc(fp2 + suffix))
|
||||||
|
|
||||||
raise
|
raise
|
||||||
|
|
||||||
@@ -744,10 +832,13 @@ class HttpCli(object):
|
|||||||
errmsg = "ERROR: " + errmsg
|
errmsg = "ERROR: " + errmsg
|
||||||
status = "ERROR"
|
status = "ERROR"
|
||||||
|
|
||||||
msg = "{0} // {1} bytes // {2:.3f} MiB/s\n".format(status, sz_total, spd)
|
msg = "{} // {} bytes // {:.3f} MiB/s\n".format(status, sz_total, spd)
|
||||||
|
|
||||||
for sz, sha512 in files:
|
for sz, sha512, ofn, lfn in files:
|
||||||
msg += "sha512: {0} // {1} bytes\n".format(sha512[:56], sz)
|
vpath = self.vpath + "/" + lfn
|
||||||
|
msg += 'sha512: {} // {} bytes // <a href="/{}">{}</a>\n'.format(
|
||||||
|
sha512[:56], sz, quotep(vpath), html_escape(ofn, crlf=True)
|
||||||
|
)
|
||||||
# truncated SHA-512 prevents length extension attacks;
|
# truncated SHA-512 prevents length extension attacks;
|
||||||
# using SHA-512/224, optionally SHA-512/256 = :64
|
# using SHA-512/224, optionally SHA-512/256 = :64
|
||||||
|
|
||||||
@@ -755,32 +846,13 @@ class HttpCli(object):
|
|||||||
self.log("{} {}".format(vspd, msg))
|
self.log("{} {}".format(vspd, msg))
|
||||||
|
|
||||||
if not nullwrite:
|
if not nullwrite:
|
||||||
# TODO this is bad
|
|
||||||
log_fn = "up.{:.6f}.txt".format(t0)
|
log_fn = "up.{:.6f}.txt".format(t0)
|
||||||
with open(log_fn, "wb") as f:
|
with open(log_fn, "wb") as f:
|
||||||
f.write(
|
ft = "{}:{}".format(self.ip, self.addr[1])
|
||||||
(
|
ft = "{}\n{}\n{}\n".format(ft, msg.rstrip(), errmsg)
|
||||||
"\n".join(
|
f.write(ft.encode("utf-8"))
|
||||||
unicode(x)
|
|
||||||
for x in [
|
|
||||||
":".join(unicode(x) for x in [self.ip, self.addr[1]]),
|
|
||||||
msg.rstrip(),
|
|
||||||
]
|
|
||||||
)
|
|
||||||
+ "\n"
|
|
||||||
+ errmsg
|
|
||||||
+ "\n"
|
|
||||||
).encode("utf-8")
|
|
||||||
)
|
|
||||||
|
|
||||||
html = self.j2(
|
self.redirect(self.vpath, msg=msg, flavor="return to", click=False)
|
||||||
"msg",
|
|
||||||
h2='<a href="/{}">return to /{}</a>'.format(
|
|
||||||
quotep(self.vpath), html_escape(self.vpath)
|
|
||||||
),
|
|
||||||
pre=msg,
|
|
||||||
)
|
|
||||||
self.reply(html.encode("utf-8", "replace"))
|
|
||||||
self.parser.drop()
|
self.parser.drop()
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@@ -880,13 +952,14 @@ class HttpCli(object):
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
def _chk_lastmod(self, file_ts):
|
def _chk_lastmod(self, file_ts):
|
||||||
|
date_fmt = "%a, %d %b %Y %H:%M:%S GMT"
|
||||||
file_dt = datetime.utcfromtimestamp(file_ts)
|
file_dt = datetime.utcfromtimestamp(file_ts)
|
||||||
file_lastmod = file_dt.strftime("%a, %d %b %Y %H:%M:%S GMT")
|
file_lastmod = file_dt.strftime(date_fmt)
|
||||||
|
|
||||||
cli_lastmod = self.headers.get("if-modified-since")
|
cli_lastmod = self.headers.get("if-modified-since")
|
||||||
if cli_lastmod:
|
if cli_lastmod:
|
||||||
try:
|
try:
|
||||||
cli_dt = time.strptime(cli_lastmod, "%a, %d %b %Y %H:%M:%S GMT")
|
cli_dt = time.strptime(cli_lastmod, date_fmt)
|
||||||
cli_ts = calendar.timegm(cli_dt)
|
cli_ts = calendar.timegm(cli_dt)
|
||||||
return file_lastmod, int(file_ts) > int(cli_ts)
|
return file_lastmod, int(file_ts) > int(cli_ts)
|
||||||
except Exception as ex:
|
except Exception as ex:
|
||||||
@@ -1045,7 +1118,9 @@ class HttpCli(object):
|
|||||||
logmsg += unicode(status) + logtail
|
logmsg += unicode(status) + logtail
|
||||||
|
|
||||||
if self.mode == "HEAD" or not do_send:
|
if self.mode == "HEAD" or not do_send:
|
||||||
self.log(logmsg)
|
if self.do_log:
|
||||||
|
self.log(logmsg)
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
ret = True
|
ret = True
|
||||||
@@ -1059,7 +1134,9 @@ class HttpCli(object):
|
|||||||
logmsg += " \033[31m" + unicode(upper - remains) + "\033[0m"
|
logmsg += " \033[31m" + unicode(upper - remains) + "\033[0m"
|
||||||
|
|
||||||
spd = self._spd((upper - lower) - remains)
|
spd = self._spd((upper - lower) - remains)
|
||||||
self.log("{}, {}".format(logmsg, spd))
|
if self.do_log:
|
||||||
|
self.log("{}, {}".format(logmsg, spd))
|
||||||
|
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
def tx_zip(self, fmt, uarg, vn, rem, items, dots):
|
def tx_zip(self, fmt, uarg, vn, rem, items, dots):
|
||||||
@@ -1133,17 +1210,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)
|
||||||
@@ -1151,38 +1227,50 @@ 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)
|
if self.do_log:
|
||||||
|
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
|
||||||
|
|
||||||
self.log(logmsg + " " + unicode(len(html)))
|
if self.do_log:
|
||||||
|
self.log(logmsg + " " + unicode(len(html)))
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def tx_mounts(self):
|
def tx_mounts(self):
|
||||||
|
suf = self.urlq(rm=["h"])
|
||||||
rvol = [x + "/" if x else x for x in self.rvol]
|
rvol = [x + "/" if x else x for x in self.rvol]
|
||||||
wvol = [x + "/" if x else x for x in self.wvol]
|
wvol = [x + "/" if x else x for x in self.wvol]
|
||||||
html = self.j2("splash", this=self, rvol=rvol, wvol=wvol)
|
html = self.j2("splash", this=self, rvol=rvol, wvol=wvol, url_suf=suf)
|
||||||
self.reply(html.encode("utf-8"))
|
self.reply(html.encode("utf-8"))
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@@ -1251,7 +1339,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
|
||||||
@@ -1262,6 +1350,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)
|
||||||
@@ -1305,8 +1481,6 @@ 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()
|
||||||
@@ -1343,7 +1517,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 = "-"
|
||||||
@@ -1402,85 +1576,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)
|
||||||
|
|
||||||
html = self.j2(
|
|
||||||
"browser",
|
|
||||||
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),
|
|
||||||
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
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import re
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
import time
|
import time
|
||||||
@@ -38,6 +39,7 @@ class HttpConn(object):
|
|||||||
self.workload = 0
|
self.workload = 0
|
||||||
self.u2idx = None
|
self.u2idx = None
|
||||||
self.log_func = hsrv.log
|
self.log_func = hsrv.log
|
||||||
|
self.lf_url = re.compile(self.args.lf_url) if self.args.lf_url else None
|
||||||
self.set_rproxy()
|
self.set_rproxy()
|
||||||
|
|
||||||
def set_rproxy(self, ip=None):
|
def set_rproxy(self, ip=None):
|
||||||
@@ -87,7 +89,9 @@ class HttpConn(object):
|
|||||||
err = "need at least 4 bytes in the first packet; got {}".format(
|
err = "need at least 4 bytes in the first packet; got {}".format(
|
||||||
len(method)
|
len(method)
|
||||||
)
|
)
|
||||||
self.log(err)
|
if method:
|
||||||
|
self.log(err)
|
||||||
|
|
||||||
self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8"))
|
self.s.send(b"HTTP/1.1 400 Bad Request\r\n\r\n" + err.encode("utf-8"))
|
||||||
return
|
return
|
||||||
|
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ class HttpSrv(object):
|
|||||||
env.loader = jinja2.FileSystemLoader(os.path.join(E.mod, "web"))
|
env.loader = jinja2.FileSystemLoader(os.path.join(E.mod, "web"))
|
||||||
self.j2 = {
|
self.j2 = {
|
||||||
x: env.get_template(x + ".html")
|
x: env.get_template(x + ".html")
|
||||||
for x in ["splash", "browser", "msg", "md", "mde"]
|
for x in ["splash", "browser", "browser2", "msg", "md", "mde"]
|
||||||
}
|
}
|
||||||
|
|
||||||
cert_path = os.path.join(E.cfg, "cert.pem")
|
cert_path = os.path.join(E.cfg, "cert.pem")
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
|
import os
|
||||||
import tarfile
|
import tarfile
|
||||||
import threading
|
import threading
|
||||||
|
|
||||||
|
from .sutil import errdesc
|
||||||
from .util import Queue, fsenc
|
from .util import Queue, fsenc
|
||||||
|
|
||||||
|
|
||||||
@@ -9,9 +11,20 @@ class QFile(object):
|
|||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.q = Queue(64)
|
self.q = Queue(64)
|
||||||
|
self.bq = []
|
||||||
|
self.nq = 0
|
||||||
|
|
||||||
def write(self, buf):
|
def write(self, buf):
|
||||||
self.q.put(buf)
|
if buf is None or self.nq >= 240 * 1024:
|
||||||
|
self.q.put(b"".join(self.bq))
|
||||||
|
self.bq = []
|
||||||
|
self.nq = 0
|
||||||
|
|
||||||
|
if buf is None:
|
||||||
|
self.q.put(None)
|
||||||
|
else:
|
||||||
|
self.bq.append(buf)
|
||||||
|
self.nq += len(buf)
|
||||||
|
|
||||||
|
|
||||||
class StreamTar(object):
|
class StreamTar(object):
|
||||||
@@ -22,6 +35,7 @@ class StreamTar(object):
|
|||||||
self.co = 0
|
self.co = 0
|
||||||
self.qfile = QFile()
|
self.qfile = QFile()
|
||||||
self.fgen = fgen
|
self.fgen = fgen
|
||||||
|
self.errf = None
|
||||||
|
|
||||||
# python 3.8 changed to PAX_FORMAT as default,
|
# python 3.8 changed to PAX_FORMAT as default,
|
||||||
# waste of space and don't care about the new features
|
# waste of space and don't care about the new features
|
||||||
@@ -35,30 +49,47 @@ class StreamTar(object):
|
|||||||
def gen(self):
|
def gen(self):
|
||||||
while True:
|
while True:
|
||||||
buf = self.qfile.q.get()
|
buf = self.qfile.q.get()
|
||||||
if buf is None:
|
if not buf:
|
||||||
break
|
break
|
||||||
|
|
||||||
self.co += len(buf)
|
self.co += len(buf)
|
||||||
yield buf
|
yield buf
|
||||||
|
|
||||||
yield None
|
yield None
|
||||||
|
if self.errf:
|
||||||
|
os.unlink(self.errf["ap"])
|
||||||
|
|
||||||
|
def ser(self, f):
|
||||||
|
name = f["vp"]
|
||||||
|
src = f["ap"]
|
||||||
|
fsi = f["st"]
|
||||||
|
|
||||||
|
inf = tarfile.TarInfo(name=name)
|
||||||
|
inf.mode = fsi.st_mode
|
||||||
|
inf.size = fsi.st_size
|
||||||
|
inf.mtime = fsi.st_mtime
|
||||||
|
inf.uid = 0
|
||||||
|
inf.gid = 0
|
||||||
|
|
||||||
|
self.ci += inf.size
|
||||||
|
with open(fsenc(src), "rb", 512 * 1024) as f:
|
||||||
|
self.tar.addfile(inf, f)
|
||||||
|
|
||||||
def _gen(self):
|
def _gen(self):
|
||||||
|
errors = []
|
||||||
for f in self.fgen:
|
for f in self.fgen:
|
||||||
name = f["vp"]
|
if "err" in f:
|
||||||
src = f["ap"]
|
errors.append([f["vp"], f["err"]])
|
||||||
fsi = f["st"]
|
continue
|
||||||
|
|
||||||
inf = tarfile.TarInfo(name=name)
|
try:
|
||||||
inf.mode = fsi.st_mode
|
self.ser(f)
|
||||||
inf.size = fsi.st_size
|
except Exception as ex:
|
||||||
inf.mtime = fsi.st_mtime
|
errors.append([f["vp"], repr(ex)])
|
||||||
inf.uid = 0
|
|
||||||
inf.gid = 0
|
|
||||||
|
|
||||||
self.ci += inf.size
|
if errors:
|
||||||
with open(fsenc(src), "rb", 512 * 1024) as f:
|
self.errf = errdesc(errors)
|
||||||
self.tar.addfile(inf, f)
|
self.ser(self.errf)
|
||||||
|
|
||||||
self.tar.close()
|
self.tar.close()
|
||||||
self.qfile.q.put(None)
|
self.qfile.write(None)
|
||||||
|
|||||||
25
copyparty/sutil.py
Normal file
25
copyparty/sutil.py
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import os
|
||||||
|
import time
|
||||||
|
import tempfile
|
||||||
|
from datetime import datetime
|
||||||
|
|
||||||
|
|
||||||
|
def errdesc(errors):
|
||||||
|
report = ["copyparty failed to add the following files to the archive:", ""]
|
||||||
|
|
||||||
|
for fn, err in errors:
|
||||||
|
report.extend([" file: {}".format(fn), "error: {}".format(err), ""])
|
||||||
|
|
||||||
|
with tempfile.NamedTemporaryFile(prefix="copyparty-", delete=False) as tf:
|
||||||
|
tf_path = tf.name
|
||||||
|
tf.write("\r\n".join(report).encode("utf-8", "replace"))
|
||||||
|
|
||||||
|
dt = datetime.utcfromtimestamp(time.time())
|
||||||
|
dt = dt.strftime("%Y-%m%d-%H%M%S")
|
||||||
|
|
||||||
|
os.chmod(tf_path, 0o444)
|
||||||
|
return {
|
||||||
|
"vp": "archive-errors-{}.txt".format(dt),
|
||||||
|
"ap": tf_path,
|
||||||
|
"st": os.stat(tf_path),
|
||||||
|
}
|
||||||
@@ -1,8 +1,10 @@
|
|||||||
|
import os
|
||||||
import time
|
import time
|
||||||
import zlib
|
import zlib
|
||||||
import struct
|
import struct
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
|
|
||||||
|
from .sutil import errdesc
|
||||||
from .util import yieldfile, sanitize_fn
|
from .util import yieldfile, sanitize_fn
|
||||||
|
|
||||||
|
|
||||||
@@ -85,16 +87,20 @@ 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
|
||||||
ret += struct.pack("<HH", len(bfn), z64_len)
|
ret += struct.pack("<HH", len(bfn), z64_len)
|
||||||
|
|
||||||
if h_pos is not None:
|
if h_pos is not None:
|
||||||
# 2b comment, 2b diskno, 2b internal.attr,
|
# 2b comment, 2b diskno
|
||||||
# 4b external.attr (infozip-linux: 0000(a481|ff81)) idk
|
ret += b"\x00" * 4
|
||||||
ret += b"\x00" * 10
|
|
||||||
|
# 2b internal.attr, 4b external.attr
|
||||||
|
# infozip-macos: 0100 0000 a481 file:644
|
||||||
|
# infozip-macos: 0100 0100 0080 file:000
|
||||||
|
ret += b"\x01\x00\x00\x00\xa4\x81"
|
||||||
|
|
||||||
# 4b local-header-ofs
|
# 4b local-header-ofs
|
||||||
ret += struct.pack("<L", min(h_pos, 0xFFFFFFFF))
|
ret += struct.pack("<L", min(h_pos, 0xFFFFFFFF))
|
||||||
@@ -187,43 +193,61 @@ class StreamZip(object):
|
|||||||
self.pos += len(buf)
|
self.pos += len(buf)
|
||||||
return buf
|
return buf
|
||||||
|
|
||||||
def gen(self):
|
def ser(self, f):
|
||||||
for f in self.fgen:
|
name = f["vp"]
|
||||||
name = f["vp"]
|
src = f["ap"]
|
||||||
src = f["ap"]
|
st = f["st"]
|
||||||
st = f["st"]
|
|
||||||
|
|
||||||
sz = st.st_size
|
sz = st.st_size
|
||||||
ts = st.st_mtime + 1
|
ts = st.st_mtime + 1
|
||||||
|
|
||||||
crc = None
|
crc = None
|
||||||
if self.pre_crc:
|
if self.pre_crc:
|
||||||
crc = 0
|
crc = 0
|
||||||
for buf in yieldfile(src):
|
|
||||||
crc = zlib.crc32(buf, crc)
|
|
||||||
|
|
||||||
crc &= 0xFFFFFFFF
|
|
||||||
|
|
||||||
h_pos = self.pos
|
|
||||||
buf = gen_hdr(None, name, sz, ts, self.utf8, crc, self.pre_crc)
|
|
||||||
yield self._ct(buf)
|
|
||||||
|
|
||||||
crc = crc or 0
|
|
||||||
for buf in yieldfile(src):
|
for buf in yieldfile(src):
|
||||||
if not self.pre_crc:
|
crc = zlib.crc32(buf, crc)
|
||||||
crc = zlib.crc32(buf, crc)
|
|
||||||
|
|
||||||
yield self._ct(buf)
|
|
||||||
|
|
||||||
crc &= 0xFFFFFFFF
|
crc &= 0xFFFFFFFF
|
||||||
|
|
||||||
self.items.append([name, sz, ts, crc, h_pos])
|
h_pos = self.pos
|
||||||
|
buf = gen_hdr(None, name, sz, ts, self.utf8, crc, self.pre_crc)
|
||||||
|
yield self._ct(buf)
|
||||||
|
|
||||||
z64 = sz >= 4 * 1024 * 1024 * 1024
|
crc = crc or 0
|
||||||
|
for buf in yieldfile(src):
|
||||||
|
if not self.pre_crc:
|
||||||
|
crc = zlib.crc32(buf, crc)
|
||||||
|
|
||||||
if z64 or not self.pre_crc:
|
yield self._ct(buf)
|
||||||
buf = gen_fdesc(sz, crc, z64)
|
|
||||||
yield self._ct(buf)
|
crc &= 0xFFFFFFFF
|
||||||
|
|
||||||
|
self.items.append([name, sz, ts, crc, h_pos])
|
||||||
|
|
||||||
|
z64 = sz >= 4 * 1024 * 1024 * 1024
|
||||||
|
|
||||||
|
if z64 or not self.pre_crc:
|
||||||
|
buf = gen_fdesc(sz, crc, z64)
|
||||||
|
yield self._ct(buf)
|
||||||
|
|
||||||
|
def gen(self):
|
||||||
|
errors = []
|
||||||
|
for f in self.fgen:
|
||||||
|
if "err" in f:
|
||||||
|
errors.append([f["vp"], f["err"]])
|
||||||
|
continue
|
||||||
|
|
||||||
|
try:
|
||||||
|
for x in self.ser(f):
|
||||||
|
yield x
|
||||||
|
except Exception as ex:
|
||||||
|
errors.append([f["vp"], repr(ex)])
|
||||||
|
|
||||||
|
if errors:
|
||||||
|
errf = errdesc(errors)
|
||||||
|
print(repr(errf))
|
||||||
|
for x in self.ser(errf):
|
||||||
|
yield x
|
||||||
|
|
||||||
cdir_pos = self.pos
|
cdir_pos = self.pos
|
||||||
for name, sz, ts, crc, h_pos in self.items:
|
for name, sz, ts, crc, h_pos in self.items:
|
||||||
@@ -242,3 +266,6 @@ class StreamZip(object):
|
|||||||
|
|
||||||
ecdr, _ = gen_ecdr(self.items, cdir_pos, cdir_end)
|
ecdr, _ = gen_ecdr(self.items, cdir_pos, cdir_end)
|
||||||
yield self._ct(ecdr)
|
yield self._ct(ecdr)
|
||||||
|
|
||||||
|
if errors:
|
||||||
|
os.unlink(errf["ap"])
|
||||||
|
|||||||
@@ -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)
|
||||||
@@ -101,17 +101,18 @@ class Up2k(object):
|
|||||||
thr.daemon = True
|
thr.daemon = True
|
||||||
thr.start()
|
thr.start()
|
||||||
|
|
||||||
thr = threading.Thread(target=self._tagger)
|
|
||||||
thr.daemon = True
|
|
||||||
thr.start()
|
|
||||||
|
|
||||||
thr = threading.Thread(target=self._hasher)
|
thr = threading.Thread(target=self._hasher)
|
||||||
thr.daemon = True
|
thr.daemon = True
|
||||||
thr.start()
|
thr.start()
|
||||||
|
|
||||||
thr = threading.Thread(target=self._run_all_mtp)
|
if self.mtag:
|
||||||
thr.daemon = True
|
thr = threading.Thread(target=self._tagger)
|
||||||
thr.start()
|
thr.daemon = True
|
||||||
|
thr.start()
|
||||||
|
|
||||||
|
thr = threading.Thread(target=self._run_all_mtp)
|
||||||
|
thr.daemon = True
|
||||||
|
thr.start()
|
||||||
|
|
||||||
def log(self, msg, c=0):
|
def log(self, msg, c=0):
|
||||||
self.log_func("up2k", msg + "\033[K", c)
|
self.log_func("up2k", msg + "\033[K", c)
|
||||||
@@ -510,6 +511,7 @@ class Up2k(object):
|
|||||||
|
|
||||||
def _run_all_mtp(self):
|
def _run_all_mtp(self):
|
||||||
t0 = time.time()
|
t0 = time.time()
|
||||||
|
self.mtp_audio = {}
|
||||||
self.mtp_force = {}
|
self.mtp_force = {}
|
||||||
self.mtp_parsers = {}
|
self.mtp_parsers = {}
|
||||||
for ptop, flags in self.flags.items():
|
for ptop, flags in self.flags.items():
|
||||||
@@ -526,8 +528,9 @@ class Up2k(object):
|
|||||||
|
|
||||||
entags = self.entags[ptop]
|
entags = self.entags[ptop]
|
||||||
|
|
||||||
force = {}
|
audio = {} # [r]equire [n]ot [d]ontcare
|
||||||
timeout = {}
|
force = {} # bool
|
||||||
|
timeout = {} # int
|
||||||
parsers = {}
|
parsers = {}
|
||||||
for parser in self.flags[ptop]["mtp"]:
|
for parser in self.flags[ptop]["mtp"]:
|
||||||
orig = parser
|
orig = parser
|
||||||
@@ -535,6 +538,8 @@ class Up2k(object):
|
|||||||
if tag not in entags:
|
if tag not in entags:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
audio[tag] = "y"
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
try:
|
try:
|
||||||
bp = os.path.expanduser(parser)
|
bp = os.path.expanduser(parser)
|
||||||
@@ -548,6 +553,10 @@ class Up2k(object):
|
|||||||
arg, parser = parser.split(",", 1)
|
arg, parser = parser.split(",", 1)
|
||||||
arg = arg.lower()
|
arg = arg.lower()
|
||||||
|
|
||||||
|
if arg.startswith("a"):
|
||||||
|
audio[tag] = arg[1:]
|
||||||
|
continue
|
||||||
|
|
||||||
if arg == "f":
|
if arg == "f":
|
||||||
force[tag] = True
|
force[tag] = True
|
||||||
continue
|
continue
|
||||||
@@ -562,6 +571,8 @@ class Up2k(object):
|
|||||||
self.log("invalid argument: " + orig, 1)
|
self.log("invalid argument: " + orig, 1)
|
||||||
return
|
return
|
||||||
|
|
||||||
|
# todo audio/force => parser attributes
|
||||||
|
self.mtp_audio[ptop] = audio
|
||||||
self.mtp_force[ptop] = force
|
self.mtp_force[ptop] = force
|
||||||
self.mtp_parsers[ptop] = parsers
|
self.mtp_parsers[ptop] = parsers
|
||||||
|
|
||||||
@@ -595,8 +606,8 @@ class Up2k(object):
|
|||||||
have = cur.execute(q, (w,)).fetchall()
|
have = cur.execute(q, (w,)).fetchall()
|
||||||
have = [x[0] for x in have]
|
have = [x[0] for x in have]
|
||||||
|
|
||||||
if ".dur" not in have and ".dur" in entags:
|
parsers = self._get_parsers(ptop, have)
|
||||||
# skip non-audio
|
if not parsers:
|
||||||
to_delete[w] = True
|
to_delete[w] = True
|
||||||
n_left -= 1
|
n_left -= 1
|
||||||
continue
|
continue
|
||||||
@@ -604,10 +615,7 @@ class Up2k(object):
|
|||||||
if w in in_progress:
|
if w in in_progress:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
task_parsers = {
|
jobs.append([parsers, None, w, abspath])
|
||||||
k: v for k, v in parsers.items() if k in force or k not in have
|
|
||||||
}
|
|
||||||
jobs.append([task_parsers, None, w, abspath])
|
|
||||||
in_progress[w] = True
|
in_progress[w] = True
|
||||||
|
|
||||||
done = self._flush_mpool(wcur)
|
done = self._flush_mpool(wcur)
|
||||||
@@ -666,13 +674,32 @@ class Up2k(object):
|
|||||||
wcur.close()
|
wcur.close()
|
||||||
cur.close()
|
cur.close()
|
||||||
|
|
||||||
def _start_mpool(self):
|
def _get_parsers(self, ptop, have):
|
||||||
if WINDOWS and False:
|
try:
|
||||||
nah = open(os.devnull, "wb")
|
all_parsers = self.mtp_parsers[ptop]
|
||||||
wmic = "processid={}".format(os.getpid())
|
except:
|
||||||
wmic = ["wmic", "process", "where", wmic, "call", "setpriority"]
|
return {}
|
||||||
sp.call(wmic + ["below normal"], stdout=nah, stderr=nah)
|
|
||||||
|
|
||||||
|
audio = self.mtp_audio[ptop]
|
||||||
|
force = self.mtp_force[ptop]
|
||||||
|
entags = self.entags[ptop]
|
||||||
|
parsers = {}
|
||||||
|
for k, v in all_parsers.items():
|
||||||
|
if ".dur" in entags:
|
||||||
|
if ".dur" in have:
|
||||||
|
# is audio, require non-audio?
|
||||||
|
if audio[k] == "n":
|
||||||
|
continue
|
||||||
|
# is not audio, require audio?
|
||||||
|
elif audio[k] == "y":
|
||||||
|
continue
|
||||||
|
|
||||||
|
parsers[k] = v
|
||||||
|
|
||||||
|
parsers = {k: v for k, v in parsers.items() if k in force or k not in have}
|
||||||
|
return parsers
|
||||||
|
|
||||||
|
def _start_mpool(self):
|
||||||
# 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
|
||||||
@@ -697,12 +724,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):
|
||||||
@@ -902,7 +923,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()
|
||||||
@@ -1068,6 +1089,8 @@ class Up2k(object):
|
|||||||
with self.mutex:
|
with self.mutex:
|
||||||
job = self.registry[ptop].get(wark, None)
|
job = self.registry[ptop].get(wark, None)
|
||||||
if not job:
|
if not job:
|
||||||
|
known = " ".join([x for x in self.registry[ptop].keys()])
|
||||||
|
self.log("unknown wark [{}], known: {}".format(wark, known))
|
||||||
raise Pebkac(400, "unknown wark")
|
raise Pebkac(400, "unknown wark")
|
||||||
|
|
||||||
if chash not in job["need"]:
|
if chash not in job["need"]:
|
||||||
@@ -1107,8 +1130,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(
|
||||||
@@ -1206,9 +1230,23 @@ class Up2k(object):
|
|||||||
# raise Exception("aaa")
|
# raise Exception("aaa")
|
||||||
|
|
||||||
tnam = job["name"] + ".PARTIAL"
|
tnam = job["name"] + ".PARTIAL"
|
||||||
|
if self.args.dotpart:
|
||||||
|
tnam = "." + tnam
|
||||||
|
|
||||||
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")
|
||||||
|
|
||||||
@@ -1220,13 +1258,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
|
||||||
@@ -1296,13 +1340,9 @@ class Up2k(object):
|
|||||||
abspath = os.path.join(ptop, rd, fn)
|
abspath = os.path.join(ptop, rd, fn)
|
||||||
tags = self.mtag.get(abspath)
|
tags = self.mtag.get(abspath)
|
||||||
ntags1 = len(tags)
|
ntags1 = len(tags)
|
||||||
if self.mtp_parsers.get(ptop, {}):
|
parsers = self._get_parsers(ptop, tags)
|
||||||
parser = {
|
if parsers:
|
||||||
k: v
|
tags.update(self.mtag.get_bin(parsers, abspath))
|
||||||
for k, v in self.mtp_parsers[ptop].items()
|
|
||||||
if k in self.mtp_force[ptop] or k not in tags
|
|
||||||
}
|
|
||||||
tags.update(self.mtag.get_bin(parser, abspath))
|
|
||||||
|
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
cur = self.cur[ptop]
|
cur = self.cur[ptop]
|
||||||
@@ -1310,6 +1350,7 @@ class Up2k(object):
|
|||||||
self.log("no cursor to write tags with??", c=1)
|
self.log("no cursor to write tags with??", c=1)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
# TODO is undef if vol 404 on startup
|
||||||
entags = self.entags[ptop]
|
entags = self.entags[ptop]
|
||||||
if not entags:
|
if not entags:
|
||||||
self.log("no entags okay.jpg", c=3)
|
self.log("no entags okay.jpg", c=3)
|
||||||
|
|||||||
@@ -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
|
||||||
@@ -49,6 +49,7 @@ HTTPCODE = {
|
|||||||
200: "OK",
|
200: "OK",
|
||||||
204: "No Content",
|
204: "No Content",
|
||||||
206: "Partial Content",
|
206: "Partial Content",
|
||||||
|
302: "Found",
|
||||||
304: "Not Modified",
|
304: "Not Modified",
|
||||||
400: "Bad Request",
|
400: "Bad Request",
|
||||||
403: "Forbidden",
|
403: "Forbidden",
|
||||||
@@ -576,12 +577,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 +592,16 @@ def sanitize_fn(fn, ok=""):
|
|||||||
["|", "|"],
|
["|", "|"],
|
||||||
["?", "?"],
|
["?", "?"],
|
||||||
["*", "*"],
|
["*", "*"],
|
||||||
] if x[0] not in ok]:
|
]
|
||||||
fn = fn.replace(bad, good)
|
for a, b in [x for x in remap if x[0] not in ok]:
|
||||||
|
fn = fn.replace(a, b)
|
||||||
|
|
||||||
bad = ["con", "prn", "aux", "nul"]
|
bad.extend(["con", "prn", "aux", "nul"])
|
||||||
for n in range(1, 10):
|
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 +617,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 {
|
||||||
@@ -182,6 +180,21 @@ a, #files tbody div a:last-child {
|
|||||||
color: #840;
|
color: #840;
|
||||||
text-shadow: 0 0 .3em #b80;
|
text-shadow: 0 0 .3em #b80;
|
||||||
}
|
}
|
||||||
|
#files tbody tr.sel td {
|
||||||
|
color: #fff;
|
||||||
|
background: #925;
|
||||||
|
border-color: #c37;
|
||||||
|
}
|
||||||
|
#files tr.sel a {
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
#files tr.sel a.play {
|
||||||
|
color: #fc5;
|
||||||
|
}
|
||||||
|
#files tr.sel a.play.act {
|
||||||
|
color: #fff;
|
||||||
|
text-shadow: 0 0 1px #fff;
|
||||||
|
}
|
||||||
#blocked {
|
#blocked {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 0;
|
top: 0;
|
||||||
@@ -238,7 +251,7 @@ a, #files tbody div a:last-child {
|
|||||||
height: 100%;
|
height: 100%;
|
||||||
background: #3c3c3c;
|
background: #3c3c3c;
|
||||||
}
|
}
|
||||||
#wtoggle {
|
#wtico {
|
||||||
cursor: url(/.cpr/dd/1.png), pointer;
|
cursor: url(/.cpr/dd/1.png), pointer;
|
||||||
animation: cursor 500ms infinite;
|
animation: cursor 500ms infinite;
|
||||||
}
|
}
|
||||||
@@ -268,6 +281,52 @@ a, #files tbody div a:last-child {
|
|||||||
padding: .2em 0 0 .07em;
|
padding: .2em 0 0 .07em;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
}
|
}
|
||||||
|
#wzip, #wnp {
|
||||||
|
display: none;
|
||||||
|
margin-right: .3em;
|
||||||
|
padding-right: .3em;
|
||||||
|
border-right: .1em solid #555;
|
||||||
|
}
|
||||||
|
#wnp a {
|
||||||
|
position: relative;
|
||||||
|
font-size: .47em;
|
||||||
|
margin: 0 .1em;
|
||||||
|
top: -.4em;
|
||||||
|
}
|
||||||
|
#wnp a+a {
|
||||||
|
margin-left: .33em;
|
||||||
|
}
|
||||||
|
#wtoggle,
|
||||||
|
#wtoggle * {
|
||||||
|
line-height: 1em;
|
||||||
|
}
|
||||||
|
#wtoggle.np {
|
||||||
|
width: 5.5em;
|
||||||
|
}
|
||||||
|
#wtoggle.sel {
|
||||||
|
width: 6.4em;
|
||||||
|
}
|
||||||
|
#wtoggle.sel #wzip,
|
||||||
|
#wtoggle.np #wnp {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
#wtoggle.sel.np #wnp {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
#wzip a {
|
||||||
|
font-size: .4em;
|
||||||
|
padding: 0 .3em;
|
||||||
|
margin: -.3em .2em;
|
||||||
|
position: relative;
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
#wzip a+a {
|
||||||
|
margin-left: .8em;
|
||||||
|
}
|
||||||
|
#wtoggle.sel #wzip #selzip {
|
||||||
|
top: -.6em;
|
||||||
|
padding: .4em .3em;
|
||||||
|
}
|
||||||
#barpos,
|
#barpos,
|
||||||
#barbuf {
|
#barbuf {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
@@ -311,10 +370,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;
|
||||||
@@ -324,6 +383,9 @@ a, #files tbody div a:last-child {
|
|||||||
bottom: -3.2em;
|
bottom: -3.2em;
|
||||||
height: 3.2em;
|
height: 3.2em;
|
||||||
}
|
}
|
||||||
|
#pvol {
|
||||||
|
max-width: 9em;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -375,6 +437,7 @@ a, #files tbody div a:last-child {
|
|||||||
padding: .3em .6em;
|
padding: .3em .6em;
|
||||||
border-radius: .3em;
|
border-radius: .3em;
|
||||||
border-width: .15em 0;
|
border-width: .15em 0;
|
||||||
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
.opbox {
|
.opbox {
|
||||||
background: #2d2d2d;
|
background: #2d2d2d;
|
||||||
@@ -455,19 +518,15 @@ 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;
|
||||||
}
|
}
|
||||||
#tree {
|
#tree {
|
||||||
display: none;
|
display: none;
|
||||||
position: fixed;
|
position: absolute;
|
||||||
left: 0;
|
left: 0;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
top: 7em;
|
top: 7em;
|
||||||
padding-top: .2em;
|
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
-ms-scroll-chaining: none;
|
-ms-scroll-chaining: none;
|
||||||
overscroll-behavior-y: none;
|
overscroll-behavior-y: none;
|
||||||
@@ -476,9 +535,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;
|
||||||
}
|
}
|
||||||
@@ -517,6 +574,7 @@ input[type="checkbox"]:checked+label {
|
|||||||
#detree {
|
#detree {
|
||||||
padding: .3em .5em;
|
padding: .3em .5em;
|
||||||
font-size: 1.5em;
|
font-size: 1.5em;
|
||||||
|
line-height: 1.5em;
|
||||||
}
|
}
|
||||||
#tree ul,
|
#tree ul,
|
||||||
#tree li {
|
#tree li {
|
||||||
@@ -598,7 +656,8 @@ input[type="checkbox"]:checked+label {
|
|||||||
#files td.min a {
|
#files td.min a {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
#files tr.play td {
|
#files tr.play td,
|
||||||
|
#files tr.play div a {
|
||||||
background: #fc4;
|
background: #fc4;
|
||||||
border-color: transparent;
|
border-color: transparent;
|
||||||
color: #400;
|
color: #400;
|
||||||
@@ -652,3 +711,199 @@ 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;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
html.light {
|
||||||
|
color: #333;
|
||||||
|
background: #eee;
|
||||||
|
text-shadow: none;
|
||||||
|
}
|
||||||
|
html.light #ops,
|
||||||
|
html.light .opbox,
|
||||||
|
html.light #srch_form {
|
||||||
|
background: #f7f7f7;
|
||||||
|
box-shadow: 0 0 .3em #ddd;
|
||||||
|
border-color: #f7f7f7;
|
||||||
|
}
|
||||||
|
html.light #ops a.act {
|
||||||
|
box-shadow: 0 .2em .2em #ccc;
|
||||||
|
background: #fff;
|
||||||
|
border-color: #07a;
|
||||||
|
padding-top: .4em;
|
||||||
|
}
|
||||||
|
html.light #op_cfg h3 {
|
||||||
|
border-color: #ccc;
|
||||||
|
}
|
||||||
|
html.light .tglbtn,
|
||||||
|
html.light #tree > a + a {
|
||||||
|
color: #666;
|
||||||
|
background: #ddd;
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
html.light .tglbtn:hover,
|
||||||
|
html.light #tree > a + a:hover {
|
||||||
|
background: #caf;
|
||||||
|
}
|
||||||
|
html.light .tglbtn.on,
|
||||||
|
html.light #tree > a + a.on {
|
||||||
|
background: #4a0;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
html.light #srv_info {
|
||||||
|
color: #c83;
|
||||||
|
text-shadow: 1px 1px 0 #fff;
|
||||||
|
}
|
||||||
|
html.light #srv_info span {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
html.light #treeul a+a {
|
||||||
|
background: inherit;
|
||||||
|
color: #06a;
|
||||||
|
}
|
||||||
|
html.light #treeul a.hl {
|
||||||
|
background: #07a;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
html.light #tree li {
|
||||||
|
border-color: #ddd #fff #f7f7f7 #fff;
|
||||||
|
}
|
||||||
|
html.light #tree ul {
|
||||||
|
border-color: #ccc;
|
||||||
|
}
|
||||||
|
html.light a,
|
||||||
|
html.light #ops a,
|
||||||
|
html.light #files tbody div a:last-child {
|
||||||
|
color: #06a;
|
||||||
|
}
|
||||||
|
html.light #files tbody {
|
||||||
|
background: #f7f7f7;
|
||||||
|
}
|
||||||
|
html.light #files {
|
||||||
|
box-shadow: 0 0 .3em #ccc;
|
||||||
|
}
|
||||||
|
html.light #files thead th {
|
||||||
|
background: #eee;
|
||||||
|
}
|
||||||
|
html.light #files tr td {
|
||||||
|
border-top: 1px solid #ddd;
|
||||||
|
}
|
||||||
|
html.light #files td {
|
||||||
|
border-bottom: 1px solid #f7f7f7;
|
||||||
|
}
|
||||||
|
html.light #files tbody tr:last-child td {
|
||||||
|
border-bottom: .2em solid #ccc;
|
||||||
|
}
|
||||||
|
html.light #files td:nth-child(2n) {
|
||||||
|
color: #d38;
|
||||||
|
}
|
||||||
|
html.light #files tr:hover td {
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
html.light #files tbody a.play {
|
||||||
|
color: #c0f;
|
||||||
|
}
|
||||||
|
html.light tr.play td {
|
||||||
|
background: #fc5;
|
||||||
|
}
|
||||||
|
html.light tr.play a {
|
||||||
|
color: #406;
|
||||||
|
}
|
||||||
|
html.light #files th:hover .cfg,
|
||||||
|
html.light #files th.min .cfg {
|
||||||
|
background: #ccc;
|
||||||
|
}
|
||||||
|
html.light #files > thead > tr > th.min span {
|
||||||
|
background: linear-gradient(90deg, rgba(204,204,204,0), rgba(204,204,204,0.5) 70%, #ccc);
|
||||||
|
}
|
||||||
|
html.light #blocked {
|
||||||
|
background: #eee;
|
||||||
|
}
|
||||||
|
html.light #blk_play a,
|
||||||
|
html.light #blk_abrt a {
|
||||||
|
background: #fff;
|
||||||
|
box-shadow: 0 .2em .4em #ddd;
|
||||||
|
}
|
||||||
|
html.light #widget a {
|
||||||
|
color: #fc5;
|
||||||
|
}
|
||||||
|
html.light #files tr.sel:hover td {
|
||||||
|
background: #c37;
|
||||||
|
}
|
||||||
|
html.light #files tr.sel td {
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
html.light #files tr.sel a {
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
html.light #files tr.sel a.play.act {
|
||||||
|
color: #fb0;
|
||||||
|
}
|
||||||
|
html.light input[type="checkbox"] + label {
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
html.light .opview input[type="text"] {
|
||||||
|
background: #fff;
|
||||||
|
color: #333;
|
||||||
|
box-shadow: 0 0 2px #888;
|
||||||
|
border-color: #38d;
|
||||||
|
}
|
||||||
|
html.light #ops:hover #opdesc {
|
||||||
|
background: #fff;
|
||||||
|
box-shadow: 0 .3em 1em #ccc;
|
||||||
|
}
|
||||||
|
html.light #opdesc code {
|
||||||
|
background: #060;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
html.light #u2tab a>span,
|
||||||
|
html.light #files td div span {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
html.light #path {
|
||||||
|
background: #f7f7f7;
|
||||||
|
text-shadow: none;
|
||||||
|
box-shadow: 0 0 .3em #bbb;
|
||||||
|
}
|
||||||
|
html.light #path a {
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
html.light #path a:not(:last-child)::after {
|
||||||
|
border-color: #ccc;
|
||||||
|
background: none;
|
||||||
|
border-width: .1em .1em 0 0;
|
||||||
|
margin: -.2em .3em -.2em -.3em;
|
||||||
|
}
|
||||||
|
html.light #path a:hover {
|
||||||
|
background: none;
|
||||||
|
color: #60a;
|
||||||
|
}
|
||||||
|
html.light #files tbody div a {
|
||||||
|
color: #d38;
|
||||||
|
}
|
||||||
|
html.light #files a:hover,
|
||||||
|
html.light #files tr.sel a:hover {
|
||||||
|
color: #000;
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
html.light #tree {
|
||||||
|
scrollbar-color: #a70 #ddd;
|
||||||
|
}
|
||||||
|
html.light #tree::-webkit-scrollbar-track,
|
||||||
|
html.light #tree::-webkit-scrollbar {
|
||||||
|
background: #ddd;
|
||||||
|
}
|
||||||
|
#tree::-webkit-scrollbar-thumb {
|
||||||
|
background: #da0;
|
||||||
|
}
|
||||||
@@ -13,15 +13,15 @@
|
|||||||
<body>
|
<body>
|
||||||
<div id="ops">
|
<div id="ops">
|
||||||
<a href="#" data-dest="" data-desc="close submenu">---</a>
|
<a href="#" data-dest="" data-desc="close submenu">---</a>
|
||||||
<a href="#" data-perm="read" data-dest="search" data-desc="search for files by attributes, path/name, music tags, or any combination of those.<br /><br /><code>foo bar</code> = must contain both foo and bar,<br /><code>foo -bar</code> = must contain foo but not bar,<br /><code>^yana .opus$</code> = must start with yana and have the opus extension">🔎</a>
|
|
||||||
{%- if have_up2k_idx %}
|
{%- if have_up2k_idx %}
|
||||||
|
<a href="#" data-perm="read" data-dest="search" data-desc="search for files by attributes, path/name, music tags, or any combination of those.<br /><br /><code>foo bar</code> = must contain both foo and bar,<br /><code>foo -bar</code> = must contain foo but not bar,<br /><code>^yana .opus$</code> = must start with yana and have the opus extension">🔎</a>
|
||||||
<a href="#" data-dest="up2k" data-desc="up2k: upload files (if you have write-access) or toggle into the search-mode and drag files onto the search button to see if they exist somewhere on the server">🚀</a>
|
<a href="#" data-dest="up2k" data-desc="up2k: upload files (if you have write-access) or toggle into the search-mode and drag files onto the search button to see if they exist somewhere on the server">🚀</a>
|
||||||
{%- else %}
|
{%- else %}
|
||||||
<a href="#" data-perm="write" data-dest="up2k" data-desc="up2k: upload files with resume support (close your browser and drop the same files in later)">🚀</a>
|
<a href="#" data-perm="write" data-dest="up2k" data-desc="up2k: upload files with resume support (close your browser and drop the same files in later)">🚀</a>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
<a href="#" data-perm="write" data-dest="bup" data-desc="bup: basic uploader, even supports netscape 4.0">🎈</a>
|
<a href="#" data-perm="write" data-dest="bup" data-desc="bup: basic uploader, even supports netscape 4.0">🎈</a>
|
||||||
<a href="#" data-perm="write" data-dest="mkdir" data-desc="mkdir: create a new directory">📂</a>
|
<a href="#" data-perm="write" data-dest="mkdir" data-desc="mkdir: create a new directory">📂</a>
|
||||||
<a href="#" data-perm="write" data-dest="new_md" data-desc="new-md: create a new markdown document">📝</a>
|
<a href="#" data-perm="read write" data-dest="new_md" data-desc="new-md: create a new markdown document">📝</a>
|
||||||
<a href="#" data-perm="write" data-dest="msg" data-desc="msg: send a message to the server log">📟</a>
|
<a href="#" data-perm="write" data-dest="msg" data-desc="msg: send a message to the server log">📟</a>
|
||||||
<a href="#" data-dest="cfg" data-desc="configuration options">⚙️</a>
|
<a href="#" data-dest="cfg" data-desc="configuration options">⚙️</a>
|
||||||
<div id="opdesc"></div>
|
<div id="opdesc"></div>
|
||||||
@@ -39,14 +39,17 @@
|
|||||||
{%- include 'upload.html' %}
|
{%- include 'upload.html' %}
|
||||||
|
|
||||||
<div id="op_cfg" class="opview opbox">
|
<div id="op_cfg" class="opview opbox">
|
||||||
<h3>key notation</h3>
|
<h3>switches</h3>
|
||||||
<div id="key_notation"></div>
|
<div>
|
||||||
|
<a id="tooltips" class="tglbtn" href="#">tooltips</a>
|
||||||
|
<a id="lightmode" class="tglbtn" href="#">lightmode</a>
|
||||||
|
</div>
|
||||||
{%- if have_zip %}
|
{%- if have_zip %}
|
||||||
<h3>folder download</h3>
|
<h3>folder download</h3>
|
||||||
<div id="arc_fmt"></div>
|
<div id="arc_fmt"></div>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
<h3>tooltips</h3>
|
<h3>key notation</h3>
|
||||||
<div><a id="tooltips" class="tglbtn" href="#">enable</a></div>
|
<div id="key_notation"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h1 id="path">
|
<h1 id="path">
|
||||||
@@ -72,7 +75,7 @@
|
|||||||
<table id="files">
|
<table id="files">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th></th>
|
<th name="lead"><span>c</span></th>
|
||||||
<th name="href"><span>File Name</span></th>
|
<th name="href"><span>File Name</span></th>
|
||||||
<th name="sz" sort="int"><span>Size</span></th>
|
<th name="sz" sort="int"><span>Size</span></th>
|
||||||
{%- for k in taglist %}
|
{%- for k in taglist %}
|
||||||
@@ -111,15 +114,7 @@
|
|||||||
<div id="srv_info"><span>{{ srv_info }}</span></div>
|
<div id="srv_info"><span>{{ srv_info }}</span></div>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
|
|
||||||
<div id="widget">
|
<div id="widget"></div>
|
||||||
<div id="wtoggle">♫</div>
|
|
||||||
<div id="widgeti">
|
|
||||||
<div id="pctl"><a href="#" id="bprev">⏮</a><a href="#" id="bplay">▶</a><a href="#" id="bnext">⏭</a></div>
|
|
||||||
<canvas id="pvol" width="288" height="38"></canvas>
|
|
||||||
<canvas id="barpos"></canvas>
|
|
||||||
<canvas id="barbuf"></canvas>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
var tag_order_cfg = {{ tag_order }};
|
var tag_order_cfg = {{ tag_order }};
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
60
copyparty/web/browser2.html
Normal file
60
copyparty/web/browser2.html
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>{{ title }}</title>
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=0.8">
|
||||||
|
<style>
|
||||||
|
html{font-family:sans-serif}
|
||||||
|
td{border:1px solid #999;border-width:1px 1px 0 0;padding:0 5px}
|
||||||
|
a{display:block}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
{%- if srv_info %}
|
||||||
|
<p><span>{{ srv_info }}</span></p>
|
||||||
|
{%- endif %}
|
||||||
|
|
||||||
|
{%- if have_b_u %}
|
||||||
|
<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}">
|
||||||
|
<input type="hidden" name="act" value="bput" />
|
||||||
|
<input type="file" name="f" multiple /><br />
|
||||||
|
<input type="submit" value="start upload" />
|
||||||
|
</form>
|
||||||
|
<br />
|
||||||
|
{%- endif %}
|
||||||
|
|
||||||
|
{%- if logues[0] %}
|
||||||
|
<div>{{ logues[0] }}</div><br />
|
||||||
|
{%- endif %}
|
||||||
|
|
||||||
|
<table id="files">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th name="lead"><span>c</span></th>
|
||||||
|
<th name="href"><span>File Name</span></th>
|
||||||
|
<th name="sz" sort="int"><span>Size</span></th>
|
||||||
|
<th name="ts"><span>Date</span></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr><td></td><td><a href="../{{ url_suf }}">parent folder</a></td><td>-</td><td>-</td></tr>
|
||||||
|
|
||||||
|
{%- for f in files %}
|
||||||
|
<tr><td>{{ f.lead }}</td><td><a href="{{ f.href }}{{ url_suf }}">{{ f.name|e }}</a></td><td>{{ f.sz }}</td><td>{{ f.dt }}</td></tr>
|
||||||
|
{%- endfor %}
|
||||||
|
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
{%- if logues[1] %}
|
||||||
|
<div>{{ logues[1] }}</div><br />
|
||||||
|
{%- endif %}
|
||||||
|
|
||||||
|
<h2><a href="{{ url_suf }}{{ url_suf and '&' or '?' }}h">control-panel</a></h2>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -50,6 +50,9 @@ pre code:last-child {
|
|||||||
pre code::before {
|
pre code::before {
|
||||||
content: counter(precode);
|
content: counter(precode);
|
||||||
-webkit-user-select: none;
|
-webkit-user-select: none;
|
||||||
|
-moz-user-select: none;
|
||||||
|
-ms-user-select: none;
|
||||||
|
user-select: none;
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
text-align: right;
|
text-align: right;
|
||||||
font-size: .75em;
|
font-size: .75em;
|
||||||
@@ -591,12 +594,3 @@ blink {
|
|||||||
color: #940;
|
color: #940;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
*[data-ln]:before {
|
|
||||||
content: attr(data-ln);
|
|
||||||
font-size: .8em;
|
|
||||||
margin: 0 .4em;
|
|
||||||
color: #f0c;
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
@@ -138,10 +138,10 @@ var md_opt = {
|
|||||||
document.documentElement.setAttribute("class", dark ? "dark" : "");
|
document.documentElement.setAttribute("class", dark ? "dark" : "");
|
||||||
btn.innerHTML = "go " + (dark ? "light" : "dark");
|
btn.innerHTML = "go " + (dark ? "light" : "dark");
|
||||||
if (window.localStorage)
|
if (window.localStorage)
|
||||||
localStorage.setItem('darkmode', dark ? 1 : 0);
|
localStorage.setItem('lightmode', dark ? 0 : 1);
|
||||||
};
|
};
|
||||||
btn.onclick = toggle;
|
btn.onclick = toggle;
|
||||||
if (window.localStorage && localStorage.getItem('darkmode') == 1)
|
if (window.localStorage && localStorage.getItem('lightmode') != 1)
|
||||||
toggle();
|
toggle();
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
|||||||
@@ -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;
|
||||||
}
|
}
|
||||||
@@ -31,12 +31,12 @@ var md_opt = {
|
|||||||
|
|
||||||
var lightswitch = (function () {
|
var lightswitch = (function () {
|
||||||
var fun = function () {
|
var fun = function () {
|
||||||
var dark = !!!document.documentElement.getAttribute("class");
|
var dark = !document.documentElement.getAttribute("class");
|
||||||
document.documentElement.setAttribute("class", dark ? "dark" : "");
|
document.documentElement.setAttribute("class", dark ? "dark" : "");
|
||||||
if (window.localStorage)
|
if (window.localStorage)
|
||||||
localStorage.setItem('darkmode', dark ? 1 : 0);
|
localStorage.setItem('lightmode', dark ? 0 : 1);
|
||||||
};
|
};
|
||||||
if (window.localStorage && localStorage.getItem('darkmode') == 1)
|
if (window.localStorage && localStorage.getItem('lightmode') != 1)
|
||||||
fun();
|
fun();
|
||||||
|
|
||||||
return fun;
|
return fun;
|
||||||
|
|||||||
@@ -71,7 +71,7 @@ var mde = (function () {
|
|||||||
})();
|
})();
|
||||||
|
|
||||||
function set_jumpto() {
|
function set_jumpto() {
|
||||||
document.querySelector('.editor-preview-side').onclick = jumpto;
|
QS('.editor-preview-side').onclick = jumpto;
|
||||||
}
|
}
|
||||||
|
|
||||||
function jumpto(ev) {
|
function jumpto(ev) {
|
||||||
@@ -94,7 +94,7 @@ function md_changed(mde, on_srv) {
|
|||||||
window.md_saved = mde.value();
|
window.md_saved = mde.value();
|
||||||
|
|
||||||
var md_now = mde.value();
|
var md_now = mde.value();
|
||||||
var save_btn = document.querySelector('.editor-toolbar button.save');
|
var save_btn = QS('.editor-toolbar button.save');
|
||||||
|
|
||||||
if (md_now == window.md_saved)
|
if (md_now == window.md_saved)
|
||||||
save_btn.classList.add('disabled');
|
save_btn.classList.add('disabled');
|
||||||
@@ -105,7 +105,7 @@ function md_changed(mde, on_srv) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function save(mde) {
|
function save(mde) {
|
||||||
var save_btn = document.querySelector('.editor-toolbar button.save');
|
var save_btn = QS('.editor-toolbar button.save');
|
||||||
if (save_btn.classList.contains('disabled')) {
|
if (save_btn.classList.contains('disabled')) {
|
||||||
alert('there is nothing to save');
|
alert('there is nothing to save');
|
||||||
return;
|
return;
|
||||||
@@ -212,7 +212,7 @@ function save_chk() {
|
|||||||
last_modified = this.lastmod;
|
last_modified = this.lastmod;
|
||||||
md_changed(this.mde, true);
|
md_changed(this.mde, true);
|
||||||
|
|
||||||
var ok = document.createElement('div');
|
var ok = mknod('div');
|
||||||
ok.setAttribute('style', 'font-size:6em;font-family:serif;font-weight:bold;color:#cf6;background:#444;border-radius:.3em;padding:.6em 0;position:fixed;top:30%;left:calc(50% - 2em);width:4em;text-align:center;z-index:9001;transition:opacity 0.2s ease-in-out;opacity:1');
|
ok.setAttribute('style', 'font-size:6em;font-family:serif;font-weight:bold;color:#cf6;background:#444;border-radius:.3em;padding:.6em 0;position:fixed;top:30%;left:calc(50% - 2em);width:4em;text-align:center;z-index:9001;transition:opacity 0.2s ease-in-out;opacity:1');
|
||||||
ok.innerHTML = 'OK✔️';
|
ok.innerHTML = 'OK✔️';
|
||||||
var parent = ebi('m');
|
var parent = ebi('m');
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ html,body,tr,th,td,#files,a {
|
|||||||
background: none;
|
background: none;
|
||||||
font-weight: inherit;
|
font-weight: inherit;
|
||||||
font-size: inherit;
|
font-size: inherit;
|
||||||
padding: none;
|
padding: 0;
|
||||||
border: none;
|
border: none;
|
||||||
}
|
}
|
||||||
html {
|
html {
|
||||||
@@ -20,8 +20,8 @@ body {
|
|||||||
padding-bottom: 5em;
|
padding-bottom: 5em;
|
||||||
}
|
}
|
||||||
#box {
|
#box {
|
||||||
padding: .5em 1em;
|
padding: .5em 1em;
|
||||||
background: #2c2c2c;
|
background: #2c2c2c;
|
||||||
}
|
}
|
||||||
pre {
|
pre {
|
||||||
font-family: monospace, monospace;
|
font-family: monospace, monospace;
|
||||||
|
|||||||
@@ -13,23 +13,27 @@
|
|||||||
<div id="wrap">
|
<div id="wrap">
|
||||||
<p>hello {{ this.uname }}</p>
|
<p>hello {{ this.uname }}</p>
|
||||||
|
|
||||||
|
{%- if rvol %}
|
||||||
<h1>you can browse these:</h1>
|
<h1>you can browse these:</h1>
|
||||||
<ul>
|
<ul>
|
||||||
{% for mp in rvol %}
|
{% for mp in rvol %}
|
||||||
<li><a href="/{{ mp }}">/{{ mp }}</a></li>
|
<li><a href="/{{ mp }}{{ url_suf }}">/{{ mp }}</a></li>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</ul>
|
</ul>
|
||||||
|
{%- endif %}
|
||||||
|
|
||||||
|
{%- if wvol %}
|
||||||
<h1>you can upload to:</h1>
|
<h1>you can upload to:</h1>
|
||||||
<ul>
|
<ul>
|
||||||
{% for mp in wvol %}
|
{% for mp in wvol %}
|
||||||
<li><a href="/{{ mp }}">/{{ mp }}</a></li>
|
<li><a href="/{{ mp }}{{ url_suf }}">/{{ mp }}</a></li>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</ul>
|
</ul>
|
||||||
|
{%- endif %}
|
||||||
|
|
||||||
<h1>login for more:</h1>
|
<h1>login for more:</h1>
|
||||||
<ul>
|
<ul>
|
||||||
<form method="post" enctype="multipart/form-data" action="/">
|
<form method="post" enctype="multipart/form-data" action="/{{ url_suf }}">
|
||||||
<input type="hidden" name="act" value="login" />
|
<input type="hidden" name="act" value="login" />
|
||||||
<input type="password" name="cppwd" />
|
<input type="password" name="cppwd" />
|
||||||
<input type="submit" value="Login" />
|
<input type="submit" value="Login" />
|
||||||
@@ -38,7 +42,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<script>
|
<script>
|
||||||
|
|
||||||
if (window.localStorage && localStorage.getItem('darkmode') == 1)
|
if (window.localStorage && localStorage.getItem('lightmode') != 1)
|
||||||
document.documentElement.setAttribute("class", "dark");
|
document.documentElement.setAttribute("class", "dark");
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -19,6 +19,11 @@
|
|||||||
color: #f87;
|
color: #f87;
|
||||||
padding: .5em;
|
padding: .5em;
|
||||||
}
|
}
|
||||||
|
#u2err.msg {
|
||||||
|
color: #999;
|
||||||
|
padding: .5em;
|
||||||
|
font-size: .9em;
|
||||||
|
}
|
||||||
#u2btn {
|
#u2btn {
|
||||||
color: #eee;
|
color: #eee;
|
||||||
background: #555;
|
background: #555;
|
||||||
@@ -47,6 +52,11 @@
|
|||||||
margin: -1.5em 0;
|
margin: -1.5em 0;
|
||||||
padding: .8em 0;
|
padding: .8em 0;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
max-width: 12em;
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
#u2conf #u2btn_cw {
|
||||||
|
text-align: right;
|
||||||
}
|
}
|
||||||
#u2notbtn {
|
#u2notbtn {
|
||||||
display: none;
|
display: none;
|
||||||
@@ -72,6 +82,7 @@
|
|||||||
}
|
}
|
||||||
#u2tab td:nth-child(2) {
|
#u2tab td:nth-child(2) {
|
||||||
width: 5em;
|
width: 5em;
|
||||||
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
#u2tab td:nth-child(3) {
|
#u2tab td:nth-child(3) {
|
||||||
width: 40%;
|
width: 40%;
|
||||||
@@ -80,9 +91,46 @@
|
|||||||
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 {
|
||||||
|
padding: 1em 0 .3em 1em;
|
||||||
|
margin: 1.5em auto -2.5em auto;
|
||||||
|
white-space: nowrap;
|
||||||
|
text-align: center;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
#u2cards.w {
|
||||||
|
width: 45em;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
#u2cards a {
|
||||||
|
padding: .2em 1em;
|
||||||
|
border: 1px solid #777;
|
||||||
|
border-width: 0 0 1px 0;
|
||||||
|
background: linear-gradient(to bottom, #333, #222);
|
||||||
|
}
|
||||||
|
#u2cards a:first-child {
|
||||||
|
border-radius: .4em 0 0 0;
|
||||||
|
}
|
||||||
|
#u2cards a:last-child {
|
||||||
|
border-radius: 0 .4em 0 0;
|
||||||
|
}
|
||||||
|
#u2cards a.act {
|
||||||
|
padding-bottom: .5em;
|
||||||
|
border-width: 1px 1px .1em 1px;
|
||||||
|
border-radius: .3em .3em 0 0;
|
||||||
|
margin-left: -1px;
|
||||||
|
background: linear-gradient(to bottom, #464, #333 80%);
|
||||||
|
box-shadow: 0 -.17em .67em #280;
|
||||||
|
border-color: #7c5 #583 #333 #583;
|
||||||
|
position: relative;
|
||||||
|
color: #fd7;
|
||||||
|
}
|
||||||
|
#u2cards span {
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
#u2conf {
|
#u2conf {
|
||||||
margin: 1em auto;
|
margin: 1em auto;
|
||||||
width: 30em;
|
width: 30em;
|
||||||
@@ -99,12 +147,16 @@
|
|||||||
outline: none;
|
outline: none;
|
||||||
}
|
}
|
||||||
#u2conf .txtbox {
|
#u2conf .txtbox {
|
||||||
width: 4em;
|
width: 3em;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
background: #444;
|
background: #444;
|
||||||
border: 1px solid #777;
|
border: 1px solid #777;
|
||||||
font-size: 1.2em;
|
font-size: 1.2em;
|
||||||
padding: .15em 0;
|
padding: .15em 0;
|
||||||
|
height: 1.05em;
|
||||||
|
}
|
||||||
|
#u2conf .txtbox.err {
|
||||||
|
background: #922;
|
||||||
}
|
}
|
||||||
#u2conf a {
|
#u2conf a {
|
||||||
color: #fff;
|
color: #fff;
|
||||||
@@ -113,13 +165,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;
|
||||||
@@ -130,7 +181,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"] {
|
||||||
@@ -170,12 +220,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;
|
||||||
@@ -193,24 +244,6 @@
|
|||||||
.prog {
|
.prog {
|
||||||
font-family: monospace;
|
font-family: monospace;
|
||||||
}
|
}
|
||||||
.prog>div {
|
|
||||||
display: inline-block;
|
|
||||||
position: relative;
|
|
||||||
overflow: hidden;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
height: 1.1em;
|
|
||||||
margin-bottom: -.15em;
|
|
||||||
box-shadow: -1px -1px 0 inset rgba(255,255,255,0.1);
|
|
||||||
}
|
|
||||||
.prog>div>div {
|
|
||||||
width: 0%;
|
|
||||||
position: absolute;
|
|
||||||
left: 0;
|
|
||||||
top: 0;
|
|
||||||
bottom: 0;
|
|
||||||
background: #0a0;
|
|
||||||
}
|
|
||||||
#u2tab a>span {
|
#u2tab a>span {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
font-style: italic;
|
font-style: italic;
|
||||||
@@ -221,3 +254,44 @@
|
|||||||
float: right;
|
float: right;
|
||||||
margin-bottom: -.3em;
|
margin-bottom: -.3em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
html.light #u2btn {
|
||||||
|
box-shadow: .4em .4em 0 #ccc;
|
||||||
|
}
|
||||||
|
html.light #u2cards span {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
html.light #u2cards a {
|
||||||
|
background: linear-gradient(to bottom, #eee, #fff);
|
||||||
|
}
|
||||||
|
html.light #u2cards a.act {
|
||||||
|
color: #037;
|
||||||
|
background: inherit;
|
||||||
|
box-shadow: 0 -.17em .67em #0ad;
|
||||||
|
border-color: #09c #05a #eee #05a;
|
||||||
|
}
|
||||||
|
html.light #u2conf .txtbox {
|
||||||
|
background: #fff;
|
||||||
|
color: #444;
|
||||||
|
}
|
||||||
|
html.light #u2conf .txtbox.err {
|
||||||
|
background: #f96;
|
||||||
|
color: #300;
|
||||||
|
}
|
||||||
|
html.light #u2cdesc {
|
||||||
|
background: #fff;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
html.light #op_up2k.srch #u2btn {
|
||||||
|
border-color: #a80;
|
||||||
|
}
|
||||||
|
html.light #u2foot {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
html.light #u2tab tbody tr:hover td {
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
|
|
||||||
<div id="op_bup" class="opview opbox act">
|
<div id="op_bup" class="opview opbox act">
|
||||||
<div id="u2err"></div>
|
<div id="u2err"></div>
|
||||||
<form method="post" enctype="multipart/form-data" accept-charset="utf-8">
|
<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}">
|
||||||
<input type="hidden" name="act" value="bput" />
|
<input type="hidden" name="act" value="bput" />
|
||||||
<input type="file" name="f" multiple><br />
|
<input type="file" name="f" multiple><br />
|
||||||
<input type="submit" value="start upload">
|
<input type="submit" value="start upload">
|
||||||
@@ -9,7 +9,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="op_mkdir" class="opview opbox act">
|
<div id="op_mkdir" class="opview opbox act">
|
||||||
<form method="post" enctype="multipart/form-data" accept-charset="utf-8">
|
<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}">
|
||||||
<input type="hidden" name="act" value="mkdir" />
|
<input type="hidden" name="act" value="mkdir" />
|
||||||
<input type="text" name="name" size="30">
|
<input type="text" name="name" size="30">
|
||||||
<input type="submit" value="mkdir">
|
<input type="submit" value="mkdir">
|
||||||
@@ -17,15 +17,15 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="op_new_md" class="opview opbox">
|
<div id="op_new_md" class="opview opbox">
|
||||||
<form method="post" enctype="multipart/form-data" accept-charset="utf-8">
|
<form method="post" enctype="multipart/form-data" accept-charset="utf-8" action="{{ url_suf }}">
|
||||||
<input type="hidden" name="act" value="new_md" />
|
<input type="hidden" name="act" value="new_md" />
|
||||||
<input type="text" name="name" size="30">
|
<input type="text" name="name" size="30">
|
||||||
<input type="submit" value="create doc">
|
<input type="submit" value="create doc">
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="op_msg" class="opview opbox">
|
<div id="op_msg" class="opview opbox act">
|
||||||
<form method="post" enctype="application/x-www-form-urlencoded" accept-charset="utf-8">
|
<form method="post" enctype="application/x-www-form-urlencoded" accept-charset="utf-8" action="{{ url_suf }}">
|
||||||
<input type="text" name="msg" size="30">
|
<input type="text" name="msg" size="30">
|
||||||
<input type="submit" value="send msg">
|
<input type="submit" value="send msg">
|
||||||
</form>
|
</form>
|
||||||
@@ -36,7 +36,7 @@
|
|||||||
|
|
||||||
<table id="u2conf">
|
<table id="u2conf">
|
||||||
<tr>
|
<tr>
|
||||||
<td>parallel uploads</td>
|
<td><br />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><br />
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
@@ -79,14 +79,25 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div id="u2cards">
|
||||||
|
<a href="#" act="ok">ok <span>0</span></a><a
|
||||||
|
href="#" act="ng">ng <span>0</span></a><a
|
||||||
|
href="#" act="done">done <span>0</span></a><a
|
||||||
|
href="#" act="bz" class="act">busy <span>0</span></a><a
|
||||||
|
href="#" act="q">que <span>0</span></a>
|
||||||
|
</div>
|
||||||
|
|
||||||
<table id="u2tab">
|
<table id="u2tab">
|
||||||
<tr>
|
<thead>
|
||||||
<td>filename</td>
|
<tr>
|
||||||
<td>status</td>
|
<td>filename</td>
|
||||||
<td>progress<a href="#" id="u2cleanup">cleanup</a></td>
|
<td>status</td>
|
||||||
</tr>
|
<td>progress<a href="#" id="u2cleanup">cleanup</a></td>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody></tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
<p id="u2foot"></p>
|
<p id="u2foot"></p>
|
||||||
<p 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" data-perm="write">( you can use the <a href="#" id="u2nope">basic uploader</a> if you don't need lastmod timestamps, resumable uploads, or progress bars )</p>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,5 +1,15 @@
|
|||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
|
if (!window['console'])
|
||||||
|
window['console'] = {
|
||||||
|
"log": function (msg) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
var clickev = window.Touch ? 'touchstart' : 'click',
|
||||||
|
ANDROID = /(android)/i.test(navigator.userAgent);
|
||||||
|
|
||||||
|
|
||||||
// error handler for mobile devices
|
// error handler for mobile devices
|
||||||
function hcroak(msg) {
|
function hcroak(msg) {
|
||||||
document.body.innerHTML = msg;
|
document.body.innerHTML = msg;
|
||||||
@@ -40,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;
|
||||||
@@ -80,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;
|
||||||
|
|
||||||
@@ -110,7 +122,85 @@ function crc32(str) {
|
|||||||
crc = (crc >>> 8) ^ crctab[(crc ^ str.charCodeAt(i)) & 0xFF];
|
crc = (crc >>> 8) ^ crctab[(crc ^ str.charCodeAt(i)) & 0xFF];
|
||||||
}
|
}
|
||||||
return ((crc ^ (-1)) >>> 0).toString(16);
|
return ((crc ^ (-1)) >>> 0).toString(16);
|
||||||
};
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function clmod(obj, cls, add) {
|
||||||
|
var re = new RegExp('\\s*\\b' + cls + '\\s*\\b', 'g');
|
||||||
|
if (add == 't')
|
||||||
|
add = !re.test(obj.className);
|
||||||
|
|
||||||
|
obj.className = obj.className.replace(re, ' ') + (add ? ' ' + cls : '');
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function sortfiles(nodes) {
|
||||||
|
var sopts = jread('fsort', [["lead", -1, ""], ["href", 1, ""]]);
|
||||||
|
|
||||||
|
try {
|
||||||
|
var is_srch = false;
|
||||||
|
if (nodes[0]['rp']) {
|
||||||
|
is_srch = true;
|
||||||
|
for (var b = 0, bb = nodes.length; b < bb; b++)
|
||||||
|
nodes[b].ext = nodes[b].rp.split('.').pop();
|
||||||
|
for (var b = 0; b < sopts.length; b++)
|
||||||
|
if (sopts[b][0] == 'href')
|
||||||
|
sopts[b][0] = 'rp';
|
||||||
|
}
|
||||||
|
for (var a = sopts.length - 1; a >= 0; a--) {
|
||||||
|
var name = sopts[a][0], rev = sopts[a][1], typ = sopts[a][2];
|
||||||
|
if (!name)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (name.indexOf('tags/') === 0) {
|
||||||
|
name = name.slice(5);
|
||||||
|
for (var b = 0, bb = nodes.length; b < bb; b++)
|
||||||
|
nodes[b]._sv = nodes[b].tags[name];
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
for (var b = 0, bb = nodes.length; b < bb; b++) {
|
||||||
|
var v = nodes[b][name];
|
||||||
|
|
||||||
|
if ((v + '').indexOf('<a ') === 0)
|
||||||
|
v = v.split('>')[1];
|
||||||
|
else if (name == "href" && v)
|
||||||
|
v = uricom_dec(v)[0]
|
||||||
|
|
||||||
|
nodes[b]._sv = v;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var onodes = nodes.map(function (x) { return x; });
|
||||||
|
nodes.sort(function (n1, n2) {
|
||||||
|
var v1 = n1._sv,
|
||||||
|
v2 = n2._sv;
|
||||||
|
|
||||||
|
if (v1 === undefined) {
|
||||||
|
if (v2 === undefined) {
|
||||||
|
return onodes.indexOf(n1) - onodes.indexOf(n2);
|
||||||
|
}
|
||||||
|
return -1 * rev;
|
||||||
|
}
|
||||||
|
if (v2 === undefined) return 1 * rev;
|
||||||
|
|
||||||
|
var ret = rev * (typ == 'int' ? (v1 - v2) : (v1.localeCompare(v2)));
|
||||||
|
if (ret === 0)
|
||||||
|
ret = onodes.indexOf(n1) - onodes.indexOf(n2);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
for (var b = 0, bb = nodes.length; b < bb; b++) {
|
||||||
|
delete nodes[b]._sv;
|
||||||
|
if (is_srch)
|
||||||
|
delete nodes[b].ext;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (ex) {
|
||||||
|
console.log("failed to apply sort config: " + ex);
|
||||||
|
}
|
||||||
|
return nodes;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
function sortTable(table, col, cb) {
|
function sortTable(table, col, cb) {
|
||||||
@@ -186,9 +276,8 @@ 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;
|
||||||
}
|
}
|
||||||
@@ -203,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--)
|
||||||
obj[a].classList.remove('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--)
|
||||||
obj[a].classList.remove('act');
|
clmod(obj[a], 'act');
|
||||||
|
|
||||||
if (dest) {
|
if (dest) {
|
||||||
var ui = ebi('op_' + dest);
|
var ui = ebi('op_' + dest);
|
||||||
ui.classList.add('act');
|
clmod(ui, 'act', true);
|
||||||
document.querySelector('#ops>a[data-dest=' + dest + ']').classList.add('act');
|
QS('#ops>a[data-dest=' + dest + ']').className += " act";
|
||||||
|
|
||||||
var fn = window['goto_' + dest];
|
var fn = window['goto_' + dest];
|
||||||
if (fn)
|
if (fn)
|
||||||
@@ -408,8 +497,7 @@ function bcfg_upd_ui(name, val) {
|
|||||||
if (o.getAttribute('type') == 'checkbox')
|
if (o.getAttribute('type') == 'checkbox')
|
||||||
o.checked = val;
|
o.checked = val;
|
||||||
else if (o) {
|
else if (o) {
|
||||||
var fun = val ? 'add' : 'remove';
|
clmod(o, 'on', val);
|
||||||
o.classList[fun]('on');
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -32,9 +32,13 @@ r
|
|||||||
|
|
||||||
# and a folder where anyone can upload
|
# and a folder where anyone can upload
|
||||||
# but nobody can see the contents
|
# but nobody can see the contents
|
||||||
|
# and set the e2d flag to enable the uploads database
|
||||||
|
# and set the nodupe flag to reject duplicate uploads
|
||||||
/home/ed/inc
|
/home/ed/inc
|
||||||
/dump
|
/dump
|
||||||
w
|
w
|
||||||
|
c e2d
|
||||||
|
c nodupe
|
||||||
|
|
||||||
# this entire config file can be replaced with these arguments:
|
# this entire config file can be replaced with these arguments:
|
||||||
# -u ed:123 -u k:k -v .::r:aed -v priv:priv:rk:aed -v /home/ed/Music:music:r -v /home/ed/inc:dump:w
|
# -u ed:123 -u k:k -v .::r:aed -v priv:priv:rk:aed -v /home/ed/Music:music:r -v /home/ed/inc:dump:w
|
||||||
|
|||||||
32
docs/minimal-up2k.html
Normal file
32
docs/minimal-up2k.html
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
<!--
|
||||||
|
save this as .epilogue.html inside a write-only folder to declutter the UI, makes it look like
|
||||||
|
https://user-images.githubusercontent.com/241032/118311195-dd6ca380-b4ef-11eb-86f3-75a3ff2e1332.png
|
||||||
|
-->
|
||||||
|
|
||||||
|
<style>
|
||||||
|
|
||||||
|
/* make the up2k ui REALLY minimal by hiding a bunch of stuff: */
|
||||||
|
|
||||||
|
#ops, #tree, #path, #wrap>h2:last-child, /* main tabs and navigators (tree/breadcrumbs) */
|
||||||
|
|
||||||
|
#u2cleanup, #u2conf tr:first-child>td[rowspan]:not(#u2btn_cw), /* most of the config options */
|
||||||
|
|
||||||
|
#u2cards /* and the upload progress tabs */
|
||||||
|
|
||||||
|
{display: none !important} /* do it! */
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* add some margins because now it's weird */
|
||||||
|
.opview {margin-top: 2.5em}
|
||||||
|
#op_up2k {margin-top: 3em}
|
||||||
|
|
||||||
|
/* and embiggen the upload button */
|
||||||
|
#u2conf #u2btn, #u2btn {padding:1.5em 0}
|
||||||
|
|
||||||
|
/* adjust the button area a bit */
|
||||||
|
#u2conf.has_btn {width: 35em !important; margin: 5em auto}
|
||||||
|
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<a href="#" onclick="this.parentNode.innerHTML='';">show advanced options</a>
|
||||||
@@ -73,6 +73,13 @@ shab64() { sp=$1; f="$2"; v=0; sz=$(stat -c%s "$f"); while true; do w=$((v+sp*10
|
|||||||
command -v gdate && date() { gdate "$@"; }; while true; do t=$(date +%s.%N); (time wget http://127.0.0.1:3923/?ls -qO- | jq -C '.files[]|{sz:.sz,ta:.tags.artist,tb:.tags.".bpm"}|del(.[]|select(.==null))' | awk -F\" '/"/{t[$2]++} END {for (k in t){v=t[k];p=sprintf("%" (v+1) "s",v);gsub(/ /,"#",p);printf "\033[36m%s\033[33m%s ",k,p}}') 2>&1 | awk -v ts=$t 'NR==1{t1=$0} NR==2{sub(/.*0m/,"");sub(/s$/,"");t2=$0;c=2; if(t2>0.3){c=3} if(t2>0.8){c=1} } END{sub(/[0-9]{6}$/,"",ts);printf "%s \033[3%dm%s %s\033[0m\n",ts,c,t2,t1}'; sleep 0.1 || break; done
|
command -v gdate && date() { gdate "$@"; }; while true; do t=$(date +%s.%N); (time wget http://127.0.0.1:3923/?ls -qO- | jq -C '.files[]|{sz:.sz,ta:.tags.artist,tb:.tags.".bpm"}|del(.[]|select(.==null))' | awk -F\" '/"/{t[$2]++} END {for (k in t){v=t[k];p=sprintf("%" (v+1) "s",v);gsub(/ /,"#",p);printf "\033[36m%s\033[33m%s ",k,p}}') 2>&1 | awk -v ts=$t 'NR==1{t1=$0} NR==2{sub(/.*0m/,"");sub(/s$/,"");t2=$0;c=2; if(t2>0.3){c=3} if(t2>0.8){c=1} } END{sub(/[0-9]{6}$/,"",ts);printf "%s \033[3%dm%s %s\033[0m\n",ts,c,t2,t1}'; sleep 0.1 || break; done
|
||||||
|
|
||||||
|
|
||||||
|
##
|
||||||
|
## js oneliners
|
||||||
|
|
||||||
|
# get all up2k search result URLs
|
||||||
|
var t=[]; var b=document.location.href.split('#')[0].slice(0, -1); document.querySelectorAll('#u2tab .prog a').forEach((x) => {t.push(b+encodeURI(x.getAttribute("href")))}); console.log(t.join("\n"));
|
||||||
|
|
||||||
|
|
||||||
##
|
##
|
||||||
## sqlite3 stuff
|
## sqlite3 stuff
|
||||||
|
|
||||||
@@ -83,6 +90,9 @@ sqlite3 up2k.db 'select mt1.w, mt1.k, mt1.v, mt2.v from mt mt1 inner join mt mt2
|
|||||||
time sqlite3 up2k.db 'select mt1.w from mt mt1 inner join mt mt2 on mt1.w = mt2.w where mt1.k = +mt2.k and mt1.rowid != mt2.rowid' > warks
|
time sqlite3 up2k.db 'select mt1.w from mt mt1 inner join mt mt2 on mt1.w = mt2.w where mt1.k = +mt2.k and mt1.rowid != mt2.rowid' > warks
|
||||||
cat warks | while IFS= read -r x; do sqlite3 up2k.db "delete from mt where w = '$x'"; done
|
cat warks | while IFS= read -r x; do sqlite3 up2k.db "delete from mt where w = '$x'"; done
|
||||||
|
|
||||||
|
# dump all dbs
|
||||||
|
find -iname up2k.db | while IFS= read -r x; do sqlite3 "$x" 'select substr(w,1,12), rd, fn from up' | sed -r 's/\|/ \| /g' | while IFS= read -r y; do printf '%s | %s\n' "$x" "$y"; done; done
|
||||||
|
|
||||||
|
|
||||||
##
|
##
|
||||||
## media
|
## media
|
||||||
@@ -126,6 +136,16 @@ pip install virtualenv
|
|||||||
# readme toc
|
# readme toc
|
||||||
cat README.md | awk '!/^#/{next} {lv=length($1);sub(/[^ ]+ /,"");bab=$0;gsub(/ /,"-",bab)} {printf "%" ((lv-1)*4+1) "s [%s](#%s)\n", "*",$0,bab}'
|
cat README.md | awk '!/^#/{next} {lv=length($1);sub(/[^ ]+ /,"");bab=$0;gsub(/ /,"-",bab)} {printf "%" ((lv-1)*4+1) "s [%s](#%s)\n", "*",$0,bab}'
|
||||||
|
|
||||||
|
# fix firefox phantom breakpoints,
|
||||||
|
# suggestions from bugtracker, doesnt work (debugger is not attachable)
|
||||||
|
devtools settings >> advanced >> enable browser chrome debugging + enable remote debugging
|
||||||
|
burger > developer >> browser toolbox (ctrl-alt-shift-i)
|
||||||
|
iframe btn topright >> chrome://devtools/content/debugger/index.html
|
||||||
|
dbg.asyncStore.pendingBreakpoints = {}
|
||||||
|
|
||||||
|
# fix firefox phantom breakpoints
|
||||||
|
about:config >> devtools.debugger.prefs-schema-version = -1
|
||||||
|
|
||||||
|
|
||||||
##
|
##
|
||||||
## http 206
|
## http 206
|
||||||
@@ -151,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) {
|
||||||
|
|||||||
@@ -45,11 +45,13 @@ pybin=$(command -v python3 || command -v python) || {
|
|||||||
exit 1
|
exit 1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
use_gz=
|
||||||
do_sh=1
|
do_sh=1
|
||||||
do_py=1
|
do_py=1
|
||||||
while [ ! -z "$1" ]; do
|
while [ ! -z "$1" ]; do
|
||||||
[ "$1" = clean ] && clean=1 && shift && continue
|
[ "$1" = clean ] && clean=1 && shift && continue
|
||||||
[ "$1" = re ] && repack=1 && shift && continue
|
[ "$1" = re ] && repack=1 && shift && continue
|
||||||
|
[ "$1" = gz ] && use_gz=1 && shift && continue
|
||||||
[ "$1" = no-ogv ] && no_ogv=1 && shift && continue
|
[ "$1" = no-ogv ] && no_ogv=1 && shift && continue
|
||||||
[ "$1" = no-cm ] && no_cm=1 && shift && continue
|
[ "$1" = no-cm ] && no_cm=1 && shift && continue
|
||||||
[ "$1" = no-sh ] && do_sh= && shift && continue
|
[ "$1" = no-sh ] && do_sh= && shift && continue
|
||||||
@@ -115,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\.]+$' && {
|
||||||
@@ -161,7 +163,7 @@ find .. -type f \( -name .DS_Store -or -name ._.DS_Store \) -delete
|
|||||||
find .. -type f -name ._\* | while IFS= read -r f; do cmp <(printf '\x00\x05\x16') <(head -c 3 -- "$f") && rm -f -- "$f"; done
|
find .. -type f -name ._\* | while IFS= read -r f; do cmp <(printf '\x00\x05\x16') <(head -c 3 -- "$f") && rm -f -- "$f"; done
|
||||||
|
|
||||||
echo use smol web deps
|
echo use smol web deps
|
||||||
rm -f copyparty/web/deps/*.full.*
|
rm -f copyparty/web/deps/*.full.* copyparty/web/Makefile
|
||||||
|
|
||||||
# it's fine dw
|
# it's fine dw
|
||||||
grep -lE '\.full\.(js|css)' copyparty/web/* |
|
grep -lE '\.full\.(js|css)' copyparty/web/* |
|
||||||
@@ -197,23 +199,34 @@ 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
|
||||||
|
pe=bz2
|
||||||
|
[ $use_gz ] && pc=gzip && pe=gz
|
||||||
|
|
||||||
echo compressing tar
|
echo compressing tar
|
||||||
# detect best level; bzip2 -7 is usually better than -9
|
# detect best level; bzip2 -7 is usually better than -9
|
||||||
[ $do_py ] && { for n in {2..9}; do cp tar t.$n; bzip2 -$n t.$n & done; wait; mv -v $(ls -1S t.*.bz2 | tail -n 1) tar.bz2; }
|
[ $do_py ] && { for n in {2..9}; do cp tar t.$n; $pc -$n t.$n & done; wait; mv -v $(ls -1S t.*.$pe | tail -n 1) tar.bz2; }
|
||||||
[ $do_sh ] && { for n in {2..9}; do cp tar t.$n; xz -ze$n t.$n & done; wait; mv -v $(ls -1S t.*.xz | tail -n 1) tar.xz; }
|
[ $do_sh ] && { for n in {2..9}; do cp tar t.$n; xz -ze$n t.$n & done; wait; mv -v $(ls -1S t.*.xz | tail -n 1) tar.xz; }
|
||||||
rm t.* || true
|
rm t.* || true
|
||||||
exts=()
|
exts=()
|
||||||
|
|
||||||
|
|
||||||
[ $do_sh ] && {
|
[ $do_sh ] && {
|
||||||
exts+=(sh)
|
exts+=(.sh)
|
||||||
echo creating unix sfx
|
echo creating unix sfx
|
||||||
(
|
(
|
||||||
sed "s/PACK_TS/$ts/; s/PACK_HTS/$hts/; s/CPP_VER/$ver/" <../scripts/sfx.sh |
|
sed "s/PACK_TS/$ts/; s/PACK_HTS/$hts/; s/CPP_VER/$ver/" <../scripts/sfx.sh |
|
||||||
@@ -224,17 +237,30 @@ echo creating unix sfx
|
|||||||
|
|
||||||
|
|
||||||
[ $do_py ] && {
|
[ $do_py ] && {
|
||||||
exts+=(py)
|
echo creating generic sfx
|
||||||
echo creating generic sfx
|
|
||||||
$pybin ../scripts/sfx.py --sfx-make tar.bz2 $ver $ts
|
py=../scripts/sfx.py
|
||||||
mv sfx.out $sfx_out.py
|
suf=
|
||||||
chmod 755 $sfx_out.*
|
[ $use_gz ] && {
|
||||||
|
sed -r 's/"r:bz2"/"r:gz"/' <$py >$py.t
|
||||||
|
py=$py.t
|
||||||
|
suf=-gz
|
||||||
|
}
|
||||||
|
|
||||||
|
$pybin $py --sfx-make tar.bz2 $ver $ts
|
||||||
|
mv sfx.out $sfx_out$suf.py
|
||||||
|
|
||||||
|
exts+=($suf.py)
|
||||||
|
[ $use_gz ] &&
|
||||||
|
rm $py
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
chmod 755 $sfx_out*
|
||||||
|
|
||||||
printf "done:\n"
|
printf "done:\n"
|
||||||
for ext in ${exts[@]}; do
|
for ext in ${exts[@]}; do
|
||||||
printf " %s\n" "$(realpath $sfx_out)."$ext
|
printf " %s\n" "$(realpath $sfx_out)"$ext
|
||||||
done
|
done
|
||||||
|
|
||||||
# apk add bash python3 tar xz bzip2
|
# apk add bash python3 tar xz bzip2
|
||||||
|
|||||||
187
scripts/sfx.py
187
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, runpy, tarfile, hashlib, platform, tempfile, traceback
|
import re, os, sys, time, shutil, signal, threading, tarfile, hashlib, platform, tempfile, traceback
|
||||||
|
import subprocess as sp
|
||||||
|
|
||||||
"""
|
"""
|
||||||
run me with any version of python, i will unpack and run copyparty
|
run me with any version of python, i will unpack and run copyparty
|
||||||
@@ -26,21 +27,21 @@ CKSUM = None
|
|||||||
STAMP = None
|
STAMP = None
|
||||||
|
|
||||||
PY2 = sys.version_info[0] == 2
|
PY2 = sys.version_info[0] == 2
|
||||||
|
WINDOWS = sys.platform in ["win32", "msys"]
|
||||||
sys.dont_write_bytecode = True
|
sys.dont_write_bytecode = True
|
||||||
me = os.path.abspath(os.path.realpath(__file__))
|
me = os.path.abspath(os.path.realpath(__file__))
|
||||||
cpp = None
|
|
||||||
|
|
||||||
|
|
||||||
def eprint(*args, **kwargs):
|
def eprint(*a, **ka):
|
||||||
kwargs["file"] = sys.stderr
|
ka["file"] = sys.stderr
|
||||||
print(*args, **kwargs)
|
print(*a, **ka)
|
||||||
|
|
||||||
|
|
||||||
def msg(*args, **kwargs):
|
def msg(*a, **ka):
|
||||||
if args:
|
if a:
|
||||||
args = ["[SFX]", args[0]] + list(args[1:])
|
a = ["[SFX]", a[0]] + list(a[1:])
|
||||||
|
|
||||||
eprint(*args, **kwargs)
|
eprint(*a, **ka)
|
||||||
|
|
||||||
|
|
||||||
# skip 1
|
# skip 1
|
||||||
@@ -155,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 [
|
||||||
@@ -208,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():
|
||||||
@@ -221,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):
|
||||||
@@ -232,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:
|
||||||
@@ -271,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():
|
||||||
@@ -306,46 +307,57 @@ def get_payload():
|
|||||||
if ofs < 0:
|
if ofs < 0:
|
||||||
raise Exception("could not find archive marker")
|
raise Exception("could not find archive marker")
|
||||||
|
|
||||||
# start reading from the final b"\n"
|
# start at final b"\n"
|
||||||
fpos = ofs + len(ptn) - 3
|
fpos = ofs + len(ptn) - 3
|
||||||
# msg("tar found at", fpos)
|
|
||||||
f.seek(fpos)
|
f.seek(fpos)
|
||||||
dpos = 0
|
dpos = 0
|
||||||
leftovers = b""
|
rem = b""
|
||||||
while True:
|
while True:
|
||||||
rbuf = f.read(1024 * 32)
|
rbuf = f.read(1024 * 32)
|
||||||
if rbuf:
|
if rbuf:
|
||||||
buf = leftovers + rbuf
|
buf = rem + rbuf
|
||||||
ofs = buf.rfind(b"\n")
|
ofs = buf.rfind(b"\n")
|
||||||
if len(buf) <= 4:
|
if len(buf) <= 4:
|
||||||
leftovers = buf
|
rem = buf
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if ofs >= len(buf) - 4:
|
if ofs >= len(buf) - 4:
|
||||||
leftovers = buf[ofs:]
|
rem = buf[ofs:]
|
||||||
buf = buf[:ofs]
|
buf = buf[:ofs]
|
||||||
else:
|
else:
|
||||||
leftovers = b"\n# "
|
rem = b"\n# "
|
||||||
else:
|
else:
|
||||||
buf = leftovers
|
buf = rem
|
||||||
|
|
||||||
fpos += len(buf) + 1
|
fpos += len(buf) + 1
|
||||||
buf = (
|
for a, b in [[b"\n# ", b""], [b"\n#r", b"\r"], [b"\n#n", b"\n"]]:
|
||||||
buf.replace(b"\n# ", b"")
|
buf = buf.replace(a, b)
|
||||||
.replace(b"\n#r", b"\r")
|
|
||||||
.replace(b"\n#n", b"\n")
|
|
||||||
)
|
|
||||||
dpos += len(buf) - 1
|
|
||||||
|
|
||||||
|
dpos += len(buf) - 1
|
||||||
yield buf
|
yield buf
|
||||||
|
|
||||||
if not rbuf:
|
if not rbuf:
|
||||||
break
|
break
|
||||||
|
|
||||||
|
|
||||||
|
def utime(top):
|
||||||
|
i = 0
|
||||||
|
files = [os.path.join(dp, p) for dp, dd, df in os.walk(top) for p in dd + df]
|
||||||
|
while WINDOWS:
|
||||||
|
t = int(time.time())
|
||||||
|
if i:
|
||||||
|
msg("utime {}, {}".format(i, t))
|
||||||
|
|
||||||
|
for f in files:
|
||||||
|
os.utime(f, (t, t))
|
||||||
|
|
||||||
|
i += 1
|
||||||
|
time.sleep(78123)
|
||||||
|
|
||||||
|
|
||||||
def confirm(rv):
|
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()
|
||||||
@@ -355,37 +367,59 @@ 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()
|
||||||
|
|
||||||
# "systemd-tmpfiles-clean.timer"?? HOW do you even come up with this shit
|
# block systemd-tmpfiles-clean.timer
|
||||||
try:
|
try:
|
||||||
import fcntl
|
import fcntl
|
||||||
|
|
||||||
fd = os.open(tmp, os.O_RDONLY)
|
fd = os.open(tmp, os.O_RDONLY)
|
||||||
fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB)
|
fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB)
|
||||||
tmp = os.readlink(tmp) # can't flock a symlink, even with O_NOFOLLOW
|
except Exception as ex:
|
||||||
except:
|
if not WINDOWS:
|
||||||
pass
|
msg("\033[31mflock:", repr(ex))
|
||||||
|
|
||||||
|
t = threading.Thread(target=utime, args=(tmp,))
|
||||||
|
t.daemon = True
|
||||||
|
t.start()
|
||||||
|
|
||||||
ld = [tmp, os.path.join(tmp, "dep-j2")]
|
ld = [tmp, os.path.join(tmp, "dep-j2")]
|
||||||
if j2ver:
|
if j2:
|
||||||
del ld[-1]
|
del ld[-1]
|
||||||
|
|
||||||
|
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
|
||||||
runpy.run_module(str("copyparty"), run_name=str("__main__"))
|
|
||||||
except SystemExit as ex:
|
p()
|
||||||
if ex.code:
|
|
||||||
confirm(ex.code)
|
|
||||||
except:
|
def run_s(ld):
|
||||||
confirm(1)
|
# fmt: off
|
||||||
|
c = "import sys,runpy;" + "".join(['sys.path.insert(0,r"' + x + '");' for x in ld]) + 'runpy.run_module("copyparty",run_name="__main__")'
|
||||||
|
c = [str(x) for x in [sys.executable, "-c", c] + list(sys.argv[1:])]
|
||||||
|
# fmt: on
|
||||||
|
msg("\n", c, "\n")
|
||||||
|
p = sp.Popen(c)
|
||||||
|
|
||||||
|
def bye(*a):
|
||||||
|
p.send_signal(signal.SIGINT)
|
||||||
|
|
||||||
|
signal.signal(signal.SIGTERM, bye)
|
||||||
|
p.wait()
|
||||||
|
|
||||||
|
raise SystemExit(p.returncode)
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
@@ -419,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__":
|
||||||
|
|||||||
@@ -17,14 +17,15 @@ __license__ = "MIT"
|
|||||||
__url__ = "https://github.com/9001/copyparty/"
|
__url__ = "https://github.com/9001/copyparty/"
|
||||||
|
|
||||||
|
|
||||||
def get_spd(nbyte, nsec):
|
def get_spd(nbyte, nfiles, nsec):
|
||||||
if not nsec:
|
if not nsec:
|
||||||
return "0.000 MB 0.000 sec 0.000 MB/s"
|
return "0.000 MB 0 files 0.000 sec 0.000 MB/s 0.000 f/s"
|
||||||
|
|
||||||
mb = nbyte / (1024 * 1024.0)
|
mb = nbyte / (1024 * 1024.0)
|
||||||
spd = mb / nsec
|
spd = mb / nsec
|
||||||
|
nspd = nfiles / nsec
|
||||||
|
|
||||||
return f"{mb:.3f} MB {nsec:.3f} sec {spd:.3f} MB/s"
|
return f"{mb:.3f} MB {nfiles} files {nsec:.3f} sec {spd:.3f} MB/s {nspd:.3f} f/s"
|
||||||
|
|
||||||
|
|
||||||
class Inf(object):
|
class Inf(object):
|
||||||
@@ -36,6 +37,7 @@ class Inf(object):
|
|||||||
self.mtx_reports = threading.Lock()
|
self.mtx_reports = threading.Lock()
|
||||||
|
|
||||||
self.n_byte = 0
|
self.n_byte = 0
|
||||||
|
self.n_file = 0
|
||||||
self.n_sec = 0
|
self.n_sec = 0
|
||||||
self.n_done = 0
|
self.n_done = 0
|
||||||
self.t0 = t0
|
self.t0 = t0
|
||||||
@@ -63,7 +65,8 @@ class Inf(object):
|
|||||||
continue
|
continue
|
||||||
|
|
||||||
msgs = msgs[-64:]
|
msgs = msgs[-64:]
|
||||||
msgs = [f"{get_spd(self.n_byte, self.n_sec)} {x}" for x in msgs]
|
spd = get_spd(self.n_byte, len(self.reports), self.n_sec)
|
||||||
|
msgs = [f"{spd} {x}" for x in msgs]
|
||||||
print("\n".join(msgs))
|
print("\n".join(msgs))
|
||||||
|
|
||||||
def report(self, fn, n_byte, n_sec):
|
def report(self, fn, n_byte, n_sec):
|
||||||
@@ -131,8 +134,9 @@ def main():
|
|||||||
|
|
||||||
num_threads = 8
|
num_threads = 8
|
||||||
read_sz = 32 * 1024
|
read_sz = 32 * 1024
|
||||||
|
targs = (q, inf, read_sz)
|
||||||
for _ in range(num_threads):
|
for _ in range(num_threads):
|
||||||
thr = threading.Thread(target=worker, args=(q, inf, read_sz,))
|
thr = threading.Thread(target=worker, args=targs)
|
||||||
thr.daemon = True
|
thr.daemon = True
|
||||||
thr.start()
|
thr.start()
|
||||||
|
|
||||||
@@ -151,14 +155,14 @@ def main():
|
|||||||
log = inf.reports
|
log = inf.reports
|
||||||
log.sort()
|
log.sort()
|
||||||
for nbyte, nsec, fn in log[-64:]:
|
for nbyte, nsec, fn in log[-64:]:
|
||||||
print(f"{get_spd(nbyte, nsec)} {fn}")
|
spd = get_spd(nbyte, len(log), nsec)
|
||||||
|
print(f"{spd} {fn}")
|
||||||
|
|
||||||
print()
|
print()
|
||||||
print("\n".join(inf.errors))
|
print("\n".join(inf.errors))
|
||||||
|
|
||||||
print(get_spd(inf.n_byte, t2 - t0))
|
print(get_spd(inf.n_byte, len(log), t2 - t0))
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
main()
|
main()
|
||||||
|
|
||||||
|
|||||||
33
tests/run.py
Executable file
33
tests/run.py
Executable file
@@ -0,0 +1,33 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import runpy
|
||||||
|
|
||||||
|
host = sys.argv[1]
|
||||||
|
sys.argv = sys.argv[:1] + sys.argv[2:]
|
||||||
|
sys.path.insert(0, ".")
|
||||||
|
|
||||||
|
|
||||||
|
def rp():
|
||||||
|
runpy.run_module("unittest", run_name="__main__")
|
||||||
|
|
||||||
|
|
||||||
|
if host == "vmprof":
|
||||||
|
rp()
|
||||||
|
|
||||||
|
elif host == "cprofile":
|
||||||
|
import cProfile
|
||||||
|
import pstats
|
||||||
|
|
||||||
|
log_fn = "cprofile.log"
|
||||||
|
cProfile.run("rp()", log_fn)
|
||||||
|
p = pstats.Stats(log_fn)
|
||||||
|
p.sort_stats(pstats.SortKey.CUMULATIVE).print_stats(64)
|
||||||
|
|
||||||
|
|
||||||
|
"""
|
||||||
|
python3.9 tests/run.py cprofile -v tests/test_httpcli.py
|
||||||
|
|
||||||
|
python3.9 -m pip install --user vmprof
|
||||||
|
python3.9 -m vmprof --lines -o vmprof.log tests/run.py vmprof -v tests/test_httpcli.py
|
||||||
|
"""
|
||||||
202
tests/test_httpcli.py
Normal file
202
tests/test_httpcli.py
Normal file
@@ -0,0 +1,202 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# coding: utf-8
|
||||||
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
|
import io
|
||||||
|
import os
|
||||||
|
import time
|
||||||
|
import shutil
|
||||||
|
import pprint
|
||||||
|
import tarfile
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
from argparse import Namespace
|
||||||
|
from copyparty.authsrv import AuthSrv
|
||||||
|
from copyparty.httpcli import HttpCli
|
||||||
|
|
||||||
|
from tests import util as tu
|
||||||
|
|
||||||
|
|
||||||
|
def hdr(query):
|
||||||
|
h = "GET /{} HTTP/1.1\r\nCookie: cppwd=o\r\nConnection: close\r\n\r\n"
|
||||||
|
return h.format(query).encode("utf-8")
|
||||||
|
|
||||||
|
|
||||||
|
class Cfg(Namespace):
|
||||||
|
def __init__(self, a=[], v=[], c=None):
|
||||||
|
super(Cfg, self).__init__(
|
||||||
|
a=a,
|
||||||
|
v=v,
|
||||||
|
c=c,
|
||||||
|
ed=False,
|
||||||
|
no_zip=False,
|
||||||
|
no_scandir=False,
|
||||||
|
no_sendfile=True,
|
||||||
|
nih=True,
|
||||||
|
mtp=[],
|
||||||
|
mte="a",
|
||||||
|
**{k: False for k in "e2d e2ds e2dsa e2t e2ts e2tsr".split()}
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TestHttpCli(unittest.TestCase):
|
||||||
|
def test(self):
|
||||||
|
td = os.path.join(tu.get_ramdisk(), "vfs")
|
||||||
|
try:
|
||||||
|
shutil.rmtree(td)
|
||||||
|
except OSError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
os.mkdir(td)
|
||||||
|
os.chdir(td)
|
||||||
|
|
||||||
|
self.dtypes = ["ra", "ro", "rx", "wa", "wo", "wx", "aa", "ao", "ax"]
|
||||||
|
self.can_read = ["ra", "ro", "aa", "ao"]
|
||||||
|
self.can_write = ["wa", "wo", "aa", "ao"]
|
||||||
|
self.fn = "g{:x}g".format(int(time.time() * 3))
|
||||||
|
|
||||||
|
allfiles = []
|
||||||
|
allvols = []
|
||||||
|
for top in self.dtypes:
|
||||||
|
allvols.append(top)
|
||||||
|
allfiles.append("/".join([top, self.fn]))
|
||||||
|
for s1 in self.dtypes:
|
||||||
|
p = "/".join([top, s1])
|
||||||
|
allvols.append(p)
|
||||||
|
allfiles.append(p + "/" + self.fn)
|
||||||
|
allfiles.append(p + "/n/" + self.fn)
|
||||||
|
for s2 in self.dtypes:
|
||||||
|
p = "/".join([top, s1, "n", s2])
|
||||||
|
os.makedirs(p)
|
||||||
|
allvols.append(p)
|
||||||
|
allfiles.append(p + "/" + self.fn)
|
||||||
|
|
||||||
|
for fp in allfiles:
|
||||||
|
with open(fp, "w") as f:
|
||||||
|
f.write("ok {}\n".format(fp))
|
||||||
|
|
||||||
|
for top in self.dtypes:
|
||||||
|
vcfg = []
|
||||||
|
for vol in allvols:
|
||||||
|
if not vol.startswith(top):
|
||||||
|
continue
|
||||||
|
|
||||||
|
mode = vol[-2]
|
||||||
|
usr = vol[-1]
|
||||||
|
if usr == "a":
|
||||||
|
usr = ""
|
||||||
|
|
||||||
|
if "/" not in vol:
|
||||||
|
vol += "/"
|
||||||
|
|
||||||
|
top, sub = vol.split("/", 1)
|
||||||
|
vcfg.append("{0}/{1}:{1}:{2}{3}".format(top, sub, mode, usr))
|
||||||
|
|
||||||
|
pprint.pprint(vcfg)
|
||||||
|
|
||||||
|
self.args = Cfg(v=vcfg, a=["o:o", "x:x"])
|
||||||
|
self.auth = AuthSrv(self.args, self.log)
|
||||||
|
vfiles = [x for x in allfiles if x.startswith(top)]
|
||||||
|
for fp in vfiles:
|
||||||
|
rok, wok = self.can_rw(fp)
|
||||||
|
furl = fp.split("/", 1)[1]
|
||||||
|
durl = furl.rsplit("/", 1)[0] if "/" in furl else ""
|
||||||
|
|
||||||
|
# file download
|
||||||
|
h, ret = self.curl(furl)
|
||||||
|
res = "ok " + fp in ret
|
||||||
|
print("[{}] {} {} = {}".format(fp, rok, wok, res))
|
||||||
|
if rok != res:
|
||||||
|
print("\033[33m{}\n# {}\033[0m".format(ret, furl))
|
||||||
|
self.fail()
|
||||||
|
|
||||||
|
# file browser: html
|
||||||
|
h, ret = self.curl(durl)
|
||||||
|
res = "'{}'".format(self.fn) in ret
|
||||||
|
print(res)
|
||||||
|
if rok != res:
|
||||||
|
print("\033[33m{}\n# {}\033[0m".format(ret, durl))
|
||||||
|
self.fail()
|
||||||
|
|
||||||
|
# file browser: json
|
||||||
|
url = durl + "?ls"
|
||||||
|
h, ret = self.curl(url)
|
||||||
|
res = '"{}"'.format(self.fn) in ret
|
||||||
|
print(res)
|
||||||
|
if rok != res:
|
||||||
|
print("\033[33m{}\n# {}\033[0m".format(ret, url))
|
||||||
|
self.fail()
|
||||||
|
|
||||||
|
# tar
|
||||||
|
url = durl + "?tar"
|
||||||
|
h, b = self.curl(url, True)
|
||||||
|
# with open(os.path.join(td, "tar"), "wb") as f:
|
||||||
|
# f.write(b)
|
||||||
|
try:
|
||||||
|
tar = tarfile.open(fileobj=io.BytesIO(b)).getnames()
|
||||||
|
except:
|
||||||
|
tar = []
|
||||||
|
tar = ["/".join([y for y in [top, durl, x] if y]) for x in tar]
|
||||||
|
tar = [[x] + self.can_rw(x) for x in tar]
|
||||||
|
tar_ok = [x[0] for x in tar if x[1]]
|
||||||
|
tar_ng = [x[0] for x in tar if not x[1]]
|
||||||
|
self.assertEqual([], tar_ng)
|
||||||
|
|
||||||
|
if durl.split("/")[-1] in self.can_read:
|
||||||
|
ref = [x for x in vfiles if self.in_dive(top + "/" + durl, x)]
|
||||||
|
for f in ref:
|
||||||
|
print("{}: {}".format("ok" if f in tar_ok else "NG", f))
|
||||||
|
ref.sort()
|
||||||
|
tar_ok.sort()
|
||||||
|
self.assertEqual(ref, tar_ok)
|
||||||
|
|
||||||
|
# stash
|
||||||
|
h, ret = self.put(url)
|
||||||
|
res = h.startswith("HTTP/1.1 200 ")
|
||||||
|
self.assertEqual(res, wok)
|
||||||
|
|
||||||
|
def can_rw(self, fp):
|
||||||
|
# lowest non-neutral folder declares permissions
|
||||||
|
expect = fp.split("/")[:-1]
|
||||||
|
for x in reversed(expect):
|
||||||
|
if x != "n":
|
||||||
|
expect = x
|
||||||
|
break
|
||||||
|
|
||||||
|
return [expect in self.can_read, expect in self.can_write]
|
||||||
|
|
||||||
|
def in_dive(self, top, fp):
|
||||||
|
# archiver bails at first inaccessible subvolume
|
||||||
|
top = top.strip("/").split("/")
|
||||||
|
fp = fp.split("/")
|
||||||
|
for f1, f2 in zip(top, fp):
|
||||||
|
if f1 != f2:
|
||||||
|
return False
|
||||||
|
|
||||||
|
for f in fp[len(top) :]:
|
||||||
|
if f == self.fn:
|
||||||
|
return True
|
||||||
|
if f not in self.can_read and f != "n":
|
||||||
|
return False
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def put(self, url):
|
||||||
|
buf = "PUT /{0} HTTP/1.1\r\nCookie: cppwd=o\r\nConnection: close\r\nContent-Length: {1}\r\n\r\nok {0}\n"
|
||||||
|
buf = buf.format(url, len(url) + 4).encode("utf-8")
|
||||||
|
conn = tu.VHttpConn(self.args, self.auth, self.log, buf)
|
||||||
|
HttpCli(conn).run()
|
||||||
|
return conn.s._reply.decode("utf-8").split("\r\n\r\n", 1)
|
||||||
|
|
||||||
|
def curl(self, url, binary=False):
|
||||||
|
conn = tu.VHttpConn(self.args, self.auth, self.log, hdr(url))
|
||||||
|
HttpCli(conn).run()
|
||||||
|
if binary:
|
||||||
|
h, b = conn.s._reply.split(b"\r\n\r\n", 1)
|
||||||
|
return [h.decode("utf-8"), b]
|
||||||
|
|
||||||
|
return conn.s._reply.decode("utf-8").split("\r\n\r\n", 1)
|
||||||
|
|
||||||
|
def log(self, src, msg, c=0):
|
||||||
|
# print(repr(msg))
|
||||||
|
pass
|
||||||
@@ -3,22 +3,24 @@
|
|||||||
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):
|
||||||
ex = {k: False for k in "e2d e2ds e2dsa e2t e2ts e2tsr mte".split()}
|
ex = {k: False for k in "e2d e2ds e2dsa e2t e2ts e2tsr".split()}
|
||||||
|
ex["mtp"] = []
|
||||||
|
ex["mte"] = "a"
|
||||||
super(Cfg, self).__init__(a=a, v=v, c=c, **ex)
|
super(Cfg, self).__init__(a=a, v=v, c=c, **ex)
|
||||||
|
|
||||||
|
|
||||||
@@ -49,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:
|
||||||
@@ -266,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