mirror of
https://github.com/9001/copyparty.git
synced 2025-11-04 13:53:18 +00:00
Compare commits
61 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e815c091b9 | ||
|
|
963529b7cf | ||
|
|
638a52374d | ||
|
|
d9d42b7aa2 | ||
|
|
ec7e5f36a2 | ||
|
|
56110883ea | ||
|
|
7f8d7d6006 | ||
|
|
49e4fb7e12 | ||
|
|
8dbbea473f | ||
|
|
3d375d5114 | ||
|
|
f3eae67d97 | ||
|
|
40c1b19235 | ||
|
|
ccaf0ab159 | ||
|
|
d07f147423 | ||
|
|
f5cb9f92b9 | ||
|
|
f991f74983 | ||
|
|
6b3295059e | ||
|
|
b18a07ae6b | ||
|
|
8ab03dabda | ||
|
|
5e760e35dc | ||
|
|
afbfa04514 | ||
|
|
7aace470c5 | ||
|
|
b4acb24f6a | ||
|
|
bcee8a4934 | ||
|
|
36b0718542 | ||
|
|
9a92bca45d | ||
|
|
b07445a363 | ||
|
|
a62ec0c27e | ||
|
|
57e3a2d382 | ||
|
|
b61022b374 | ||
|
|
a3e2b2ec87 | ||
|
|
a83d3f8801 | ||
|
|
90c5f2b9d2 | ||
|
|
4885653c07 | ||
|
|
21e1cd87ca | ||
|
|
81f82e8e9f | ||
|
|
c0e31851da | ||
|
|
6599c3eced | ||
|
|
5d6c61a861 | ||
|
|
1a5c66edd3 | ||
|
|
deae9fe95a | ||
|
|
abd65c6334 | ||
|
|
8137a99904 | ||
|
|
6f6f9c1f74 | ||
|
|
7b575f716f | ||
|
|
6ba6ea3572 | ||
|
|
9a22ad5ea3 | ||
|
|
beaab9778e | ||
|
|
f327bdb6b4 | ||
|
|
ae180e0f5f | ||
|
|
e3f1d19756 | ||
|
|
93c2bd6ef6 | ||
|
|
4d0e5ff6db | ||
|
|
0893f06919 | ||
|
|
46b6abde3f | ||
|
|
0696610dee | ||
|
|
edf0d3684c | ||
|
|
7af159f5f6 | ||
|
|
7f2cb6764a | ||
|
|
96495a9bf1 | ||
|
|
b2fafec5fc |
6
.vscode/tasks.json
vendored
6
.vscode/tasks.json
vendored
@@ -8,8 +8,10 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"label": "no_dbg",
|
"label": "no_dbg",
|
||||||
"command": "${config:python.pythonPath} -m copyparty -ed -emp -e2dsa -e2ts -a ed:wark -v srv::r:aed:cnodupe -v dist:dist:r ;exit 1",
|
"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"
|
||||||
|
// -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
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
70
README.md
70
README.md
@@ -13,6 +13,30 @@ turn your phone or raspi into a portable file server with resumable uploads/down
|
|||||||
* code standard: `black`
|
* code standard: `black`
|
||||||
|
|
||||||
|
|
||||||
|
## readme toc
|
||||||
|
|
||||||
|
* top
|
||||||
|
* [quickstart](#quickstart)
|
||||||
|
* [notes](#notes)
|
||||||
|
* [status](#status)
|
||||||
|
* [bugs](#bugs)
|
||||||
|
* [usage](#usage)
|
||||||
|
* [searching](#searching)
|
||||||
|
* [search configuration](#search-configuration)
|
||||||
|
* [metadata from audio files](#metadata-from-audio-files)
|
||||||
|
* [file parser plugins](#file-parser-plugins)
|
||||||
|
* [complete examples](#complete-examples)
|
||||||
|
* [client examples](#client-examples)
|
||||||
|
* [dependencies](#dependencies)
|
||||||
|
* [optional gpl stuff](#optional-gpl-stuff)
|
||||||
|
* [sfx](#sfx)
|
||||||
|
* [sfx repack](#sfx-repack)
|
||||||
|
* [install on android](#install-on-android)
|
||||||
|
* [dev env setup](#dev-env-setup)
|
||||||
|
* [how to release](#how-to-release)
|
||||||
|
* [todo](#todo)
|
||||||
|
|
||||||
|
|
||||||
## quickstart
|
## quickstart
|
||||||
|
|
||||||
download [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) and you're all set!
|
download [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) and you're all set!
|
||||||
@@ -74,6 +98,16 @@ summary: it works! you can use it! (but technically not even close to beta)
|
|||||||
* probably more, pls let me know
|
* probably more, pls let me know
|
||||||
|
|
||||||
|
|
||||||
|
# usage
|
||||||
|
|
||||||
|
the browser has the following hotkeys
|
||||||
|
* `0..9` jump to 10%..90%
|
||||||
|
* `U/O` skip 10sec back/forward
|
||||||
|
* `J/L` prev/next song
|
||||||
|
* `I/K` prev/next folder
|
||||||
|
* `P` parent folder
|
||||||
|
|
||||||
|
|
||||||
# searching
|
# 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:
|
||||||
@@ -100,19 +134,24 @@ through arguments:
|
|||||||
* `-e2tsr` deletes all existing tags, so a full reindex
|
* `-e2tsr` deletes all existing tags, so a full reindex
|
||||||
|
|
||||||
the same arguments can be set as volume flags, in addition to `d2d` and `d2t` for disabling:
|
the same arguments can be set as volume flags, in addition to `d2d` and `d2t` for disabling:
|
||||||
* `-v ~/music::ce2dsa:ce2tsr` does a full reindex of everything on startup
|
* `-v ~/music::r:ce2dsa:ce2tsr` does a full reindex of everything on startup
|
||||||
* `-v ~/music::cd2d` disables **all** indexing, even if any `-e2*` are on
|
* `-v ~/music::r:cd2d` disables **all** indexing, even if any `-e2*` are on
|
||||||
* `-v ~/music::cd2t` disables all `-e2t*` (tags), does not affect `-e2d*`
|
* `-v ~/music::r:cd2t` disables all `-e2t*` (tags), does not affect `-e2d*`
|
||||||
|
|
||||||
`e2tsr` is probably always overkill, since `e2ds`/`e2dsa` would pick up any file modifications and cause `e2ts` to reindex those
|
`e2tsr` is probably always overkill, since `e2ds`/`e2dsa` would pick up any file modifications and cause `e2ts` to reindex those
|
||||||
|
|
||||||
|
|
||||||
|
## metadata from audio files
|
||||||
|
|
||||||
`-mte` decides which tags to index and display in the browser (and also the display order), this can be changed per-volume:
|
`-mte` decides which tags to index and display in the browser (and also the display order), this can be changed per-volume:
|
||||||
* `-v ~/music::cmte=title,artist` indexes and displays *title* followed by *artist*
|
* `-v ~/music::r:cmte=title,artist` indexes and displays *title* followed by *artist*
|
||||||
|
|
||||||
if you add/remove a tag from `mte` you will need to run with `-e2tsr` once to rebuild the database, otherwise only new files will be affected
|
if you add/remove a tag from `mte` you will need to run with `-e2tsr` once to rebuild the database, otherwise only new files will be affected
|
||||||
|
|
||||||
`-mtm` can be used to add or redefine a metadata mapping, say you have media files with `foo` and `bar` tags and you want them to display as `qux` in the browser (preferring `foo` if both are present), then do `-mtm qux=foo,bar` and now you can `-mte artist,title,qux`
|
`-mtm` can be used to add or redefine a metadata mapping, say you have media files with `foo` and `bar` tags and you want them to display as `qux` in the browser (preferring `foo` if both are present), then do `-mtm qux=foo,bar` and now you can `-mte artist,title,qux`
|
||||||
|
|
||||||
|
tags that start with a `.` such as `.bpm` and `.dur`(ation) indicate numeric value
|
||||||
|
|
||||||
see the beautiful mess of a dictionary in [mtag.py](https://github.com/9001/copyparty/blob/master/copyparty/mtag.py) for the default mappings (should cover mp3,opus,flac,m4a,wav,aif,)
|
see the beautiful mess of a dictionary in [mtag.py](https://github.com/9001/copyparty/blob/master/copyparty/mtag.py) for the default mappings (should cover mp3,opus,flac,m4a,wav,aif,)
|
||||||
|
|
||||||
`--no-mutagen` disables mutagen and uses ffprobe instead, which...
|
`--no-mutagen` disables mutagen and uses ffprobe instead, which...
|
||||||
@@ -122,6 +161,21 @@ see the beautiful mess of a dictionary in [mtag.py](https://github.com/9001/copy
|
|||||||
* more importantly runs ffprobe on incoming files which is bad if your ffmpeg has a cve
|
* more importantly runs ffprobe on incoming files which is bad if your ffmpeg has a cve
|
||||||
|
|
||||||
|
|
||||||
|
## file parser plugins
|
||||||
|
|
||||||
|
copyparty can invoke external programs to collect additional metadata for files using `mtp` (as argument or volume flag), there is a default timeout of 30sec
|
||||||
|
|
||||||
|
* `-mtp .bpm=~/bin/audio-bpm.py` will execute `~/bin/audio-bpm.py` with the audio file as argument 1 to provide the `.bpm` tag, if that does not exist in the audio metadata
|
||||||
|
* `-mtp key=f,t5,~/bin/audio-key.py` uses `~/bin/audio-key.py` to get the `key` tag, replacing any existing metadata tag (`f,`), aborting if it takes longer than 5sec (`t5,`)
|
||||||
|
* `-v ~/music::r:cmtp=.bpm=~/bin/audio-bpm.py:cmtp=key=f,t5,~/bin/audio-key.py` both as a per-volume config wow this is getting ugly
|
||||||
|
|
||||||
|
|
||||||
|
## complete examples
|
||||||
|
|
||||||
|
* read-only music server with bpm and key scanning
|
||||||
|
`python copyparty-sfx.py -v /mnt/nas/music:/music:r -e2dsa -e2ts -mtp .bpm=f,audio-bpm.py -mtp key=f,audio-key.py`
|
||||||
|
|
||||||
|
|
||||||
# client examples
|
# client examples
|
||||||
|
|
||||||
* javascript: dump some state into a file (two separate examples)
|
* javascript: dump some state into a file (two separate examples)
|
||||||
@@ -158,6 +212,13 @@ copyparty returns a truncated sha512sum of your PUT/POST as base64; you can gene
|
|||||||
* `Pillow` (requires py2.7 or py3.5+)
|
* `Pillow` (requires py2.7 or py3.5+)
|
||||||
|
|
||||||
|
|
||||||
|
## optional gpl stuff
|
||||||
|
|
||||||
|
some bundled tools have copyleft dependencies, see [./bin/#mtag](bin/#mtag)
|
||||||
|
|
||||||
|
these are standalone and will never be imported / evaluated by copyparty
|
||||||
|
|
||||||
|
|
||||||
# sfx
|
# sfx
|
||||||
|
|
||||||
currently there are two self-contained binaries:
|
currently there are two self-contained binaries:
|
||||||
@@ -212,6 +273,7 @@ pip install black bandit pylint flake8 # vscode tooling
|
|||||||
in the `scripts` folder:
|
in the `scripts` folder:
|
||||||
|
|
||||||
* run `make -C deps-docker` to build all dependencies
|
* run `make -C deps-docker` to build all dependencies
|
||||||
|
* `git tag v1.2.3 && git push origin --tags`
|
||||||
* create github release with `make-tgz-release.sh`
|
* create github release with `make-tgz-release.sh`
|
||||||
* upload to pypi with `make-pypi-release.(sh|bat)`
|
* upload to pypi with `make-pypi-release.(sh|bat)`
|
||||||
* create sfx with `make-sfx.sh`
|
* create sfx with `make-sfx.sh`
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# copyparty-fuse.py
|
# [`copyparty-fuse.py`](copyparty-fuse.py)
|
||||||
* mount a copyparty server as a local filesystem (read-only)
|
* mount a copyparty server as a local filesystem (read-only)
|
||||||
* **supports Windows!** -- expect `194 MiB/s` sequential read
|
* **supports Windows!** -- expect `194 MiB/s` sequential read
|
||||||
* **supports Linux** -- expect `117 MiB/s` sequential read
|
* **supports Linux** -- expect `117 MiB/s` sequential read
|
||||||
@@ -29,7 +29,7 @@ you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releas
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
# copyparty-fuse🅱️.py
|
# [`copyparty-fuse🅱️.py`](copyparty-fuseb.py)
|
||||||
* mount a copyparty server as a local filesystem (read-only)
|
* mount a copyparty server as a local filesystem (read-only)
|
||||||
* does the same thing except more correct, `samba` approves
|
* does the same thing except more correct, `samba` approves
|
||||||
* **supports Linux** -- expect `18 MiB/s` (wait what)
|
* **supports Linux** -- expect `18 MiB/s` (wait what)
|
||||||
@@ -37,5 +37,11 @@ you could replace winfsp with [dokan](https://github.com/dokan-dev/dokany/releas
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
# copyparty-fuse-streaming.py
|
# [`copyparty-fuse-streaming.py`](copyparty-fuse-streaming.py)
|
||||||
* pretend this doesn't exist
|
* pretend this doesn't exist
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# [`mtag/`](mtag/)
|
||||||
|
* standalone programs which perform misc. file analysis
|
||||||
|
* copyparty can Popen programs like these during file indexing to collect additional metadata
|
||||||
|
|||||||
@@ -1008,6 +1008,12 @@ def main():
|
|||||||
log = null_log
|
log = null_log
|
||||||
dbg = null_log
|
dbg = null_log
|
||||||
|
|
||||||
|
if ar.a and ar.a.startswith("$"):
|
||||||
|
fn = ar.a[1:]
|
||||||
|
log("reading password from file [{}]".format(fn))
|
||||||
|
with open(fn, "rb") as f:
|
||||||
|
ar.a = f.read().decode("utf-8").strip()
|
||||||
|
|
||||||
if WINDOWS:
|
if WINDOWS:
|
||||||
os.system("rem")
|
os.system("rem")
|
||||||
|
|
||||||
|
|||||||
34
bin/mtag/README.md
Normal file
34
bin/mtag/README.md
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
standalone programs which take an audio file as argument
|
||||||
|
|
||||||
|
some of these rely on libraries which are not MIT-compatible
|
||||||
|
|
||||||
|
* [audio-bpm.py](./audio-bpm.py) detects the BPM of music using the BeatRoot Vamp Plugin; imports GPL2
|
||||||
|
* [audio-key.py](./audio-key.py) detects the melodic key of music using the Mixxx fork of keyfinder; imports GPL3
|
||||||
|
|
||||||
|
|
||||||
|
# dependencies
|
||||||
|
|
||||||
|
run [`install-deps.sh`](install-deps.sh) to build/install most dependencies required by these programs (supports windows/linux/macos)
|
||||||
|
|
||||||
|
*alternatively* (or preferably) use packages from your distro instead, then you'll need at least these:
|
||||||
|
|
||||||
|
* from distro: `numpy vamp-plugin-sdk beatroot-vamp mixxx-keyfinder ffmpeg`
|
||||||
|
* from pypy: `keyfinder vamp`
|
||||||
|
|
||||||
|
|
||||||
|
# usage from copyparty
|
||||||
|
|
||||||
|
`copyparty -e2dsa -e2ts -mtp key=f,audio-key.py -mtp .bpm=f,audio-bpm.py`
|
||||||
|
|
||||||
|
* `f,` makes the detected value replace any existing values
|
||||||
|
* the `.` in `.bpm` indicates numeric value
|
||||||
|
* assumes the python files are in the folder you're launching copyparty from, replace the filename with a relative/absolute path if that's not the case
|
||||||
|
* `mtp` modules will not run if a file has existing tags in the db, so clear out the tags with `-e2tsr` the first time you launch with new `mtp` options
|
||||||
|
|
||||||
|
|
||||||
|
## usage with volume-flags
|
||||||
|
|
||||||
|
instead of affecting all volumes, you can set the options for just one volume like so:
|
||||||
|
```
|
||||||
|
copyparty -v /mnt/nas/music:/music:r:cmtp=key=f,audio-key.py:cmtp=.bpm=f,audio-bpm.py:ce2dsa:ce2ts
|
||||||
|
```
|
||||||
69
bin/mtag/audio-bpm.py
Executable file
69
bin/mtag/audio-bpm.py
Executable file
@@ -0,0 +1,69 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import vamp
|
||||||
|
import tempfile
|
||||||
|
import numpy as np
|
||||||
|
import subprocess as sp
|
||||||
|
|
||||||
|
from copyparty.util import fsenc
|
||||||
|
|
||||||
|
"""
|
||||||
|
dep: vamp
|
||||||
|
dep: beatroot-vamp
|
||||||
|
dep: ffmpeg
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def det(tf):
|
||||||
|
# fmt: off
|
||||||
|
sp.check_call([
|
||||||
|
"ffmpeg",
|
||||||
|
"-nostdin",
|
||||||
|
"-hide_banner",
|
||||||
|
"-v", "fatal",
|
||||||
|
"-ss", "13",
|
||||||
|
"-y", "-i", fsenc(sys.argv[1]),
|
||||||
|
"-ac", "1",
|
||||||
|
"-ar", "22050",
|
||||||
|
"-t", "300",
|
||||||
|
"-f", "f32le",
|
||||||
|
tf
|
||||||
|
])
|
||||||
|
# fmt: on
|
||||||
|
|
||||||
|
with open(tf, "rb") as f:
|
||||||
|
d = np.fromfile(f, dtype=np.float32)
|
||||||
|
try:
|
||||||
|
# 98% accuracy on jcore
|
||||||
|
c = vamp.collect(d, 22050, "beatroot-vamp:beatroot")
|
||||||
|
cl = c["list"]
|
||||||
|
except:
|
||||||
|
# fallback; 73% accuracy
|
||||||
|
plug = "vamp-example-plugins:fixedtempo"
|
||||||
|
c = vamp.collect(d, 22050, plug, parameters={"maxdflen": 40})
|
||||||
|
print(c["list"][0]["label"].split(" ")[0])
|
||||||
|
return
|
||||||
|
|
||||||
|
# throws if detection failed:
|
||||||
|
bpm = float(cl[-1]["timestamp"] - cl[1]["timestamp"])
|
||||||
|
bpm = round(60 * ((len(cl) - 1) / bpm), 2)
|
||||||
|
print(f"{bpm:.2f}")
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
with tempfile.NamedTemporaryFile(suffix=".pcm", delete=False) as f:
|
||||||
|
f.write(b"h")
|
||||||
|
tf = f.name
|
||||||
|
|
||||||
|
try:
|
||||||
|
det(tf)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
finally:
|
||||||
|
os.unlink(tf)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
18
bin/mtag/audio-key.py
Executable file
18
bin/mtag/audio-key.py
Executable file
@@ -0,0 +1,18 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import keyfinder
|
||||||
|
|
||||||
|
"""
|
||||||
|
dep: github/mixxxdj/libkeyfinder
|
||||||
|
dep: pypi/keyfinder
|
||||||
|
dep: ffmpeg
|
||||||
|
|
||||||
|
note: cannot fsenc
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
try:
|
||||||
|
print(keyfinder.key(sys.argv[1]).camelot())
|
||||||
|
except:
|
||||||
|
pass
|
||||||
265
bin/mtag/install-deps.sh
Executable file
265
bin/mtag/install-deps.sh
Executable file
@@ -0,0 +1,265 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
|
||||||
|
|
||||||
|
# install dependencies for audio-*.py
|
||||||
|
#
|
||||||
|
# linux: requires {python3,ffmpeg,fftw}-dev py3-{wheel,pip} py3-numpy{,-dev} vamp-sdk-dev patchelf
|
||||||
|
# win64: requires msys2-mingw64 environment
|
||||||
|
# macos: requires macports
|
||||||
|
#
|
||||||
|
# has the following manual dependencies, especially on mac:
|
||||||
|
# https://www.vamp-plugins.org/pack.html
|
||||||
|
#
|
||||||
|
# installs stuff to the following locations:
|
||||||
|
# ~/pe/
|
||||||
|
# whatever your python uses for --user packages
|
||||||
|
#
|
||||||
|
# does the following terrible things:
|
||||||
|
# modifies the keyfinder python lib to load the .so in ~/pe
|
||||||
|
|
||||||
|
|
||||||
|
linux=1
|
||||||
|
|
||||||
|
win=
|
||||||
|
[ ! -z "$MSYSTEM" ] || [ -e /msys2.exe ] && {
|
||||||
|
[ "$MSYSTEM" = MINGW64 ] || {
|
||||||
|
echo windows detected, msys2-mingw64 required
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
pacman -S --needed mingw-w64-x86_64-{ffmpeg,python,python-pip,vamp-plugin-sdk}
|
||||||
|
win=1
|
||||||
|
linux=
|
||||||
|
}
|
||||||
|
|
||||||
|
mac=
|
||||||
|
[ $(uname -s) = Darwin ] && {
|
||||||
|
#pybin="$(printf '%s\n' /opt/local/bin/python* | (sed -E 's/(.*\/[^/0-9]+)([0-9]?[^/]*)$/\2 \1/' || cat) | (sort -nr || cat) | (sed -E 's/([^ ]*) (.*)/\2\1/' || cat) | grep -E '/(python|pypy)[0-9\.-]*$' | head -n 1)"
|
||||||
|
pybin=/opt/local/bin/python3.9
|
||||||
|
[ -e "$pybin" ] || {
|
||||||
|
echo mac detected, python3 from macports required
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
pkgs='ffmpeg python39 py39-wheel'
|
||||||
|
ninst=$(port installed | awk '/^ /{print$1}' | sort | uniq | grep -E '^('"$(echo "$pkgs" | tr ' ' '|')"')$' | wc -l)
|
||||||
|
[ $ninst -eq 3 ] || {
|
||||||
|
sudo port install $pkgs
|
||||||
|
}
|
||||||
|
mac=1
|
||||||
|
linux=
|
||||||
|
}
|
||||||
|
|
||||||
|
hash -r
|
||||||
|
|
||||||
|
[ $mac ] || {
|
||||||
|
command -v python3 && pybin=python3 || pybin=python
|
||||||
|
}
|
||||||
|
|
||||||
|
$pybin -m pip install --user numpy
|
||||||
|
|
||||||
|
|
||||||
|
command -v gnutar && tar() { gnutar "$@"; }
|
||||||
|
command -v gtar && tar() { gtar "$@"; }
|
||||||
|
command -v gsed && sed() { gsed "$@"; }
|
||||||
|
|
||||||
|
|
||||||
|
need() {
|
||||||
|
command -v $1 >/dev/null || {
|
||||||
|
echo need $1
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
need cmake
|
||||||
|
need ffmpeg
|
||||||
|
need $pybin
|
||||||
|
#need patchelf
|
||||||
|
|
||||||
|
|
||||||
|
td="$(mktemp -d)"
|
||||||
|
cln() {
|
||||||
|
rm -rf "$td"
|
||||||
|
}
|
||||||
|
trap cln EXIT
|
||||||
|
cd "$td"
|
||||||
|
pwd
|
||||||
|
|
||||||
|
|
||||||
|
dl_text() {
|
||||||
|
command -v curl >/dev/null && exec curl "$@"
|
||||||
|
exec wget -O- "$@"
|
||||||
|
}
|
||||||
|
dl_files() {
|
||||||
|
local yolo= ex=
|
||||||
|
[ $1 = "yolo" ] && yolo=1 && ex=k && shift
|
||||||
|
command -v curl >/dev/null && exec curl -${ex}JOL "$@"
|
||||||
|
|
||||||
|
[ $yolo ] && ex=--no-check-certificate
|
||||||
|
exec wget --trust-server-names $ex "$@"
|
||||||
|
}
|
||||||
|
export -f dl_files
|
||||||
|
|
||||||
|
|
||||||
|
github_tarball() {
|
||||||
|
dl_text "$1" |
|
||||||
|
tee json |
|
||||||
|
(
|
||||||
|
# prefer jq if available
|
||||||
|
jq -r '.tarball_url' ||
|
||||||
|
|
||||||
|
# fallback to awk (sorry)
|
||||||
|
awk -F\" '/"tarball_url": "/ {print$4}'
|
||||||
|
) |
|
||||||
|
tee /dev/stderr |
|
||||||
|
tr -d '\r' | tr '\n' '\0' |
|
||||||
|
xargs -0 bash -c 'dl_files "$@"' _
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
gitlab_tarball() {
|
||||||
|
dl_text "$1" |
|
||||||
|
tee json |
|
||||||
|
(
|
||||||
|
# prefer jq if available
|
||||||
|
jq -r '.[0].assets.sources[]|select(.format|test("tar.gz")).url' ||
|
||||||
|
|
||||||
|
# fallback to abomination
|
||||||
|
tr \" '\n' | grep -E '\.tar\.gz$' | head -n 1
|
||||||
|
) |
|
||||||
|
tee /dev/stderr |
|
||||||
|
tr -d '\r' | tr '\n' '\0' |
|
||||||
|
tee links |
|
||||||
|
xargs -0 bash -c 'dl_files "$@"' _
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
install_keyfinder() {
|
||||||
|
# windows support:
|
||||||
|
# use msys2 in mingw-w64 mode
|
||||||
|
# pacman -S --needed mingw-w64-x86_64-{ffmpeg,python}
|
||||||
|
|
||||||
|
github_tarball https://api.github.com/repos/mixxxdj/libkeyfinder/releases/latest
|
||||||
|
|
||||||
|
tar -xf mixxxdj-libkeyfinder-*
|
||||||
|
rm -- *.tar.gz
|
||||||
|
cd mixxxdj-libkeyfinder*
|
||||||
|
|
||||||
|
h="$HOME"
|
||||||
|
so="lib/libkeyfinder.so"
|
||||||
|
memes=()
|
||||||
|
|
||||||
|
[ $win ] &&
|
||||||
|
so="bin/libkeyfinder.dll" &&
|
||||||
|
h="$(printf '%s\n' "$USERPROFILE" | tr '\\' '/')" &&
|
||||||
|
memes+=(-G "MinGW Makefiles" -DBUILD_TESTING=OFF)
|
||||||
|
|
||||||
|
[ $mac ] &&
|
||||||
|
so="lib/libkeyfinder.dylib"
|
||||||
|
|
||||||
|
cmake -DCMAKE_INSTALL_PREFIX="$h/pe/keyfinder" "${memes[@]}" -S . -B build
|
||||||
|
cmake --build build --parallel $(nproc || echo 4)
|
||||||
|
cmake --install build
|
||||||
|
|
||||||
|
libpath="$h/pe/keyfinder/$so"
|
||||||
|
[ $linux ] && [ ! -e "$libpath" ] &&
|
||||||
|
so=lib64/libkeyfinder.so
|
||||||
|
|
||||||
|
libpath="$h/pe/keyfinder/$so"
|
||||||
|
[ -e "$libpath" ] || {
|
||||||
|
echo "so not found at $sop"
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
# rm -rf /Users/ed/Library/Python/3.9/lib/python/site-packages/*keyfinder*
|
||||||
|
CFLAGS="-I$h/pe/keyfinder/include -I/opt/local/include" \
|
||||||
|
LDFLAGS="-L$h/pe/keyfinder/lib -L$h/pe/keyfinder/lib64 -L/opt/local/lib" \
|
||||||
|
PKG_CONFIG_PATH=/c/msys64/mingw64/lib/pkgconfig \
|
||||||
|
$pybin -m pip install --user keyfinder
|
||||||
|
|
||||||
|
pypath="$($pybin -c 'import keyfinder; print(keyfinder.__file__)')"
|
||||||
|
for pyso in "${pypath%/*}"/*.so; do
|
||||||
|
[ -e "$pyso" ] || break
|
||||||
|
patchelf --set-rpath "${libpath%/*}" "$pyso" ||
|
||||||
|
echo "WARNING: patchelf failed (only fatal on musl-based distros)"
|
||||||
|
done
|
||||||
|
|
||||||
|
mv "$pypath"{,.bak}
|
||||||
|
(
|
||||||
|
printf 'import ctypes\nctypes.cdll.LoadLibrary("%s")\n' "$libpath"
|
||||||
|
cat "$pypath.bak"
|
||||||
|
) >"$pypath"
|
||||||
|
|
||||||
|
echo
|
||||||
|
echo libkeyfinder successfully installed to the following locations:
|
||||||
|
echo " $libpath"
|
||||||
|
echo " $pypath"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
have_beatroot() {
|
||||||
|
$pybin -c 'import vampyhost, sys; plugs = vampyhost.list_plugins(); sys.exit(0 if "beatroot-vamp:beatroot" in plugs else 1)'
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
install_vamp() {
|
||||||
|
# windows support:
|
||||||
|
# use msys2 in mingw-w64 mode
|
||||||
|
# pacman -S --needed mingw-w64-x86_64-{ffmpeg,python,python-pip,vamp-plugin-sdk}
|
||||||
|
|
||||||
|
$pybin -m pip install --user vamp
|
||||||
|
|
||||||
|
have_beatroot || {
|
||||||
|
printf '\033[33mcould not find the vamp beatroot plugin, building from source\033[0m\n'
|
||||||
|
(dl_files yolo https://code.soundsoftware.ac.uk/attachments/download/885/beatroot-vamp-v1.0.tar.gz)
|
||||||
|
sha512sum -c <(
|
||||||
|
echo "1f444d1d58ccf565c0adfe99f1a1aa62789e19f5071e46857e2adfbc9d453037bc1c4dcb039b02c16240e9b97f444aaff3afb625c86aa2470233e711f55b6874 -"
|
||||||
|
) <beatroot-vamp-v1.0.tar.gz
|
||||||
|
tar -xf beatroot-vamp-v1.0.tar.gz
|
||||||
|
cd beatroot-vamp-v1.0
|
||||||
|
make -f Makefile.linux -j4
|
||||||
|
# /home/ed/vamp /home/ed/.vamp /usr/local/lib/vamp
|
||||||
|
mkdir ~/vamp
|
||||||
|
cp -pv beatroot-vamp.* ~/vamp/
|
||||||
|
}
|
||||||
|
|
||||||
|
have_beatroot &&
|
||||||
|
printf '\033[32mfound the vamp beatroot plugin, nice\033[0m\n' ||
|
||||||
|
printf '\033[31mWARNING: could not find the vamp beatroot plugin, please install it for optimal results\033[0m\n'
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# not in use because it kinda segfaults, also no windows support
|
||||||
|
install_soundtouch() {
|
||||||
|
gitlab_tarball https://gitlab.com/api/v4/projects/soundtouch%2Fsoundtouch/releases
|
||||||
|
|
||||||
|
tar -xvf soundtouch-*
|
||||||
|
rm -- *.tar.gz
|
||||||
|
cd soundtouch-*
|
||||||
|
|
||||||
|
# https://github.com/jrising/pysoundtouch
|
||||||
|
./bootstrap
|
||||||
|
./configure --enable-integer-samples CXXFLAGS="-fPIC" --prefix="$HOME/pe/soundtouch"
|
||||||
|
make -j$(nproc || echo 4)
|
||||||
|
make install
|
||||||
|
|
||||||
|
CFLAGS=-I$HOME/pe/soundtouch/include/ \
|
||||||
|
LDFLAGS=-L$HOME/pe/soundtouch/lib \
|
||||||
|
$pybin -m pip install --user git+https://github.com/snowxmas/pysoundtouch.git
|
||||||
|
|
||||||
|
pypath="$($pybin -c 'import importlib; print(importlib.util.find_spec("soundtouch").origin)')"
|
||||||
|
libpath="$(echo "$HOME/pe/soundtouch/lib/")"
|
||||||
|
patchelf --set-rpath "$libpath" "$pypath"
|
||||||
|
|
||||||
|
echo
|
||||||
|
echo soundtouch successfully installed to the following locations:
|
||||||
|
echo " $libpath"
|
||||||
|
echo " $pypath"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
[ "$1" = keyfinder ] && { install_keyfinder; exit $?; }
|
||||||
|
[ "$1" = soundtouch ] && { install_soundtouch; exit $?; }
|
||||||
|
[ "$1" = vamp ] && { install_vamp; exit $?; }
|
||||||
|
|
||||||
|
echo no args provided, installing keyfinder and vamp
|
||||||
|
install_keyfinder
|
||||||
|
install_vamp
|
||||||
8
bin/mtag/sleep.py
Normal file
8
bin/mtag/sleep.py
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
|
||||||
|
import time
|
||||||
|
import random
|
||||||
|
|
||||||
|
v = random.random() * 6
|
||||||
|
time.sleep(v)
|
||||||
|
print(f"{v:.2f}")
|
||||||
@@ -12,16 +12,19 @@ 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
|
||||||
import argparse
|
import argparse
|
||||||
|
import threading
|
||||||
|
import traceback
|
||||||
from textwrap import dedent
|
from textwrap import dedent
|
||||||
|
|
||||||
from .__init__ import E, WINDOWS, VT100, PY2
|
from .__init__ import E, WINDOWS, VT100, PY2
|
||||||
from .__version__ import S_VERSION, S_BUILD_DT, CODENAME
|
from .__version__ import S_VERSION, S_BUILD_DT, CODENAME
|
||||||
from .svchub import SvcHub
|
from .svchub import SvcHub
|
||||||
from .util import py_desc, align_tab
|
from .util import py_desc, align_tab, IMPLICATIONS
|
||||||
|
|
||||||
HAVE_SSL = True
|
HAVE_SSL = True
|
||||||
try:
|
try:
|
||||||
@@ -164,6 +167,16 @@ def configure_ssl_ciphers(al):
|
|||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
|
|
||||||
|
|
||||||
|
def sighandler(signal=None, frame=None):
|
||||||
|
msg = [""] * 5
|
||||||
|
for th in threading.enumerate():
|
||||||
|
msg.append(str(th))
|
||||||
|
msg.extend(traceback.format_stack(sys._current_frames()[th.ident]))
|
||||||
|
|
||||||
|
msg.append("\n")
|
||||||
|
print("\n".join(msg))
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
time.strptime("19970815", "%Y%m%d") # python#7980
|
time.strptime("19970815", "%Y%m%d") # python#7980
|
||||||
if WINDOWS:
|
if WINDOWS:
|
||||||
@@ -241,6 +254,7 @@ 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")
|
||||||
@@ -249,7 +263,7 @@ def main():
|
|||||||
ap.add_argument("-nid", action="store_true", help="no info disk-usage")
|
ap.add_argument("-nid", action="store_true", help="no info disk-usage")
|
||||||
ap.add_argument("--no-sendfile", action="store_true", help="disable sendfile (for debugging)")
|
ap.add_argument("--no-sendfile", action="store_true", help="disable sendfile (for debugging)")
|
||||||
ap.add_argument("--no-scandir", action="store_true", help="disable scandir (for debugging)")
|
ap.add_argument("--no-scandir", action="store_true", help="disable scandir (for debugging)")
|
||||||
ap.add_argument("--urlform", 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")
|
||||||
|
|
||||||
ap2 = ap.add_argument_group('database options')
|
ap2 = ap.add_argument_group('database options')
|
||||||
@@ -264,11 +278,13 @@ def main():
|
|||||||
ap2.add_argument("-mtm", metavar="M=t,t,t", action="append", type=str, help="add/replace metadata mapping")
|
ap2.add_argument("-mtm", metavar="M=t,t,t", action="append", type=str, help="add/replace metadata mapping")
|
||||||
ap2.add_argument("-mte", metavar="M,M,M", type=str, help="tags to index/display (comma-sep.)",
|
ap2.add_argument("-mte", metavar="M,M,M", type=str, help="tags to index/display (comma-sep.)",
|
||||||
default="circle,album,.tn,artist,title,.bpm,key,.dur,.q")
|
default="circle,album,.tn,artist,title,.bpm,key,.dur,.q")
|
||||||
|
ap2.add_argument("-mtp", metavar="M=[f,]bin", action="append", type=str, help="read tag M using bin")
|
||||||
|
ap2.add_argument("--srch-time", metavar="SEC", type=int, default=30, help="search deadline")
|
||||||
|
|
||||||
ap2 = ap.add_argument_group('SSL/TLS options')
|
ap2 = ap.add_argument_group('SSL/TLS options')
|
||||||
ap2.add_argument("--http-only", action="store_true", help="disable ssl/tls")
|
ap2.add_argument("--http-only", action="store_true", help="disable ssl/tls")
|
||||||
ap2.add_argument("--https-only", action="store_true", help="disable plaintext")
|
ap2.add_argument("--https-only", action="store_true", help="disable plaintext")
|
||||||
ap2.add_argument("--ssl-ver", type=str, help="ssl/tls versions to allow")
|
ap2.add_argument("--ssl-ver", metavar="LIST", type=str, help="ssl/tls versions to allow")
|
||||||
ap2.add_argument("--ciphers", metavar="LIST", help="set allowed ciphers")
|
ap2.add_argument("--ciphers", metavar="LIST", help="set allowed ciphers")
|
||||||
ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info")
|
ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info")
|
||||||
ap2.add_argument("--ssl-log", metavar="PATH", help="log master secrets")
|
ap2.add_argument("--ssl-log", metavar="PATH", help="log master secrets")
|
||||||
@@ -277,13 +293,7 @@ def main():
|
|||||||
# fmt: on
|
# fmt: on
|
||||||
|
|
||||||
# propagate implications
|
# propagate implications
|
||||||
for k1, k2 in [
|
for k1, k2 in IMPLICATIONS:
|
||||||
["e2dsa", "e2ds"],
|
|
||||||
["e2ds", "e2d"],
|
|
||||||
["e2tsr", "e2ts"],
|
|
||||||
["e2ts", "e2t"],
|
|
||||||
["e2t", "e2d"],
|
|
||||||
]:
|
|
||||||
if getattr(al, k1):
|
if getattr(al, k1):
|
||||||
setattr(al, k2, True)
|
setattr(al, k2, True)
|
||||||
|
|
||||||
@@ -312,6 +322,8 @@ def main():
|
|||||||
+ " (if you crash with codec errors then that is why)"
|
+ " (if you crash with codec errors then that is why)"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# signal.signal(signal.SIGINT, sighandler)
|
||||||
|
|
||||||
SvcHub(al).run()
|
SvcHub(al).run()
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
|
|
||||||
VERSION = (0, 9, 5)
|
VERSION = (0, 9, 10)
|
||||||
CODENAME = "the strongest music server"
|
CODENAME = "the strongest music server"
|
||||||
BUILD_DT = (2021, 3, 7)
|
BUILD_DT = (2021, 3, 21)
|
||||||
|
|
||||||
S_VERSION = ".".join(map(str, VERSION))
|
S_VERSION = ".".join(map(str, VERSION))
|
||||||
S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT)
|
S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT)
|
||||||
|
|||||||
@@ -1,12 +1,13 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import os
|
|
||||||
import re
|
import re
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
import threading
|
import threading
|
||||||
|
|
||||||
from .__init__ import PY2, WINDOWS
|
from .__init__ import PY2, WINDOWS
|
||||||
from .util import undot, Pebkac, fsdec, fsenc, statdir, nuprint
|
from .util import IMPLICATIONS, undot, Pebkac, fsdec, fsenc, statdir, nuprint
|
||||||
|
|
||||||
|
|
||||||
class VFS(object):
|
class VFS(object):
|
||||||
@@ -200,16 +201,39 @@ class AuthSrv(object):
|
|||||||
continue
|
continue
|
||||||
|
|
||||||
lvl, uname = ln.split(" ")
|
lvl, uname = ln.split(" ")
|
||||||
if lvl in "ra":
|
self._read_vol_str(
|
||||||
mread[vol_dst].append(uname)
|
lvl, uname, mread[vol_dst], mwrite[vol_dst], mflags[vol_dst]
|
||||||
if lvl in "wa":
|
)
|
||||||
mwrite[vol_dst].append(uname)
|
|
||||||
|
def _read_vol_str(self, lvl, uname, mr, mw, mf):
|
||||||
if lvl == "c":
|
if lvl == "c":
|
||||||
cval = True
|
cval = True
|
||||||
if "=" in uname:
|
if "=" in uname:
|
||||||
uname, cval = uname.split("=", 1)
|
uname, cval = uname.split("=", 1)
|
||||||
|
|
||||||
mflags[vol_dst][uname] = cval
|
self._read_volflag(mf, uname, cval, False)
|
||||||
|
return
|
||||||
|
|
||||||
|
if uname == "":
|
||||||
|
uname = "*"
|
||||||
|
|
||||||
|
if lvl in "ra":
|
||||||
|
mr.append(uname)
|
||||||
|
|
||||||
|
if lvl in "wa":
|
||||||
|
mw.append(uname)
|
||||||
|
|
||||||
|
def _read_volflag(self, flags, name, value, is_list):
|
||||||
|
if name not in ["mtp"]:
|
||||||
|
flags[name] = value
|
||||||
|
return
|
||||||
|
|
||||||
|
if not is_list:
|
||||||
|
value = [value]
|
||||||
|
elif not value:
|
||||||
|
return
|
||||||
|
|
||||||
|
flags[name] = flags.get(name, []) + value
|
||||||
|
|
||||||
def reload(self):
|
def reload(self):
|
||||||
"""
|
"""
|
||||||
@@ -232,7 +256,7 @@ class AuthSrv(object):
|
|||||||
|
|
||||||
if self.args.v:
|
if self.args.v:
|
||||||
# list of src:dst:permset:permset:...
|
# list of src:dst:permset:permset:...
|
||||||
# permset is [rwa]username
|
# permset is [rwa]username or [c]flag
|
||||||
for v_str in self.args.v:
|
for v_str in self.args.v:
|
||||||
m = self.re_vol.match(v_str)
|
m = self.re_vol.match(v_str)
|
||||||
if not m:
|
if not m:
|
||||||
@@ -249,22 +273,7 @@ class AuthSrv(object):
|
|||||||
|
|
||||||
perms = perms.split(":")
|
perms = perms.split(":")
|
||||||
for (lvl, uname) in [[x[0], x[1:]] for x in perms]:
|
for (lvl, uname) in [[x[0], x[1:]] for x in perms]:
|
||||||
if lvl == "c":
|
self._read_vol_str(lvl, uname, mread[dst], mwrite[dst], mflags[dst])
|
||||||
cval = True
|
|
||||||
if "=" in uname:
|
|
||||||
uname, cval = uname.split("=", 1)
|
|
||||||
|
|
||||||
mflags[dst][uname] = cval
|
|
||||||
continue
|
|
||||||
|
|
||||||
if uname == "":
|
|
||||||
uname = "*"
|
|
||||||
|
|
||||||
if lvl in "ra":
|
|
||||||
mread[dst].append(uname)
|
|
||||||
|
|
||||||
if lvl in "wa":
|
|
||||||
mwrite[dst].append(uname)
|
|
||||||
|
|
||||||
if self.args.c:
|
if self.args.c:
|
||||||
for cfg_fn in self.args.c:
|
for cfg_fn in self.args.c:
|
||||||
@@ -310,6 +319,8 @@ class AuthSrv(object):
|
|||||||
)
|
)
|
||||||
raise Exception("invalid config")
|
raise Exception("invalid config")
|
||||||
|
|
||||||
|
all_mte = {}
|
||||||
|
errors = False
|
||||||
for vol in vfs.all_vols.values():
|
for vol in vfs.all_vols.values():
|
||||||
if (self.args.e2ds and vol.uwrite) or self.args.e2dsa:
|
if (self.args.e2ds and vol.uwrite) or self.args.e2dsa:
|
||||||
vol.flags["e2ds"] = True
|
vol.flags["e2ds"] = True
|
||||||
@@ -321,10 +332,60 @@ class AuthSrv(object):
|
|||||||
if getattr(self.args, k):
|
if getattr(self.args, k):
|
||||||
vol.flags[k] = True
|
vol.flags[k] = True
|
||||||
|
|
||||||
|
for k1, k2 in IMPLICATIONS:
|
||||||
|
if k1 in vol.flags:
|
||||||
|
vol.flags[k2] = True
|
||||||
|
|
||||||
# default tag-list if unset
|
# default tag-list if unset
|
||||||
if "mte" not in vol.flags:
|
if "mte" not in vol.flags:
|
||||||
vol.flags["mte"] = self.args.mte
|
vol.flags["mte"] = self.args.mte
|
||||||
|
|
||||||
|
# append parsers from argv to volume-flags
|
||||||
|
self._read_volflag(vol.flags, "mtp", self.args.mtp, True)
|
||||||
|
|
||||||
|
# verify tags mentioned by -mt[mp] are used by -mte
|
||||||
|
local_mtp = {}
|
||||||
|
local_only_mtp = {}
|
||||||
|
for a in vol.flags.get("mtp", []) + vol.flags.get("mtm", []):
|
||||||
|
a = a.split("=")[0]
|
||||||
|
local_mtp[a] = True
|
||||||
|
local = True
|
||||||
|
for b in self.args.mtp or []:
|
||||||
|
b = b.split("=")[0]
|
||||||
|
if a == b:
|
||||||
|
local = False
|
||||||
|
|
||||||
|
if local:
|
||||||
|
local_only_mtp[a] = True
|
||||||
|
|
||||||
|
local_mte = {}
|
||||||
|
for a in vol.flags.get("mte", "").split(","):
|
||||||
|
local = True
|
||||||
|
all_mte[a] = True
|
||||||
|
local_mte[a] = True
|
||||||
|
for b in self.args.mte.split(","):
|
||||||
|
if not a or not b:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if a == b:
|
||||||
|
local = False
|
||||||
|
|
||||||
|
for mtp in local_only_mtp.keys():
|
||||||
|
if mtp not in local_mte:
|
||||||
|
m = 'volume "/{}" defines metadata tag "{}", but doesnt use it in "-mte" (or with "cmte" in its volume-flags)'
|
||||||
|
self.log(m.format(vol.vpath, mtp), 1)
|
||||||
|
errors = True
|
||||||
|
|
||||||
|
for mtp in self.args.mtp or []:
|
||||||
|
mtp = mtp.split("=")[0]
|
||||||
|
if mtp not in all_mte:
|
||||||
|
m = 'metadata tag "{}" is defined by "-mtm" or "-mtp", but is not used by "-mte" (or by any "cmte" volume-flag)'
|
||||||
|
self.log(m.format(mtp), 1)
|
||||||
|
errors = True
|
||||||
|
|
||||||
|
if errors:
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
v, _ = vfs.get("/", "*", False, True)
|
v, _ = vfs.get("/", "*", False, True)
|
||||||
if self.warn_anonwrite and os.getcwd() == v.realpath:
|
if self.warn_anonwrite and os.getcwd() == v.realpath:
|
||||||
|
|||||||
@@ -73,7 +73,9 @@ class MpWorker(object):
|
|||||||
if PY2:
|
if PY2:
|
||||||
sck = pickle.loads(sck) # nosec
|
sck = pickle.loads(sck) # nosec
|
||||||
|
|
||||||
|
if self.args.log_conn:
|
||||||
self.log("%s %s" % addr, "|%sC-qpop" % ("-" * 4,), c="1;30")
|
self.log("%s %s" % addr, "|%sC-qpop" % ("-" * 4,), c="1;30")
|
||||||
|
|
||||||
self.httpsrv.accept(sck, addr)
|
self.httpsrv.accept(sck, addr)
|
||||||
|
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
|
|||||||
@@ -28,7 +28,9 @@ class BrokerThr(object):
|
|||||||
def put(self, want_retval, dest, *args):
|
def put(self, want_retval, dest, *args):
|
||||||
if dest == "httpconn":
|
if dest == "httpconn":
|
||||||
sck, addr = args
|
sck, addr = args
|
||||||
|
if self.args.log_conn:
|
||||||
self.log("%s %s" % addr, "|%sC-qpop" % ("-" * 4,), c="1;30")
|
self.log("%s %s" % addr, "|%sC-qpop" % ("-" * 4,), c="1;30")
|
||||||
|
|
||||||
self.httpsrv.accept(sck, addr)
|
self.httpsrv.accept(sck, addr)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
|
|||||||
@@ -45,7 +45,7 @@ class HttpCli(object):
|
|||||||
self.log_func(self.log_src, msg, c)
|
self.log_func(self.log_src, msg, c)
|
||||||
|
|
||||||
def _check_nonfatal(self, ex):
|
def _check_nonfatal(self, ex):
|
||||||
return ex.code < 400 or ex.code == 404
|
return ex.code < 400 or ex.code in [404, 429]
|
||||||
|
|
||||||
def _assert_safe_rem(self, rem):
|
def _assert_safe_rem(self, rem):
|
||||||
# sanity check to prevent any disasters
|
# sanity check to prevent any disasters
|
||||||
@@ -450,19 +450,30 @@ class HttpCli(object):
|
|||||||
|
|
||||||
idx = self.conn.get_u2idx()
|
idx = self.conn.get_u2idx()
|
||||||
t0 = time.time()
|
t0 = time.time()
|
||||||
|
if idx.p_end:
|
||||||
|
penalty = 0.7
|
||||||
|
t_idle = t0 - idx.p_end
|
||||||
|
if idx.p_dur > 0.7 and t_idle < penalty:
|
||||||
|
m = "rate-limit ({:.1f} sec), cost {:.2f}, idle {:.2f}"
|
||||||
|
raise Pebkac(429, m.format(penalty, idx.p_dur, t_idle))
|
||||||
|
|
||||||
if "srch" in body:
|
if "srch" in body:
|
||||||
# search by up2k hashlist
|
# search by up2k hashlist
|
||||||
vbody = copy.deepcopy(body)
|
vbody = copy.deepcopy(body)
|
||||||
vbody["hash"] = len(vbody["hash"])
|
vbody["hash"] = len(vbody["hash"])
|
||||||
self.log("qj: " + repr(vbody))
|
self.log("qj: " + repr(vbody))
|
||||||
hits = idx.fsearch(vols, body)
|
hits = idx.fsearch(vols, body)
|
||||||
self.log("q#: {} ({:.2f}s)".format(repr(hits), time.time() - t0))
|
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))
|
||||||
hits, taglist = idx.search(vols, body)
|
hits, taglist = idx.search(vols, body)
|
||||||
self.log("q#: {} ({:.2f}s)".format(len(hits), time.time() - t0))
|
msg = len(hits)
|
||||||
|
|
||||||
|
idx.p_end = time.time()
|
||||||
|
idx.p_dur = idx.p_end - t0
|
||||||
|
self.log("q#: {} ({:.2f}s)".format(msg, idx.p_dur))
|
||||||
|
|
||||||
order = []
|
order = []
|
||||||
cfg = self.args.mte.split(",")
|
cfg = self.args.mte.split(",")
|
||||||
@@ -1282,20 +1293,20 @@ class HttpCli(object):
|
|||||||
args = s3enc(idx.mem_cur, rd, fn)
|
args = s3enc(idx.mem_cur, rd, fn)
|
||||||
r = icur.execute(q, args).fetchone()
|
r = icur.execute(q, args).fetchone()
|
||||||
|
|
||||||
|
tags = {}
|
||||||
|
f["tags"] = tags
|
||||||
|
|
||||||
if not r:
|
if not r:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
w = r[0][:16]
|
w = r[0][:16]
|
||||||
tags = {}
|
|
||||||
q = "select k, v from mt where w = ? and k != 'x'"
|
q = "select k, v from mt where w = ? and k != 'x'"
|
||||||
for k, v in icur.execute(q, (w,)):
|
for k, v in icur.execute(q, (w,)):
|
||||||
taglist[k] = True
|
taglist[k] = True
|
||||||
tags[k] = v
|
tags[k] = v
|
||||||
|
|
||||||
f["tags"] = tags
|
|
||||||
|
|
||||||
if icur:
|
if icur:
|
||||||
taglist = [k for k in self.args.mte.split(",") if k in taglist]
|
taglist = [k for k in vn.flags["mte"].split(",") if k in taglist]
|
||||||
for f in dirs:
|
for f in dirs:
|
||||||
f["tags"] = {}
|
f["tags"] = {}
|
||||||
|
|
||||||
@@ -1368,7 +1379,7 @@ class HttpCli(object):
|
|||||||
ts=ts,
|
ts=ts,
|
||||||
perms=json.dumps(perms),
|
perms=json.dumps(perms),
|
||||||
taglist=taglist,
|
taglist=taglist,
|
||||||
tag_order=json.dumps(self.args.mte.split(",")),
|
tag_order=json.dumps(vn.flags["mte"].split(",")),
|
||||||
have_up2k_idx=("e2d" in vn.flags),
|
have_up2k_idx=("e2d" in vn.flags),
|
||||||
have_tags_idx=("e2t" in vn.flags),
|
have_tags_idx=("e2t" in vn.flags),
|
||||||
logues=logues,
|
logues=logues,
|
||||||
|
|||||||
@@ -38,7 +38,9 @@ class HttpSrv(object):
|
|||||||
|
|
||||||
def accept(self, sck, addr):
|
def accept(self, sck, addr):
|
||||||
"""takes an incoming tcp connection and creates a thread to handle it"""
|
"""takes an incoming tcp connection and creates a thread to handle it"""
|
||||||
|
if self.args.log_conn:
|
||||||
self.log("%s %s" % addr, "|%sC-cthr" % ("-" * 5,), c="1;30")
|
self.log("%s %s" % addr, "|%sC-cthr" % ("-" * 5,), c="1;30")
|
||||||
|
|
||||||
thr = threading.Thread(target=self.thr_client, args=(sck, addr))
|
thr = threading.Thread(target=self.thr_client, args=(sck, addr))
|
||||||
thr.daemon = True
|
thr.daemon = True
|
||||||
thr.start()
|
thr.start()
|
||||||
@@ -66,11 +68,15 @@ class HttpSrv(object):
|
|||||||
thr.start()
|
thr.start()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
if self.args.log_conn:
|
||||||
self.log("%s %s" % addr, "|%sC-crun" % ("-" * 6,), c="1;30")
|
self.log("%s %s" % addr, "|%sC-crun" % ("-" * 6,), c="1;30")
|
||||||
|
|
||||||
cli.run()
|
cli.run()
|
||||||
|
|
||||||
finally:
|
finally:
|
||||||
|
if self.args.log_conn:
|
||||||
self.log("%s %s" % addr, "|%sC-cdone" % ("-" * 7,), c="1;30")
|
self.log("%s %s" % addr, "|%sC-cdone" % ("-" * 7,), c="1;30")
|
||||||
|
|
||||||
try:
|
try:
|
||||||
sck.shutdown(socket.SHUT_RDWR)
|
sck.shutdown(socket.SHUT_RDWR)
|
||||||
sck.close()
|
sck.close()
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import shutil
|
|||||||
import subprocess as sp
|
import subprocess as sp
|
||||||
|
|
||||||
from .__init__ import PY2, WINDOWS
|
from .__init__ import PY2, WINDOWS
|
||||||
from .util import fsenc, fsdec
|
from .util import fsenc, fsdec, REKOBO_LKEY
|
||||||
|
|
||||||
if not PY2:
|
if not PY2:
|
||||||
unicode = str
|
unicode = str
|
||||||
@@ -151,6 +151,12 @@ class MTag(object):
|
|||||||
v = v.split("/")[0].strip().lstrip("0")
|
v = v.split("/")[0].strip().lstrip("0")
|
||||||
ret[k] = v or 0
|
ret[k] = v or 0
|
||||||
|
|
||||||
|
# normalize key notation to rkeobo
|
||||||
|
okey = ret.get("key")
|
||||||
|
if okey:
|
||||||
|
key = okey.replace(" ", "").replace("maj", "").replace("min", "m")
|
||||||
|
ret["key"] = REKOBO_LKEY.get(key.lower(), okey)
|
||||||
|
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
def compare(self, abspath):
|
def compare(self, abspath):
|
||||||
@@ -225,7 +231,7 @@ class MTag(object):
|
|||||||
"""
|
"""
|
||||||
note:
|
note:
|
||||||
tags which contain newline will be truncated on first \n,
|
tags which contain newline will be truncated on first \n,
|
||||||
ffmpeg emits \n and spacepads the : to align visually
|
ffprobe emits \n and spacepads the : to align visually
|
||||||
note:
|
note:
|
||||||
the Stream ln always mentions Audio: if audio
|
the Stream ln always mentions Audio: if audio
|
||||||
the Stream ln usually has kb/s, is more accurate
|
the Stream ln usually has kb/s, is more accurate
|
||||||
@@ -295,7 +301,7 @@ class MTag(object):
|
|||||||
sec *= 60
|
sec *= 60
|
||||||
sec += int(f)
|
sec += int(f)
|
||||||
except:
|
except:
|
||||||
self.log("invalid timestr from ffmpeg: [{}]".format(tstr), c=3)
|
self.log("invalid timestr from ffprobe: [{}]".format(tstr), c=3)
|
||||||
|
|
||||||
ret[".dur"] = sec
|
ret[".dur"] = sec
|
||||||
m = ptn_br1.search(ln)
|
m = ptn_br1.search(ln)
|
||||||
@@ -312,3 +318,30 @@ class MTag(object):
|
|||||||
ret = {k: [0, v] for k, v in ret.items()}
|
ret = {k: [0, v] for k, v in ret.items()}
|
||||||
|
|
||||||
return self.normalize_tags(ret, md)
|
return self.normalize_tags(ret, md)
|
||||||
|
|
||||||
|
def get_bin(self, parsers, abspath):
|
||||||
|
pypath = os.path.abspath(os.path.dirname(os.path.dirname(__file__)))
|
||||||
|
pypath = [str(pypath)] + [str(x) for x in sys.path if x]
|
||||||
|
pypath = str(os.pathsep.join(pypath))
|
||||||
|
env = os.environ.copy()
|
||||||
|
env["PYTHONPATH"] = pypath
|
||||||
|
|
||||||
|
ret = {}
|
||||||
|
for tagname, (binpath, timeout) in parsers.items():
|
||||||
|
try:
|
||||||
|
cmd = [sys.executable, binpath, abspath]
|
||||||
|
args = {"env": env, "timeout": timeout}
|
||||||
|
|
||||||
|
if WINDOWS:
|
||||||
|
args["creationflags"] = 0x4000
|
||||||
|
else:
|
||||||
|
cmd = ["nice"] + cmd
|
||||||
|
|
||||||
|
cmd = [fsenc(x) for x in cmd]
|
||||||
|
v = sp.check_output(cmd, **args).strip()
|
||||||
|
if v:
|
||||||
|
ret[tagname] = v.decode("utf-8")
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
return ret
|
||||||
|
|||||||
@@ -68,16 +68,21 @@ class TcpSrv(object):
|
|||||||
self.log("tcpsrv", "listening @ {0}:{1}".format(ip, port))
|
self.log("tcpsrv", "listening @ {0}:{1}".format(ip, port))
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
|
if self.args.log_conn:
|
||||||
self.log("tcpsrv", "|%sC-ncli" % ("-" * 1,), c="1;30")
|
self.log("tcpsrv", "|%sC-ncli" % ("-" * 1,), c="1;30")
|
||||||
|
|
||||||
if self.num_clients.v >= self.args.nc:
|
if self.num_clients.v >= self.args.nc:
|
||||||
time.sleep(0.1)
|
time.sleep(0.1)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
if self.args.log_conn:
|
||||||
self.log("tcpsrv", "|%sC-acc1" % ("-" * 2,), c="1;30")
|
self.log("tcpsrv", "|%sC-acc1" % ("-" * 2,), c="1;30")
|
||||||
|
|
||||||
ready, _, _ = select.select(self.srv, [], [])
|
ready, _, _ = select.select(self.srv, [], [])
|
||||||
for srv in ready:
|
for srv in ready:
|
||||||
sck, addr = srv.accept()
|
sck, addr = srv.accept()
|
||||||
sip, sport = srv.getsockname()
|
sip, sport = srv.getsockname()
|
||||||
|
if self.args.log_conn:
|
||||||
self.log(
|
self.log(
|
||||||
"%s %s" % addr,
|
"%s %s" % addr,
|
||||||
"|{}C-acc2 \033[0;36m{} \033[3{}m{}".format(
|
"|{}C-acc2 \033[0;36m{} \033[3{}m{}".format(
|
||||||
@@ -85,6 +90,7 @@ class TcpSrv(object):
|
|||||||
),
|
),
|
||||||
c="1;30",
|
c="1;30",
|
||||||
)
|
)
|
||||||
|
|
||||||
self.num_clients.add()
|
self.num_clients.add()
|
||||||
self.hub.broker.put(False, "httpconn", sck, addr)
|
self.hub.broker.put(False, "httpconn", sck, addr)
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,13 @@
|
|||||||
# 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 time
|
||||||
|
import threading
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
|
|
||||||
from .util import u8safe
|
from .util import u8safe, s3dec, html_escape, Pebkac
|
||||||
from .up2k import up2k_wark_from_hashlist
|
from .up2k import up2k_wark_from_hashlist
|
||||||
|
|
||||||
|
|
||||||
@@ -19,6 +22,7 @@ class U2idx(object):
|
|||||||
def __init__(self, args, log_func):
|
def __init__(self, args, log_func):
|
||||||
self.args = args
|
self.args = args
|
||||||
self.log_func = log_func
|
self.log_func = log_func
|
||||||
|
self.timeout = args.srch_time
|
||||||
|
|
||||||
if not HAVE_SQLITE3:
|
if not HAVE_SQLITE3:
|
||||||
self.log("could not load sqlite3; searchign wqill be disabled")
|
self.log("could not load sqlite3; searchign wqill be disabled")
|
||||||
@@ -28,6 +32,9 @@ class U2idx(object):
|
|||||||
self.mem_cur = sqlite3.connect(":memory:")
|
self.mem_cur = sqlite3.connect(":memory:")
|
||||||
self.mem_cur.execute(r"create table a (b text)")
|
self.mem_cur.execute(r"create table a (b text)")
|
||||||
|
|
||||||
|
self.p_end = None
|
||||||
|
self.p_dur = 0
|
||||||
|
|
||||||
def log(self, msg, c=0):
|
def log(self, msg, c=0):
|
||||||
self.log_func("u2idx", msg, c)
|
self.log_func("u2idx", msg, c)
|
||||||
|
|
||||||
@@ -43,7 +50,10 @@ class U2idx(object):
|
|||||||
uq = "substr(w,1,16) = ? and w = ?"
|
uq = "substr(w,1,16) = ? and w = ?"
|
||||||
uv = [wark[:16], wark]
|
uv = [wark[:16], wark]
|
||||||
|
|
||||||
return self.run_query(vols, uq, uv, "", [])[0]
|
try:
|
||||||
|
return self.run_query(vols, uq, uv, {})[0]
|
||||||
|
except Exception as ex:
|
||||||
|
raise Pebkac(500, repr(ex))
|
||||||
|
|
||||||
def get_cur(self, ptop):
|
def get_cur(self, ptop):
|
||||||
cur = self.cur.get(ptop)
|
cur = self.cur.get(ptop)
|
||||||
@@ -73,17 +83,64 @@ class U2idx(object):
|
|||||||
|
|
||||||
uq, uv = _sqlize(qobj)
|
uq, uv = _sqlize(qobj)
|
||||||
|
|
||||||
tq = ""
|
|
||||||
tv = []
|
|
||||||
qobj = {}
|
qobj = {}
|
||||||
if "tags" in body:
|
if "tags" in body:
|
||||||
_conv_txt(qobj, body, "tags", "mt.v")
|
_conv_txt(qobj, body, "tags", "mt.v")
|
||||||
tq, tv = _sqlize(qobj)
|
|
||||||
|
|
||||||
return self.run_query(vols, uq, uv, tq, tv)
|
if "adv" in body:
|
||||||
|
_conv_adv(qobj, body, "adv")
|
||||||
|
|
||||||
def run_query(self, vols, uq, uv, tq, tv):
|
try:
|
||||||
self.log("qs: {} {} , {} {}".format(uq, repr(uv), tq, repr(tv)))
|
return self.run_query(vols, uq, uv, qobj)
|
||||||
|
except Exception as ex:
|
||||||
|
raise Pebkac(500, repr(ex))
|
||||||
|
|
||||||
|
def run_query(self, vols, uq, uv, targs):
|
||||||
|
self.log("qs: {} {} , {}".format(uq, repr(uv), repr(targs)))
|
||||||
|
|
||||||
|
done_flag = []
|
||||||
|
self.active_id = "{:.6f}_{}".format(
|
||||||
|
time.time(), threading.current_thread().ident
|
||||||
|
)
|
||||||
|
thr = threading.Thread(
|
||||||
|
target=self.terminator,
|
||||||
|
args=(
|
||||||
|
self.active_id,
|
||||||
|
done_flag,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
thr.daemon = True
|
||||||
|
thr.start()
|
||||||
|
|
||||||
|
if not targs:
|
||||||
|
if not uq:
|
||||||
|
q = "select * from up"
|
||||||
|
v = ()
|
||||||
|
else:
|
||||||
|
q = "select * from up where " + uq
|
||||||
|
v = tuple(uv)
|
||||||
|
else:
|
||||||
|
q = "select up.* from up"
|
||||||
|
keycmp = "substr(up.w,1,16)"
|
||||||
|
where = []
|
||||||
|
v = []
|
||||||
|
ctr = 0
|
||||||
|
for tq, tv in sorted(targs.items()):
|
||||||
|
ctr += 1
|
||||||
|
tq = tq.split("\n")[0]
|
||||||
|
keycmp2 = "mt{}.w".format(ctr)
|
||||||
|
q += " inner join mt mt{} on {} = {}".format(ctr, keycmp, keycmp2)
|
||||||
|
keycmp = keycmp2
|
||||||
|
where.append(tq.replace("mt.", keycmp[:-1]))
|
||||||
|
v.append(tv)
|
||||||
|
|
||||||
|
if uq:
|
||||||
|
where.append(uq)
|
||||||
|
v.extend(uv)
|
||||||
|
|
||||||
|
q += " where " + (" and ".join(where))
|
||||||
|
|
||||||
|
# self.log("q2: {} {}".format(q, repr(v)))
|
||||||
|
|
||||||
ret = []
|
ret = []
|
||||||
lim = 1000
|
lim = 1000
|
||||||
@@ -93,18 +150,7 @@ class U2idx(object):
|
|||||||
if not cur:
|
if not cur:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if not tq:
|
self.active_cur = cur
|
||||||
if not uq:
|
|
||||||
q = "select * from up"
|
|
||||||
v = ()
|
|
||||||
else:
|
|
||||||
q = "select * from up where " + uq
|
|
||||||
v = tuple(uv)
|
|
||||||
else:
|
|
||||||
# naive assumption: tags first
|
|
||||||
q = "select up.* from up inner join mt on substr(up.w,1,16) = mt.w where {}"
|
|
||||||
q = q.format(" and ".join([tq, uq]) if uq else tq)
|
|
||||||
v = tuple(tv + uv)
|
|
||||||
|
|
||||||
sret = []
|
sret = []
|
||||||
c = cur.execute(q, v)
|
c = cur.execute(q, v)
|
||||||
@@ -124,17 +170,35 @@ class U2idx(object):
|
|||||||
w = hit["w"]
|
w = hit["w"]
|
||||||
del hit["w"]
|
del hit["w"]
|
||||||
tags = {}
|
tags = {}
|
||||||
q = "select k, v from mt where w = ? and k != 'x'"
|
q2 = "select k, v from mt where w = ? and k != 'x'"
|
||||||
for k, v in cur.execute(q, (w,)):
|
for k, v2 in cur.execute(q2, (w,)):
|
||||||
taglist[k] = True
|
taglist[k] = True
|
||||||
tags[k] = v
|
tags[k] = v2
|
||||||
|
|
||||||
hit["tags"] = tags
|
hit["tags"] = tags
|
||||||
|
|
||||||
ret.extend(sret)
|
ret.extend(sret)
|
||||||
|
|
||||||
|
done_flag.append(True)
|
||||||
|
self.active_id = None
|
||||||
|
|
||||||
|
# undupe hits from multiple metadata keys
|
||||||
|
if len(ret) > 1:
|
||||||
|
ret = [ret[0]] + [
|
||||||
|
y for x, y in zip(ret[:-1], ret[1:]) if x["rp"] != y["rp"]
|
||||||
|
]
|
||||||
|
|
||||||
return ret, list(taglist.keys())
|
return ret, list(taglist.keys())
|
||||||
|
|
||||||
|
def terminator(self, identifier, done_flag):
|
||||||
|
for _ in range(self.timeout):
|
||||||
|
time.sleep(1)
|
||||||
|
if done_flag:
|
||||||
|
return
|
||||||
|
|
||||||
|
if identifier == self.active_id:
|
||||||
|
self.active_cur.connection.interrupt()
|
||||||
|
|
||||||
|
|
||||||
def _open(ptop):
|
def _open(ptop):
|
||||||
db_path = os.path.join(ptop, ".hist", "up2k.db")
|
db_path = os.path.join(ptop, ".hist", "up2k.db")
|
||||||
@@ -190,6 +254,23 @@ def _conv_txt(q, body, k, sql):
|
|||||||
q[qk + "\n" + v] = u8safe(v)
|
q[qk + "\n" + v] = u8safe(v)
|
||||||
|
|
||||||
|
|
||||||
|
def _conv_adv(q, body, k):
|
||||||
|
ptn = re.compile(r"^(\.?[a-z]+) *(==?|!=|<=?|>=?) *(.*)$")
|
||||||
|
|
||||||
|
parts = body[k].split(" ")
|
||||||
|
parts = [x.strip() for x in parts if x.strip()]
|
||||||
|
|
||||||
|
for part in parts:
|
||||||
|
m = ptn.match(part)
|
||||||
|
if not m:
|
||||||
|
p = html_escape(part)
|
||||||
|
raise Pebkac(400, "invalid argument [" + p + "]")
|
||||||
|
|
||||||
|
k, op, v = m.groups()
|
||||||
|
qk = "mt.k = '{}' and mt.v {} ?".format(k, op)
|
||||||
|
q[qk + "\n" + v] = u8safe(v)
|
||||||
|
|
||||||
|
|
||||||
def _sqlize(qobj):
|
def _sqlize(qobj):
|
||||||
keys = []
|
keys = []
|
||||||
values = []
|
values = []
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import base64
|
|||||||
import hashlib
|
import hashlib
|
||||||
import threading
|
import threading
|
||||||
import traceback
|
import traceback
|
||||||
|
import subprocess as sp
|
||||||
from copy import deepcopy
|
from copy import deepcopy
|
||||||
|
|
||||||
from .__init__ import WINDOWS
|
from .__init__ import WINDOWS
|
||||||
@@ -28,6 +29,7 @@ from .util import (
|
|||||||
s3enc,
|
s3enc,
|
||||||
s3dec,
|
s3dec,
|
||||||
statdir,
|
statdir,
|
||||||
|
s2hms,
|
||||||
)
|
)
|
||||||
from .mtag import MTag
|
from .mtag import MTag
|
||||||
from .authsrv import AuthSrv
|
from .authsrv import AuthSrv
|
||||||
@@ -64,7 +66,7 @@ class Up2k(object):
|
|||||||
self.flags = {}
|
self.flags = {}
|
||||||
self.cur = {}
|
self.cur = {}
|
||||||
self.mtag = None
|
self.mtag = None
|
||||||
self.n_mtag_tags_added = -1
|
self.pending_tags = None
|
||||||
|
|
||||||
self.mem_cur = None
|
self.mem_cur = None
|
||||||
self.sqlite_ver = None
|
self.sqlite_ver = None
|
||||||
@@ -107,6 +109,10 @@ class Up2k(object):
|
|||||||
thr.daemon = True
|
thr.daemon = True
|
||||||
thr.start()
|
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)
|
||||||
|
|
||||||
@@ -219,6 +225,10 @@ class Up2k(object):
|
|||||||
|
|
||||||
_, flags = self._expr_idx_filter(flags)
|
_, flags = self._expr_idx_filter(flags)
|
||||||
|
|
||||||
|
a = "\033[0;36m{}:\033[1;30m{}"
|
||||||
|
a = [a.format(k, v) for k, v in sorted(flags.items())]
|
||||||
|
self.log(" ".join(a) + "\033[0m")
|
||||||
|
|
||||||
reg = {}
|
reg = {}
|
||||||
path = os.path.join(ptop, ".hist", "up2k.snap")
|
path = os.path.join(ptop, ".hist", "up2k.snap")
|
||||||
if "e2d" in flags and os.path.exists(path):
|
if "e2d" in flags and os.path.exists(path):
|
||||||
@@ -435,18 +445,7 @@ class Up2k(object):
|
|||||||
|
|
||||||
mpool = False
|
mpool = False
|
||||||
if self.mtag.prefer_mt and not self.args.no_mtag_mt:
|
if self.mtag.prefer_mt and not self.args.no_mtag_mt:
|
||||||
# mp.pool.ThreadPool and concurrent.futures.ThreadPoolExecutor
|
mpool = self._start_mpool()
|
||||||
# both do crazy runahead so lets reinvent another wheel
|
|
||||||
nw = os.cpu_count() if hasattr(os, "cpu_count") else 4
|
|
||||||
if self.n_mtag_tags_added == -1:
|
|
||||||
self.log("using {}x {}".format(nw, self.mtag.backend))
|
|
||||||
self.n_mtag_tags_added = 0
|
|
||||||
|
|
||||||
mpool = Queue(nw)
|
|
||||||
for _ in range(nw):
|
|
||||||
thr = threading.Thread(target=self._tag_thr, args=(mpool,))
|
|
||||||
thr.daemon = True
|
|
||||||
thr.start()
|
|
||||||
|
|
||||||
c2 = cur.connection.cursor()
|
c2 = cur.connection.cursor()
|
||||||
c3 = cur.connection.cursor()
|
c3 = cur.connection.cursor()
|
||||||
@@ -457,19 +456,21 @@ class Up2k(object):
|
|||||||
if c2.execute(q, (w[:16],)).fetchone():
|
if c2.execute(q, (w[:16],)).fetchone():
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
if "mtp" in flags:
|
||||||
|
q = "insert into mt values (?,'t:mtp','a')"
|
||||||
|
c2.execute(q, (w[:16],))
|
||||||
|
|
||||||
if rd.startswith("//") or fn.startswith("//"):
|
if rd.startswith("//") or fn.startswith("//"):
|
||||||
rd, fn = s3dec(rd, fn)
|
rd, fn = s3dec(rd, fn)
|
||||||
|
|
||||||
abspath = os.path.join(ptop, rd, fn)
|
abspath = os.path.join(ptop, rd, fn)
|
||||||
self.pp.msg = "c{} {}".format(n_left, abspath)
|
self.pp.msg = "c{} {}".format(n_left, abspath)
|
||||||
args = c3, entags, w, abspath
|
args = [entags, w, abspath]
|
||||||
if not mpool:
|
if not mpool:
|
||||||
n_tags = self._tag_file(*args)
|
n_tags = self._tag_file(c3, *args)
|
||||||
else:
|
else:
|
||||||
mpool.put(args)
|
mpool.put(["mtag"] + args)
|
||||||
with self.mutex:
|
n_tags = len(self._flush_mpool(c3))
|
||||||
n_tags = self.n_mtag_tags_added
|
|
||||||
self.n_mtag_tags_added = 0
|
|
||||||
|
|
||||||
n_add += n_tags
|
n_add += n_tags
|
||||||
n_buf += n_tags
|
n_buf += n_tags
|
||||||
@@ -481,17 +482,220 @@ class Up2k(object):
|
|||||||
last_write = time.time()
|
last_write = time.time()
|
||||||
n_buf = 0
|
n_buf = 0
|
||||||
|
|
||||||
if mpool:
|
self._stop_mpool(mpool, c3)
|
||||||
for _ in range(mpool.maxsize):
|
|
||||||
mpool.put(None)
|
|
||||||
|
|
||||||
mpool.join()
|
|
||||||
|
|
||||||
c3.close()
|
c3.close()
|
||||||
c2.close()
|
c2.close()
|
||||||
|
|
||||||
return n_add, n_rm, True
|
return n_add, n_rm, True
|
||||||
|
|
||||||
|
def _flush_mpool(self, wcur):
|
||||||
|
with self.mutex:
|
||||||
|
ret = []
|
||||||
|
for x in self.pending_tags:
|
||||||
|
self._tag_file(wcur, *x)
|
||||||
|
ret.append(x[1])
|
||||||
|
|
||||||
|
self.pending_tags = []
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def _run_all_mtp(self):
|
||||||
|
t0 = time.time()
|
||||||
|
self.mtp_force = {}
|
||||||
|
self.mtp_parsers = {}
|
||||||
|
for ptop, flags in self.flags.items():
|
||||||
|
if "mtp" in flags:
|
||||||
|
self._run_one_mtp(ptop)
|
||||||
|
|
||||||
|
td = time.time() - t0
|
||||||
|
msg = "mtp finished in {:.2f} sec ({})"
|
||||||
|
self.log(msg.format(td, s2hms(td, True)))
|
||||||
|
|
||||||
|
def _run_one_mtp(self, ptop):
|
||||||
|
db_path = os.path.join(ptop, ".hist", "up2k.db")
|
||||||
|
sz0 = os.path.getsize(db_path) // 1024
|
||||||
|
|
||||||
|
entags = self.entags[ptop]
|
||||||
|
|
||||||
|
force = {}
|
||||||
|
timeout = {}
|
||||||
|
parsers = {}
|
||||||
|
for parser in self.flags[ptop]["mtp"]:
|
||||||
|
orig = parser
|
||||||
|
tag, parser = parser.split("=", 1)
|
||||||
|
if tag not in entags:
|
||||||
|
continue
|
||||||
|
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
|
bp = os.path.expanduser(parser)
|
||||||
|
if os.path.exists(bp):
|
||||||
|
parsers[tag] = [bp, timeout.get(tag, 30)]
|
||||||
|
break
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
arg, parser = parser.split(",", 1)
|
||||||
|
arg = arg.lower()
|
||||||
|
|
||||||
|
if arg == "f":
|
||||||
|
force[tag] = True
|
||||||
|
continue
|
||||||
|
|
||||||
|
if arg.startswith("t"):
|
||||||
|
timeout[tag] = int(arg[1:])
|
||||||
|
continue
|
||||||
|
|
||||||
|
raise Exception()
|
||||||
|
|
||||||
|
except:
|
||||||
|
self.log("invalid argument: " + orig, 1)
|
||||||
|
return
|
||||||
|
|
||||||
|
self.mtp_force[ptop] = force
|
||||||
|
self.mtp_parsers[ptop] = parsers
|
||||||
|
|
||||||
|
q = "select count(w) from mt where k = 't:mtp'"
|
||||||
|
with self.mutex:
|
||||||
|
cur = self.cur[ptop]
|
||||||
|
cur = cur.connection.cursor()
|
||||||
|
wcur = cur.connection.cursor()
|
||||||
|
n_left = cur.execute(q).fetchone()[0]
|
||||||
|
|
||||||
|
mpool = self._start_mpool()
|
||||||
|
batch_sz = mpool.maxsize * 3
|
||||||
|
t_prev = time.time()
|
||||||
|
n_prev = n_left
|
||||||
|
n_done = 0
|
||||||
|
to_delete = {}
|
||||||
|
in_progress = {}
|
||||||
|
while True:
|
||||||
|
with self.mutex:
|
||||||
|
q = "select w from mt where k = 't:mtp' limit ?"
|
||||||
|
warks = cur.execute(q, (batch_sz,)).fetchall()
|
||||||
|
warks = [x[0] for x in warks]
|
||||||
|
jobs = []
|
||||||
|
for w in warks:
|
||||||
|
q = "select rd, fn from up where substr(w,1,16)=? limit 1"
|
||||||
|
rd, fn = cur.execute(q, (w,)).fetchone()
|
||||||
|
rd, fn = s3dec(rd, fn)
|
||||||
|
abspath = os.path.join(ptop, rd, fn)
|
||||||
|
|
||||||
|
q = "select k from mt where w = ?"
|
||||||
|
have = cur.execute(q, (w,)).fetchall()
|
||||||
|
have = [x[0] for x in have]
|
||||||
|
|
||||||
|
if ".dur" not in have and ".dur" in entags:
|
||||||
|
# skip non-audio
|
||||||
|
to_delete[w] = True
|
||||||
|
n_left -= 1
|
||||||
|
continue
|
||||||
|
|
||||||
|
if w in in_progress:
|
||||||
|
continue
|
||||||
|
|
||||||
|
task_parsers = {
|
||||||
|
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
|
||||||
|
|
||||||
|
done = self._flush_mpool(wcur)
|
||||||
|
|
||||||
|
with self.mutex:
|
||||||
|
for w in done:
|
||||||
|
to_delete[w] = True
|
||||||
|
in_progress.pop(w)
|
||||||
|
n_done += 1
|
||||||
|
|
||||||
|
for w in to_delete.keys():
|
||||||
|
q = "delete from mt where w = ? and k = 't:mtp'"
|
||||||
|
cur.execute(q, (w,))
|
||||||
|
|
||||||
|
to_delete = {}
|
||||||
|
|
||||||
|
if not warks:
|
||||||
|
break
|
||||||
|
|
||||||
|
if not jobs:
|
||||||
|
continue
|
||||||
|
|
||||||
|
try:
|
||||||
|
now = time.time()
|
||||||
|
s = ((now - t_prev) / (n_prev - n_left)) * n_left
|
||||||
|
h, s = divmod(s, 3600)
|
||||||
|
m, s = divmod(s, 60)
|
||||||
|
n_prev = n_left
|
||||||
|
t_prev = now
|
||||||
|
except:
|
||||||
|
h = 1
|
||||||
|
m = 1
|
||||||
|
|
||||||
|
msg = "mtp: {} done, {} left, eta {}h {:02d}m"
|
||||||
|
with self.mutex:
|
||||||
|
msg = msg.format(n_done, n_left, int(h), int(m))
|
||||||
|
self.log(msg, c=6)
|
||||||
|
|
||||||
|
for j in jobs:
|
||||||
|
n_left -= 1
|
||||||
|
mpool.put(j)
|
||||||
|
|
||||||
|
with self.mutex:
|
||||||
|
cur.connection.commit()
|
||||||
|
|
||||||
|
done = self._stop_mpool(mpool, wcur)
|
||||||
|
with self.mutex:
|
||||||
|
for w in done:
|
||||||
|
q = "delete from mt where w = ? and k = 't:mtp'"
|
||||||
|
cur.execute(q, (w,))
|
||||||
|
|
||||||
|
cur.connection.commit()
|
||||||
|
if n_done:
|
||||||
|
self.vac(cur, db_path, n_done, 0, sz0)
|
||||||
|
|
||||||
|
wcur.close()
|
||||||
|
cur.close()
|
||||||
|
|
||||||
|
def _start_mpool(self):
|
||||||
|
if WINDOWS and False:
|
||||||
|
nah = open(os.devnull, "wb")
|
||||||
|
wmic = "processid={}".format(os.getpid())
|
||||||
|
wmic = ["wmic", "process", "where", wmic, "call", "setpriority"]
|
||||||
|
sp.call(wmic + ["below normal"], stdout=nah, stderr=nah)
|
||||||
|
|
||||||
|
# mp.pool.ThreadPool and concurrent.futures.ThreadPoolExecutor
|
||||||
|
# both do crazy runahead so lets reinvent another wheel
|
||||||
|
nw = os.cpu_count() if hasattr(os, "cpu_count") else 4
|
||||||
|
if self.pending_tags is None:
|
||||||
|
self.log("using {}x {}".format(nw, self.mtag.backend))
|
||||||
|
self.pending_tags = []
|
||||||
|
|
||||||
|
mpool = Queue(nw)
|
||||||
|
for _ in range(nw):
|
||||||
|
thr = threading.Thread(target=self._tag_thr, args=(mpool,))
|
||||||
|
thr.daemon = True
|
||||||
|
thr.start()
|
||||||
|
|
||||||
|
return mpool
|
||||||
|
|
||||||
|
def _stop_mpool(self, mpool, wcur):
|
||||||
|
if not mpool:
|
||||||
|
return
|
||||||
|
|
||||||
|
for _ in range(mpool.maxsize):
|
||||||
|
mpool.put(None)
|
||||||
|
|
||||||
|
mpool.join()
|
||||||
|
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
|
||||||
|
|
||||||
def _tag_thr(self, q):
|
def _tag_thr(self, q):
|
||||||
while True:
|
while True:
|
||||||
task = q.get()
|
task = q.get()
|
||||||
@@ -500,25 +704,48 @@ class Up2k(object):
|
|||||||
return
|
return
|
||||||
|
|
||||||
try:
|
try:
|
||||||
write_cur, entags, wark, abspath = task
|
parser, entags, wark, abspath = task
|
||||||
|
if parser == "mtag":
|
||||||
tags = self.mtag.get(abspath)
|
tags = self.mtag.get(abspath)
|
||||||
|
else:
|
||||||
|
tags = self.mtag.get_bin(parser, abspath)
|
||||||
|
vtags = [
|
||||||
|
"\033[36m{} \033[33m{}".format(k, v) for k, v in tags.items()
|
||||||
|
]
|
||||||
|
self.log("{}\033[0m [{}]".format(" ".join(vtags), abspath))
|
||||||
|
|
||||||
with self.mutex:
|
with self.mutex:
|
||||||
n = self._tag_file(write_cur, entags, wark, abspath, tags)
|
self.pending_tags.append([entags, wark, abspath, tags])
|
||||||
self.n_mtag_tags_added += n
|
|
||||||
except:
|
except:
|
||||||
ex = traceback.format_exc()
|
ex = traceback.format_exc()
|
||||||
|
if parser == "mtag":
|
||||||
|
parser = self.mtag.backend
|
||||||
|
|
||||||
msg = "{} failed to read tags from {}:\n{}"
|
msg = "{} failed to read tags from {}:\n{}"
|
||||||
self.log(msg.format(self.mtag.backend, abspath, ex), c=3)
|
self.log(msg.format(parser, abspath, ex), c=3)
|
||||||
|
|
||||||
q.task_done()
|
q.task_done()
|
||||||
|
|
||||||
def _tag_file(self, write_cur, entags, wark, abspath, tags=None):
|
def _tag_file(self, write_cur, entags, wark, abspath, tags=None):
|
||||||
tags = tags or self.mtag.get(abspath)
|
if tags is None:
|
||||||
|
tags = self.mtag.get(abspath)
|
||||||
|
|
||||||
|
if entags:
|
||||||
tags = {k: v for k, v in tags.items() if k in entags}
|
tags = {k: v for k, v in tags.items() if k in entags}
|
||||||
if not tags:
|
if not tags:
|
||||||
# indicate scanned without tags
|
# indicate scanned without tags
|
||||||
tags = {"x": 0}
|
tags = {"x": 0}
|
||||||
|
|
||||||
|
if not tags:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
for k in tags.keys():
|
||||||
|
q = "delete from mt where w = ? and ({})".format(
|
||||||
|
" or ".join(["k = ?"] * len(tags))
|
||||||
|
)
|
||||||
|
args = [wark[:16]] + list(tags.keys())
|
||||||
|
write_cur.execute(q, tuple(args))
|
||||||
|
|
||||||
ret = 0
|
ret = 0
|
||||||
for k, v in tags.items():
|
for k, v in tags.items():
|
||||||
q = "insert into mt values (?,?,?)"
|
q = "insert into mt values (?,?,?)"
|
||||||
@@ -529,6 +756,7 @@ class Up2k(object):
|
|||||||
|
|
||||||
def _orz(self, db_path):
|
def _orz(self, db_path):
|
||||||
return sqlite3.connect(db_path, check_same_thread=False).cursor()
|
return sqlite3.connect(db_path, check_same_thread=False).cursor()
|
||||||
|
# x.set_trace_callback(trace)
|
||||||
|
|
||||||
def _open_db(self, db_path):
|
def _open_db(self, db_path):
|
||||||
existed = os.path.exists(db_path)
|
existed = os.path.exists(db_path)
|
||||||
@@ -1053,8 +1281,20 @@ class Up2k(object):
|
|||||||
def _tagger(self):
|
def _tagger(self):
|
||||||
while True:
|
while True:
|
||||||
ptop, wark, rd, fn = self.tagq.get()
|
ptop, wark, rd, fn = self.tagq.get()
|
||||||
|
if "e2t" not in self.flags[ptop]:
|
||||||
|
continue
|
||||||
|
|
||||||
abspath = os.path.join(ptop, rd, fn)
|
abspath = os.path.join(ptop, rd, fn)
|
||||||
self.log("tagging " + abspath)
|
tags = self.mtag.get(abspath)
|
||||||
|
ntags1 = len(tags)
|
||||||
|
if self.mtp_parsers.get(ptop, {}):
|
||||||
|
parser = {
|
||||||
|
k: v
|
||||||
|
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]
|
||||||
if not cur:
|
if not cur:
|
||||||
@@ -1066,11 +1306,11 @@ class Up2k(object):
|
|||||||
self.log("no entags okay.jpg", c=3)
|
self.log("no entags okay.jpg", c=3)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if "e2t" in self.flags[ptop]:
|
self._tag_file(cur, entags, wark, abspath, tags)
|
||||||
self._tag_file(cur, entags, wark, abspath)
|
|
||||||
|
|
||||||
cur.connection.commit()
|
cur.connection.commit()
|
||||||
|
|
||||||
|
self.log("tagged {} ({}+{})".format(abspath, ntags1, len(tags) - ntags1))
|
||||||
|
|
||||||
def _hasher(self):
|
def _hasher(self):
|
||||||
while True:
|
while True:
|
||||||
ptop, rd, fn = self.hashq.get()
|
ptop, rd, fn = self.hashq.get()
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import select
|
|||||||
import struct
|
import struct
|
||||||
import hashlib
|
import hashlib
|
||||||
import platform
|
import platform
|
||||||
|
import traceback
|
||||||
import threading
|
import threading
|
||||||
import mimetypes
|
import mimetypes
|
||||||
import contextlib
|
import contextlib
|
||||||
@@ -56,11 +57,58 @@ HTTPCODE = {
|
|||||||
413: "Payload Too Large",
|
413: "Payload Too Large",
|
||||||
416: "Requested Range Not Satisfiable",
|
416: "Requested Range Not Satisfiable",
|
||||||
422: "Unprocessable Entity",
|
422: "Unprocessable Entity",
|
||||||
|
429: "Too Many Requests",
|
||||||
500: "Internal Server Error",
|
500: "Internal Server Error",
|
||||||
501: "Not Implemented",
|
501: "Not Implemented",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
IMPLICATIONS = [
|
||||||
|
["e2dsa", "e2ds"],
|
||||||
|
["e2ds", "e2d"],
|
||||||
|
["e2tsr", "e2ts"],
|
||||||
|
["e2ts", "e2t"],
|
||||||
|
["e2t", "e2d"],
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
REKOBO_KEY = {
|
||||||
|
v: ln.split(" ", 1)[0]
|
||||||
|
for ln in """
|
||||||
|
1B 6d B
|
||||||
|
2B 7d Gb F#
|
||||||
|
3B 8d Db C#
|
||||||
|
4B 9d Ab G#
|
||||||
|
5B 10d Eb D#
|
||||||
|
6B 11d Bb A#
|
||||||
|
7B 12d F
|
||||||
|
8B 1d C
|
||||||
|
9B 2d G
|
||||||
|
10B 3d D
|
||||||
|
11B 4d A
|
||||||
|
12B 5d E
|
||||||
|
1A 6m Abm G#m
|
||||||
|
2A 7m Ebm D#m
|
||||||
|
3A 8m Bbm A#m
|
||||||
|
4A 9m Fm
|
||||||
|
5A 10m Cm
|
||||||
|
6A 11m Gm
|
||||||
|
7A 12m Dm
|
||||||
|
8A 1m Am
|
||||||
|
9A 2m Em
|
||||||
|
10A 3m Bm
|
||||||
|
11A 4m Gbm F#m
|
||||||
|
12A 5m Dbm C#m
|
||||||
|
""".strip().split(
|
||||||
|
"\n"
|
||||||
|
)
|
||||||
|
for v in ln.strip().split(" ")[1:]
|
||||||
|
if v
|
||||||
|
}
|
||||||
|
|
||||||
|
REKOBO_LKEY = {k.lower(): v for k, v in REKOBO_KEY.items()}
|
||||||
|
|
||||||
|
|
||||||
class Counter(object):
|
class Counter(object):
|
||||||
def __init__(self, v=0):
|
def __init__(self, v=0):
|
||||||
self.v = v
|
self.v = v
|
||||||
@@ -139,6 +187,31 @@ def nuprint(msg):
|
|||||||
uprint("{}\n".format(msg))
|
uprint("{}\n".format(msg))
|
||||||
|
|
||||||
|
|
||||||
|
def rice_tid():
|
||||||
|
tid = threading.current_thread().ident
|
||||||
|
c = struct.unpack(b"B" * 5, struct.pack(b">Q", tid)[-5:])
|
||||||
|
return "".join("\033[1;37;48;5;{}m{:02x}".format(x, x) for x in c) + "\033[0m"
|
||||||
|
|
||||||
|
|
||||||
|
def trace(*args, **kwargs):
|
||||||
|
t = time.time()
|
||||||
|
stack = "".join(
|
||||||
|
"\033[36m{}\033[33m{}".format(x[0].split(os.sep)[-1][:-3], x[1])
|
||||||
|
for x in traceback.extract_stack()[3:-1]
|
||||||
|
)
|
||||||
|
parts = ["{:.6f}".format(t), rice_tid(), stack]
|
||||||
|
|
||||||
|
if args:
|
||||||
|
parts.append(repr(args))
|
||||||
|
|
||||||
|
if kwargs:
|
||||||
|
parts.append(repr(kwargs))
|
||||||
|
|
||||||
|
msg = "\033[0m ".join(parts)
|
||||||
|
# _tracebuf.append(msg)
|
||||||
|
nuprint(msg)
|
||||||
|
|
||||||
|
|
||||||
@contextlib.contextmanager
|
@contextlib.contextmanager
|
||||||
def ren_open(fname, *args, **kwargs):
|
def ren_open(fname, *args, **kwargs):
|
||||||
fdir = kwargs.pop("fdir", None)
|
fdir = kwargs.pop("fdir", None)
|
||||||
@@ -477,6 +550,16 @@ def get_spd(nbyte, t0, t=None):
|
|||||||
return "{} \033[0m{}/s\033[0m".format(s1, s2)
|
return "{} \033[0m{}/s\033[0m".format(s1, s2)
|
||||||
|
|
||||||
|
|
||||||
|
def s2hms(s, optional_h=False):
|
||||||
|
s = int(s)
|
||||||
|
h, s = divmod(s, 3600)
|
||||||
|
m, s = divmod(s, 60)
|
||||||
|
if not h and optional_h:
|
||||||
|
return "{}:{:02}".format(m, s)
|
||||||
|
|
||||||
|
return "{}:{:02}:{:02}".format(h, m, s)
|
||||||
|
|
||||||
|
|
||||||
def undot(path):
|
def undot(path):
|
||||||
ret = []
|
ret = []
|
||||||
for node in path.split("/"):
|
for node in path.split("/"):
|
||||||
@@ -837,7 +920,11 @@ def chkcmd(*argv):
|
|||||||
def gzip_orig_sz(fn):
|
def gzip_orig_sz(fn):
|
||||||
with open(fsenc(fn), "rb") as f:
|
with open(fsenc(fn), "rb") as f:
|
||||||
f.seek(-4, 2)
|
f.seek(-4, 2)
|
||||||
return struct.unpack(b"I", f.read(4))[0]
|
rv = f.read(4)
|
||||||
|
try:
|
||||||
|
return struct.unpack(b"I", rv)[0]
|
||||||
|
except:
|
||||||
|
return struct.unpack("I", rv)[0]
|
||||||
|
|
||||||
|
|
||||||
def py_desc():
|
def py_desc():
|
||||||
@@ -847,7 +934,11 @@ def py_desc():
|
|||||||
if ofs > 0:
|
if ofs > 0:
|
||||||
py_ver = py_ver[:ofs]
|
py_ver = py_ver[:ofs]
|
||||||
|
|
||||||
|
try:
|
||||||
bitness = struct.calcsize(b"P") * 8
|
bitness = struct.calcsize(b"P") * 8
|
||||||
|
except:
|
||||||
|
bitness = struct.calcsize("P") * 8
|
||||||
|
|
||||||
host_os = platform.system()
|
host_os = platform.system()
|
||||||
compiler = platform.python_compiler()
|
compiler = platform.python_compiler()
|
||||||
|
|
||||||
|
|||||||
@@ -405,7 +405,7 @@ input[type="checkbox"]:checked+label {
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
#op_search table {
|
#srch_form {
|
||||||
border: 1px solid #3a3a3a;
|
border: 1px solid #3a3a3a;
|
||||||
box-shadow: 0 0 1em #222 inset;
|
box-shadow: 0 0 1em #222 inset;
|
||||||
background: #2d2d2d;
|
background: #2d2d2d;
|
||||||
@@ -414,14 +414,25 @@ input[type="checkbox"]:checked+label {
|
|||||||
margin-bottom: 0;
|
margin-bottom: 0;
|
||||||
padding: 0 .5em .5em 0;
|
padding: 0 .5em .5em 0;
|
||||||
}
|
}
|
||||||
|
#srch_form table {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
#srch_form td {
|
#srch_form td {
|
||||||
padding: .6em .6em;
|
padding: .6em .6em;
|
||||||
}
|
}
|
||||||
|
#srch_form td:first-child {
|
||||||
|
width: 3em;
|
||||||
|
padding-right: .2em;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
#op_search input {
|
#op_search input {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
}
|
}
|
||||||
#srch_q {
|
#srch_q {
|
||||||
white-space: pre;
|
white-space: pre;
|
||||||
|
color: #f80;
|
||||||
|
height: 1em;
|
||||||
|
margin: .2em 0 -1em 1.6em;
|
||||||
}
|
}
|
||||||
#files td div span {
|
#files td div span {
|
||||||
color: #fff;
|
color: #fff;
|
||||||
@@ -482,6 +493,7 @@ input[type="checkbox"]:checked+label {
|
|||||||
left: -1.7em;
|
left: -1.7em;
|
||||||
width: calc(100% + 1.3em);
|
width: calc(100% + 1.3em);
|
||||||
}
|
}
|
||||||
|
.tglbtn,
|
||||||
#tree>a+a {
|
#tree>a+a {
|
||||||
padding: .2em .4em;
|
padding: .2em .4em;
|
||||||
font-size: 1.2em;
|
font-size: 1.2em;
|
||||||
@@ -492,9 +504,11 @@ input[type="checkbox"]:checked+label {
|
|||||||
position: relative;
|
position: relative;
|
||||||
top: -.2em;
|
top: -.2em;
|
||||||
}
|
}
|
||||||
|
.tglbtn:hover,
|
||||||
#tree>a+a:hover {
|
#tree>a+a:hover {
|
||||||
background: #805;
|
background: #805;
|
||||||
}
|
}
|
||||||
|
.tglbtn.on,
|
||||||
#tree>a+a.on {
|
#tree>a+a.on {
|
||||||
background: #fc4;
|
background: #fc4;
|
||||||
color: #400;
|
color: #400;
|
||||||
@@ -601,7 +615,10 @@ input[type="checkbox"]:checked+label {
|
|||||||
max-width: none;
|
max-width: none;
|
||||||
margin-right: 1.5em;
|
margin-right: 1.5em;
|
||||||
}
|
}
|
||||||
#key_notation>span {
|
#op_cfg>div>a {
|
||||||
|
line-height: 2em;
|
||||||
|
}
|
||||||
|
#op_cfg>div>span {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
padding: .2em .4em;
|
padding: .2em .4em;
|
||||||
}
|
}
|
||||||
@@ -624,6 +641,9 @@ input[type="checkbox"]:checked+label {
|
|||||||
top: 6em;
|
top: 6em;
|
||||||
right: 1.5em;
|
right: 1.5em;
|
||||||
}
|
}
|
||||||
|
#ops:hover #opdesc.off {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
#opdesc code {
|
#opdesc code {
|
||||||
background: #3c3c3c;
|
background: #3c3c3c;
|
||||||
padding: .2em .3em;
|
padding: .2em .3em;
|
||||||
|
|||||||
@@ -29,9 +29,9 @@
|
|||||||
|
|
||||||
<div id="op_search" class="opview">
|
<div id="op_search" class="opview">
|
||||||
{%- if have_tags_idx %}
|
{%- if have_tags_idx %}
|
||||||
<table id="srch_form" class="tags"></table>
|
<div id="srch_form" class="tags"></div>
|
||||||
{%- else %}
|
{%- else %}
|
||||||
<table id="srch_form"></table>
|
<div id="srch_form"></div>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
<div id="srch_q"></div>
|
<div id="srch_q"></div>
|
||||||
</div>
|
</div>
|
||||||
@@ -41,6 +41,10 @@
|
|||||||
<div id="op_cfg" class="opview opbox">
|
<div id="op_cfg" class="opview opbox">
|
||||||
<h3>key notation</h3>
|
<h3>key notation</h3>
|
||||||
<div id="key_notation"></div>
|
<div id="key_notation"></div>
|
||||||
|
<h3>tooltips</h3>
|
||||||
|
<div>
|
||||||
|
<a id="tooltips" class="tglbtn" href="#">enable</a>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h1 id="path">
|
<h1 id="path">
|
||||||
@@ -54,7 +58,7 @@
|
|||||||
<a href="#" id="detree">🍞...</a>
|
<a href="#" id="detree">🍞...</a>
|
||||||
<a href="#" step="2" id="twobytwo">+</a>
|
<a href="#" step="2" id="twobytwo">+</a>
|
||||||
<a href="#" step="-2" id="twig">–</a>
|
<a href="#" step="-2" id="twig">–</a>
|
||||||
<a href="#" id="dyntree">a</a>
|
<a href="#" class="tglbtn" id="dyntree">a</a>
|
||||||
<ul id="treeul"></ul>
|
<ul id="treeul"></ul>
|
||||||
<div id="thx_ff"> </div>
|
<div id="thx_ff"> </div>
|
||||||
</div>
|
</div>
|
||||||
@@ -67,17 +71,17 @@
|
|||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th></th>
|
<th></th>
|
||||||
<th><span>File Name</span></th>
|
<th name="href"><span>File Name</span></th>
|
||||||
<th sort="int"><span>Size</span></th>
|
<th name="sz" sort="int"><span>Size</span></th>
|
||||||
{%- for k in taglist %}
|
{%- for k in taglist %}
|
||||||
{%- if k.startswith('.') %}
|
{%- if k.startswith('.') %}
|
||||||
<th sort="int"><span>{{ k[1:] }}</span></th>
|
<th name="tags/{{ k }}" sort="int"><span>{{ k[1:] }}</span></th>
|
||||||
{%- else %}
|
{%- else %}
|
||||||
<th><span>{{ k[0]|upper }}{{ k[1:] }}</span></th>
|
<th name="tags/{{ k }}"><span>{{ k[0]|upper }}{{ k[1:] }}</span></th>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
{%- endfor %}
|
{%- endfor %}
|
||||||
<th><span>T</span></th>
|
<th name="ext"><span>T</span></th>
|
||||||
<th><span>Date</span></th>
|
<th name="ts"><span>Date</span></th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
|
|||||||
@@ -6,22 +6,19 @@ function dbg(msg) {
|
|||||||
ebi('path').innerHTML = msg;
|
ebi('path').innerHTML = msg;
|
||||||
}
|
}
|
||||||
|
|
||||||
makeSortable(ebi('files'));
|
|
||||||
|
|
||||||
|
|
||||||
// extract songs + add play column
|
// extract songs + add play column
|
||||||
function init_mp() {
|
function MPlayer() {
|
||||||
var tracks = [];
|
this.id = new Date().getTime();
|
||||||
var ret = {
|
this.au = null;
|
||||||
'au': null,
|
this.au_native = null;
|
||||||
'au_native': null,
|
this.au_ogvjs = null;
|
||||||
'au_ogvjs': null,
|
this.cover_url = '';
|
||||||
'tracks': tracks,
|
this.tracks = {};
|
||||||
'cover_url': ''
|
this.order = [];
|
||||||
};
|
|
||||||
var re_audio = /\.(opus|ogg|m4a|aac|mp3|wav|flac)$/i;
|
|
||||||
|
|
||||||
var trs = ebi('files').getElementsByTagName('tbody')[0].getElementsByTagName('tr');
|
var re_audio = /\.(opus|ogg|m4a|aac|mp3|wav|flac)$/i;
|
||||||
|
var trs = document.querySelectorAll('#files tbody tr');
|
||||||
for (var a = 0, aa = trs.length; a < aa; a++) {
|
for (var a = 0, aa = trs.length; a < aa; a++) {
|
||||||
var tds = trs[a].getElementsByTagName('td');
|
var tds = trs[a].getElementsByTagName('td');
|
||||||
var link = tds[1].getElementsByTagName('a');
|
var link = tds[1].getElementsByTagName('a');
|
||||||
@@ -30,37 +27,48 @@ function init_mp() {
|
|||||||
|
|
||||||
var m = re_audio.exec(url);
|
var m = re_audio.exec(url);
|
||||||
if (m) {
|
if (m) {
|
||||||
var ntrack = tracks.length;
|
var tid = link.getAttribute('id');
|
||||||
tracks.push(url);
|
this.order.push(tid);
|
||||||
|
this.tracks[tid] = url;
|
||||||
tds[0].innerHTML = '<a id="trk' + ntrack + '" href="#trk' + ntrack + '" class="play">play</a></td>';
|
tds[0].innerHTML = '<a id="a' + tid + '" href="#a' + tid + '" class="play">play</a></td>';
|
||||||
|
ebi('a' + tid).onclick = ev_play;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (var a = 0, aa = tracks.length; a < aa; a++)
|
this.vol = sread('vol');
|
||||||
ebi('trk' + a).onclick = ev_play;
|
if (this.vol !== null)
|
||||||
|
this.vol = parseFloat(this.vol);
|
||||||
ret.vol = sread('vol');
|
|
||||||
if (ret.vol !== null)
|
|
||||||
ret.vol = parseFloat(ret.vol);
|
|
||||||
else
|
else
|
||||||
ret.vol = 0.5;
|
this.vol = 0.5;
|
||||||
|
|
||||||
ret.expvol = function () {
|
this.expvol = function () {
|
||||||
return 0.5 * ret.vol + 0.5 * ret.vol * ret.vol;
|
return 0.5 * this.vol + 0.5 * this.vol * this.vol;
|
||||||
};
|
};
|
||||||
|
|
||||||
ret.setvol = function (vol) {
|
this.setvol = function (vol) {
|
||||||
ret.vol = Math.max(Math.min(vol, 1), 0);
|
this.vol = Math.max(Math.min(vol, 1), 0);
|
||||||
swrite('vol', vol);
|
swrite('vol', vol);
|
||||||
|
|
||||||
if (ret.au)
|
if (this.au)
|
||||||
ret.au.volume = ret.expvol();
|
this.au.volume = this.expvol();
|
||||||
};
|
};
|
||||||
|
|
||||||
return ret;
|
this.read_order = function () {
|
||||||
|
var order = [];
|
||||||
|
var links = document.querySelectorAll('#files>tbody>tr>td:nth-child(1)>a');
|
||||||
|
for (var a = 0, aa = links.length; a < aa; a++) {
|
||||||
|
var tid = links[a].getAttribute('id');
|
||||||
|
if (!tid || tid.indexOf('af-') !== 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
order.push(tid.slice(1));
|
||||||
}
|
}
|
||||||
var mp = init_mp();
|
this.order = order;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
addcrc();
|
||||||
|
var mp = new MPlayer();
|
||||||
|
makeSortable(ebi('files'), mp.read_order.bind(mp));
|
||||||
|
|
||||||
|
|
||||||
// toggle player widget
|
// toggle player widget
|
||||||
@@ -180,10 +188,31 @@ var pbar = (function () {
|
|||||||
pctx.setTransform(scale, 0, 0, scale, 0, 0);
|
pctx.setTransform(scale, 0, 0, scale, 0, 0);
|
||||||
pctx.clearRect(0, 0, sw, sh);
|
pctx.clearRect(0, 0, sw, sh);
|
||||||
|
|
||||||
|
pctx.fillStyle = 'rgba(204,255,128,0.15)';
|
||||||
|
for (var p = 1, mins = mp.au.duration / 10; p <= mins; p++)
|
||||||
|
pctx.fillRect(Math.floor(sm * p * 10), 0, 2, sh);
|
||||||
|
|
||||||
|
pctx.fillStyle = '#9b7';
|
||||||
|
pctx.fillStyle = 'rgba(192,255,96,0.5)';
|
||||||
|
for (var p = 1, mins = mp.au.duration / 60; p <= mins; p++)
|
||||||
|
pctx.fillRect(Math.floor(sm * p * 60), 0, 2, sh);
|
||||||
|
|
||||||
var w = 8;
|
var w = 8;
|
||||||
var x = sm * mp.au.currentTime;
|
var x = sm * mp.au.currentTime;
|
||||||
pctx.fillStyle = '#573'; pctx.fillRect((x - w / 2) - 1, 0, w + 2, sh);
|
pctx.fillStyle = '#573'; pctx.fillRect((x - w / 2) - 1, 0, w + 2, sh);
|
||||||
pctx.fillStyle = '#dfc'; pctx.fillRect((x - w / 2), 0, 8, sh);
|
pctx.fillStyle = '#dfc'; pctx.fillRect((x - w / 2), 0, 8, sh);
|
||||||
|
|
||||||
|
pctx.fillStyle = '#fff';
|
||||||
|
pctx.font = '1em sans-serif';
|
||||||
|
var txt = s2ms(mp.au.duration);
|
||||||
|
var tw = pctx.measureText(txt).width;
|
||||||
|
pctx.fillText(txt, sw - (tw + 8), sh / 3 * 2);
|
||||||
|
|
||||||
|
txt = s2ms(mp.au.currentTime);
|
||||||
|
tw = pctx.measureText(txt).width;
|
||||||
|
var gw = pctx.measureText("88:88::").width;
|
||||||
|
var xt = x < sw / 2 ? (x + 8) : (Math.min(sw - gw, x - 8) - tw);
|
||||||
|
pctx.fillText(txt, xt, sh / 3 * 2);
|
||||||
};
|
};
|
||||||
return r;
|
return r;
|
||||||
})();
|
})();
|
||||||
@@ -277,18 +306,41 @@ var vbar = (function () {
|
|||||||
})();
|
})();
|
||||||
|
|
||||||
|
|
||||||
// hook up the widget buttons
|
function seek_au_mul(mul) {
|
||||||
(function () {
|
if (mp.au)
|
||||||
var bskip = function (n) {
|
seek_au_sec(mp.au.duration * mul);
|
||||||
|
}
|
||||||
|
|
||||||
|
function seek_au_sec(seek) {
|
||||||
|
if (!mp.au)
|
||||||
|
return;
|
||||||
|
|
||||||
|
console.log('seek: ' + seek);
|
||||||
|
if (!isFinite(seek))
|
||||||
|
return;
|
||||||
|
|
||||||
|
mp.au.currentTime = seek;
|
||||||
|
|
||||||
|
if (mp.au === mp.au_native)
|
||||||
|
// hack: ogv.js breaks on .play() during playback
|
||||||
|
mp.au.play();
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
function song_skip(n) {
|
||||||
var tid = null;
|
var tid = null;
|
||||||
if (mp.au)
|
if (mp.au)
|
||||||
tid = mp.au.tid;
|
tid = mp.au.tid;
|
||||||
|
|
||||||
if (tid !== null)
|
if (tid !== null)
|
||||||
play(tid + n);
|
play(mp.order.indexOf(tid) + n);
|
||||||
else
|
else
|
||||||
play(0);
|
play(mp.order[0]);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// hook up the widget buttons
|
||||||
|
(function () {
|
||||||
ebi('bplay').onclick = function (e) {
|
ebi('bplay').onclick = function (e) {
|
||||||
ev(e);
|
ev(e);
|
||||||
if (mp.au) {
|
if (mp.au) {
|
||||||
@@ -302,31 +354,20 @@ var vbar = (function () {
|
|||||||
};
|
};
|
||||||
ebi('bprev').onclick = function (e) {
|
ebi('bprev').onclick = function (e) {
|
||||||
ev(e);
|
ev(e);
|
||||||
bskip(-1);
|
song_skip(-1);
|
||||||
};
|
};
|
||||||
ebi('bnext').onclick = function (e) {
|
ebi('bnext').onclick = function (e) {
|
||||||
ev(e);
|
ev(e);
|
||||||
bskip(1);
|
song_skip(1);
|
||||||
};
|
};
|
||||||
ebi('barpos').onclick = function (e) {
|
ebi('barpos').onclick = function (e) {
|
||||||
if (!mp.au) {
|
if (!mp.au) {
|
||||||
//dbg((new Date()).getTime());
|
|
||||||
return play(0);
|
return play(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
var rect = pbar.pcan.getBoundingClientRect();
|
var rect = pbar.pcan.getBoundingClientRect();
|
||||||
var x = e.clientX - rect.left;
|
var x = e.clientX - rect.left;
|
||||||
var mul = x * 1.0 / rect.width;
|
seek_au_mul(x * 1.0 / rect.width);
|
||||||
var seek = mp.au.duration * mul;
|
|
||||||
console.log('seek: ' + seek);
|
|
||||||
if (!isFinite(seek))
|
|
||||||
return;
|
|
||||||
|
|
||||||
mp.au.currentTime = seek;
|
|
||||||
|
|
||||||
if (mp.au === mp.au_native)
|
|
||||||
// hack: ogv.js breaks on .play() during playback
|
|
||||||
mp.au.play();
|
|
||||||
};
|
};
|
||||||
})();
|
})();
|
||||||
|
|
||||||
@@ -359,7 +400,7 @@ var vbar = (function () {
|
|||||||
var len = mp.au.duration;
|
var len = mp.au.duration;
|
||||||
if (pos > 0 && pos > len - 0.1) {
|
if (pos > 0 && pos > len - 0.1) {
|
||||||
last_skip_url = mp.au.src;
|
last_skip_url = mp.au.src;
|
||||||
play(mp.au.tid + 1);
|
song_skip(1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -372,7 +413,7 @@ var vbar = (function () {
|
|||||||
// event from play button next to a file in the list
|
// event from play button next to a file in the list
|
||||||
function ev_play(e) {
|
function ev_play(e) {
|
||||||
ev(e);
|
ev(e);
|
||||||
play(parseInt(this.getAttribute('id').substr(3)));
|
play(this.getAttribute('id').slice(1));
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -394,18 +435,24 @@ catch (ex) { }
|
|||||||
|
|
||||||
// plays the tid'th audio file on the page
|
// plays the tid'th audio file on the page
|
||||||
function play(tid, call_depth) {
|
function play(tid, call_depth) {
|
||||||
if (mp.tracks.length == 0)
|
if (mp.order.length == 0)
|
||||||
return alert('no audio found wait what');
|
return alert('no audio found wait what');
|
||||||
|
|
||||||
while (tid >= mp.tracks.length)
|
var tn = tid;
|
||||||
tid -= mp.tracks.length;
|
if ((tn + '').indexOf('f-') === 0)
|
||||||
|
tn = mp.order.indexOf(tn);
|
||||||
|
|
||||||
while (tid < 0)
|
while (tn >= mp.order.length)
|
||||||
tid += mp.tracks.length;
|
tn -= mp.order.length;
|
||||||
|
|
||||||
|
while (tn < 0)
|
||||||
|
tn += mp.order.length;
|
||||||
|
|
||||||
|
tid = mp.order[tn];
|
||||||
|
|
||||||
if (mp.au) {
|
if (mp.au) {
|
||||||
mp.au.pause();
|
mp.au.pause();
|
||||||
setclass('trk' + mp.au.tid, 'play');
|
setclass('a' + mp.au.tid, 'play');
|
||||||
}
|
}
|
||||||
|
|
||||||
// ogv.js breaks on .play() unless directly user-triggered
|
// ogv.js breaks on .play() unless directly user-triggered
|
||||||
@@ -449,7 +496,7 @@ function play(tid, call_depth) {
|
|||||||
mp.au.tid = tid;
|
mp.au.tid = tid;
|
||||||
mp.au.src = url;
|
mp.au.src = url;
|
||||||
mp.au.volume = mp.expvol();
|
mp.au.volume = mp.expvol();
|
||||||
var oid = 'trk' + tid;
|
var oid = 'a' + tid;
|
||||||
setclass(oid, 'play act');
|
setclass(oid, 'play act');
|
||||||
var trs = ebi('files').getElementsByTagName('tbody')[0].getElementsByTagName('tr');
|
var trs = ebi('files').getElementsByTagName('tbody')[0].getElementsByTagName('tr');
|
||||||
for (var a = 0, aa = trs.length; a < aa; a++) {
|
for (var a = 0, aa = trs.length; a < aa; a++) {
|
||||||
@@ -480,8 +527,8 @@ function play(tid, call_depth) {
|
|||||||
catch (ex) {
|
catch (ex) {
|
||||||
alert('playback failed: ' + ex);
|
alert('playback failed: ' + ex);
|
||||||
}
|
}
|
||||||
setclass('trk' + mp.au.tid, 'play');
|
setclass(oid, 'play');
|
||||||
setTimeout('play(' + (mp.au.tid + 1) + ');', 500);
|
setTimeout('song_skip(1));', 500);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -560,12 +607,73 @@ function autoplay_blocked() {
|
|||||||
// autoplay linked track
|
// autoplay linked track
|
||||||
(function () {
|
(function () {
|
||||||
var v = location.hash;
|
var v = location.hash;
|
||||||
if (v && v.length > 4 && v.indexOf('#trk') === 0)
|
if (v && v.length == 12 && v.indexOf('#af-') === 0)
|
||||||
play(parseInt(v.substr(4)));
|
play(v.slice(2));
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
|
||||||
//widget.open();
|
function tree_neigh(n) {
|
||||||
|
var links = document.querySelectorAll('#treeul li>a+a');
|
||||||
|
if (!links.length) {
|
||||||
|
alert('switch to the tree for that');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var act = -1;
|
||||||
|
for (var a = 0, aa = links.length; a < aa; a++) {
|
||||||
|
if (links[a].getAttribute('class') == 'hl') {
|
||||||
|
act = a;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a += n;
|
||||||
|
if (a < 0)
|
||||||
|
a = links.length - 1;
|
||||||
|
if (a >= links.length)
|
||||||
|
a = 0;
|
||||||
|
|
||||||
|
links[a].click();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function tree_up() {
|
||||||
|
var act = document.querySelector('#treeul a.hl');
|
||||||
|
if (!act) {
|
||||||
|
alert('switch to the tree for that');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (act.previousSibling.textContent == '-')
|
||||||
|
return act.previousSibling.click();
|
||||||
|
|
||||||
|
act.parentNode.parentNode.parentNode.getElementsByTagName('a')[1].click();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
document.onkeydown = function (e) {
|
||||||
|
if (document.activeElement != document.body && document.activeElement.nodeName.toLowerCase() != 'a')
|
||||||
|
return;
|
||||||
|
|
||||||
|
var k = e.code, pos = -1;
|
||||||
|
if (k.indexOf('Digit') === 0)
|
||||||
|
pos = parseInt(k.slice(-1)) * 0.1;
|
||||||
|
|
||||||
|
if (pos !== -1)
|
||||||
|
return seek_au_mul(pos);
|
||||||
|
|
||||||
|
var n = k == 'KeyJ' ? -1 : k == 'KeyL' ? 1 : 0;
|
||||||
|
if (n !== 0)
|
||||||
|
return song_skip(n);
|
||||||
|
|
||||||
|
n = k == 'KeyU' ? -10 : k == 'KeyO' ? 10 : 0;
|
||||||
|
if (n !== 0)
|
||||||
|
return mp.au ? seek_au_sec(mp.au.currentTime + n) : true;
|
||||||
|
|
||||||
|
n = k == 'KeyI' ? -1 : k == 'KeyK' ? 1 : 0;
|
||||||
|
if (n !== 0)
|
||||||
|
return tree_neigh(n);
|
||||||
|
|
||||||
|
if (k == 'KeyP')
|
||||||
|
return tree_up();
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
// search
|
// search
|
||||||
@@ -588,15 +696,19 @@ function autoplay_blocked() {
|
|||||||
];
|
];
|
||||||
var oldcfg = [];
|
var oldcfg = [];
|
||||||
|
|
||||||
if (document.querySelector('#srch_form.tags'))
|
if (document.querySelector('#srch_form.tags')) {
|
||||||
sconf.push(["tags",
|
sconf.push(["tags",
|
||||||
["tags", "tags", "tags contains (^=start, end=$)", "46"]
|
["tags", "tags", "tags contains (^=start, end=$)", "46"]
|
||||||
]);
|
]);
|
||||||
|
sconf.push(["adv.",
|
||||||
|
["adv", "adv", "key>=1A key<=2B .bpm>165", "46"]
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
var html = [];
|
var trs = [];
|
||||||
var orig_html = null;
|
var orig_html = null;
|
||||||
for (var a = 0; a < sconf.length; a++) {
|
for (var a = 0; a < sconf.length; a++) {
|
||||||
html.push('<tr><td><br />' + sconf[a][0] + '</td>');
|
var html = ['<tr><td><br />' + sconf[a][0] + '</td>'];
|
||||||
for (var b = 1; b < 3; b++) {
|
for (var b = 1; b < 3; b++) {
|
||||||
var hn = "srch_" + sconf[a][b][0];
|
var hn = "srch_" + sconf[a][b][0];
|
||||||
var csp = (sconf[a].length == 2) ? 2 : 1;
|
var csp = (sconf[a].length == 2) ? 2 : 1;
|
||||||
@@ -609,6 +721,11 @@ function autoplay_blocked() {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
html.push('</tr>');
|
html.push('</tr>');
|
||||||
|
trs.push(html);
|
||||||
|
}
|
||||||
|
var html = [];
|
||||||
|
for (var a = 0; a < trs.length; a += 2) {
|
||||||
|
html.push('<table>' + (trs[a].concat(trs[a + 1])).join('\n') + '</table>');
|
||||||
}
|
}
|
||||||
ebi('srch_form').innerHTML = html.join('\n');
|
ebi('srch_form').innerHTML = html.join('\n');
|
||||||
|
|
||||||
@@ -617,7 +734,14 @@ function autoplay_blocked() {
|
|||||||
o[a].oninput = ev_search_input;
|
o[a].oninput = ev_search_input;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function srch_msg(err, txt) {
|
||||||
|
var o = ebi('srch_q');
|
||||||
|
o.textContent = txt;
|
||||||
|
o.style.color = err ? '#f09' : '#c90';
|
||||||
|
}
|
||||||
|
|
||||||
var search_timeout;
|
var search_timeout;
|
||||||
|
var search_in_progress = 0;
|
||||||
|
|
||||||
function ev_search_input() {
|
function ev_search_input() {
|
||||||
var v = this.value;
|
var v = this.value;
|
||||||
@@ -627,10 +751,14 @@ function autoplay_blocked() {
|
|||||||
chk.checked = ((v + '').length > 0);
|
chk.checked = ((v + '').length > 0);
|
||||||
}
|
}
|
||||||
clearTimeout(search_timeout);
|
clearTimeout(search_timeout);
|
||||||
|
var now = new Date().getTime();
|
||||||
|
if (now - search_in_progress > 30 * 1000)
|
||||||
search_timeout = setTimeout(do_search, 100);
|
search_timeout = setTimeout(do_search, 100);
|
||||||
}
|
}
|
||||||
|
|
||||||
function do_search() {
|
function do_search() {
|
||||||
|
search_in_progress = new Date().getTime();
|
||||||
|
srch_msg(false, "searching...");
|
||||||
clearTimeout(search_timeout);
|
clearTimeout(search_timeout);
|
||||||
var params = {};
|
var params = {};
|
||||||
var o = document.querySelectorAll('#op_search input[type="text"]');
|
var o = document.querySelectorAll('#op_search input[type="text"]');
|
||||||
@@ -654,9 +782,16 @@ function autoplay_blocked() {
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
if (this.status !== 200) {
|
if (this.status !== 200) {
|
||||||
alert("http " + this.status + ": " + this.responseText);
|
var msg = this.responseText;
|
||||||
|
if (msg.indexOf('<pre>') === 0)
|
||||||
|
msg = msg.slice(5);
|
||||||
|
|
||||||
|
srch_msg(true, "http " + this.status + ": " + msg);
|
||||||
|
search_in_progress = 0;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
search_in_progress = 0;
|
||||||
|
srch_msg(false, '');
|
||||||
|
|
||||||
var res = JSON.parse(this.responseText),
|
var res = JSON.parse(this.responseText),
|
||||||
tagord = res.tag_order;
|
tagord = res.tag_order;
|
||||||
@@ -674,6 +809,7 @@ function autoplay_blocked() {
|
|||||||
ebi('path').style.display = 'none';
|
ebi('path').style.display = 'none';
|
||||||
ebi('tree').style.display = 'none';
|
ebi('tree').style.display = 'none';
|
||||||
ebi('wrap').style.marginLeft = '0';
|
ebi('wrap').style.marginLeft = '0';
|
||||||
|
treectl.hidden = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
var html = mk_files_header(tagord);
|
var html = mk_files_header(tagord);
|
||||||
@@ -696,7 +832,7 @@ function autoplay_blocked() {
|
|||||||
var k = tagord[b],
|
var k = tagord[b],
|
||||||
v = r.tags[k] || "";
|
v = r.tags[k] || "";
|
||||||
|
|
||||||
if (k == "dur") {
|
if (k == ".dur") {
|
||||||
var sv = s2ms(v);
|
var sv = s2ms(v);
|
||||||
nodes[nodes.length - 1] += '</td><td sortv="' + v + '">' + sv;
|
nodes[nodes.length - 1] += '</td><td sortv="' + v + '">' + sv;
|
||||||
continue;
|
continue;
|
||||||
@@ -716,6 +852,7 @@ function autoplay_blocked() {
|
|||||||
ofiles.innerHTML = html.join('\n');
|
ofiles.innerHTML = html.join('\n');
|
||||||
ofiles.setAttribute("ts", this.ts);
|
ofiles.setAttribute("ts", this.ts);
|
||||||
filecols.set_style();
|
filecols.set_style();
|
||||||
|
mukey.render();
|
||||||
reload_browser();
|
reload_browser();
|
||||||
|
|
||||||
ebi('unsearch').onclick = unsearch;
|
ebi('unsearch').onclick = unsearch;
|
||||||
@@ -726,6 +863,7 @@ function autoplay_blocked() {
|
|||||||
ebi('path').style.display = oldcfg[0];
|
ebi('path').style.display = oldcfg[0];
|
||||||
ebi('tree').style.display = oldcfg[1];
|
ebi('tree').style.display = oldcfg[1];
|
||||||
ebi('wrap').style.marginLeft = oldcfg[2];
|
ebi('wrap').style.marginLeft = oldcfg[2];
|
||||||
|
treectl.hidden = false;
|
||||||
oldcfg = [];
|
oldcfg = [];
|
||||||
ebi('files').innerHTML = orig_html;
|
ebi('files').innerHTML = orig_html;
|
||||||
orig_html = null;
|
orig_html = null;
|
||||||
@@ -735,13 +873,18 @@ function autoplay_blocked() {
|
|||||||
|
|
||||||
|
|
||||||
var treectl = (function () {
|
var treectl = (function () {
|
||||||
|
var treectl = {
|
||||||
|
"hidden": false
|
||||||
|
};
|
||||||
var dyn = bcfg_get('dyntree', true);
|
var dyn = bcfg_get('dyntree', true);
|
||||||
var treesz = icfg_get('treesz', 16);
|
var treesz = icfg_get('treesz', 16);
|
||||||
treesz = Math.min(Math.max(treesz, 4), 50);
|
treesz = Math.min(Math.max(treesz, 4), 50);
|
||||||
console.log('treesz [' + treesz + ']');
|
console.log('treesz [' + treesz + ']');
|
||||||
|
var entreed = false;
|
||||||
|
|
||||||
function entree(e) {
|
function entree(e) {
|
||||||
ev(e);
|
ev(e);
|
||||||
|
entreed = true;
|
||||||
ebi('path').style.display = 'none';
|
ebi('path').style.display = 'none';
|
||||||
|
|
||||||
var tree = ebi('tree');
|
var tree = ebi('tree');
|
||||||
@@ -749,22 +892,29 @@ var treectl = (function () {
|
|||||||
|
|
||||||
swrite('entreed', 'tree');
|
swrite('entreed', 'tree');
|
||||||
get_tree("", get_evpath(), true);
|
get_tree("", get_evpath(), true);
|
||||||
|
window.addEventListener('scroll', onscroll);
|
||||||
|
window.addEventListener('resize', onresize);
|
||||||
onresize();
|
onresize();
|
||||||
}
|
}
|
||||||
|
|
||||||
function detree(e) {
|
function detree(e) {
|
||||||
ev(e);
|
ev(e);
|
||||||
|
entreed = false;
|
||||||
ebi('tree').style.display = 'none';
|
ebi('tree').style.display = 'none';
|
||||||
ebi('path').style.display = 'inline-block';
|
ebi('path').style.display = 'inline-block';
|
||||||
ebi('wrap').style.marginLeft = '0';
|
ebi('wrap').style.marginLeft = '0';
|
||||||
swrite('entreed', 'na');
|
swrite('entreed', 'na');
|
||||||
|
window.removeEventListener('resize', onresize);
|
||||||
|
window.removeEventListener('scroll', onscroll);
|
||||||
}
|
}
|
||||||
|
|
||||||
function onscroll() {
|
function onscroll() {
|
||||||
|
if (!entreed || treectl.hidden)
|
||||||
|
return;
|
||||||
|
|
||||||
var top = ebi('wrap').getBoundingClientRect().top;
|
var top = ebi('wrap').getBoundingClientRect().top;
|
||||||
ebi('tree').style.top = Math.max(0, parseInt(top)) + 'px';
|
ebi('tree').style.top = Math.max(0, parseInt(top)) + 'px';
|
||||||
}
|
}
|
||||||
window.addEventListener('scroll', onscroll);
|
|
||||||
|
|
||||||
function periodic() {
|
function periodic() {
|
||||||
onscroll();
|
onscroll();
|
||||||
@@ -773,6 +923,9 @@ var treectl = (function () {
|
|||||||
periodic();
|
periodic();
|
||||||
|
|
||||||
function onresize(e) {
|
function onresize(e) {
|
||||||
|
if (!entreed || treectl.hidden)
|
||||||
|
return;
|
||||||
|
|
||||||
var q = '#tree';
|
var q = '#tree';
|
||||||
var nq = 0;
|
var nq = 0;
|
||||||
while (dyn) {
|
while (dyn) {
|
||||||
@@ -786,7 +939,6 @@ var treectl = (function () {
|
|||||||
ebi('wrap').style.marginLeft = w + 'em';
|
ebi('wrap').style.marginLeft = w + 'em';
|
||||||
onscroll();
|
onscroll();
|
||||||
}
|
}
|
||||||
window.addEventListener('resize', onresize);
|
|
||||||
|
|
||||||
function get_tree(top, dst, rst) {
|
function get_tree(top, dst, rst) {
|
||||||
var xhr = new XMLHttpRequest();
|
var xhr = new XMLHttpRequest();
|
||||||
@@ -933,7 +1085,33 @@ var treectl = (function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ebi('srv_info').innerHTML = '<span>' + res.srvinf + '</span>';
|
ebi('srv_info').innerHTML = '<span>' + res.srvinf + '</span>';
|
||||||
var nodes = res.dirs.concat(res.files);
|
var nodes = res.dirs.concat(res.files),
|
||||||
|
sopts = jread('fsort', []);
|
||||||
|
|
||||||
|
try {
|
||||||
|
for (var a = sopts.length - 1; a >= 0; a--) {
|
||||||
|
var name = sopts[a][0], rev = sopts[a][1], typ = sopts[a][2];
|
||||||
|
if (name.indexOf('tags/') == -1) {
|
||||||
|
nodes.sort(function (v1, v2) {
|
||||||
|
if (!v1[name]) return -1 * rev;
|
||||||
|
if (!v2[name]) return 1 * rev;
|
||||||
|
return rev * (typ == 'int' ? (v1[name] - v2[name]) : (v1[name].localeCompare(v2[name])));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
name = name.slice(5);
|
||||||
|
nodes.sort(function (v1, v2) {
|
||||||
|
if (!v1.tags[name]) return -1 * rev;
|
||||||
|
if (!v2.tags[name]) return 1 * rev;
|
||||||
|
return rev * (typ == 'int' ? (v1.tags[name] - v2.tags[name]) : (v1.tags[name].localeCompare(v2.tags[name])));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (ex) {
|
||||||
|
console.log("failed to apply sort config: " + ex);
|
||||||
|
}
|
||||||
|
|
||||||
var top = this.top;
|
var top = this.top;
|
||||||
var html = mk_files_header(res.taglist);
|
var html = mk_files_header(res.taglist);
|
||||||
html.push('<tbody>');
|
html.push('<tbody>');
|
||||||
@@ -946,10 +1124,7 @@ var treectl = (function () {
|
|||||||
var k = res.taglist[b],
|
var k = res.taglist[b],
|
||||||
v = (r.tags || {})[k] || "";
|
v = (r.tags || {})[k] || "";
|
||||||
|
|
||||||
if (k[0] == '.')
|
if (k == ".dur") {
|
||||||
k = k.slice(1);
|
|
||||||
|
|
||||||
if (k == "dur") {
|
|
||||||
var sv = s2ms(v);
|
var sv = s2ms(v);
|
||||||
ln[ln.length - 1] += '</td><td sortv="' + v + '">' + sv;
|
ln[ln.length - 1] += '</td><td sortv="' + v + '">' + sv;
|
||||||
continue;
|
continue;
|
||||||
@@ -1049,9 +1224,8 @@ var treectl = (function () {
|
|||||||
hist_replace(get_evpath() + window.location.hash);
|
hist_replace(get_evpath() + window.location.hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
treectl.onscroll = onscroll;
|
||||||
"onscroll": onscroll
|
return treectl;
|
||||||
}
|
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
|
||||||
@@ -1107,27 +1281,47 @@ function apply_perms(perms) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function find_file_col(txt) {
|
||||||
|
var tds = ebi('files').tHead.getElementsByTagName('th');
|
||||||
|
var i = -1;
|
||||||
|
var min = false;
|
||||||
|
for (var a = 0; a < tds.length; a++) {
|
||||||
|
var spans = tds[a].getElementsByTagName('span');
|
||||||
|
if (spans.length && spans[0].textContent == txt) {
|
||||||
|
min = tds[a].getAttribute('class').indexOf('min') !== -1;
|
||||||
|
i = a;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i == -1)
|
||||||
|
return;
|
||||||
|
|
||||||
|
return [i, min];
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
function mk_files_header(taglist) {
|
function mk_files_header(taglist) {
|
||||||
var html = [
|
var html = [
|
||||||
'<thead>',
|
'<thead>',
|
||||||
'<th></th>',
|
'<th></th>',
|
||||||
'<th><span>File Name</span></th>',
|
'<th name="href"><span>File Name</span></th>',
|
||||||
'<th sort="int"><span>Size</span></th>'
|
'<th name="sz" sort="int"><span>Size</span></th>'
|
||||||
];
|
];
|
||||||
for (var a = 0; a < taglist.length; a++) {
|
for (var a = 0; a < taglist.length; a++) {
|
||||||
var tag = taglist[a];
|
var tag = taglist[a];
|
||||||
var c1 = tag.slice(0, 1).toUpperCase();
|
var c1 = tag.slice(0, 1).toUpperCase();
|
||||||
tag = c1 + tag.slice(1);
|
tag = c1 + tag.slice(1);
|
||||||
if (c1 == '.')
|
if (c1 == '.')
|
||||||
tag = '<th sort="int"><span>' + tag.slice(1);
|
tag = '<th name="tags/' + tag + '" sort="int"><span>' + tag.slice(1);
|
||||||
else
|
else
|
||||||
tag = '<th><span>' + tag;
|
tag = '<th name="tags/' + tag + '"><span>' + tag;
|
||||||
|
|
||||||
html.push(tag + '</span></th>');
|
html.push(tag + '</span></th>');
|
||||||
}
|
}
|
||||||
html = html.concat([
|
html = html.concat([
|
||||||
'<th><span>T</span></th>',
|
'<th name="ext"><span>T</span></th>',
|
||||||
'<th><span>Date</span></th>',
|
'<th name="ts"><span>Date</span></th>',
|
||||||
'</thead>',
|
'</thead>',
|
||||||
]);
|
]);
|
||||||
return html;
|
return html;
|
||||||
@@ -1204,6 +1398,21 @@ var filecols = (function () {
|
|||||||
set_style();
|
set_style();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
var ci = find_file_col('dur'),
|
||||||
|
i = ci[0],
|
||||||
|
min = ci[1],
|
||||||
|
rows = ebi('files').tBodies[0].rows;
|
||||||
|
|
||||||
|
if (!min)
|
||||||
|
for (var a = 0, aa = rows.length; a < aa; a++) {
|
||||||
|
var c = rows[a].cells[i];
|
||||||
|
if (c)
|
||||||
|
var v = c.textContent = s2ms(c.textContent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (ex) { }
|
||||||
|
|
||||||
return {
|
return {
|
||||||
"add_btns": add_btns,
|
"add_btns": add_btns,
|
||||||
"set_style": set_style,
|
"set_style": set_style,
|
||||||
@@ -1255,7 +1464,7 @@ var mukey = (function () {
|
|||||||
ev(e);
|
ev(e);
|
||||||
var notation = this.getAttribute('value');
|
var notation = this.getAttribute('value');
|
||||||
load_notation(notation);
|
load_notation(notation);
|
||||||
render();
|
try_render();
|
||||||
}
|
}
|
||||||
|
|
||||||
function load_notation(notation) {
|
function load_notation(notation) {
|
||||||
@@ -1270,32 +1479,37 @@ var mukey = (function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function render() {
|
function render() {
|
||||||
var tds = ebi('files').tHead.getElementsByTagName('th');
|
var ci = find_file_col('Key'),
|
||||||
var i = -1;
|
i = ci[0],
|
||||||
var min = false;
|
min = ci[1],
|
||||||
for (var a = 0; a < tds.length; a++) {
|
rows = ebi('files').tBodies[0].rows;
|
||||||
var spans = tds[a].getElementsByTagName('span');
|
|
||||||
if (spans.length && spans[0].textContent == 'Key') {
|
|
||||||
min = tds[a].getAttribute('class').indexOf('min') !== -1;
|
|
||||||
i = a;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (i == -1)
|
|
||||||
return;
|
|
||||||
|
|
||||||
var rows = ebi('files').tBodies[0].rows;
|
|
||||||
|
|
||||||
if (min)
|
if (min)
|
||||||
for (var a = 0, aa = rows.length; a < aa; a++) {
|
for (var a = 0, aa = rows.length; a < aa; a++) {
|
||||||
var v = rows[a].cells[i].getAttribute('html');
|
var c = rows[a].cells[i];
|
||||||
rows[a].cells[i].setAttribute('html', map[v] || v);
|
if (!c)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
var v = c.getAttribute('html');
|
||||||
|
c.setAttribute('html', map[v] || v);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
for (var a = 0, aa = rows.length; a < aa; a++) {
|
for (var a = 0, aa = rows.length; a < aa; a++) {
|
||||||
var v = rows[a].cells[i].textContent;
|
var c = rows[a].cells[i];
|
||||||
rows[a].cells[i].textContent = map[v] || v;
|
if (!c)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
var v = c.textContent;
|
||||||
|
c.textContent = map[v] || v;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function try_render() {
|
||||||
|
try {
|
||||||
|
render();
|
||||||
|
}
|
||||||
|
catch (ex) {
|
||||||
|
console.log("key notation failed: " + ex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1309,20 +1523,39 @@ var mukey = (function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
"render": render
|
"render": try_render
|
||||||
};
|
};
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
|
||||||
|
function addcrc() {
|
||||||
|
var links = document.querySelectorAll('#files>tbody>tr>td:nth-child(2)>a');
|
||||||
|
for (var a = 0, aa = links.length; a < aa; a++)
|
||||||
|
if (!links[a].getAttribute('id'))
|
||||||
|
links[a].setAttribute('id', 'f-' + crc32(links[a].textContent));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
(function () {
|
(function () {
|
||||||
|
var tt = bcfg_get('tooltips', true);
|
||||||
|
|
||||||
function set_tooltip(e) {
|
function set_tooltip(e) {
|
||||||
ev(e);
|
ev(e);
|
||||||
ebi('opdesc').innerHTML = this.getAttribute('data-desc');
|
var o = ebi('opdesc');
|
||||||
|
o.innerHTML = this.getAttribute('data-desc');
|
||||||
|
o.setAttribute('class', tt ? '' : 'off');
|
||||||
}
|
}
|
||||||
|
|
||||||
var btns = document.querySelectorAll('#ops, #ops>a');
|
var btns = document.querySelectorAll('#ops, #ops>a');
|
||||||
for (var a = 0; a < btns.length; a++) {
|
for (var a = 0; a < btns.length; a++) {
|
||||||
btns[a].onmouseenter = set_tooltip;
|
btns[a].onmouseenter = set_tooltip;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ebi('tooltips').onclick = function (e) {
|
||||||
|
ev(e);
|
||||||
|
tt = !tt;
|
||||||
|
bcfg_set('tooltips', tt);
|
||||||
|
};
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
|
||||||
@@ -1332,9 +1565,18 @@ function ev_row_tgl(e) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function reload_mp() {
|
||||||
|
if (mp && mp.au) {
|
||||||
|
mp.au.pause();
|
||||||
|
mp.au = null;
|
||||||
|
}
|
||||||
|
widget.close();
|
||||||
|
mp = new MPlayer();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
function reload_browser(not_mp) {
|
function reload_browser(not_mp) {
|
||||||
filecols.set_style();
|
filecols.set_style();
|
||||||
makeSortable(ebi('files'));
|
|
||||||
|
|
||||||
var parts = get_evpath().split('/');
|
var parts = get_evpath().split('/');
|
||||||
var rm = document.querySelectorAll('#path>a+a+a');
|
var rm = document.querySelectorAll('#path>a+a+a');
|
||||||
@@ -1346,7 +1588,7 @@ function reload_browser(not_mp) {
|
|||||||
link += parts[a] + '/';
|
link += parts[a] + '/';
|
||||||
var o = document.createElement('a');
|
var o = document.createElement('a');
|
||||||
o.setAttribute('href', link);
|
o.setAttribute('href', link);
|
||||||
o.innerHTML = parts[a];
|
o.textContent = uricom_dec(parts[a])[0];
|
||||||
ebi('path').appendChild(o);
|
ebi('path').appendChild(o);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1359,12 +1601,9 @@ function reload_browser(not_mp) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!not_mp) {
|
if (!not_mp) {
|
||||||
if (mp && mp.au) {
|
addcrc();
|
||||||
mp.au.pause();
|
reload_mp();
|
||||||
mp.au = null;
|
makeSortable(ebi('files'), mp.read_order.bind(mp));
|
||||||
}
|
|
||||||
widget.close();
|
|
||||||
mp = init_mp();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (window['up2k'])
|
if (window['up2k'])
|
||||||
|
|||||||
@@ -91,7 +91,29 @@ function import_js(url, cb) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function sortTable(table, col) {
|
var crctab = (function () {
|
||||||
|
var c, tab = [];
|
||||||
|
for (var n = 0; n < 256; n++) {
|
||||||
|
c = n;
|
||||||
|
for (var k = 0; k < 8; k++) {
|
||||||
|
c = ((c & 1) ? (0xEDB88320 ^ (c >>> 1)) : (c >>> 1));
|
||||||
|
}
|
||||||
|
tab[n] = c;
|
||||||
|
}
|
||||||
|
return tab;
|
||||||
|
})();
|
||||||
|
|
||||||
|
|
||||||
|
function crc32(str) {
|
||||||
|
var crc = 0 ^ (-1);
|
||||||
|
for (var i = 0; i < str.length; i++) {
|
||||||
|
crc = (crc >>> 8) ^ crctab[(crc ^ str.charCodeAt(i)) & 0xFF];
|
||||||
|
}
|
||||||
|
return ((crc ^ (-1)) >>> 0).toString(16);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
function sortTable(table, col, cb) {
|
||||||
var tb = table.tBodies[0],
|
var tb = table.tBodies[0],
|
||||||
th = table.tHead.rows[0].cells,
|
th = table.tHead.rows[0].cells,
|
||||||
tr = Array.prototype.slice.call(tb.rows, 0),
|
tr = Array.prototype.slice.call(tb.rows, 0),
|
||||||
@@ -100,6 +122,27 @@ function sortTable(table, col) {
|
|||||||
th[a].className = th[a].className.replace(/ *sort-?1 */, " ");
|
th[a].className = th[a].className.replace(/ *sort-?1 */, " ");
|
||||||
th[col].className += ' sort' + reverse;
|
th[col].className += ' sort' + reverse;
|
||||||
var stype = th[col].getAttribute('sort');
|
var stype = th[col].getAttribute('sort');
|
||||||
|
try {
|
||||||
|
var nrules = [], rules = jread("fsort", []);
|
||||||
|
rules.unshift([th[col].getAttribute('name'), reverse, stype || '']);
|
||||||
|
for (var a = 0; a < rules.length; a++) {
|
||||||
|
var add = true;
|
||||||
|
for (var b = 0; b < a; b++)
|
||||||
|
if (rules[a][0] == rules[b][0])
|
||||||
|
add = false;
|
||||||
|
|
||||||
|
if (add)
|
||||||
|
nrules.push(rules[a]);
|
||||||
|
|
||||||
|
if (nrules.length >= 10)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
jwrite("fsort", nrules);
|
||||||
|
}
|
||||||
|
catch (ex) {
|
||||||
|
console.log("failed to persist sort rules, resetting: " + ex);
|
||||||
|
jwrite("fsort", null);
|
||||||
|
}
|
||||||
var vl = [];
|
var vl = [];
|
||||||
for (var a = 0; a < tr.length; a++) {
|
for (var a = 0; a < tr.length; a++) {
|
||||||
var cell = tr[a].cells[col];
|
var cell = tr[a].cells[col];
|
||||||
@@ -127,8 +170,9 @@ function sortTable(table, col) {
|
|||||||
return reverse * (a.localeCompare(b));
|
return reverse * (a.localeCompare(b));
|
||||||
});
|
});
|
||||||
for (i = 0; i < tr.length; ++i) tb.appendChild(tr[vl[i][1]]);
|
for (i = 0; i < tr.length; ++i) tb.appendChild(tr[vl[i][1]]);
|
||||||
|
if (cb) cb();
|
||||||
}
|
}
|
||||||
function makeSortable(table) {
|
function makeSortable(table, cb) {
|
||||||
var th = table.tHead, i;
|
var th = table.tHead, i;
|
||||||
th && (th = th.rows[0]) && (th = th.cells);
|
th && (th = th.rows[0]) && (th = th.cells);
|
||||||
if (th) i = th.length;
|
if (th) i = th.length;
|
||||||
@@ -136,7 +180,7 @@ function makeSortable(table) {
|
|||||||
while (--i >= 0) (function (i) {
|
while (--i >= 0) (function (i) {
|
||||||
th[i].onclick = function (e) {
|
th[i].onclick = function (e) {
|
||||||
ev(e);
|
ev(e);
|
||||||
sortTable(table, i);
|
sortTable(table, i, cb);
|
||||||
};
|
};
|
||||||
}(i));
|
}(i));
|
||||||
}
|
}
|
||||||
@@ -273,6 +317,7 @@ function unix2iso(ts) {
|
|||||||
|
|
||||||
|
|
||||||
function s2ms(s) {
|
function s2ms(s) {
|
||||||
|
s = Math.floor(s);
|
||||||
var m = Math.floor(s / 60);
|
var m = Math.floor(s / 60);
|
||||||
return m + ":" + ("0" + (s - m * 60)).slice(-2);
|
return m + ":" + ("0" + (s - m * 60)).slice(-2);
|
||||||
}
|
}
|
||||||
@@ -359,8 +404,10 @@ 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) {
|
||||||
o.setAttribute('class', val ? 'on' : '');
|
var fun = val ? 'add' : 'remove';
|
||||||
|
o.classList[fun]('on');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
242
docs/music-analysis.sh
Normal file
242
docs/music-analysis.sh
Normal file
@@ -0,0 +1,242 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
echo please dont actually run this as a scriopt
|
||||||
|
exit 1
|
||||||
|
|
||||||
|
|
||||||
|
# dependency-heavy, not particularly good fit
|
||||||
|
pacman -S llvm10
|
||||||
|
python3 -m pip install --user librosa
|
||||||
|
git clone https://github.com/librosa/librosa.git
|
||||||
|
|
||||||
|
|
||||||
|
# correct bpm for tracks with bad tags
|
||||||
|
br='
|
||||||
|
/Trip Trip Trip\(Hardcore Edit\).mp3/ {v=176}
|
||||||
|
/World!!.BIG_SOS/ {v=175}
|
||||||
|
/\/08\..*\(BIG_SOS Bootleg\)\.mp3/ {v=175}
|
||||||
|
/もってけ!セーラ服.Asterisk DnB/ {v=175}
|
||||||
|
/Rondo\(Asterisk DnB Re.mp3/ {v=175}
|
||||||
|
/Ray Nautica 175 Edit/ {v=175;x="thunk"}
|
||||||
|
/TOKIMEKI Language.Jauz/ {v=174}
|
||||||
|
/YUPPUN Hardcore Remix\).mp3/ {v=174;x="keeps drifting"}
|
||||||
|
/(èâAâï.î╧ûδ|バーチャリアル.狐耶)J-Core Remix\).mp3/ {v=172;x="hard"}
|
||||||
|
/lucky train..Freezer/ {v=170}
|
||||||
|
/Alf zero Bootleg ReMix/ {v=170}
|
||||||
|
/Prisoner of Love.Kacky/ {v=170}
|
||||||
|
/火炎 .Qota/ {v=170}
|
||||||
|
/\(hu-zin Bootleg\)\.mp3/ {v=170}
|
||||||
|
/15. STRAIGHT BET\(Milynn Bootleg\)\.mp3/ {v=170}
|
||||||
|
/\/13.*\(Milynn Bootleg\)\.mp3/ {v=167;x="way hard"}
|
||||||
|
/COLOR PLANET .10SAI . nijikon Remix\)\.mp3/ {v=165}
|
||||||
|
/11\. (朝はご飯派|Æ⌐é═é▓ö╤öh)\.mp3/ {v=162}
|
||||||
|
/09\. Where.s the core/ {v=160}
|
||||||
|
/PLANET\(Koushif Jersey Club Bootleg\)remaster.mp3/ {v=160;x="starts ez turns bs"}
|
||||||
|
/kened Soul - Madeon x Angel Beats!.mp3/ {v=160}
|
||||||
|
/Dear Moments\(Mother Harlot Bootleg\)\.mp3/ {v=150}
|
||||||
|
/POWER.Ringos UKG/ {v=140}
|
||||||
|
/ブルー・フィールド\(Ringos UKG Remix\).mp3/ {v=135}
|
||||||
|
/プラチナジェット.Ringo Remix..mp3/ {v=131.2}
|
||||||
|
/Mirrorball Love \(TKM Bootleg Mix\).mp3/ {v=130}
|
||||||
|
/Photon Melodies \(TKM Bootleg Mix\).mp3/ {v=128}
|
||||||
|
/Trap of Love \(TKM Bootleg Mix\).mp3/ {v=128}
|
||||||
|
/One Step \(TKM Bootleg Mix\)\.mp3/ {v=126}
|
||||||
|
/04 (トリカムイ岩|âgâèâJâÇâCèΓ).mp3/ {v=125}
|
||||||
|
/Get your Wish \(NAWN REMIX\)\.mp3/ {v=95}
|
||||||
|
/Flicker .Nitro Fun/ {v=92}
|
||||||
|
/\/14\..*suicat Remix/ {v=85.5;x="tricky"}
|
||||||
|
/Yanagi Nagi - Harumodoki \(EO Remix\)\.mp3/ {v=150}
|
||||||
|
/Azure - Nicology\.mp3/ {v=128;x="off by 5 how"}
|
||||||
|
'
|
||||||
|
|
||||||
|
|
||||||
|
# afun host, collects/grades the results
|
||||||
|
runfun() { cores=8; touch run; rm -f /dev/shm/mres.*; t00=$(date +%s); tbc() { bc | sed -r 's/(\.[0-9]{2}).*/\1/'; }; for ((core=0; core<$cores; core++)); do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db 'select dur.w, dur.v, bpm.v from mt bpm join mt dur on bpm.w = dur.w where bpm.k = ".bpm" and dur.k = ".dur" order by dur.w' | uniq -w16 | while IFS=\| read w dur bpm; do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db "select rd, fn from up where substr(w,1,16) = '$w'" | sed -r "s/^/$bpm /"; done | grep mir/cr | tr \| / | awk '{v=$1;sub(/[^ ]+ /,"")} '"$br"' {printf "%s %s\n",v,$0}' | while read bpm fn; do [ -e run ] || break; n=$((n+1)); ncore=$((n%cores)); [ $ncore -eq $core ] || continue; t0=$(date +%s.%N); (afun || exit 1; t=$(date +%s.%N); td=$(echo "scale=3; $t - $t0" | tbc); bd=$(echo "scale=3; $bpm / $py" | tbc); printf '%4s sec, %4s orig, %6s py, %4s div, %s\n' $td $bpm $py $bd "$fn") | tee -a /dev/shm/mres.$ncore; rv=${PIPESTATUS[0]}; [ $rv -eq 0 ] || { echo "FAULT($rv): $fn"; }; done & done; wait 2>/dev/null; cat /dev/shm/mres.* | awk 'function prt(c) {printf "\033[3%sm%s\033[0m\n",c,$0} $8!="div,"{next} $5!~/^[0-9\.]+/{next} {meta=$3;det=$5;div=meta/det} div<0.7{det/=2} div>1.3{det*=2} {idet=sprintf("%.0f",det)} {idiff=idet-meta} meta>idet{idiff=meta-idet} idiff==0{n0++;prt(6);next} idiff==1{n1++;prt(3);next} idiff>10{nx++;prt(1);next} {n10++;prt(5)} END {printf "ok: %d 1off: %2s (%3s) 10off: %2s (%3s) fail: %2s\n",n0,n1,n0+n1,n10,n0+n1+n10,nx}'; te=$(date +%s); echo $((te-t00)) sec spent; }
|
||||||
|
|
||||||
|
|
||||||
|
# ok: 8 1off: 62 ( 70) 10off: 86 (156) fail: 25 # 105 sec, librosa @ 8c archvm on 3700x w10
|
||||||
|
# ok: 4 1off: 59 ( 63) 10off: 65 (128) fail: 53 # using original tags (bad)
|
||||||
|
afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -t 60 /dev/shm/$core.wav || return 1; py="$(/home/ed/src/librosa/examples/beat_tracker.py /dev/shm/$core.wav x 2>&1 | awk 'BEGIN {v=1} /^Estimated tempo: /{v=$3} END {print v}')"; } runfun
|
||||||
|
|
||||||
|
|
||||||
|
# ok: 119 1off: 5 (124) 10off: 8 (132) fail: 49 # 51 sec, vamp-example-fixedtempo
|
||||||
|
# ok: 109 1off: 4 (113) 10off: 9 (122) fail: 59 # bad-tags
|
||||||
|
afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 22050 -f f32le /dev/shm/$core.pcm || return 1; py="$(python3 -c 'import vamp; import numpy as np; f = open("/dev/shm/'$core'.pcm", "rb"); d = np.fromfile(f, dtype=np.float32); c = vamp.collect(d, 22050, "vamp-example-plugins:fixedtempo", parameters={"maxdflen":40}); print(c["list"][0]["label"].split(" ")[0])')"; }; runfun
|
||||||
|
|
||||||
|
|
||||||
|
# ok: 102 1off: 61 (163) 10off: 12 (175) fail: 6 # 61 sec, vamp-qm-tempotracker
|
||||||
|
# ok: 80 1off: 48 (128) 10off: 11 (139) fail: 42 # bad-tags
|
||||||
|
afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 22050 -f f32le /dev/shm/$core.pcm || return 1; py="$(python3 -c 'import vamp; import numpy as np; f = open("/dev/shm/'$core'.pcm", "rb"); d = np.fromfile(f, dtype=np.float32); c = vamp.collect(d, 22050, "qm-vamp-plugins:qm-tempotracker", parameters={"inputtempo":150}); v = [float(x["label"].split(" ")[0]) for x in c["list"] if x["label"]]; v = list(sorted(v))[len(v)//4:-len(v)//4]; print(round(sum(v) / len(v), 1))')"; }; runfun
|
||||||
|
|
||||||
|
|
||||||
|
# ok: 133 1off: 32 (165) 10off: 12 (177) fail: 3 # 51 sec, vamp-beatroot
|
||||||
|
# ok: 101 1off: 22 (123) 10off: 16 (139) fail: 39 # bad-tags
|
||||||
|
# note: some tracks fully fail to analyze (unlike the others which always provide a guess)
|
||||||
|
afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 22050 -f f32le /dev/shm/$core.pcm || return 1; py="$(python3 -c 'import vamp; import numpy as np; f = open("/dev/shm/'$core'.pcm", "rb"); d = np.fromfile(f, dtype=np.float32); c = vamp.collect(d, 22050, "beatroot-vamp:beatroot"); cl=c["list"]; print(round(60*((len(cl)-1)/(float(cl[-1]["timestamp"]-cl[1]["timestamp"]))), 2))')"; }; runfun
|
||||||
|
|
||||||
|
|
||||||
|
# ok: 124 1off: 9 (133) 10off: 40 (173) fail: 8 # 231 sec, essentia/full
|
||||||
|
# ok: 109 1off: 8 (117) 10off: 22 (139) fail: 42 # bad-tags
|
||||||
|
afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 /dev/shm/$core.wav || return 1; py="$(python3 -c 'import essentia; import essentia.standard as es; fe, fef = es.MusicExtractor(lowlevelStats=["mean", "stdev"], rhythmStats=["mean", "stdev"], tonalStats=["mean", "stdev"])("/dev/shm/'$core'.wav"); print("{:.2f}".format(fe["rhythm.bpm"]))')"; }; runfun
|
||||||
|
|
||||||
|
|
||||||
|
# ok: 113 1off: 18 (131) 10off: 46 (177) fail: 4 # 134 sec, essentia/re2013
|
||||||
|
# ok: 101 1off: 15 (116) 10off: 26 (142) fail: 39 # bad-tags
|
||||||
|
afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 /dev/shm/$core.wav || return 1; py="$(python3 -c 'from essentia.standard import *; a=MonoLoader(filename="/dev/shm/'$core'.wav")(); bpm,beats,confidence,_,intervals=RhythmExtractor2013(method="multifeature")(a); print("{:.2f}".format(bpm))')"; }; runfun
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
########################################################################
|
||||||
|
##
|
||||||
|
## key detectyion
|
||||||
|
##
|
||||||
|
########################################################################
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# console scriptlet reusing keytabs from browser.js
|
||||||
|
var m=''; for (var a=0; a<24; a++) m += 's/\\|(' + maps["traktor_sharps"][a].trim() + "|" + maps["rekobo_classic"][a].trim() + "|" + maps["traktor_musical"][a].trim() + "|" + maps["traktor_open"][a].trim() + ')$/|' + maps["rekobo_alnum"][a].trim() + '/;'; console.log(m);
|
||||||
|
|
||||||
|
|
||||||
|
# translate to camelot
|
||||||
|
re='s/\|(B|B|B|6d)$/|1B/;s/\|(F#|F#|Gb|7d)$/|2B/;s/\|(C#|Db|Db|8d)$/|3B/;s/\|(G#|Ab|Ab|9d)$/|4B/;s/\|(D#|Eb|Eb|10d)$/|5B/;s/\|(A#|Bb|Bb|11d)$/|6B/;s/\|(F|F|F|12d)$/|7B/;s/\|(C|C|C|1d)$/|8B/;s/\|(G|G|G|2d)$/|9B/;s/\|(D|D|D|3d)$/|10B/;s/\|(A|A|A|4d)$/|11B/;s/\|(E|E|E|5d)$/|12B/;s/\|(G#m|Abm|Abm|6m)$/|1A/;s/\|(D#m|Ebm|Ebm|7m)$/|2A/;s/\|(A#m|Bbm|Bbm|8m)$/|3A/;s/\|(Fm|Fm|Fm|9m)$/|4A/;s/\|(Cm|Cm|Cm|10m)$/|5A/;s/\|(Gm|Gm|Gm|11m)$/|6A/;s/\|(Dm|Dm|Dm|12m)$/|7A/;s/\|(Am|Am|Am|1m)$/|8A/;s/\|(Em|Em|Em|2m)$/|9A/;s/\|(Bm|Bm|Bm|3m)$/|10A/;s/\|(F#m|F#m|Gbm|4m)$/|11A/;s/\|(C#m|Dbm|Dbm|5m)$/|12A/;'
|
||||||
|
|
||||||
|
|
||||||
|
# runner/wrapper
|
||||||
|
runfun() { cores=8; touch run; tbc() { bc | sed -r 's/(\.[0-9]{2}).*/\1/'; }; for ((core=0; core<$cores; core++)); do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db 'select dur.w, dur.v, key.v from mt key join mt dur on key.w = dur.w where key.k = "key" and dur.k = ".dur" order by dur.w' | uniq -w16 | grep -vE '(Off-Key|None)$' | sed -r "s/ //g;$re" | uniq -w16 | while IFS=\| read w dur bpm; do sqlite3 /mnt/Users/ed/Music/.hist/up2k.db "select rd, fn from up where substr(w,1,16) = '$w'" | sed -r "s/^/$bpm /"; done| grep mir/cr | tr \| / | while read key fn; do [ -e run ] || break; n=$((n+1)); ncore=$((n%cores)); [ $ncore -eq $core ] || continue; t0=$(date +%s.%N); (afun || exit 1; t=$(date +%s.%N); td=$(echo "scale=3; $t - $t0" | tbc); [ "$key" = "$py" ] && c=2 || c=5; printf '%4s sec, %4s orig, \033[3%dm%4s py,\033[0m %s\n' $td "$key" $c "$py" "$fn") || break; done & done; time wait 2>/dev/null; }
|
||||||
|
|
||||||
|
|
||||||
|
# ok: 26 1off: 10 2off: 1 fail: 3 # 15 sec, keyfinder
|
||||||
|
afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 -t 60 /dev/shm/$core.wav || break; py="$(python3 -c 'import sys; import keyfinder; print(keyfinder.key(sys.argv[1]).camelot())' "/dev/shm/$core.wav")"; }; runfun
|
||||||
|
|
||||||
|
|
||||||
|
# https://github.com/MTG/essentia/raw/master/src/examples/tutorial/example_key_by_steps_streaming.py
|
||||||
|
# https://essentia.upf.edu/reference/std_Key.html # edma edmm braw bgate
|
||||||
|
sed -ri 's/^(key = Key\().*/\1profileType="bgate")/' example_key_by_steps_streaming.py
|
||||||
|
afun() { ffmpeg -hide_banner -v fatal -nostdin -ss $((dur/3)) -y -i /mnt/Users/ed/Music/"$fn" -ac 1 -ar 44100 -t 60 /dev/shm/$core.wav || break; py="$(python3 example_key_by_steps_streaming.py /dev/shm/$core.{wav,yml} 2>/dev/null | sed -r "s/ major//;s/ minor/m/;s/^/|/;$re;s/.//")"; }; runfun
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
########################################################################
|
||||||
|
##
|
||||||
|
## misc
|
||||||
|
##
|
||||||
|
########################################################################
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
python3 -m pip install --user vamp
|
||||||
|
|
||||||
|
import librosa
|
||||||
|
d, r = librosa.load('/dev/shm/0.wav')
|
||||||
|
d.dtype
|
||||||
|
# dtype('float32')
|
||||||
|
d.shape
|
||||||
|
# (1323000,)
|
||||||
|
d
|
||||||
|
# array([-1.9614939e-08, 1.8037968e-08, -1.4106059e-08, ...,
|
||||||
|
# 1.2024145e-01, 2.7462116e-01, 1.6202132e-01], dtype=float32)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
import vamp
|
||||||
|
c = vamp.collect(d, r, "vamp-example-plugins:fixedtempo")
|
||||||
|
c
|
||||||
|
# {'list': [{'timestamp': 0.005804988, 'duration': 9.999092971, 'label': '110.0 bpm', 'values': array([109.98116], dtype=float32)}]}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
ffmpeg -ss 48 -i /mnt/Users/ed/Music/mir/cr-a/'I Beg You(ths Bootleg).wav' -ac 1 -ar 22050 -f f32le -t 60 /dev/shm/f32.pcm
|
||||||
|
|
||||||
|
import numpy as np
|
||||||
|
f = open('/dev/shm/f32.pcm', 'rb')
|
||||||
|
d = np.fromfile(f, dtype=np.float32)
|
||||||
|
d
|
||||||
|
array([-0.17803933, -0.27206388, -0.41586545, ..., -0.04940119,
|
||||||
|
-0.0267825 , -0.03564296], dtype=float32)
|
||||||
|
|
||||||
|
d = np.reshape(d, [1, -1])
|
||||||
|
d
|
||||||
|
array([[-0.17803933, -0.27206388, -0.41586545, ..., -0.04940119,
|
||||||
|
-0.0267825 , -0.03564296]], dtype=float32)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
import vampyhost
|
||||||
|
print("\n".join(vampyhost.list_plugins()))
|
||||||
|
|
||||||
|
mvamp:marsyas_bextract_centroid
|
||||||
|
mvamp:marsyas_bextract_lpcc
|
||||||
|
mvamp:marsyas_bextract_lsp
|
||||||
|
mvamp:marsyas_bextract_mfcc
|
||||||
|
mvamp:marsyas_bextract_rolloff
|
||||||
|
mvamp:marsyas_bextract_scf
|
||||||
|
mvamp:marsyas_bextract_sfm
|
||||||
|
mvamp:marsyas_bextract_zero_crossings
|
||||||
|
mvamp:marsyas_ibt
|
||||||
|
mvamp:zerocrossing
|
||||||
|
qm-vamp-plugins:qm-adaptivespectrogram
|
||||||
|
qm-vamp-plugins:qm-barbeattracker
|
||||||
|
qm-vamp-plugins:qm-chromagram
|
||||||
|
qm-vamp-plugins:qm-constantq
|
||||||
|
qm-vamp-plugins:qm-dwt
|
||||||
|
qm-vamp-plugins:qm-keydetector
|
||||||
|
qm-vamp-plugins:qm-mfcc
|
||||||
|
qm-vamp-plugins:qm-onsetdetector
|
||||||
|
qm-vamp-plugins:qm-segmenter
|
||||||
|
qm-vamp-plugins:qm-similarity
|
||||||
|
qm-vamp-plugins:qm-tempotracker
|
||||||
|
qm-vamp-plugins:qm-tonalchange
|
||||||
|
qm-vamp-plugins:qm-transcription
|
||||||
|
vamp-aubio:aubiomelenergy
|
||||||
|
vamp-aubio:aubiomfcc
|
||||||
|
vamp-aubio:aubionotes
|
||||||
|
vamp-aubio:aubioonset
|
||||||
|
vamp-aubio:aubiopitch
|
||||||
|
vamp-aubio:aubiosilence
|
||||||
|
vamp-aubio:aubiospecdesc
|
||||||
|
vamp-aubio:aubiotempo
|
||||||
|
vamp-example-plugins:amplitudefollower
|
||||||
|
vamp-example-plugins:fixedtempo
|
||||||
|
vamp-example-plugins:percussiononsets
|
||||||
|
vamp-example-plugins:powerspectrum
|
||||||
|
vamp-example-plugins:spectralcentroid
|
||||||
|
vamp-example-plugins:zerocrossing
|
||||||
|
vamp-rubberband:rubberband
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
plug = vampyhost.load_plugin("vamp-example-plugins:fixedtempo", 22050, 0)
|
||||||
|
plug.info
|
||||||
|
{'apiVersion': 2, 'pluginVersion': 1, 'identifier': 'fixedtempo', 'name': 'Simple Fixed Tempo Estimator', 'description': 'Study a short section of audio and estimate its tempo, assuming the tempo is constant', 'maker': 'Vamp SDK Example Plugins', 'copyright': 'Code copyright 2008 Queen Mary, University of London. Freely redistributable (BSD license)'}
|
||||||
|
plug = vampyhost.load_plugin("qm-vamp-plugins:qm-tempotracker", 22050, 0)
|
||||||
|
from pprint import pprint; pprint(plug.parameters)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
for c in plug.parameters: print("{} \033[36m{} [\033[33m{}\033[36m] = {}\033[0m".format(c["identifier"], c["name"], "\033[36m, \033[33m".join(c["valueNames"]), c["valueNames"][int(c["defaultValue"])])) if "valueNames" in c else print("{} \033[36m{} [\033[33m{}..{}\033[36m] = {}\033[0m".format(c["identifier"], c["name"], c["minValue"], c["maxValue"], c["defaultValue"]))
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
beatroot-vamp:beatroot
|
||||||
|
cl=c["list"]; 60*((len(cl)-1)/(float(cl[-1]["timestamp"]-cl[1]["timestamp"])))
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
ffmpeg -ss 48 -i /mnt/Users/ed/Music/mir/cr-a/'I Beg You(ths Bootleg).wav' -ac 1 -ar 22050 -f f32le -t 60 /dev/shm/f32.pcm
|
||||||
|
# 128 bpm, key 5A Cm
|
||||||
|
|
||||||
|
import vamp
|
||||||
|
import numpy as np
|
||||||
|
f = open('/dev/shm/f32.pcm', 'rb')
|
||||||
|
d = np.fromfile(f, dtype=np.float32)
|
||||||
|
c = vamp.collect(d, 22050, "vamp-example-plugins:fixedtempo", parameters={"maxdflen":40})
|
||||||
|
c["list"][0]["label"]
|
||||||
|
# 127.6 bpm
|
||||||
|
|
||||||
|
c = vamp.collect(d, 22050, "qm-vamp-plugins:qm-tempotracker", parameters={"inputtempo":150})
|
||||||
|
print("\n".join([v["label"] for v in c["list"] if v["label"]]))
|
||||||
|
v = [float(x["label"].split(' ')[0]) for x in c["list"] if x["label"]]
|
||||||
|
v = list(sorted(v))[len(v)//4:-len(v)//4]
|
||||||
|
v = sum(v) / len(v)
|
||||||
|
# 128.1 bpm
|
||||||
|
|
||||||
@@ -67,6 +67,33 @@ wget -S --header='Accept-Encoding: gzip' -U 'MSIE 6.0; SV1' http://127.0.0.1:392
|
|||||||
shab64() { sp=$1; f="$2"; v=0; sz=$(stat -c%s "$f"); while true; do w=$((v+sp*1024*1024)); printf $(tail -c +$((v+1)) "$f" | head -c $((w-v)) | sha512sum | cut -c-64 | sed -r 's/ .*//;s/(..)/\\x\1/g') | base64 -w0 | cut -c-43 | tr '+/' '-_'; v=$w; [ $v -lt $sz ] || break; done; }
|
shab64() { sp=$1; f="$2"; v=0; sz=$(stat -c%s "$f"); while true; do w=$((v+sp*1024*1024)); printf $(tail -c +$((v+1)) "$f" | head -c $((w-v)) | sha512sum | cut -c-64 | sed -r 's/ .*//;s/(..)/\\x\1/g') | base64 -w0 | cut -c-43 | tr '+/' '-_'; v=$w; [ $v -lt $sz ] || break; done; }
|
||||||
|
|
||||||
|
|
||||||
|
##
|
||||||
|
## poll url for performance issues
|
||||||
|
|
||||||
|
command -v gdate && date() { gdate "$@"; }; while true; do t=$(date +%s.%N); (time wget http://127.0.0.1:3923/?ls -qO- | jq -C '.files[]|{sz:.sz,ta:.tags.artist,tb:.tags.".bpm"}|del(.[]|select(.==null))' | awk -F\" '/"/{t[$2]++} END {for (k in t){v=t[k];p=sprintf("%" (v+1) "s",v);gsub(/ /,"#",p);printf "\033[36m%s\033[33m%s ",k,p}}') 2>&1 | awk -v ts=$t 'NR==1{t1=$0} NR==2{sub(/.*0m/,"");sub(/s$/,"");t2=$0;c=2; if(t2>0.3){c=3} if(t2>0.8){c=1} } END{sub(/[0-9]{6}$/,"",ts);printf "%s \033[3%dm%s %s\033[0m\n",ts,c,t2,t1}'; sleep 0.1 || break; done
|
||||||
|
|
||||||
|
|
||||||
|
##
|
||||||
|
## sqlite3 stuff
|
||||||
|
|
||||||
|
# find dupe metadata keys
|
||||||
|
sqlite3 up2k.db 'select mt1.w, mt1.k, mt1.v, mt2.v from mt mt1 inner join mt mt2 on mt1.w = mt2.w where mt1.k = mt2.k and mt1.rowid != mt2.rowid'
|
||||||
|
|
||||||
|
# partial reindex by deleting all tags for a list of files
|
||||||
|
time sqlite3 up2k.db 'select mt1.w from mt mt1 inner join mt mt2 on mt1.w = mt2.w where mt1.k = +mt2.k and mt1.rowid != mt2.rowid' > warks
|
||||||
|
cat warks | while IFS= read -r x; do sqlite3 up2k.db "delete from mt where w = '$x'"; done
|
||||||
|
|
||||||
|
|
||||||
|
##
|
||||||
|
## media
|
||||||
|
|
||||||
|
# split track into test files
|
||||||
|
e=6; s=10; d=~/dev/copyparty/srv/aus; n=1; p=0; e=$((e*60)); rm -rf $d; mkdir $d; while true; do ffmpeg -hide_banner -ss $p -i 'nervous_testpilot - office.mp3' -c copy -t $s $d/$(printf %04d $n).mp3; n=$((n+1)); p=$((p+s)); [ $p -gt $e ] && break; done
|
||||||
|
|
||||||
|
-v srv/aus:aus:r:ce2dsa:ce2ts:cmtp=fgsfds=bin/mtag/sleep.py
|
||||||
|
sqlite3 .hist/up2k.db 'select * from mt where k="fgsfds" or k="t:mtp"' | tee /dev/stderr | wc -l
|
||||||
|
|
||||||
|
|
||||||
##
|
##
|
||||||
## vscode
|
## vscode
|
||||||
|
|
||||||
@@ -96,6 +123,9 @@ for d in /usr /var; do find $d -type f -size +30M 2>/dev/null; done | while IFS=
|
|||||||
brew install python@2
|
brew install python@2
|
||||||
pip install virtualenv
|
pip install virtualenv
|
||||||
|
|
||||||
|
# readme toc
|
||||||
|
cat README.md | awk '!/^#/{next} {lv=length($1);sub(/[^ ]+ /,"");bab=$0;gsub(/ /,"-",bab)} {printf "%" ((lv-1)*4+1) "s [%s](#%s)\n", "*",$0,bab}'
|
||||||
|
|
||||||
|
|
||||||
##
|
##
|
||||||
## http 206
|
## http 206
|
||||||
|
|||||||
@@ -20,6 +20,7 @@ set -e
|
|||||||
# -rwxr-xr-x 0 ed ed 183808 Nov 19 00:43 copyparty-extras/sfx-lite/copyparty-sfx.py
|
# -rwxr-xr-x 0 ed ed 183808 Nov 19 00:43 copyparty-extras/sfx-lite/copyparty-sfx.py
|
||||||
|
|
||||||
|
|
||||||
|
command -v gnutar && tar() { gnutar "$@"; }
|
||||||
command -v gtar && tar() { gtar "$@"; }
|
command -v gtar && tar() { gtar "$@"; }
|
||||||
command -v gsed && sed() { gsed "$@"; }
|
command -v gsed && sed() { gsed "$@"; }
|
||||||
td="$(mktemp -d)"
|
td="$(mktemp -d)"
|
||||||
@@ -29,11 +30,11 @@ pwd
|
|||||||
|
|
||||||
|
|
||||||
dl_text() {
|
dl_text() {
|
||||||
command -v curl && exec curl "$@"
|
command -v curl >/dev/null && exec curl "$@"
|
||||||
exec wget -O- "$@"
|
exec wget -O- "$@"
|
||||||
}
|
}
|
||||||
dl_files() {
|
dl_files() {
|
||||||
command -v curl && exec curl -L --remote-name-all "$@"
|
command -v curl >/dev/null && exec curl -L --remote-name-all "$@"
|
||||||
exec wget "$@"
|
exec wget "$@"
|
||||||
}
|
}
|
||||||
export -f dl_files
|
export -f dl_files
|
||||||
|
|||||||
@@ -29,6 +29,10 @@ gtar=$(command -v gtar || command -v gnutar) || true
|
|||||||
command -v grealpath >/dev/null &&
|
command -v grealpath >/dev/null &&
|
||||||
realpath() { grealpath "$@"; }
|
realpath() { grealpath "$@"; }
|
||||||
}
|
}
|
||||||
|
pybin=$(command -v python3 || command -v python) || {
|
||||||
|
echo need python
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
|
||||||
[ -e copyparty/__main__.py ] || cd ..
|
[ -e copyparty/__main__.py ] || cd ..
|
||||||
[ -e copyparty/__main__.py ] ||
|
[ -e copyparty/__main__.py ] ||
|
||||||
@@ -169,10 +173,11 @@ done
|
|||||||
sed -r '/edit2">edit \(fancy/d' <$f >t && tmv "$f"
|
sed -r '/edit2">edit \(fancy/d' <$f >t && tmv "$f"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[ $repack ] ||
|
||||||
find | grep -E '\.py$' |
|
find | grep -E '\.py$' |
|
||||||
grep -vE '__version__' |
|
grep -vE '__version__' |
|
||||||
tr '\n' '\0' |
|
tr '\n' '\0' |
|
||||||
xargs -0 python ../scripts/uncomment.py
|
xargs -0 $pybin ../scripts/uncomment.py
|
||||||
|
|
||||||
f=dep-j2/jinja2/constants.py
|
f=dep-j2/jinja2/constants.py
|
||||||
awk '/^LOREM_IPSUM_WORDS/{o=1;print "LOREM_IPSUM_WORDS = u\"a\"";next} !o; /"""/{o=0}' <$f >t
|
awk '/^LOREM_IPSUM_WORDS/{o=1;print "LOREM_IPSUM_WORDS = u\"a\"";next} !o; /"""/{o=0}' <$f >t
|
||||||
@@ -180,7 +185,7 @@ tmv "$f"
|
|||||||
|
|
||||||
# up2k goes from 28k to 22k laff
|
# up2k goes from 28k to 22k laff
|
||||||
echo entabbening
|
echo entabbening
|
||||||
find | grep -E '\.(js|css|html|py)$' | while IFS= read -r f; do
|
find | grep -E '\.(js|css|html)$' | while IFS= read -r f; do
|
||||||
unexpand -t 4 --first-only <"$f" >t
|
unexpand -t 4 --first-only <"$f" >t
|
||||||
tmv "$f"
|
tmv "$f"
|
||||||
done
|
done
|
||||||
@@ -206,7 +211,7 @@ echo creating unix sfx
|
|||||||
) >$sfx_out.sh
|
) >$sfx_out.sh
|
||||||
|
|
||||||
echo creating generic sfx
|
echo creating generic sfx
|
||||||
python ../scripts/sfx.py --sfx-make tar.bz2 $ver $ts
|
$pybin ../scripts/sfx.py --sfx-make tar.bz2 $ver $ts
|
||||||
mv sfx.out $sfx_out.py
|
mv sfx.out $sfx_out.py
|
||||||
chmod 755 $sfx_out.*
|
chmod 755 $sfx_out.*
|
||||||
|
|
||||||
@@ -214,5 +219,5 @@ printf "done:\n"
|
|||||||
printf " %s\n" "$(realpath $sfx_out)."{sh,py}
|
printf " %s\n" "$(realpath $sfx_out)."{sh,py}
|
||||||
# rm -rf *
|
# rm -rf *
|
||||||
|
|
||||||
# tar -tvf ../sfx/tar | sed -r 's/(.* ....-..-.. ..:.. )(.*)/\2 `` \1/' | sort | sed -r 's/(.*) `` (.*)/\2 \1/'| less
|
# apk add bash python3 tar xz bzip2
|
||||||
# for n in {1..9}; do tar -tf tar | grep -vE '/$' | sed -r 's/(.*)\.(.*)/\2.\1/' | sort | sed -r 's/([^\.]+)\.(.*)/\2.\1/' | tar -cT- | bzip2 -c$n | wc -c; done
|
# while true; do ./make-sfx.sh; for f in ..//dist/copyparty-sfx.{sh,py}; do mv $f $f.$(wc -c <$f | awk '{print$1}'); done; done
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
# coding: utf-8
|
# coding: latin-1
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import os, sys, time, shutil, signal, tarfile, hashlib, platform, tempfile
|
import os, sys, time, shutil, runpy, tarfile, hashlib, platform, tempfile, traceback
|
||||||
import subprocess as sp
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
run me with any version of python, i will unpack and run copyparty
|
run me with any version of python, i will unpack and run copyparty
|
||||||
@@ -344,20 +343,24 @@ def get_payload():
|
|||||||
break
|
break
|
||||||
|
|
||||||
|
|
||||||
def confirm():
|
def confirm(rv):
|
||||||
msg()
|
msg()
|
||||||
|
msg(traceback.format_exc())
|
||||||
msg("*** hit enter to exit ***")
|
msg("*** hit enter to exit ***")
|
||||||
try:
|
try:
|
||||||
raw_input() if PY2 else input()
|
raw_input() if PY2 else input()
|
||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
sys.exit(rv)
|
||||||
|
|
||||||
|
|
||||||
def run(tmp, j2ver):
|
def run(tmp, j2ver):
|
||||||
global cpp
|
global cpp
|
||||||
|
|
||||||
msg("jinja2:", j2ver or "bundled")
|
msg("jinja2:", j2ver or "bundled")
|
||||||
msg("sfxdir:", tmp)
|
msg("sfxdir:", tmp)
|
||||||
|
msg()
|
||||||
|
|
||||||
# "systemd-tmpfiles-clean.timer"?? HOW do you even come up with this shit
|
# "systemd-tmpfiles-clean.timer"?? HOW do you even come up with this shit
|
||||||
try:
|
try:
|
||||||
@@ -373,30 +376,16 @@ def run(tmp, j2ver):
|
|||||||
if j2ver:
|
if j2ver:
|
||||||
del ld[-1]
|
del ld[-1]
|
||||||
|
|
||||||
cmd = (
|
for x in ld:
|
||||||
"import sys, runpy; "
|
sys.path.insert(0, x)
|
||||||
+ "".join(['sys.path.insert(0, r"' + x + '"); ' for x in ld])
|
|
||||||
+ 'runpy.run_module("copyparty", run_name="__main__")'
|
|
||||||
)
|
|
||||||
cmd = [sys.executable, "-c", cmd] + list(sys.argv[1:])
|
|
||||||
|
|
||||||
cmd = [str(x) for x in cmd]
|
|
||||||
msg("\n", cmd, "\n")
|
|
||||||
cpp = sp.Popen(cmd)
|
|
||||||
try:
|
try:
|
||||||
cpp.wait()
|
runpy.run_module(str("copyparty"), run_name=str("__main__"))
|
||||||
|
except SystemExit as ex:
|
||||||
|
if ex.code:
|
||||||
|
confirm(ex.code)
|
||||||
except:
|
except:
|
||||||
cpp.wait()
|
confirm(1)
|
||||||
|
|
||||||
if cpp.returncode != 0:
|
|
||||||
confirm()
|
|
||||||
|
|
||||||
sys.exit(cpp.returncode)
|
|
||||||
|
|
||||||
|
|
||||||
def bye(sig, frame):
|
|
||||||
if cpp is not None:
|
|
||||||
cpp.terminate()
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
@@ -430,8 +419,6 @@ def main():
|
|||||||
|
|
||||||
# skip 0
|
# skip 0
|
||||||
|
|
||||||
signal.signal(signal.SIGTERM, bye)
|
|
||||||
|
|
||||||
tmp = unpack()
|
tmp = unpack()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@@ -439,7 +426,7 @@ def main():
|
|||||||
except:
|
except:
|
||||||
j2ver = None
|
j2ver = None
|
||||||
|
|
||||||
return run(tmp, j2ver)
|
run(tmp, j2ver)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|||||||
Reference in New Issue
Block a user