From 3cde1f3be205f4a6fe333191d19910678c78f866 Mon Sep 17 00:00:00 2001 From: ed Date: Tue, 29 Jul 2025 17:13:34 +0000 Subject: [PATCH 001/174] docker-compose: PYTHONUNBUFFERED=1 almost zero performance impact with podman in kitty --- docs/examples/docker/basic-docker-compose/docker-compose.yml | 5 ++++- docs/examples/docker/idp-authelia-traefik/docker-compose.yml | 3 +++ .../examples/docker/idp-authentik-traefik/docker-compose.yml | 3 +++ 3 files changed, 10 insertions(+), 1 deletion(-) diff --git a/docs/examples/docker/basic-docker-compose/docker-compose.yml b/docs/examples/docker/basic-docker-compose/docker-compose.yml index cfac59b6..dd3c1f9a 100644 --- a/docs/examples/docker/basic-docker-compose/docker-compose.yml +++ b/docs/examples/docker/basic-docker-compose/docker-compose.yml @@ -10,9 +10,12 @@ services: - ./:/cfg:z - /path/to/your/fileshare/top/folder:/w:z - # enabling mimalloc by replacing "NOPE" with "2" will make some stuff twice as fast, but everything will use twice as much ram: environment: LD_PRELOAD: /usr/lib/libmimalloc-secure.so.NOPE + # enable mimalloc by replacing "NOPE" with "2" for a nice speed-boost (will use twice as much ram) + + PYTHONUNBUFFERED: 1 + # ensures log-messages are not delayed (but can reduce speed a tiny bit) stop_grace_period: 15s # thumbnailer is allowed to continue finishing up for 10s after the shutdown signal healthcheck: diff --git a/docs/examples/docker/idp-authelia-traefik/docker-compose.yml b/docs/examples/docker/idp-authelia-traefik/docker-compose.yml index 5fe25a15..9ebd73ba 100644 --- a/docs/examples/docker/idp-authelia-traefik/docker-compose.yml +++ b/docs/examples/docker/idp-authelia-traefik/docker-compose.yml @@ -27,6 +27,9 @@ services: LD_PRELOAD: /usr/lib/libmimalloc-secure.so.NOPE # enable mimalloc by replacing "NOPE" with "2" for a nice speed-boost (will use twice as much ram) + PYTHONUNBUFFERED: 1 + # ensures log-messages are not delayed (but can reduce speed a tiny bit) + authelia: image: authelia/authelia:v4.38.0-beta3 # the config files in the authelia folder use the new syntax container_name: idp_authelia diff --git a/docs/examples/docker/idp-authentik-traefik/docker-compose.yml b/docs/examples/docker/idp-authentik-traefik/docker-compose.yml index ee10f0f9..7ddf1a9d 100644 --- a/docs/examples/docker/idp-authentik-traefik/docker-compose.yml +++ b/docs/examples/docker/idp-authentik-traefik/docker-compose.yml @@ -27,6 +27,9 @@ services: LD_PRELOAD: /usr/lib/libmimalloc-secure.so.NOPE # enable mimalloc by replacing "NOPE" with "2" for a nice speed-boost (will use twice as much ram) + PYTHONUNBUFFERED: 1 + # ensures log-messages are not delayed (but can reduce speed a tiny bit) + traefik: image: traefik:v2.11 container_name: traefik From fbf17be203c9d260996de7d6bd3d196554e89ab7 Mon Sep 17 00:00:00 2001 From: ed Date: Tue, 29 Jul 2025 18:14:51 +0000 Subject: [PATCH 002/174] apply unlist to navpane too --- copyparty/web/browser.js | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index ce204757..f1368912 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -7946,6 +7946,17 @@ var treectl = (function () { return toast.err(30, "bad ?tree reply;\nexpected json, got this:\n\n" + esc(this.responseText + '')); } r.rendertree(res, this.ts, this.top, this.dst, this.rst); + + if (r.lsc && r.lsc.unlist) + r.prunetree(r.lsc); + }; + + r.prunetree = function (res) { + var ptn = new RegExp(res.unlist); + var els = QSA('#treeul li>a+a'); + for (var a = els.length - 1; a >= 0; a--) + if (ptn.exec(els[a].textContent) && !els[a].className) + els[a].closest('ul').removeChild(els[a].closest('li')); }; r.rendertree = function (res, ts, top0, dst, rst) { @@ -8233,6 +8244,8 @@ var treectl = (function () { } r.rendertree({ "a": dirs }, this.ts, ".", get_evpath() + (dk ? '?k=' + dk : '')); + if (res.unlist) + r.prunetree(res); } r.gentab(this.top, res); @@ -8314,7 +8327,7 @@ var treectl = (function () { if (res.unlist) { var ptn = new RegExp(res.unlist); for (var a = nodes.length - 1; a >= 0; a--) - if (ptn.exec(nodes[a].href.split('?')[0])) + if (ptn.exec(uricom_dec(nodes[a].href.split('?')[0]))) nodes.splice(a, 1); } nodes = sortfiles(nodes); From 5c6341e99fb11b7bf272beed5d2a56ce0639be65 Mon Sep 17 00:00:00 2001 From: ed Date: Tue, 29 Jul 2025 20:03:42 +0000 Subject: [PATCH 003/174] disk-info: both free+total on windows too (#272) --- copyparty/httpcli.py | 8 ++++---- copyparty/util.py | 13 +++++++++---- 2 files changed, 13 insertions(+), 8 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 2976a7c1..9052acc0 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -6157,13 +6157,13 @@ class HttpCli(object): self.log("#wow #whoa") if not self.args.nid: - free, total, _ = get_df(abspath, False) - if total is not None: + free, total, zs = get_df(abspath, False) + if total: h1 = humansize(free or 0) h2 = humansize(total) srv_info.append("{} free of {}".format(h1, h2)) - elif free is not None: - srv_info.append(humansize(free, True) + " free") + elif zs: + self.log("diskfree(%r): %s" % (abspath, zs), 3) srv_infot = " // ".join(srv_info) diff --git a/copyparty/util.py b/copyparty/util.py index 0b2dbcc4..86cf046b 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -2662,7 +2662,7 @@ def wunlink(log: "NamedLogger", abspath: str, flags: dict[str, Any]) -> bool: return _fs_mvrm(log, abspath, "", False, flags) -def get_df(abspath: str, prune: bool) -> tuple[Optional[int], Optional[int], str]: +def get_df(abspath: str, prune: bool) -> tuple[int, int, str]: try: ap = fsenc(abspath) while prune and not os.path.isdir(ap) and BOS_SEP in ap: @@ -2673,17 +2673,22 @@ def get_df(abspath: str, prune: bool) -> tuple[Optional[int], Optional[int], str assert ctypes # type: ignore # !rm abspath = fsdec(ap) bfree = ctypes.c_ulonglong(0) + btotal = ctypes.c_ulonglong(0) + bavail = ctypes.c_ulonglong(0) ctypes.windll.kernel32.GetDiskFreeSpaceExW( # type: ignore - ctypes.c_wchar_p(abspath), None, None, ctypes.pointer(bfree) + ctypes.c_wchar_p(abspath), + ctypes.pointer(bavail), + ctypes.pointer(btotal), + ctypes.pointer(bfree), ) - return (bfree.value, None, "") + return (bavail.value, btotal.value, "") else: sv = os.statvfs(ap) free = sv.f_frsize * sv.f_bfree total = sv.f_frsize * sv.f_blocks return (free, total, "") except Exception as ex: - return (None, None, repr(ex)) + return (0, 0, repr(ex)) if not ANYWIN and not MACOS: From 4988a55ea58f1db7eff0dfbf2d5e53d783eb08f2 Mon Sep 17 00:00:00 2001 From: ed Date: Tue, 29 Jul 2025 20:07:11 +0000 Subject: [PATCH 004/174] webdav: send diskfree; closes #272 --- copyparty/httpcli.py | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 9052acc0..8bb12870 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -1575,6 +1575,18 @@ class HttpCli(object): self.log("inaccessible: %r" % ("/" + self.vpath,)) raise Pebkac(401, "authenticate") + if "quota-available-bytes" in props and not self.args.nid: + bfree, btot, _ = get_df(vn.realpath, False) + if btot: + df = { + "quota-available-bytes": str(bfree), + "quota-used-bytes": str(btot - bfree), + } + else: + df = {} + else: + df = {} + fgen = itertools.chain([topdir], fgen) vtop = vjoin(self.args.R, vjoin(vn.vpath, rem)) @@ -1617,6 +1629,9 @@ class HttpCli(object): ap = os.path.join(tap, x["vp"]) pvs["getcontenttype"] = html_escape(guess_mime(rp, ap)) pvs["getcontentlength"] = str(st.st_size) + elif df: + pvs.update(df) + df = {} for k, v in pvs.items(): if k not in props: From c3cc2ddeaea4d2c762bce0335b6637ce341861cd Mon Sep 17 00:00:00 2001 From: Jo <141064017+Arklaum@users.noreply.github.com> Date: Tue, 29 Jul 2025 21:24:17 +0100 Subject: [PATCH 005/174] diskfree without root-reserved space (#285) Signed-off-by: Jo <141064017+Arklaum@users.noreply.github.com> --- copyparty/util.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/util.py b/copyparty/util.py index 86cf046b..14768e6e 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -2684,7 +2684,7 @@ def get_df(abspath: str, prune: bool) -> tuple[int, int, str]: return (bavail.value, btotal.value, "") else: sv = os.statvfs(ap) - free = sv.f_frsize * sv.f_bfree + free = sv.f_frsize * sv.f_bavail total = sv.f_frsize * sv.f_blocks return (free, total, "") except Exception as ex: From 053de61907fa3d83a9c6b0dd699c8e2b973dd3ae Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 17:26:58 +0000 Subject: [PATCH 006/174] explain what Leeloo Dallas is doing here (closes #316) also makes rejections from IdP auths less confusing; it was handled by the config-parser throwing "invalid config" --- copyparty/authsrv.py | 19 +++++++++++++++++++ copyparty/httpcli.py | 5 ++++- 2 files changed, 23 insertions(+), 1 deletion(-) diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index bdc65640..03dea807 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -70,6 +70,25 @@ if PY2: LEELOO_DALLAS = "leeloo_dallas" +## +## you might be curious what Leeloo Dallas is doing here, so let me explain: +## +## certain daemonic tasks, namely: +## * deletion of expired files, running on a timer +## * deletion of sidecar files, initiated by plugins +## need to skip the usual permission-checks to do their thing, +## so we let Leeloo handle these +## +## and also, the smb-server has really shitty support for user-accounts +## so one popular way to avoid issues is by running copyparty without users; +## this makes all smb-clients identify as LD to gain unrestricted access +## +## Leeloo, being a fictional character from The Fifth Element, +## obviously does not exist and will never be able to access any copyparty +## instances from the outside (the username is rejected at every entrypoint) +## +## thanks for coming to my ted talk + SEE_LOG = "see log for details" SEESLOG = " (see serverlog for details)" diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 8bb12870..f90f4d93 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -33,7 +33,7 @@ except: from .__init__ import ANYWIN, PY2, RES, TYPE_CHECKING, EnvParams, unicode from .__version__ import S_VERSION -from .authsrv import VFS # typechk +from .authsrv import LEELOO_DALLAS, VFS # typechk from .bos import bos from .star import StreamTar from .stolen.qrcodegen import QrCode, qr2svg @@ -622,6 +622,9 @@ class HttpCli(object): ) or self.args.idp_h_key in self.headers if trusted_key and trusted_xff: + if idp_usr.lower() == LEELOO_DALLAS: + self.loud_reply("send her back", status=403) + return False self.asrv.idp_checkin(self.conn.hsrv.broker, idp_usr, idp_grp) else: if not trusted_key: From a9d07c63ed82b11c172009a767c6dbb4f1f78d03 Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 18:02:11 +0000 Subject: [PATCH 007/174] disable libmagic on windows; probably closes #276 --- README.md | 1 + copyparty/util.py | 4 +++- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 05661ed4..8dad60c6 100644 --- a/README.md +++ b/README.md @@ -2226,6 +2226,7 @@ force-enable features with known issues on your OS/env by setting any of the fo | env-var | what it does | | ------------------------ | ------------ | | `PRTY_FORCE_MP` | force-enable multiprocessing (real multithreading) on MacOS and other broken platforms | +| `PRTY_FORCE_MAGIC` | use [magic](https://pypi.org/project/python-magic/) on Windows (you will segfault) | # packages diff --git a/copyparty/util.py b/copyparty/util.py index 14768e6e..cf07ed5a 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -155,7 +155,9 @@ except: HAVE_PSUTIL = False try: - if os.environ.get("PRTY_NO_MAGIC"): + if os.environ.get("PRTY_NO_MAGIC") or ( + ANYWIN and not os.environ.get("PRTY_FORCE_MAGIC") + ): raise Exception() import magic From f19599886532acb8420ab52d3ed5ba5e1649c0da Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 19:35:00 +0000 Subject: [PATCH 008/174] per-volume uid/gid; closes #265 --- README.md | 21 +++++++++++++++++++ copyparty/__main__.py | 2 ++ copyparty/authsrv.py | 17 ++++++++++++++- copyparty/bos/bos.py | 17 ++++++++++++--- copyparty/cfg.py | 4 ++++ copyparty/ftpd.py | 8 ++++---- copyparty/httpcli.py | 48 +++++++++++++++++++++---------------------- copyparty/smbd.py | 2 +- copyparty/tftpd.py | 9 +++++--- copyparty/th_srv.py | 4 ++-- copyparty/up2k.py | 17 ++++++++------- copyparty/util.py | 19 ++++++++++++----- tests/util.py | 3 +++ 13 files changed, 120 insertions(+), 51 deletions(-) diff --git a/README.md b/README.md index 8dad60c6..83c47875 100644 --- a/README.md +++ b/README.md @@ -80,6 +80,7 @@ made in Norway 🇳🇴 * [periodic rescan](#periodic-rescan) - filesystem monitoring * [upload rules](#upload-rules) - set upload rules using volflags * [compress uploads](#compress-uploads) - files can be autocompressed on upload + * [chmod and chown](#chmod-and-chown) - per-volume filesystem-permissions and ownership * [other flags](#other-flags) * [database location](#database-location) - in-volume (`.hist/up2k.db`, default) or somewhere else * [metadata from audio files](#metadata-from-audio-files) - set `-e2t` to index tags on upload @@ -1649,6 +1650,26 @@ some examples, allows (but does not force) gz compression if client uploads to `/inc?pk` or `/inc?gz` or `/inc?gz=4` +## chmod and chown + +per-volume filesystem-permissions and ownership + +by default: +* all folders are chmod 755 +* files are usually chmod 644 (umask-defined) +* user/group is whatever copyparty is running as + +this can be configured per-volume: +* volflag `chmod_f` sets file permissions; default=`644` (usually) +* volflag `chmod_d` sets directory permissions; default=`755` +* volflag `uid` sets the owner user-id +* volflag `gid` sets the owner group-id + +notes: +* `gid` can only be set to one of the groups which the copyparty process is a member of +* `uid` can only be set if copyparty is running as root (i appreciate your faith) + + ## other flags * `:c,magic` enables filetype detection for nameless uploads, same as `--magic` diff --git a/copyparty/__main__.py b/copyparty/__main__.py index f6c20b8c..461a57e7 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1053,6 +1053,8 @@ def add_upload(ap): ap2.add_argument("--use-fpool", action="store_true", help="force file-handle pooling, even when it might be dangerous (multiprocessing, filesystems lacking sparse-files support, ...)") ap2.add_argument("--chmod-f", metavar="UGO", type=u, default="", help="unix file permissions to use when creating files; default is probably 644 (OS-decided), see --help-chmod. Examples: [\033[32m644\033[0m] = owner-RW + all-R, [\033[32m755\033[0m] = owner-RWX + all-RX, [\033[32m777\033[0m] = full-yolo (volflag=chmod_f)") ap2.add_argument("--chmod-d", metavar="UGO", type=u, default="755", help="unix file permissions to use when creating directories; see --help-chmod. Examples: [\033[32m755\033[0m] = owner-RW + all-R, [\033[32m777\033[0m] = full-yolo (volflag=chmod_d)") + ap2.add_argument("--uid", metavar="N", type=int, default=-1, help="unix user-id to chown new files/folders to; default = -1 = do-not-change (volflag=uid)") + ap2.add_argument("--gid", metavar="N", type=int, default=-1, help="unix group-id to chown new files/folders to; default = -1 = do-not-change (volflag=gid)") ap2.add_argument("--dedup", action="store_true", help="enable symlink-based upload deduplication (volflag=dedup)") ap2.add_argument("--safe-dedup", metavar="N", type=int, default=50, help="how careful to be when deduplicating files; [\033[32m1\033[0m] = just verify the filesize, [\033[32m50\033[0m] = verify file contents have not been altered (volflag=safededup)") ap2.add_argument("--hardlink", action="store_true", help="enable hardlink-based dedup; will fallback on symlinks when that is impossible (across filesystems) (volflag=hardlink)") diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 03dea807..89cab889 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -140,6 +140,8 @@ class Lim(object): self.reg: Optional[dict[str, dict[str, Any]]] = None # up2k registry self.chmod_d = 0o755 + self.uid = self.gid = -1 + self.chown = False self.nups: dict[str, list[float]] = {} # num tracker self.bups: dict[str, list[tuple[float, int]]] = {} # byte tracker list @@ -302,6 +304,8 @@ class Lim(object): # no branches yet; make one sub = os.path.join(path, "0") bos.mkdir(sub, self.chmod_d) + if self.chown: + os.chown(sub, self.uid, self.gid) else: # try newest branch only sub = os.path.join(path, str(dirs[-1])) @@ -317,6 +321,8 @@ class Lim(object): # make a branch sub = os.path.join(path, str(dirs[-1] + 1)) bos.mkdir(sub, self.chmod_d) + if self.chown: + os.chown(sub, self.uid, self.gid) ret = self.dive(sub, lvs - 1) if ret is None: raise Pebkac(500, "rotation bug") @@ -2181,7 +2187,7 @@ class AuthSrv(object): if vf not in vol.flags: vol.flags[vf] = getattr(self.args, ga) - zs = "forget_ip nrand tail_who u2abort u2ow ups_who zip_who" + zs = "forget_ip gid nrand tail_who u2abort u2ow uid ups_who zip_who" for k in zs.split(): if k in vol.flags: vol.flags[k] = int(vol.flags[k]) @@ -2218,8 +2224,17 @@ class AuthSrv(object): if (is_d and zi != 0o755) or not is_d: free_umask = True + vol.flags.pop("chown", None) + if vol.flags["uid"] != -1 or vol.flags["gid"] != -1: + vol.flags["chown"] = True + vol.flags.pop("fperms", None) + if "chown" in vol.flags or vol.flags.get("chmod_f"): + vol.flags["fperms"] = True if vol.lim: vol.lim.chmod_d = vol.flags["chmod_d"] + vol.lim.chown = "chown" in vol.flags + vol.lim.uid = vol.flags["uid"] + vol.lim.gid = vol.flags["gid"] if vol.flags.get("og"): self.args.uqe = True diff --git a/copyparty/bos/bos.py b/copyparty/bos/bos.py index 3d98d0df..6c876e04 100644 --- a/copyparty/bos/bos.py +++ b/copyparty/bos/bos.py @@ -9,8 +9,11 @@ from . import path as path if True: # pylint: disable=using-constant-test from typing import Any, Optional -_ = (path,) -__all__ = ["path"] +MKD_755 = {"chmod_d": 0o755} +MKD_700 = {"chmod_d": 0o700} + +_ = (path, MKD_755, MKD_700) +__all__ = ["path", "MKD_755", "MKD_700"] # grep -hRiE '(^|[^a-zA-Z_\.-])os\.' . | gsed -r 's/ /\n/g;s/\(/(\n/g' | grep -hRiE '(^|[^a-zA-Z_\.-])os\.' | sort | uniq -c # printf 'os\.(%s)' "$(grep ^def bos/__init__.py | gsed -r 's/^def //;s/\(.*//' | tr '\n' '|' | gsed -r 's/.$//')" @@ -20,11 +23,15 @@ def chmod(p: str, mode: int) -> None: return os.chmod(fsenc(p), mode) +def chown(p: str, uid: int, gid: int) -> None: + return os.chown(fsenc(p), uid, gid) + + def listdir(p: str = ".") -> list[str]: return [fsdec(x) for x in os.listdir(fsenc(p))] -def makedirs(name: str, mode: int = 0o755, exist_ok: bool = True) -> bool: +def makedirs(name: str, vf: dict[str, Any] = MKD_755, exist_ok: bool = True) -> bool: # os.makedirs does 777 for all but leaf; this does mode on all todo = [] bname = fsenc(name) @@ -37,9 +44,13 @@ def makedirs(name: str, mode: int = 0o755, exist_ok: bool = True) -> bool: if not exist_ok: os.mkdir(bname) # to throw return False + mode = vf["chmod_d"] + chown = "chown" in vf for zb in todo[::-1]: try: os.mkdir(zb, mode) + if chown: + os.chown(zb, vf["uid"], vf["gid"]) except: if os.path.isdir(zb): continue diff --git a/copyparty/cfg.py b/copyparty/cfg.py index cee8214b..2f75ab28 100644 --- a/copyparty/cfg.py +++ b/copyparty/cfg.py @@ -114,6 +114,8 @@ def vf_vmap() -> dict[str, str]: "unlist", "u2abort", "u2ts", + "uid", + "gid", "ups_who", "zip_who", "zipmaxn", @@ -175,6 +177,8 @@ flagcats = { "nodupe": "rejects existing files (instead of linking/cloning them)", "chmod_d=755": "unix-permission for new dirs/folders", "chmod_f=644": "unix-permission for new files", + "uid=573": "change owner of new files/folders to unix-user 573", + "gid=999": "change owner of new files/folders to unix-group 999", "sparse": "force use of sparse files, mainly for s3-backed storage", "nosparse": "deny use of sparse files, mainly for slow storage", "daw": "enable full WebDAV write support (dangerous);\nPUT-operations will now \033[1;31mOVERWRITE\033[0;35m existing files", diff --git a/copyparty/ftpd.py b/copyparty/ftpd.py index c464450e..2f45c3f4 100644 --- a/copyparty/ftpd.py +++ b/copyparty/ftpd.py @@ -31,6 +31,7 @@ from .util import ( relchk, runhook, sanitize_fn, + set_fperms, vjoin, wunlink, ) @@ -262,8 +263,8 @@ class FtpFs(AbstractedFS): wunlink(self.log, ap, VF_CAREFUL) ret = open(fsenc(ap), mode, self.args.iobuf) - if w and "chmod_f" in vfs.flags: - os.fchmod(ret.fileno(), vfs.flags["chmod_f"]) + if w and "fperms" in vfs.flags: + set_fperms(ret, vfs.flags) return ret @@ -297,8 +298,7 @@ class FtpFs(AbstractedFS): def mkdir(self, path: str) -> None: ap, vfs, _ = self.rv2a(path, w=True) - chmod = vfs.flags["chmod_d"] - bos.makedirs(ap, chmod) # filezilla expects this + bos.makedirs(ap, vf=vfs.flags) # filezilla expects this def listdir(self, path: str) -> list[str]: vpath = join(self.cwd, path) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index f90f4d93..e4bfe45a 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -103,6 +103,7 @@ from .util import ( sanitize_vpath, sendfile_kern, sendfile_py, + set_fperms, stat_resource, ub64dec, ub64enc, @@ -2086,7 +2087,7 @@ class HttpCli(object): fdir, fn = os.path.split(fdir) rem, _ = vsplit(rem) - bos.makedirs(fdir, vfs.flags["chmod_d"]) + bos.makedirs(fdir, vf=vfs.flags) open_ka: dict[str, Any] = {"fun": open} open_a = ["wb", self.args.iobuf] @@ -2144,9 +2145,7 @@ class HttpCli(object): if nameless: fn = vfs.flags["put_name2"].format(now=time.time(), cip=self.dip()) - params = {"suffix": suffix, "fdir": fdir} - if "chmod_f" in vfs.flags: - params["chmod"] = vfs.flags["chmod_f"] + params = {"suffix": suffix, "fdir": fdir, "vf": vfs.flags} if self.args.nw: params = {} fn = os.devnull @@ -2195,7 +2194,7 @@ class HttpCli(object): if self.args.nw: fn = os.devnull else: - bos.makedirs(fdir, vfs.flags["chmod_d"]) + bos.makedirs(fdir, vf=vfs.flags) path = os.path.join(fdir, fn) if not nameless: self.vpath = vjoin(self.vpath, fn) @@ -2327,7 +2326,7 @@ class HttpCli(object): if self.args.hook_v: log_reloc(self.log, hr["reloc"], x, path, vp, fn, vfs, rem) fdir, self.vpath, fn, (vfs, rem) = x - bos.makedirs(fdir, vfs.flags["chmod_d"]) + bos.makedirs(fdir, vf=vfs.flags) path2 = os.path.join(fdir, fn) atomic_move(self.log, path, path2, vfs.flags) path = path2 @@ -2613,7 +2612,7 @@ class HttpCli(object): dst = vfs.canonical(rem) try: if not bos.path.isdir(dst): - bos.makedirs(dst, vfs.flags["chmod_d"]) + bos.makedirs(dst, vf=vfs.flags) except OSError as ex: self.log("makedirs failed %r" % (dst,)) if not bos.path.isdir(dst): @@ -3060,7 +3059,7 @@ class HttpCli(object): raise Pebkac(405, 'folder "/%s" already exists' % (vpath,)) try: - bos.makedirs(fn, vfs.flags["chmod_d"]) + bos.makedirs(fn, vf=vfs.flags) except OSError as ex: if ex.errno == errno.EACCES: raise Pebkac(500, "the server OS denied write-access") @@ -3102,8 +3101,8 @@ class HttpCli(object): with open(fsenc(fn), "wb") as f: f.write(b"`GRUNNUR`\n") - if "chmod_f" in vfs.flags: - os.fchmod(f.fileno(), vfs.flags["chmod_f"]) + if "fperms" in vfs.flags: + set_fperms(f, vfs.flags) vpath = "{}/{}".format(self.vpath, sanitized).lstrip("/") self.redirect(vpath, "?edit") @@ -3177,7 +3176,7 @@ class HttpCli(object): ) upload_vpath = "{}/{}".format(vfs.vpath, rem).strip("/") if not nullwrite: - bos.makedirs(fdir_base, vfs.flags["chmod_d"]) + bos.makedirs(fdir_base, vf=vfs.flags) rnd, lifetime, xbu, xau = self.upload_flags(vfs) zs = self.uparam.get("want") or self.headers.get("accept") or "" @@ -3210,7 +3209,7 @@ class HttpCli(object): if rnd: fname = rand_name(fdir, fname, rnd) - open_args = {"fdir": fdir, "suffix": suffix} + open_args = {"fdir": fdir, "suffix": suffix, "vf": vfs.flags} if "replace" in self.uparam: if not self.can_delete: @@ -3272,11 +3271,8 @@ class HttpCli(object): else: open_args["fdir"] = fdir - if "chmod_f" in vfs.flags: - open_args["chmod"] = vfs.flags["chmod_f"] - if p_file and not nullwrite: - bos.makedirs(fdir, vfs.flags["chmod_d"]) + bos.makedirs(fdir, vf=vfs.flags) # reserve destination filename f, fname = ren_open(fname, "wb", fdir=fdir, suffix=suffix) @@ -3380,7 +3376,7 @@ class HttpCli(object): if nullwrite: fdir = ap2 = "" else: - bos.makedirs(fdir, vfs.flags["chmod_d"]) + bos.makedirs(fdir, vf=vfs.flags) atomic_move(self.log, abspath, ap2, vfs.flags) abspath = ap2 sz = bos.path.getsize(abspath) @@ -3501,8 +3497,8 @@ class HttpCli(object): ft = "{}:{}".format(self.ip, self.addr[1]) ft = "{}\n{}\n{}\n".format(ft, msg.rstrip(), errmsg) f.write(ft.encode("utf-8")) - if "chmod_f" in vfs.flags: - os.fchmod(f.fileno(), vfs.flags["chmod_f"]) + if "fperms" in vfs.flags: + set_fperms(f, vfs.flags) except Exception as ex: suf = "\nfailed to write the upload report: {}".format(ex) @@ -3553,7 +3549,7 @@ class HttpCli(object): lim = vfs.get_dbv(rem)[0].lim if lim: fp, rp = lim.all(self.ip, rp, clen, vfs.realpath, fp, self.conn.hsrv.broker) - bos.makedirs(fp, vfs.flags["chmod_d"]) + bos.makedirs(fp, vf=vfs.flags) fp = os.path.join(fp, fn) rem = "{}/{}".format(rp, fn).strip("/") @@ -3621,15 +3617,17 @@ class HttpCli(object): zs = ub64enc(zb).decode("ascii")[:24].lower() dp = "%s/md/%s/%s/%s" % (dbv.histpath, zs[:2], zs[2:4], zs) self.log("moving old version to %s/%s" % (dp, mfile2)) - if bos.makedirs(dp, vfs.flags["chmod_d"]): + if bos.makedirs(dp, vf=vfs.flags): with open(os.path.join(dp, "dir.txt"), "wb") as f: f.write(afsenc(vrd)) - if "chmod_f" in vfs.flags: - os.fchmod(f.fileno(), vfs.flags["chmod_f"]) + if "fperms" in vfs.flags: + set_fperms(f, vfs.flags) elif hist_cfg == "s": dp = os.path.join(mdir, ".hist") try: bos.mkdir(dp, vfs.flags["chmod_d"]) + if "chown" in vfs.flags: + bos.chown(dp, vfs.flags["uid"], vfs.flags["gid"]) hidedir(dp) except: pass @@ -3668,8 +3666,8 @@ class HttpCli(object): wunlink(self.log, fp, vfs.flags) with open(fsenc(fp), "wb", self.args.iobuf) as f: - if "chmod_f" in vfs.flags: - os.fchmod(f.fileno(), vfs.flags["chmod_f"]) + if "fperms" in vfs.flags: + set_fperms(f, vfs.flags) sz, sha512, _ = hashcopy(p_data, f, None, 0, self.args.s_wr_slp) if lim: diff --git a/copyparty/smbd.py b/copyparty/smbd.py index d5098de5..2b9b3d77 100644 --- a/copyparty/smbd.py +++ b/copyparty/smbd.py @@ -320,7 +320,7 @@ class SMB(object): self.hub.up2k.handle_mv(uname, "1.7.6.2", vp1, vp2) try: - bos.makedirs(ap2, vfs2.flags["chmod_d"]) + bos.makedirs(ap2, vf=vfs2.flags) except: pass diff --git a/copyparty/tftpd.py b/copyparty/tftpd.py index 82ef3726..6f5726b3 100644 --- a/copyparty/tftpd.py +++ b/copyparty/tftpd.py @@ -45,6 +45,7 @@ from .util import ( exclude_dotfiles, min_ex, runhook, + set_fperms, undot, vjoin, vsplit, @@ -388,8 +389,8 @@ class Tftpd(object): a = (self.args.iobuf,) ret = open(ap, mode, *a, **ka) - if wr and "chmod_f" in vfs.flags: - os.fchmod(ret.fileno(), vfs.flags["chmod_f"]) + if wr and "fperms" in vfs.flags: + set_fperms(ret, vfs.flags) return ret @@ -398,7 +399,9 @@ class Tftpd(object): if "*" not in vfs.axs.uwrite: yeet("blocked mkdir; folder not world-writable: /%s" % (vpath,)) - return bos.mkdir(ap, vfs.flags["chmod_d"]) + bos.mkdir(ap, vfs.flags["chmod_d"]) + if "chown" in vfs.flags: + bos.chown(ap, vfs.flags["uid"], vfs.flags["gid"]) def _unlink(self, vpath: str) -> None: # return bos.unlink(self._v2a("stat", vpath, *a)[1]) diff --git a/copyparty/th_srv.py b/copyparty/th_srv.py index e00c04e9..67413c10 100644 --- a/copyparty/th_srv.py +++ b/copyparty/th_srv.py @@ -269,8 +269,8 @@ class ThumbSrv(object): self.log("joined waiting room for %r" % (tpath,)) except: thdir = os.path.dirname(tpath) - chmod = 0o700 if self.args.free_umask else 0o755 - bos.makedirs(os.path.join(thdir, "w"), chmod) + chmod = bos.MKD_700 if self.args.free_umask else bos.MKD_755 + bos.makedirs(os.path.join(thdir, "w"), vf=chmod) inf_path = os.path.join(thdir, "dir.txt") if not bos.path.exists(inf_path): diff --git a/copyparty/up2k.py b/copyparty/up2k.py index 907347f5..49bf21b7 100644 --- a/copyparty/up2k.py +++ b/copyparty/up2k.py @@ -916,7 +916,7 @@ class Up2k(object): for vol in vols: try: # mkdir gonna happen at snap anyways; - bos.makedirs(vol.realpath, vol.flags["chmod_d"]) + bos.makedirs(vol.realpath, vf=vol.flags) dir_is_empty(self.log_func, not self.args.no_scandir, vol.realpath) except Exception as ex: self.volstate[vol.vpath] = "OFFLINE (cannot access folder)" @@ -3309,7 +3309,7 @@ class Up2k(object): reg, "up2k._get_volsize", ) - bos.makedirs(ap2, vfs.flags["chmod_d"]) + bos.makedirs(ap2, vf=vfs.flags) vfs.lim.nup(cj["addr"]) vfs.lim.bup(cj["addr"], cj["size"]) @@ -3445,7 +3445,7 @@ class Up2k(object): "wb", fdir=fdir, suffix="-%.6f-%s" % (ts, dip), - chmod=vf.get("chmod_f", -1), + vf=vf, ) f.close() return ret @@ -4304,7 +4304,7 @@ class Up2k(object): self.log(t, 1) raise Pebkac(405, t) - bos.makedirs(os.path.dirname(dabs), dvn.flags["chmod_d"]) + bos.makedirs(os.path.dirname(dabs), vf=dvn.flags) c1, w, ftime_, fsize_, ip, at = self._find_from_vpath( svn_dbv.realpath, srem_dbv @@ -4480,7 +4480,10 @@ class Up2k(object): vp = vjoin(dvp, rem) try: dvn, drem = self.vfs.get(vp, uname, False, True) - bos.mkdir(dvn.canonical(drem), dvn.flags["chmod_d"]) + dap = dvn.canonical(drem) + bos.mkdir(dap, dvn.flags["chmod_d"]) + if "chown" in dvn.flags: + bos.chown(dap, dvn.flags["uid"], dvn.flags["gid"]) except: pass @@ -4550,7 +4553,7 @@ class Up2k(object): is_xvol = svn.realpath != dvn.realpath - bos.makedirs(os.path.dirname(dabs), dvn.flags["chmod_d"]) + bos.makedirs(os.path.dirname(dabs), vf=dvn.flags) if is_dirlink: dlabs = absreal(sabs) @@ -5062,7 +5065,7 @@ class Up2k(object): "wb", fdir=pdir, suffix="-%.6f-%s" % (job["t0"], dip), - chmod=vf.get("chmod_f", -1), + vf=vf, ) try: abspath = djoin(pdir, job["tnam"]) diff --git a/copyparty/util.py b/copyparty/util.py index cf07ed5a..58a883d5 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -1587,7 +1587,8 @@ def ren_open(fname: str, *args: Any, **kwargs: Any) -> tuple[typing.IO[Any], str fun = kwargs.pop("fun", open) fdir = kwargs.pop("fdir", None) suffix = kwargs.pop("suffix", None) - chmod = kwargs.pop("chmod", -1) + vf = kwargs.pop("vf", None) + fperms = vf and "fperms" in vf if fname == os.devnull: return fun(fname, *args, **kwargs), fname @@ -1631,11 +1632,11 @@ def ren_open(fname: str, *args: Any, **kwargs: Any) -> tuple[typing.IO[Any], str fp2 = os.path.join(fdir, fp2) with open(fsenc(fp2), "wb") as f2: f2.write(orig_name.encode("utf-8")) - if chmod >= 0: - os.fchmod(f2.fileno(), chmod) + if fperms: + set_fperms(f2, vf) - if chmod >= 0: - os.fchmod(f.fileno(), chmod) + if fperms: + set_fperms(f, vf) return f, fname @@ -2565,6 +2566,14 @@ def lsof(log: "NamedLogger", abspath: str) -> None: log("lsof failed; " + min_ex(), 3) +def set_fperms(f: Union[typing.BinaryIO, typing.IO[Any]], vf: dict[str, Any]) -> None: + fno = f.fileno() + if "chmod_f" in vf: + os.fchmod(fno, vf["chmod_f"]) + if "chown" in vf: + os.fchown(fno, vf["uid"], vf["gid"]) + + def _fs_mvrm( log: "NamedLogger", src: str, dst: str, atomic: bool, flags: dict[str, Any] ) -> bool: diff --git a/tests/util.py b/tests/util.py index 17c1d06d..8027c372 100644 --- a/tests/util.py +++ b/tests/util.py @@ -152,6 +152,9 @@ class Cfg(Namespace): ex = "ah_cli ah_gen css_browser dbpath hist ipu js_browser js_other mime mimes no_forget no_hash no_idx nonsus_urls og_tpl og_ua ua_nodoc ua_nozip" ka.update(**{k: None for k in ex.split()}) + ex = "gid uid" + ka.update(**{k: -1 for k in ex.split()}) + ex = "hash_mt hsortn qdel safe_dedup srch_time tail_fd tail_rate u2abort u2j u2sz" ka.update(**{k: 1 for k in ex.split()}) From 6069bc9b19d9d366a0bbba45b3525932e2cc01ff Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 19:38:33 +0000 Subject: [PATCH 009/174] mention optional idp persistence --- docs/idp.md | 16 +++++++++++++--- 1 file changed, 13 insertions(+), 3 deletions(-) diff --git a/docs/idp.md b/docs/idp.md index f3dc7e8c..c27a9355 100644 --- a/docs/idp.md +++ b/docs/idp.md @@ -9,9 +9,9 @@ in the copyparty `[global]` config, specify which headers to read client info fr # important notes -## IdP volumes are forgotten on shutdown +## by default, IdP volumes are forgotten on shutdown -IdP volumes, meaning dynamically-created volumes, meaning volumes that contain `${u}` or `${g}` in their URL, will be forgotten during a server restart and then "revived" when the volume's owner sends their first request after the restart +IdP volumes, meaning dynamically-created volumes, meaning volumes that contain `${u}` or `${g}` in their URL, will (by default) be forgotten during a server restart and then "revived" when the volume's owner sends their first request after the restart until each IdP volume is revived, it will inherit the permissions of its parent volume (if any) @@ -19,7 +19,17 @@ this means that, if an IdP volume is located inside a folder that is readable by and likewise -- if the IdP volume is inside a folder that is only accessible by certain users, but the IdP volume is configured to allow access from unauthenticated users, then the contents of the volume will NOT be accessible until it is revived -until this limitation is fixed (if ever), it is recommended to place IdP volumes inside an appropriate parent volume, so they can inherit acceptable permissions until their revival; see the "strategic volumes" at the bottom of [./examples/docker/idp/copyparty.conf](./examples/docker/idp/copyparty.conf) +it is recommended to place IdP volumes inside an appropriate parent volume, so they can inherit acceptable permissions until their revival; see the "strategic volumes" at the bottom of [./examples/docker/idp/copyparty.conf](./examples/docker/idp/copyparty.conf) + +## but you can enable IdP volume persistence + +global-option `idp-store` can enable user/group persistence across restarts; + +* `idp-store: 1` (default) will log users into a database, but not actually "remember" them (the knowledge is ignored) +* `idp-store: 2` remembers usernames only +* `idp-store: 3` remembers usernames and their groups + +the reason why this is default-disabled, is because you may expect copyparty to forget about a user when you delete them from the IdP-server; this will not be the case any longer, you will need to click `view idp cache` in the controlpanel and manually remove the users you want gone ## Connecting webdav clients From e9684d402e987b7858be86952c9a971cf4b7f1ca Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 19:41:45 +0000 Subject: [PATCH 010/174] fix ipv6 cors-chk --- copyparty/httpcli.py | 17 ++++++++++------- 1 file changed, 10 insertions(+), 7 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index e4bfe45a..7d1a951b 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -1114,15 +1114,18 @@ class HttpCli(object): else: return True + host = self.host.lower() + if host.startswith("["): + if "]:" in host: + host = host.split("]:")[0] + "]" + else: + host = host.split(":")[0] + oh = self.out_headers origin = origin.lower() - good_origins = self.args.acao + [ - "%s://%s" - % ( - "https" if self.is_https else "http", - self.host.lower().split(":")[0], - ) - ] + proto = "https" if self.is_https else "http" + good_origins = self.args.acao + ["%s://%s" % (proto, host)] + if "pw" in ih or re.sub(r"(:[0-9]{1,5})?/?$", "", origin) in good_origins: good_origin = True bad_hdrs = ("",) From fb7cbc423b4832867cd9c36a156675d1a20cd908 Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 19:43:47 +0000 Subject: [PATCH 011/174] shares: move all config to webroot --- copyparty/httpcli.py | 6 +++--- copyparty/web/shares.js | 6 ++++-- 2 files changed, 7 insertions(+), 5 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 7d1a951b..0ff9c0ef 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -5659,15 +5659,15 @@ class HttpCli(object): raise Pebkac(500, "sqlite3 not found on server; sharing is disabled") raise Pebkac(500, "server busy, cannot create share; please retry in a bit") + skey = self.uparam.get("skey") or self.vpath.split("/")[-1] + if self.args.shr_v: - self.log("handle_eshare: " + self.req) + self.log("handle_eshare: " + skey) cur = idx.get_shr() if not cur: raise Pebkac(400, "huh, sharing must be disabled in the server config...") - skey = self.vpath.split("/")[-1] - rows = cur.execute("select un, t1 from sh where k = ?", (skey,)).fetchall() un = rows[0][0] if rows and rows[0] else "" diff --git a/copyparty/web/shares.js b/copyparty/web/shares.js index 86d4cc45..e3faf029 100644 --- a/copyparty/web/shares.js +++ b/copyparty/web/shares.js @@ -1,9 +1,11 @@ +var SRS = SR.trimEnd('/') + '/'; + var t = QSA('a[k]'); for (var a = 0; a < t.length; a++) t[a].onclick = rm; function rm() { - var u = SR + shr + uricom_enc(this.getAttribute('k')) + '?eshare=rm', + var u = SRS + '?eshare=rm&skey=' + uricom_enc(this.getAttribute('k')), xhr = new XHR(); xhr.open('POST', u, true); @@ -13,7 +15,7 @@ function rm() { function bump() { var k = this.closest('tr').getElementsByTagName('a')[2].getAttribute('k'), - u = SR + shr + uricom_enc(k) + '?eshare=' + this.value, + u = SRS + '?skey=' + uricom_enc(k) + '&eshare=' + this.value, xhr = new XHR(); xhr.open('POST', u, true); From 6016ec938844e37329a570f292eaeae9c3c7313e Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 20:30:18 +0000 Subject: [PATCH 012/174] connectpage: fix sharex --- copyparty/web/svcs.html | 18 +++++++++++++++--- 1 file changed, 15 insertions(+), 3 deletions(-) diff --git a/copyparty/web/svcs.html b/copyparty/web/svcs.html index c47d8403..a4b57746 100644 --- a/copyparty/web/svcs.html +++ b/copyparty/web/svcs.html @@ -240,14 +240,26 @@

ShareX

-

to upload screenshots using ShareX v12 or v15+, save this as copyparty.sxcu and run it:

+

to upload screenshots using ShareX v15+, save this as copyparty.sxcu and run it:

+ +
+                { "Version": "15.0.0", "Name": "copyparty",
+                "RequestURL": "http{{ s }}://{{ ep }}/{{ rvp }}",
+                "Headers": {
+                    {% if accs %}"pw": "{{ pw }}", {% endif %}"accept": "url"
+                },
+                "DestinationType": "ImageUploader, TextUploader, FileUploader",
+                "Body": "MultipartFormData", "URL": "{response}",
+                "RequestMethod": "POST", "FileFormName": "f" }
+            
+ +

for ShareX v12 specifically, save this as copyparty.sxcu and run it:

                 { "Name": "copyparty",
                 "RequestURL": "http{{ s }}://{{ ep }}/{{ rvp }}",
                 "Headers": {
-                    {% if accs %}"pw": "{{ pw }}",{% endif %}
-                    "accept": "url"
+                    {% if accs %}"pw": "{{ pw }}", {% endif %}"accept": "url"
                 },
                 "DestinationType": "ImageUploader, TextUploader, FileUploader",
                 "FileFormName": "f" }

From 9d32564c68124d60f0c72f05b7b09febb9083fb3 Mon Sep 17 00:00:00 2001
From: Raphael Guntersweiler 
Date: Wed, 30 Jul 2025 22:34:51 +0200
Subject: [PATCH 013/174] translate to german (#212)

* added german translation
---
 copyparty/web/browser.js | 628 ++++++++++++++++++++++++++++++++++++++-
 copyparty/web/splash.js  |  44 ++-
 2 files changed, 670 insertions(+), 2 deletions(-)

diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js
index f1368912..fc63bf9f 100644
--- a/copyparty/web/browser.js
+++ b/copyparty/web/browser.js
@@ -1884,9 +1884,635 @@ var Ls = {
 
 		"lang_set": "刷新以使更改生效?",
 	},
+	"deu": {
+		"tt": "Deutsch",
+
+		"cols": {
+			"c": "Aktionen",
+			"dur": "Dauer",
+			"q": "Qualität / Bitrate",
+			"Ac": "Audiocodec",
+			"Vc": "Videocodec",
+			"Fmt": "Format / Container",
+			"Ahash": "Audio Checksumme",
+			"Vhash": "Video Checksumme",
+			"Res": "Auflösung",
+			"T": "Dateityp",
+			"aq": "Audioqualität / Bitrate",
+			"vq": "Videoqualität / Bitrate",
+			"pixfmt": "Subsampling / Pixelstruktur",
+			"resw": "horizontale Auflösung",
+			"resh": "vertikale Auflösung",
+			"chs": "Audiokanäle",
+			"hz": "Abtastrate"
+		},
+
+		"hks": [
+			[
+				"misc",
+				["ESC", "Dinge schliessen"],
+
+				"file-manager",
+				["G", "zwischen Liste und Gitter wechseln"],
+				["T", "zwischen Vorschaubildern und Symbolen wechseln"],
+				["⇧ A/D", "Vorschaubildergrösse ändern"],
+				["STRG-K", "Auswahl löschen"],
+				["STRG-X", "Auswahl ausschneiden"],
+				["STRG-C", "Auswahl in Zwischenablage kopieren"],
+				["STRG-V", "Zwischenablage hier einfügen"],
+				["Y", "Auswahl herunterladen"],
+				["F2", "Auswahl umbenennen"],
+
+				"file-list-sel",
+				["LEER", "Dateiauswahl aktivieren"],
+				["↑/↓", "Cursor verschieben"],
+				["STRG ↑/↓", "Cursor und Bildschirm verschieben"],
+				["⇧ ↑/↓", "Vorherige / nächste Datei auswählen"],
+				["STRG-A", "Alle Dateien / Ordner auswählen"],
+			], [
+				"navigation",
+				["B", "Zwischen Brotkrumen und Navpane wechseln"],
+				["I/K", "vorheriger / nächster Ordner"],
+				["M", "übergeordneter Ordner (oder Vorherigen einklappen)"],
+				["V", "Zwischen Textdateien und Navpane wechseln"],
+				["A/D", "Grösse der Navpane ändern"],
+			], [
+				"audio-player",
+				["J/L", "Vorheriger / nächster Song"],
+				["U/O", "10 Sek. vor- / zurückspringen"],
+				["0..9", "zu 0%..90% springen"],
+				["P", "Wiedergabe / Pause"],
+				["S", "aktuell abgespielten Song auswählen"],
+				["Y", "Sing herunterladen"],
+			], [
+				"image-viewer",
+				["J/L, ←/→", "vorheriges / nächstes Bild"],
+				["Pos1/Ende", "erstes / letztes Bild"],
+				["F", "Vollbild"],
+				["R", "im Uhrzeigersinn drehen"],
+				["⇧ R", "gegen den Uhrzeigensinn drehen"],
+				["S", "Bild auswählen"],
+				["Y", "Bild herunterladen"],
+			], [
+				"video-player",
+				["U/O", "10 Sek. vor- / zurückspringen"],
+				["P/K/LEER", "Wiedergabe / Pause"],
+				["C", "continue playing next"],
+				["V", "Wiederholungs-Wiedergabe (Loop)"],
+				["M", "Stummschalten"],
+				["[ und ]", "Loop-Interval einstellen"],
+			], [
+				"textfile-viewer",
+				["I/K", "vorherige / nächste Datei"],
+				["M", "Textdatei schliessen"],
+				["E", "Textdatei bearbeiten"],
+				["S", "Textdatei auswählen (für Ausschneiden / Kopieren / Umbenennen)"],
+			]
+		],
+
+		"m_ok": "OK",
+		"m_ng": "Abbrechen",
+
+		"enable": "Aktivieren",
+		"danger": "ACHTUNG",
+		"clipped": "in Zwischenablage kopiert",
+
+		"ht_s1": "Sekunde",
+		"ht_s2": "Sekunden",
+		"ht_m1": "Minute",
+		"ht_m2": "Minuten",
+		"ht_h1": "Stunde",
+		"ht_h2": "Stunden",
+		"ht_d1": "Tag",
+		"ht_d2": "Tage",
+		"ht_and": " und ",
+
+		"goh": "Einstellungen",
+		"gop": 'zum vorherigen Ordner springen">vorh.',
+		"gou": 'zum übergeordneter Ordner springen">hoch',
+		"gon": 'zum nächsten Ordner springen">nächst.',
+		"logout": "Abmelden ",
+		"access": " Zugriff",
+		"ot_close": "Submenu schliessen",
+		"ot_search": "Dateien nach Attributen, Pfad/Name, Musiktags oder beliebiger Kombination suchen$N$N<code>foo bar</code> = muss «foo» und «bar» enthalten,$N<code>foo -bar</code> = muss «foo» aber nicht «bar» enthalten,$N<code>^yana .opus$</code> = beginnt mit «yana» und ist «opus»-Datei$N<code>"try unite"</code> = genau «try unite» enthalten$N$NDatumsformat ist iso-8601, z.B.$N<code>2009-12-31</code> oder <code>2020-09-12 23:30:00</code>",
+		"ot_unpost": "unpost: lösche deine letzten Uploads oder breche unvollständige ab",
+		"ot_bup": "bup: Basic Uploader, unterstützt sogar Neuheiten wie Netscape 4.0",
+		"ot_mkdir": "mkdir: Neuen Ordner erstellen",
+		"ot_md": "new-md: Neues Markdown-Dokument erstellen",
+		"ot_msg": "msg: Eine Nachricht an das Server-Log schicken",
+		"ot_mp": "Media Player-Optionen",
+		"ot_cfg": "Konfigurationsoptionen",
+		"ot_u2i": 'up2k: Dateien hochladen (wenn du Schreibrechte hast) oder in den Suchmodus wechseln, um zu prüfen, ob sie bereits auf dem Server existieren$N$NUploads sind fortsetzbar, multithreaded und behalten Dateizeitstempel, verbrauchen aber mehr CPU als [🎈]  (der einfache Uploader)

während Uploads wird dieses Symbol zu einem Fortschrittsanzeiger!', + "ot_u2w": 'up2k: Dateien mit Wiederaufnahme-Unterstützung hochladen (Browser schließen und später dieselben Dateien erneut hochladen)$N$Nmultithreaded, behält Dateizeitstempel, verbraucht aber mehr CPU als [🎈]  (der einfache Uploader)

während Uploads wird dieses Symbol zu einem Fortschrittsanzeiger!', + "ot_noie": 'Bitte benutze Chrome / Firefox / Edge', + + "ab_mkdir": "Ordner erstellen", + "ab_mkdoc": "Markdown Doc erstellen", + "ab_msg": "Nachricht an Server Log senden", + + "ay_path": "zu Ordnern springen", + "ay_files": "zu Dateien springen", + + "wt_ren": "ausgewählte Elemente umbenennen$NHotkey: F2", + "wt_del": "ausgewählte Elemente löschen$NHotkey: STRG-K", + "wt_cut": "ausgewählte Elemente ausschneiden <small>(um sie dann irgendwo anders einzufügen)</small>$NHotkey: STRG-X", + "wt_cpy": "ausgewählte Elemente in Zwischenablage kopieren$N(um sie dann irgendwo anders einzufügen)$NHotkey: ctrl-C", + "wt_pst": "zuvor ausgeschnittenen / kopierte Elemente einfügen$NHotkey: STRG-V", + "wt_selall": "alle Dateien auswählen$NHotkey: STRG-A (wenn Datei fokusiert)", + "wt_selinv": "Auswahl invertieren", + "wt_zip1": "Diesen Ordner als Archiv herunterladen", + "wt_selzip": "Auswahl als Archiv herunterladen", + "wt_seldl": "Auswahl als separate Dateien herunterladen$NHotkey: Y", + "wt_npirc": "kopiere Titelinfo als IRC-formattierten Text", + "wt_nptxt": "kopiere Titelinfo als Text", + "wt_m3ua": "Zu M3U-Wiedergabeliste hinzufügen (wähle später 📻copy)", + "wt_m3uc": "M3U-Wiedergabeliste in Zwischenablage kopieren", + "wt_grid": "Zwischen Gitter und Liste wechseln$NHotkey: G", + "wt_prev": "Vorheriger Titel$NHotkey: J", + "wt_play": "Wiedergabe / Pause$NHotkey: P", + "wt_next": "Nächster Titel$NHotkey: L", + + "ul_par": "Parallele Uploads:", + "ut_rand": "Zufällige Dateinamen", + "ut_u2ts": "Zuletzt geändert-Zeitstempel von$Ndeinem Dateisystem auf den Server übertragen\">📅", + "ut_ow": "Existierende Dateien auf dem Server überschreiben?$N🛡️: Nie (generiert einen neuen Dateinamen)$N🕒: Überschreiben, wenn Server-Datei älter ist als meine$N♻️: Überschreiben, wenn der Dateiinhalt anders ist", + "ut_mt": "Andere Dateien während des Uploads hashen$N$Nsolltest du deaktivieren, falls deine CPU oder Festplatte zum Flaschenhals werden könnte", + "ut_ask": 'Vor dem Upload nach Bestätigung fragen">💭', + "ut_pot": "Verbessert Upload-Geschwindigkeit$Nindem das UI weniger komplex gemacht wird", + "ut_srch": "nicht wirklich hochladen, stattdessen prüfen ob Datei bereits auf dem Server existiert (scannt alle Ordner, die du lesen kannst)", + "ut_par": "setze auf 0 zum Pausieren$N$Nerhöhe, wenn deine Verbindung langsam / instabil ist$N$lass auf 1 im LAN oder wenn die Festplatte auf dem Server ein Flaschenhals ist", + "ul_btn": "Dateien / Ordner hier
ablegen (oder klick mich)", + "ul_btnu": "U P L O A D", + "ul_btns": "S U C H E N", + + "ul_hash": "hash", + "ul_send": "senden", + "ul_done": "fertig", + "ul_idle1": "keine Uploads in der Warteschlange", + "ut_etah": "durchschnittl. <em>hashing</em> Geschw. & gesch. Restzeit", + "ut_etau": "durchschnittl. <em>upload</em> Geschw. & gesch. Restzeit", + "ut_etat": "durchschnittl. <em>total</em> Geschw. & gesch. Restzeit", + + "uct_ok": "Erfolgreich abgeschlossen", + "uct_ng": "no-good: fehlgeschlagen / abgelehnt / nicht gefunden", + "uct_done": "ok and ng zusammen", + "uct_bz": "wird gehasht oder hochgeladen", + "uct_q": "ausstehend", + + "utl_name": "Dateiname", + "utl_ulist": "Liste", + "utl_ucopy": "kopieren", + "utl_links": "Links", + "utl_stat": "Status", + "utl_prog": "Fortschritt", + + // keep short: + "utl_404": "404", + "utl_err": "Fehler", + "utl_oserr": "OS-Fehler", + "utl_found": "gefunden", + "utl_defer": "zurückstellen", + "utl_yolo": "YOLO", + "utl_done": "fertig", + + "ul_flagblk": "Die Dateien wurden zur Warteschlange hinzugefügt
jedoch ist up2k gerade in einem anderen Browsertab aktiv.
Ich warte, bis der Upload abgeschlossen ist.", + "ul_btnlk": "Die Serverkonfiguration hat diese Einstellung gesperrt", + + "udt_up": "Upload", + "udt_srch": "Suchen", + "udt_drop": "hier ablegen", + + "u_nav_m": '
okay, was gibts??
Eingabe = Dateien (1 oder mehr)\nESC = 1 Ordner (inkl. Unterordner)', + "u_nav_b": 'Dateien1 Ordner', + + "cl_opts": "Schalter", + "cl_themes": "Themes", + "cl_langs": "Sprache", + "cl_ziptype": "Ordner Download", + "cl_uopts": "up2k Schalter", + "cl_favico": "Favicon", + "cl_bigdir": "grosse Ordner", + "cl_hsort": "#sort", + "cl_keytype": "Schlüsselnotation", + "cl_hiddenc": "Spalten verstecken", + "cl_hidec": "verstecken", + "cl_reset": "zurücksetzen", + "cl_hpick": "zum Verstecken, tippe auf Spaltenüberschriften in der Tabelle unten", + "cl_hcancel": "Spaltenbearbeitung abgebrochen", + + "ct_grid": '田 Das Raster™', + "ct_ttips": '◔ ◡ ◔">ℹ️ Tooltips', + "ct_thumb": 'In Raster-Ansicht, zwischen Icons und Vorschau wechseln$NHotkey: T">🖼️ Vorschaubilder', + "ct_csel": 'Benutze STRG und UMSCHALT für Dateiauswahl in Raster-Ansicht">sel', + "ct_ihop": 'Wenn die Bildanzeige geschlossen ist, scrolle runter zu den zuletzt angesehenen Dateien">g⮯', + "ct_dots": 'Verstecke Dateien anzeigen (wenn erlaubt durch Server)">dotfiles', + "ct_qdel": 'Nur einmal fragen, wenn mehrere Dateien gelöscht werden">qdel', + "ct_dir1st": 'Ordner vor Dateien sortieren">📁 zuerst', + "ct_nsort": 'Natürliche Sortierung (für Dateinamen mit führenden Ziffern)">nsort', + "ct_readme": 'README.md in Dateiliste anzeigen">📜 readme', + "ct_idxh": 'index.html anstelle von Dateiliste anzeigen">htm', + "ct_sbars": 'Scrollbars zeigen">⟊', + + "cut_umod": "Sollte die Datei bereits auf dem Server existieren, den 'Zuletzt geändert'-Zeitstempel an deine lokale Datei anpassen (benötigt Lese- und Löschrechte)\">re📅", + + "cut_turbo": "der YOLO-Knopf, den du wahrscheinlich NICHT aktivieren willst:$N$NBenutze ihn, falls du ne Menge Zeug hochladen wolltest und aus irgendeinem Grund neustarten musstest und du so schnell wie möglich weitermachen willst.$N$Ndies ersetzt den Hash-Check mit einem einfachen "Ist die Datei auf dem Server gleich gross?", wenn die Datei also anderen Inhalt hat, wird sie NICHT nochmal hochgeladen!$N$NDu solltest dieses Feature ausschalten, sobald der Upload fertig ist und dann die gleichen Dateien nochmal "hochladen", damit der Client sie verifizieren kann.\">turbo", + + "cut_datechk": "Funktioniert nur in kombination mit dem Turbo-Knopf$N$NReduziert den YOLO-Faktor ein bisschen; prüft, ob der Zeitstempel deiner Datei mit dem auf dem Server übereinstimmt$N$Nsollte theoretisch die meisten unfertigen / korrupten Uploads erwischen, ist aber nicht zu gebrauchen, um einen Prüfdurchgang nach einem Turbo-Upload zu machen\">date-chk", + + "cut_u2sz": "Grösse (in MiB) für jeden Upload-Chunk; mit grossen Werten fliegen die Bits besser über den Atlantik. Versuche kleine Werte, wenn du eine schlechte Verbindung hast (z.B. du benutzt mobile Daten in Deutschland)", + + "cut_flag": "Stelle sicher, dass nur ein Tab auf einmal Dateien hochlädt$N -- andere Tabs müssen diese Funktion auch aktiviert haben $N -- funktioniert nur bei Tabs mit der gleichen Domäne", + + "cut_az": "Lädt Dateien in alphabetischer Reihenfolge hoch, anstatt nach Dateigrösse$N$NAlphabethische Reihenfolge kann es einfacher machen, Server-Fehler mit naktem Auge zu erkennen, macht aber Uploads über Glassfaser / LAN etwas langsamer", + + "cut_nag": "Benachrichtigung über das Betriebssystem abgeben, wenn Upload fertig ist$N(nur wenn Browser oder Tab nicht im Vordergrund ist)", + "cut_sfx": "Spielt ein Ton ab, wenn Upload fertig ist$N(nur wenn Browser oder Tab nicht im Vordergrund ist)", + + "cut_mt": "Multithreading benutzen um Datei-Hashing zu beschleunigen$N$NDies nutzt Web-Workers und benötigt$Nmehr RAM (bis zu 512 MiB extra)$N$Nbeschleunigt HTTPS 30% schneller, HTTP um 4.5x\">mt", + + "cut_wasm": "benutzt WASM anstelle des Browser-eigenen Hashers; verbessert Geschwindigkeit auf Chromium-basierten Browsern, erhöht aber die CPU-Auslastung. Viele ältere Versionen von Chrome haben Memory-Leaks, die den gesamten RAM verbrauchen und dann crashen, wenn diese Funktion aktiviert ist.\">wasm", + + "cft_text": "Favicon Text (leer lassen und neuladen zum Deaktivieren)", + "cft_fg": "Vordergrundfarbe", + "cft_bg": "Hintergrundfarbe", + + "cdt_lim": "max. Anz. Dateien, die in einem Ordner gezeigt werden sollen", + "cdt_ask": "beim Runterscrollen nach $NAktion fragen statt mehr,$NDateien zu laden", + "cdt_hsort": "Menge an Sortierregeln (<code>,sorthref</code>) in Media-URLs enthalten sein sollen. Ein Wert von 0 sorgt dafür, dass Sortierregeln in Media-URLs ignoriert werden", + + "tt_entree": "Navpane anzeigen (Ordnerbaum Sidebar)$NHotkey: B", + "tt_detree": "Breadcrumbs anzeigen$NHotkey: B", + "tt_visdir": "zu ausgewähltem Ordner scrollen", + "tt_ftree": "zw. Ordnerbaum / Textdateien wechseln$NHotkey: V", + "tt_pdock": "übergeordnete Ordner in einem angedockten Fenster oben anzeigen", + "tt_dynt": "autom. wachsen wenn Baum wächst", + "tt_wrap": "Zeilenumbruch", + "tt_hover": "Beim Hovern überlange Zeilen anzeigen$N(Scrollen funktioniert nicht ausser $N  Cursor ist im linken Gutter)", + + "ml_pmode": "am Ende des Ordners...", + "ml_btns": "cmds", + "ml_tcode": "transcodieren", + "ml_tcode2": "transcodieren zu", + "ml_tint": "färben", + "ml_eq": "Audio Equalizer", + "ml_drc": "Dynamic Range Compressor", + + "mt_loop": "Song wiederholen\">🔁", + "mt_one": "Wiedergabe nach diesem Song beenden\">1️⃣", + "mt_shuf": "Zufällige Wiedergabe im Ordner\">🔀", + "mt_aplay": "automatisch abspielen, wenn der Link, mit dem du auf den Server zugreifst, eine Titel-ID enthält$N$NDeaktivieren verhindert auch, dass die Seiten-URL bei Musikwiedergabe mit Titel-IDs aktualisiert wird, um Autoplay zu verhindern, falls diese Einstellungen verloren gehen, die URL aber bestehen bleibt\">a▶", + "mt_preload": "nächsten Titel gegen Ende vorladen für nahtlose Wiedergabe\">Vorladen", + "mt_prescan": "vor Ende des letzten Titels zum nächsten Ordner wechseln,$Ndamit der Browser die$NWiedergabe nicht stoppt\">Navigation", + "mt_fullpre": "versuchen, den gesamten Titel vorzuladen;$N✅ bei unzuverlässiger Verbindung aktivieren,$N❌ bei langsamer Verbindung deaktivieren\">vollständig", + "mt_fau": "auf Handys verhindern, dass Musik stoppt, wenn der nächste Titel nicht schnell genug vorlädt (kann zu fehlerhafter Tag-Anzeige führen)\">☕️", + "mt_waves": "Wellenform-Suchleiste:$NAudio-Amplitude in der Leiste anzeigen\">~s", + "mt_npclip": "Buttons zum Kopieren des aktuellen Titels anzeigen\">/np", + "mt_m3u_c": "Buttons zum Kopieren der$Nausgewählten Titel als m3u8-Wiedergabeliste anzeigen\">📻", + "mt_octl": "OS-Integration (Media-Hotkeys/OSD)\">os-ctl", + "mt_oseek": "Suchen via OS-Integration erlauben$N$NHinweis: auf einigen Geräten (iPhones)$Nersetzt dies den nächsten-Titel-Button\">Suchen", + "mt_oscv": "Albumcover in OSD anzeigen\">Cover", + "mt_follow": "den spielenden Titel im Blick behalten\">🎯", + "mt_compact": "kompakte Steuerelemente\">⟎", + "mt_uncache": "Cache leeren  (probier das, wenn dein Browser$Neine defekte Kopie eines Titels zwischenspeichert und sich weigert, ihn abzuspielen)\">Cache leeren", + "mt_mloop": "offenen Ordner wiederholen\">🔁 Schleife", + "mt_mnext": "nächsten Ordner laden und fortfahren\">📂 nächster", + "mt_mstop": "Wiedergabe beenden\">⏸ Stop", + "mt_cflac": "FLAC / WAV zu OPUS konvertierebn\">flac", + "mt_caac": "AAC / M4A zu OPUS konvertieren\">aac", + "mt_coth": "Convertiere alle Dateien (die nicht MP3 sind) zu OPUS\">oth", + "mt_c2opus": "Beste Wahl für Desktops, Laptops, Android\">opus", + "mt_c2owa": "opus-weba, für iOS 17.5 und neuer\">owa", + "mt_c2caf": "opus-caf, für iOS 11 bis 17\">caf", + "mt_c2mp3": "benutze dieses Format für ältere Geräte\">mp3", + "mt_c2ok": "Gute Wahl, Chef!", + "mt_c2nd": "Das ist nicht das empfohlene Ausgabeformat für dein Gerät, aber passt schon", + "mt_c2ng": "Dein Gerät scheint dieses Ausgabeformat nicht zu unterstützen, aber lass trotzdem mal probieren", + "mt_xowa": "Es gibt Bugs in iOS, die die Hintergrund-Wiedergabe mit diesem Format verhindern; bitte nutze caf oder mp3 stattdessen", + "mt_tint": "Hintergrundlevel (0-100) auf der Seekbar$Num Buffern weniger ablenkend zu machen", + "mt_eq": "Aktiviert Equalizer und Lautstärkeregelung;$N$Nboost <code>0</code> = Standard 100% Lautstärke (unverändert)$N$Nwidth <code>1  </code> = Standard Stereo (unverändert)$Nwidth <code>0.5</code> = 50% Links-Rechts-Crossfeed$Nwidth <code>0  </code> = Mono$N$Nboost <code>-0.8</code> & width <code>10</code> = Gesangsentfernung :^)$N$NDer Equalizer macht nahtlose Alben vollständig nahtlos, also lass' ihn mit allen Werten auf Null (außer width = 1) aktiviert, wenn dir das wichtig ist", + "mt_drc": "Aktiviert den Dynamic Range Compressor (Lautstärkeglättung/-begrenzung); aktiviert auch den Equalizer zum Ausgleich, setze alle EQ-Felder außer 'width' auf 0, wenn du das nicht willst$N$Nsenkt die Lautstärke von Audio über SCHWELLENWERT dB; für jedes VERHÄLTNIS dB über SCHWELLENWERT gibt es 1 dB Ausgabe, also bedeuten Standardwerte von tresh -24 und ratio 12, dass es nie lauter als -22 dB werden sollte und der Equalizer-Boost sicher auf 0.8 oder sogar 1.8 mit ATK 0 und einem großen RLS wie 90 erhöht werden kann (funktioniert nur in Firefox; in anderen Browsern ist RLS max. 1)$N$N(siehe Wikipedia, dort wird es viel besser erklärt)", + + "mb_play": "Abspielen", + "mm_hashplay": "Diese Audiodatei abspielen?", + "mm_m3u": "Drücke Eingabe/OK zum Abspielen\nDrücke ESC/Abbrechen zum Bearbeiten", + "mp_breq": "Benötigt Firefox 82+ oder Chrome 73+ oder iOS 15+", + "mm_bload": "Lädt...", + "mm_bconv": "Konvertiere zu {0}, bitte warte...", + "mm_opusen": "Dein Browser kann AAC- / M4A-Dateien nicht abspielen;\nUmwandlung zu Opus ist jetzt aktiv", + "mm_playerr": "Wiedergabefehler: ", + "mm_eabrt": "Der Wiedergabeversuch wurde abgebrochen", + "mm_enet": "Dein Internet läuft auf Edge, wa?", + "mm_edec": "Die Datei scheint beschädigt zu sein??", + "mm_esupp": "Dein Browser versteht dieses Audioformat nicht", + "mm_eunk": "Unbekannter Fehler", + "mm_e404": "Konnte Datei nicht abspielen; Fehler 404: Datei nicht gefunden.", + "mm_e403": "Konnte Datei nicht abspielen; Fehler 403: Zugriff verweigert.\n\nDrücke F5 zum Neuladen, vielleicht wurdest du abgemeldet", + "mm_e500": "Konnte Datei nicht abspielen; Fehler 500: Prüfe die Serverlogs.", + "mm_e5xx": "Konnte Datei nicht abspielen; Server Fehler ", + "mm_nof": "finde keine weiteren Audiodateien in der Nähe", + "mm_prescan": "Suche nach Musik zum Abspielen...", + "mm_scank": "Nächster Song gefunden:", + "mm_uncache": "Cache geleert; Alle Songs werden beim nächsten Abspielversuch neu heruntergeladen", + "mm_hnf": "dieser Song existiert nicht mehr", + + "im_hnf": "dieses Bild existiert nicht mehr", + + "f_empty": 'Dieser Ordner ist leer', + "f_chide": 'Dies blendet die Spalte «{0}» aus\n\nDu kannst Spalten in den Einstellungen wieder einblenden.', + "f_bigtxt": "Diese Datei ist {0} MiB gross -- Sicher, dass du sie als Text anzeigen willst?", + "f_bigtxt2": "Möchtest du stattdessen nur das Ende der Datei anzeigen? Das aktiviert ausserdem die Folgen- und Verfolgen-Funktion, welche neu hinzugefügte Textzeilen in Echtzeit anzeigt", + "fbd_more": '
zeige {0} von {1} Dateien; {2} anzeigen oder alle anzeigen
', + "fbd_all": '
zeige {0} von {1} Dateien; alle anzeigen
', + "f_anota": "nur {0} der {1} Elemente wurden ausgewählt;\num den gesamten Ordner auszuwählen, zuerst nach unten scrollen", + + "f_dls": 'die Dateilinks im aktuellen Ordner wurden\nin Downloadlinks geändert', + + "f_partial": "Um eine Datei sicher herunterzuladen, die gerade hochgeladen wird, klicke bitte die Datei mit dem gleichen Namen, aber ohne die .PARTIAL-Endung. Bitte drücke Abbrechen oder Escape, um dies zu tun.\n\nWenn du auf OK / Eingabe drückst, ignorierst du diese Warnung und lädst die .PARTIAL-Datei herunter, die ziemlich sicher beschädigte Daten enthält.", + + "ft_paste": "{0} Elemente einfügen$NHotkey: STRG-V", + "fr_eperm": 'Umbenennen fehlgeschlagen:\nDir fehlt die "Verschieben"-Berechtigung in diesem Ordner', + "fd_eperm": 'Löschen fehlgeschlagen:\nDir fehlt die "Löschen"-Berechtigung in diesem Ordner', + "fc_eperm": 'Ausschneiden fehlgeschlagen:\nDir fehlt die "Verschieben"-Berechtigung in diesem Ordner', + "fp_eperm": 'Einfügen fehlgeschlagen:\nDir fehlt die "Schreiben"-Berechtigung in diesem Ordner', + "fr_emore": "Wähle mindestens ein Element zum Umbenennen aus", + "fd_emore": "Wähle mindestens ein Element zum Löschen aus", + "fc_emore": "Wähle mindestens ein Element zum Ausschneiden aus", + "fcp_emore": "Wähle mindestens ein Element aus, um es in die Zwischenablage zu kopieren", + + "fs_sc": "Teile diesen Ordner", + "fs_ss": "Teile die ausgewählten Dateien", + "fs_just1d": "Du kannst nicht mehrere Ordner auswählen \noder Dateien und Ordner in der Auswahl mischen.", + "fs_abrt": "❌ Abbrechen", + "fs_rand": "🎲 Zufallsname", + "fs_go": "✅ Share erstellen", + "fs_name": "Name", + "fs_src": "Quelle", + "fs_pwd": "Passwort", + "fs_exp": "Ablauf", + "fs_tmin": "Minuten", + "fs_thrs": "Stunden", + "fs_tdays": "Tage", + "fs_never": "nie", + "fs_pname": "optionaler Linkname; zufällig wenn leer", + "fs_tsrc": "zu teilende Datei oder Ordner", + "fs_ppwd": "optionales Passwort", + "fs_w8": "erstelle Share...", + "fs_ok": "drücke Eingabe/OK für Zwischenablage\ndrücke ESC/Abbrechen zum Schliessen", + + "frt_dec": "Kann Fälle von beschädigten Dateien beheben\">url-decode", + "frt_rst": "Geänderte Dateinamen auf Orginale zurücksetzen\">↺ zurücksetzen", + "frt_abrt": "Abbrechen und dieses Fenster schliessen\">❌ abbrechen", + "frb_apply": "ÜBERNEHMEN", + "fr_adv": "Stapel-/Metadaten-/Musterumbenennung\">erweitert", + "fr_case": "Groß-/Kleinschreibung beachten (Regex)\">Großschreibung", + "fr_win": "Windows-kompatible Namen; ersetzt <>:"\\|?* durch japanische Fullwidth-Zeichen\">win", + "fr_slash": "Ersetzt / durch ein Zeichen, das keine neuen Ordner erstellt\">no /", + "fr_re": "Regex-Suchmuster für Originaldateinamen; Erfassungsgruppen können im Formatfeld unten als <code>(1)</code> und <code>(2)</code> usw. referenziert werden", + "fr_fmt": "inspiriert von foobar2000:$N<code>(title)</code> wird durch Songtitel ersetzt,$N<code>[(artist) - ](title)</code> überspringt [diesen] Teil falls Interpret leer$N<code>$lpad((tn),2,0)</code> füllt die Titelnummer auf 2 Ziffern auf", + "fr_pdel": "Löschen", + "fr_pnew": "Speichern als", + "fr_pname": "Gib der Vorlage einen Namen", + "fr_aborted": "Abgebrochen", + "fr_lold": "Alter Name", + "fr_lnew": "Neuer Name", + "fr_tags": "Tags für die ausgewählten Dateien (liest nur, als Referenz):", + "fr_busy": "Benenne {0} Elemente um...\n\n{1}", + "fr_efail": "Umbenennen fehlgeschlagen:\n", + "fr_nchg": "{0} der neuen Namen wurden angepasst durch win und/oder no /\n\nMöchtest du mit diesen geänderten Namen fortfahren?", + + "fd_ok": "Löschen OK", + "fd_err": "Löschen fehlgeschlagen:\n", + "fd_none": "Nichts würde gelöscht; vielleicht durch die Serverkonfiguration blockiert (xbd)?", + "fd_busy": "Lösche {0} Elemente...\n\n{1}", + "fd_warn1": "Diese {0} Elemente LÖSCHEN?", + "fd_warn2": "Ich frage das letzte Mal! Was weg ist, ist weg. Keine Chance, das rückgängig zu machen. Löschen?", + + "fc_ok": "{0} Elemente ausgeschnitten", + "fc_warn": '{0} Elemente in die Zwischenablage kopiert\n\nAber: nur dieses Browsertab kann sie einfügen\n(da deine Auswahl so abartig riesig war)', + + "fcc_ok": "{0} Elemente in die Zwischenablage kopiert", + "fcc_warn": '{0} Elemente in die Zwischenablage kopiert\n\nAber: nur dieses Browsertab kann sie einfügen\n(da deine Auswahl so abartig riesig war)', + + "fp_apply": "Diese Namen verwenden", + "fp_ecut": "Kopiere erst ein paar Dateien / Ordner, um sie einzufügen\n\nTipp: Ausschneiden und Kopieren funktioniert über Browsertabs hinweg", + "fp_ename": "{0} Elemente konnten nicht verschoben werden, weil bereits andere Dateien mit diesen Namen existieren. Gib ihnen unten neue Namen um fortzufahren, oder lass das Feld leer zum Überspringen:", + "fcp_ename": "{0} Elemente konnten nicht kopiert werden, weil bereits andere Dateien mit diesen Namen existieren. Gib ihnen unten neue Namen um fortzufahren, oder lass das Feld leer zum Überspringen:", + "fp_emore": "Es gibt noch ein paar Dateinamen, die geändert werden müssen", + "fp_ok": "Verschieben OK", + "fcp_ok": "Kopieren OK", + "fp_busy": "Verschiebe {0} Elemente...\n\n{1}", + "fcp_busy": "Kopiere {0} Elemente...\n\n{1}", + "fp_err": "Verschieben fehlgeschlagen:\n", + "fcp_err": "Kopieren fehlgeschlagen:\n", + "fp_confirm": "Diese {0} Elemente hierher verschieben?", + "fcp_confirm": "Diese {0} Elemente hierher kopieren?", + "fp_etab": 'Konnte die Zwischenablage nicht vom anderen Browsertab lesen', + "fp_name": "Lade Datei von deinem Gerät hoch. Gib ihr einen Namen:", + "fp_both_m": '
Wähle, was eingefügt werden soll
Eingabe = {0} Dateien von «{1}» verschieben\nESC = {2} Dateien von deinem Gerät hochladen', + "fcp_both_m": '
Wähle, was eingefügt werden soll
Eingabe = {0} Dateien von «{1}» kopieren\nESC = {2} Dateien von deinem Gerät hochladen', + "fp_both_b": 'VerschiebenHochladen', + "fcp_both_b": 'KopierenHochladen', + + "mk_noname": "Tipp' mal vorher lieber einen Namen in das Textfeld links, bevor du das machst :p", + + "tv_load": "Textdatei wird geladen:\n\n{0}\n\n{1}% ({2} von {3} MiB geladen)", + "tv_xe1": "Konnte Textdatei nicht laden:\n\nFehler ", + "tv_xe2": "404, Datei nicht gefunden", + "tv_lst": "Liste der Textdateien in", + "tvt_close": "Zu Ordneransicht zurück$NHotkey: M (oder Esc)\">❌ Schliessen", + "tvt_dl": "Diese Datei herunterladen$NHotkey: Y\">💾 Herunterladen", + "tvt_prev": "Vorheriges Dokument zeigen$NHotkey: i\">⬆ vorh.", + "tvt_next": "Nächstes Dokument zeigen$NHotkey: K\">⬇ nächst.", + "tvt_sel": "Wählt diese Datei aus   ( zum Ausschneiden / Kopieren / Löschen / ... )$NHotkey: S\">ausw.", + "tvt_edit": "Datei im Texteditor zum Bearbeiten öffnen$NHotkey: E\">✏️ bearb.", + "tvt_tail": "Datei auf Veränderungen überwachen; Neue Zeilen werden in Echtzeit angezeigt\">📡 folgen", + "tvt_wrap": "Zeilenumbruch\">↵", + "tvt_atail": "Automatisch nach unten scrollen\">⚓", + "tvt_ctail": "Terminal-Farben dekodieren (ANSI Escape Codes)\">🌈", + "tvt_ntail": "Scrollback limitieren (Menge an Bytes an Text, die geladen bleiben sollen)", + + "m3u_add1": "Song wurde zur M3U-Playlist hinzugefügt", + "m3u_addn": "{0} Songs zur M3U-Playlist hinzugefügt", + "m3u_clip": "M3U-Playlist in die Zwischenablage kopiert\n\nDu solltest eine neue Datei mit dem Namen something.m3u erstellen und die Playlist da rein kopieren; damit wird die Playlist abspielbar", + + "gt_vau": "nur Ton abspielen, kein Video zeigen\">🎧", + "gt_msel": "Dateiauswahl aktivieren; STRG-klicke eine Datei zum überschreiben$N$N<em>wenn aktiv: Datei / Ordner doppelklicken zum Öffnen</em>$N$NHotkey: S\">multiselect", + "gt_crop": "Vorschaubilder mittig zuschneiden\">crop", + "gt_3x": "hochauflösende Vorschaubilder\">3x", + "gt_zoom": "zoom", + "gt_chop": "kürzen", + "gt_sort": "sortieren nach", + "gt_name": "Name", + "gt_sz": "Grösse", + "gt_ts": "Datum", + "gt_ext": "Typ", + "gt_c1": "Dateinamen mehr kürzen (weniger zeigen)", + "gt_c2": "Dateinamen weniger kürzen (mehr zeigen)", + + "sm_w8": "Suche ...", + "sm_prev": "Die Suchresultate gehören zu einer vorherigen Suchanfrage:\n ", + "sl_close": "Suchresultate schliessen", + "sl_hits": "Zeige {0} Treffer", + "sl_moar": "Mehr laden", + + "s_sz": "Grösse", + "s_dt": "Datum", + "s_rd": "Pfad", + "s_fn": "Name", + "s_ta": "Tags", + "s_ua": "up@", + "s_ad": "adv.", + "s_s1": "minimum MiB", + "s_s2": "maximum MiB", + "s_d1": "min. iso8601", + "s_d2": "max. iso8601", + "s_u1": "hochgeladen nach", + "s_u2": "und/oder vor", + "s_r1": "Pfad enthält   (Leerzeichen-separiert)", + "s_f1": "Name enthält   (negieren mit -nope)", + "s_t1": "Tags enthält   (^=start, end=$)", + "s_a1": "spezifische Metadaten-Eigenschaften", + + "md_eshow": "Kann nicht rendern ", + "md_off": "[📜readme] deaktiviert in [⚙️] -- Dokument versteckt", + + "badreply": "Hab die Antwort vom Server nicht verstanden. (badreply)", + + "xhr403": "403: Zugriff verweigert\n\nVersuche, F5 zu drücken. Vielleicht wurdest du abgemeldet.", + "xhr0": "Unbekannt (wahrschenlich Verbindung zum Server verloren oder der Server ist offline)", + "cf_ok": "Sorry dafür -- Der DD" + wah + "oS-Schutz hat angeschlagen.\n\nEs sollte in etwa 30 Sekunden weitergehen.\n\nFalls nichts passiert, drück' F5, um die Seite neuzuladen", + "tl_xe1": "Konnte Unterordner nicht auflisten:\n\nFehler ", + "tl_xe2": "404: Ordner nicht gefunden", + "fl_xe1": "Konnte Dateien in Ordner nicht auflisten:\n\nFehler ", + "fl_xe2": "404: Ordner nicht gefunden", + "fd_xe1": "Konnte Unterordner nicht erstellen:\n\nFehler ", + "fd_xe2": "404: Übergeordneter Ordner nicht gefunden", + "fsm_xe1": "Konnte Nachricht nicht senden:\n\nFehler ", + "fsm_xe2": "404: Übergeordneter Ordner nicht gefunden", + "fu_xe1": "Konnte unpost-Liste nicht laden:\n\nFehler ", + "fu_xe2": "404: Datei nicht gefunden??", + + "fz_tar": "Unkomprimierte GNU TAR-Datei (Linux / Mac)", + "fz_pax": "Unkomprimierte pax-format TAR-Datei (slower)", + "fz_targz": "GNU-TAR mit gzip Level 3 Kompression$N$Nüblicherweise recht langsam,$Nbenutze stattdessen ein unkomprimiertes TAR", + "fz_tarxz": "GNU-TAR mit xz level 1 Kompression$N$Nüblicherweise recht langsam,$Nbenutze stattdessen ein unkomprimiertes TAR", + "fz_zip8": "ZIP mit UTF8-Dateinamen (könnte kaputt gehen auf Windows 7 oder älter)", + "fz_zipd": "ZIP mit traditionellen CP437-Dateinamen, für richtig alte Software", + "fz_zipc": "CP437 mit CRC32 früh berechnet,$Nfür MS-DOS PKZIP v2.04g (Oktober 1993)$N(braucht länger zum Verarbeiten, bevor der Download starten kann)", + + "un_m1": "Unten kannst du deine neusten Uploads löschen (oder Unvollständige abbrechen)", + "un_upd": "Neu laden", + "un_m4": "Oder die unten sichtbaren Dateien teilen:", + "un_ulist": "Anzeigen", + "un_ucopy": "Kopieren", + "un_flt": "Optionale Filter:  URL muss enthalten", + "un_fclr": "Filter löschen", + "un_derr": 'unpost-delete fehlgeschlagen:\n', + "un_f5": 'Etwas ist kaputt gegangen, versuche die Seite neuzuladen (drücke dazu F5)', + "un_uf5": "Sorry, aber du musst die Seite neuladen (z.B. in dem du F5 oder STRG-R drückst) bevor zu diesen Upload abbrechen kannst", + "un_nou": 'Warnung: Der Server ist grade zu beschäftigt, um unvollständige Uploads anzuzeigen; Drücke den "Neu laden"-Link in ein paar Sekunden', + "un_noc": 'Warnung: unpost von vollständig hochgeladenen Dateien ist über die Serverkonfiguration gesperrt', + "un_max": "Zeige die ersten 2000 Dateien (benutze Filter, um die gewünschten Dateien zu finden)", + "un_avail": "{0} zuletzt hochgeladene Dateien können gelöscht werden
{1} Unvollständige können abgebrochen werden", + "un_m2": "Sortiert nach Upload-Zeitpunkt; neuste zuerst:", + "un_no1": "Hoppala! Es gibt keine ausreichend aktuellen Uploads.", + "un_no2": "Pech gehabt! Kein Upload, der zu dem Filter passen würde, ist neu genug", + "un_next": "Lösche die nächsten {0} Dateien", + "un_abrt": "Abbrechen", + "un_del": "Löschen", + "un_m3": "Deine letzten Uploads werden geladen ...", + "un_busy": "Lösche {0} Dateien ...", + "un_clip": "{0} Links in die Zwischenablage kopiert", + + "u_https1": "für bessere Performance solltest du", + "u_https2": "auf HTTPS wechseln", + "u_https3": "", + "u_ancient": 'Dein Browser ist verdammt antik -- vielleicht solltest du stattdessen bup benutzen', + "u_nowork": "Benötigt Firefox 53+ oder Chrome 57+ oder iOS 11+", + "tail_2old": "Benötigt Firefox 105+ oder Chrome 71+ oder iOS 14.5+", + "u_nodrop": 'Dein Browser ist zu alt für Drag-and-Drop Uploads', + "u_notdir": "Das ist kein Ordner!\n\nDein Browser ist zu alt,\nversuch stattdessen dragdrop", + "u_uri": "Um Bilder per Drag-and-Drop aus anderen Browserfenstern hochzuladen,\nlass' sie bitte über dem grossen Upload-Button fallen", + "u_enpot": 'Zu Potato UI wechseln (kann Upload-Geschw. verbessern)', + "u_depot": 'Zu fancy UI wechseln (kann Upload-Geschw. verschlechtern)', + "u_gotpot": 'Wechsle zu Potato UI für verbesserte Upload-Geschwindigkeit,\n\nwenn du anderer Meinung bist, kannst du gerne zurück wechseln', + "u_pott": "

Dateien:   {0} fertig,   {1} fehlgeschlagen,   {2} in Bearbeitung,   {3} ausstehend

", + "u_ever": "Dies ist der Basic Uploader; up2k benötigt mind.
Chrome 21 // Firefox 13 // Edge 12 // Opera 12 // Safari 5.1", + "u_su2k": 'Dies ist der Basic Uploader; up2k ist besser', + "u_uput": 'Für Geschwindigkeit optimieren (Checksum überspringen)', + "u_ewrite": 'Du hast kein Schreibzugriff auf diesen Ordner', + "u_eread": 'Du hast kein Lesezugriff auf diesen Ordner', + "u_enoi": 'file-search ist in der Serverkonfiguration nicht aktiviert', + "u_enoow": "Überschreiben wird hier nicht funktionieren; benötige Lösch-Berechtigung", + "u_badf": 'Diese {0} Dateien (von insgesammt {1}) wurden übersprungen, wahrscheinlich wegen Dateisystem-Berechtigungen:\n\n', + "u_blankf": 'Diese {0} Dateien (von insgesammt {1}) sind leer; trotzdem hochladen?\n\n', + "u_applef": 'Diese {0} Dateien (von insgesammt {1}) sind möglicherweise unerwünscht;\nOK/Eingabe drücken, um die folgenden Dateien zu überspringen.\nDrücke Abbrechen/ESC um sie NICHT zu überspringen und diese AUCH HOCHZULADEN:\n\n', + "u_just1": '\nFunktioniert vielleicht besser, wenn du nur eine Datei auswählst', + "u_ff_many": "Falls du Linux / MacOS / Android benutzt, könnte Firefox mit dieser Menge an Dateien crashen!\nFalls das passiert, probier nochmal (oder benutz Chrome).", + "u_up_life": "Dieser Upload wird vom Server gelöscht\n{0} nachdem er abgeschlossen ist", + "u_asku": 'Diese {0} Dateien nach {1} hochladen', + "u_unpt": "Du kannst diesen Upload rückgängig machen mit dem 🧯 oben-links", + "u_bigtab": 'Versuche {0} Dateien anzuzeigen.\n\nDas könnte dein Browser crashen, bist du dir wirklich sicher?', + "u_scan": 'Scanne Dateien...', + "u_dirstuck": 'Ordner-Iterator blieb hängen beim Versuch, diese {0} Einträge zu lesen; überspringe:', + "u_etadone": 'Fertig ({0}, {1} Dateien)', + "u_etaprep": '(Upload wird vorbereitet)', + "u_hashdone": 'Hashing vollständig', + "u_hashing": 'Hash', + "u_hs": 'Wir schütteln uns die Hände ("handshaking")...', + "u_started": "Dateien werden hochgeladen; siehe [🚀]", + "u_dupdefer": "Duplikat; wird nach allen anderen Dateien verarbeitet", + "u_actx": "Klicke diesen Text um Performance-
Einbusen zu Vermeiden beim Wechsel auf andere Fenster/Tabs", + "u_fixed": "OK!  Habs repariert 👍", + "u_cuerr": "failed to upload chunk {0} of {1};\nprobably harmless, continuing\n\nfile: {2}", + "u_cuerr2": "server rejected upload (chunk {0} of {1});\nwill retry later\n\nfile: {2}\n\nerror ", + "u_ehstmp": "versuche nochmal; siehe unten-rechts", + "u_ehsfin": "Der Server hat die Anfrage zum Abschluss des Uploads abgelehnt; versuche nochmal...", + "u_ehssrch": "Der Server hat die Anfrage zur Suche abgelehnt; versuche nochmal...", + "u_ehsinit": "Der Server hat die Anfrage zum Start des Uploads abgelehnt; versuche nochmal...", + "u_eneths": "Netzwerkfehler beim Upload-Handshake; versuche nochmal...", + "u_enethd": "Netzwerkfehler beim Testen der Existenz des Ziels; versuche nochmal...", + "u_cbusy": "Der Server mag uns grade nicht mehr nach einem Netzwerkglitch, warte einen Moment...", + "u_ehsdf": "Server hat kein Speicherplatz mehr!\n\nwerde es erneut versuchen, falls jemand\ngenug Platz schafft um fortzufahren", + "u_emtleak1": "scheint, als ob dein Browser ein Memory Leak hätte;\nbitte", + "u_emtleak2": ' wechsle auf HTTPS (empfohlen) oder ', + "u_emtleak3": '', + "u_emtleakc": 'versuche folgendes:\n
  • drücke F5 um die Seite neu zu laden
  • deaktivere dann den  mt  Button in den  ⚙️ Einstellungen
  • und versuche den Upload nochmal.
Uploads werden etwas langsamer sein, aber man kann ja nicht alles haben.\nSorry für die Umstände !\n\nPS: Chrome v107 hat ein Bugfix dafür', + "u_emtleakf": 'versuche folgendes:\n
  • drücke F5 um die Seite neu zu laden
  • aktivere dann 🥔 (potato) im Upload UI
  • und versuche den Upload nochmal
\nPS: Firefox hat hoffentlich irgendwann ein Bugfix', + "u_s404": "nicht auf dem Server gefunden", + "u_expl": "erklären", + "u_maxconn": "die meisten Browser limitieren dies auf 6, aber Firefox lässt mehr zu unter connections-per-server in about:config", + "u_tu": '

WARNUNG: Turbo aktiviert,  Client könnte unvollständige Uploads verpassen und nicht wiederholen; siehe Turbo-Button Tooltip

', + "u_ts": '

WARNUNG: Turbo aktiviert,  Suchresultate können inkorrekt sein; siehe Turbo-Button Tooltip

', + "u_turbo_c": "Turbo deaktiviert in der Serverkonfiguration", + "u_turbo_g": "Turbo deaktiviert, da du keine Listen-Berechtigung\nauf diesem Volume hast", + "u_life_cfg": 'Autodelete nach min (or h)', + "u_life_est": 'Upload wird gelöscht ---', + "u_life_max": 'Dieser Ordner erzwingt eine\nmax Lebensdauer von {0}', + "u_unp_ok": 'unpost ist erlaubt für {0}', + "u_unp_ng": 'unpost wird NICHT erlaubt', + "ue_ro": 'Du hast nur Lese-Zugriff auf diesen Ordner\n\n', + "ue_nl": 'Du bist nicht angemeldet', + "ue_la": 'Du bist angemeldet als "{0}"', + "ue_sr": 'Du bist derzeit im Suchmodus\n\nWechsle zum Upload-Modus indem du auf die Lupe 🔎 klickst (neben dem grossen SUCHEN Button), und versuche den Upload nochmal.\n\nSorry', + "ue_ta": 'Versuche den Upload nochmal, sollte jetzt klappen', + "ue_ab": "Diese Datei wird gerade in einem anderen Ordner hochgeladen, dieser Upload muss zuerst abgeschlossen werden, bevor die Datei woanders hochgeladen werden kann.\n\nDu kannst den Upload abbrechen und vergessen mit dem 🧯 oben-links", + "ur_1uo": "OK: Datei erfolgreich hochgeladen", + "ur_auo": "OK: Alle {0} Dateien erfolgreich hochgeladen", + "ur_1so": "OK: Datei auf dem Server gefunden", + "ur_aso": "OK: Alle {0} Dateien auf dem Server gefunden", + "ur_1un": "Upload fehlgeschlagen, sorry", + "ur_aun": "Alle {0} Uploads fehlgeschlagen, sorry", + "ur_1sn": "Datei wurde NICHT auf dem Server gefunden", + "ur_asn": "Die {0} Dateien wurden NICHT auf dem Server gefunden", + "ur_um": "Fertig;\n{0} Uploads OK,\n{1} Uploads fehlgeschlagen, sorry", + "ur_sm": "Fertig;\n{0} Uploads gefunden auf dem Server,\n{1} Dateien NICHT gefunden auf dem Server", + + "lang_set": "Neuladen um Änderungen anzuwenden?", + }, }; -var LANGS = ["eng", "nor", "chi"]; +var LANGS = ["eng", "nor", "chi", "deu"]; if (window.langmod) langmod(); diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index f0c38b54..c2e60eb6 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -92,7 +92,49 @@ var Ls = { "ae1": "正在下载:", //m "af1": "显示最近上传的文件", //m "ag1": "查看已知 IdP 用户", //m - } + }, + + "deu": { + "a1": "Neu laden", + "b1": "Tach, wie geht's?   (Du bist nicht angemeldet)", + "c1": "Abmelden", + "d1": "Zustand", // TLNote: "d2" is the tooltip for this button + "d2": "Zeigt den Zustand aller aktiven Threads", + "e1": "Config neu laden", + "e2": "Konfigurationsdatei neu laden (Accounts/Volumes/VolFlags)$Nund scannt alle e2ds-Volumes$N$NBeachte: Jegliche Änderung an globalen Einstellungen$Nbenötigt einen Neustart zum Anwenden", + "f1": "Du kannst lesen:", + "g1": "Du kannst hochladen nach:", + "cc1": "Andere Dinge:", + "h1": "k304 deaktivieren", // TLNote: "j1" explains what k304 is + "i1": "k304 aktivieren", + "j1": "k304 trennt die Clientverbindung bei jedem HTTP 304, was Bugs mit problematischen Proxies vorbeugen kann (z.B. nicht ladenden Seiten), macht Dinge aber generell langsamer", + "k1": "Client-Einstellungen zurücksetzen", + "l1": "Melde dich an für mehr:", + "m1": "Willkommen zurück,", // TLNote: "welcome back, USERNAME" + "n1": "404 Nicht gefunden  ┐( ´ -`)┌", + "o1": 'or maybe you don\'t have access -- try a password or go home', + "p1": "403 Verboten  ~┻━┻", + "q1": 'Benutze ein Passwort oder gehe zur Homepage', + "r1": "Gehe zur Homepage", + ".s1": "Neu scannen", + "t1": "Aktion", // TLNote: this is the header above the "rescan" buttons + "u2": "time since the last server write$N( upload / rename / ... )$N$N17d = 17 days$N1h23 = 1 hour 23 minutes$N4m56 = 4 minutes 56 seconds", + "v1": "Verbinden", + "v2": "Benutze diesen Server als lokale Festplatte", + "w1": "Zu HTTPS wechseln", + "x1": "Passwort ändern", + "y1": "Shares bearbeiten", // TLNote: shows the list of folders that the user has decided to share + "z1": "Share entsperren:", // TLNote: the password prompt to see a hidden share + "ta1": "Trage zuerst dein Passwort ein", + "ta2": "Wiederhole dein Passwort zur Bestätigung:", + "ta3": "Da stimmt etwas nicht; probier's nochmal", + "aa1": "Eingehende Dateien:", + "ab1": "no304 deaktivieren", + "ac1": "no304 aktivieren", + "ad1": "Das Aktivieren von no304 deaktiviert jegliche Form von Caching; probier dies, wenn k304 nicht genug war. Dies verschwendet eine grosse Menge Netzwerk-Traffic!", + "ae1": "Aktive Downloads:", + "af1": "Zeige neue Uploads", + }, }; if (window.langmod) From 4f1eb89382ded1fe7368d920e243219cd5b40a45 Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 21:05:37 +0000 Subject: [PATCH 014/174] just moving some stuff around, not foreshadowing --- copyparty/__main__.py | 4 ++-- copyparty/svchub.py | 6 +++--- copyparty/util.py | 28 +++++++++++++++++----------- 3 files changed, 22 insertions(+), 16 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 461a57e7..f7e93b24 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -53,13 +53,13 @@ from .util import ( PYFTPD_VER, RAM_AVAIL, RAM_TOTAL, + RE_ANSI, SQLITE_VER, UNPLICATIONS, URL_BUG, URL_PRJ, Daemon, align_tab, - ansi_re, b64enc, dedent, has_resource, @@ -167,7 +167,7 @@ def lprint(*a: Any, **ka: Any) -> None: txt: str = " ".join(unicode(x) for x in a) + eol printed.append(txt) if not VT100: - txt = ansi_re.sub("", txt) + txt = RE_ANSI.sub("", txt) print(txt, end="", **ka) diff --git a/copyparty/svchub.py b/copyparty/svchub.py index 751fd566..73855bf1 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -51,6 +51,7 @@ from .util import ( HAVE_PSUTIL, HAVE_SQLITE3, HAVE_ZMQ, + RE_ANSI, URL_BUG, UTC, VERSIONS, @@ -60,7 +61,6 @@ from .util import ( HMaccas, ODict, alltrace, - ansi_re, build_netmap, expat_ver, gzip, @@ -1409,9 +1409,9 @@ class SvcHub(object): if self.no_ansi: fmt = "%s %-21s %s\n" if "\033" in msg: - msg = ansi_re.sub("", msg) + msg = RE_ANSI.sub("", msg) if "\033" in src: - src = ansi_re.sub("", src) + src = RE_ANSI.sub("", src) elif c: if isinstance(c, int): msg = "\033[3%sm%s\033[0m" % (c, msg) diff --git a/copyparty/util.py b/copyparty/util.py index 58a883d5..ae968777 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -243,7 +243,17 @@ except: BITNESS = struct.calcsize("P") * 8 -ansi_re = re.compile("\033\\[[^mK]*[mK]") +RE_ANSI = re.compile("\033\\[[^mK]*[mK]") +RE_CTYPE = re.compile(r"^content-type: *([^; ]+)", re.IGNORECASE) +RE_CDISP = re.compile(r"^content-disposition: *([^; ]+)", re.IGNORECASE) +RE_CDISP_FIELD = re.compile( + r'^content-disposition:(?: *|.*; *)name="([^"]+)"', re.IGNORECASE +) +RE_CDISP_FILE = re.compile( + r'^content-disposition:(?: *|.*; *)filename="(.*)"', re.IGNORECASE +) +RE_MEMTOTAL = re.compile("^MemTotal:.* kB") +RE_MEMAVAIL = re.compile("^MemAvailable:.* kB") BOS_SEP = ("%s" % (os.sep,)).encode("ascii") @@ -488,11 +498,11 @@ def read_ram() -> tuple[float, float]: with open("/proc/meminfo", "rb", 0x10000) as f: zsl = f.read(0x10000).decode("ascii", "replace").split("\n") - p = re.compile("^MemTotal:.* kB") + p = RE_MEMTOTAL zs = next((x for x in zsl if p.match(x))) a = int((int(zs.split()[1]) / 0x100000) * 100) / 100 - p = re.compile("^MemAvailable:.* kB") + p = RE_MEMAVAIL zs = next((x for x in zsl if p.match(x))) b = int((int(zs.split()[1]) / 0x100000) * 100) / 100 except: @@ -1698,14 +1708,10 @@ class MultipartParser(object): self.args = args self.headers = http_headers - self.re_ctype = re.compile(r"^content-type: *([^; ]+)", re.IGNORECASE) - self.re_cdisp = re.compile(r"^content-disposition: *([^; ]+)", re.IGNORECASE) - self.re_cdisp_field = re.compile( - r'^content-disposition:(?: *|.*; *)name="([^"]+)"', re.IGNORECASE - ) - self.re_cdisp_file = re.compile( - r'^content-disposition:(?: *|.*; *)filename="(.*)"', re.IGNORECASE - ) + self.re_ctype = RE_CTYPE + self.re_cdisp = RE_CDISP + self.re_cdisp_field = RE_CDISP_FIELD + self.re_cdisp_file = RE_CDISP_FILE self.boundary = b"" self.gen: Optional[ From b7ca6f4a66202ced70ebd87df169aebb41ebb81d Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 21:07:47 +0000 Subject: [PATCH 015/174] try to fix #300 the importlib stuff broke early versions of py2.7 --- copyparty/util.py | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/copyparty/util.py b/copyparty/util.py index ae968777..4e5cbfbe 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -4186,7 +4186,14 @@ def load_resource(E: EnvParams, name: str, mode="rb") -> IO[bytes]: stream = codecs.getreader(enc)(stream) return stream - return open(os.path.join(E.mod, name), mode, encoding=enc) + ap = os.path.join(E.mod, name) + + if PY2: + import codecs + + return codecs.open(ap, "r", encoding=enc) # type: ignore + + return open(ap, mode, encoding=enc) class Pebkac(Exception): From a8705e611d05eeb22be5d3d7d9ab5c020fe54c62 Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 21:19:39 +0000 Subject: [PATCH 016/174] fix GHSA-8mx2-rjh8-q3jq ; this fixes a DOM-Based XSS in the recent-uploads page: it was possible to execute arbitrary javascript by tricking someone into visiting `/?ru&filter=` huge thanks to @Ju0x for finding and reporting this! --- copyparty/httpcli.py | 3 ++- copyparty/util.py | 4 ++++ 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 0ff9c0ef..fa6d7418 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -81,6 +81,7 @@ from .util import ( html_escape, humansize, ipnorm, + json_hesc, justcopy, load_resource, loadpy, @@ -5595,7 +5596,7 @@ class HttpCli(object): self.reply(jtxt.encode("utf-8", "replace"), mime="application/json") return True - html = self.j2s("rups", this=self, v=jtxt) + html = self.j2s("rups", this=self, v=json_hesc(jtxt)) self.reply(html.encode("utf-8"), status=200) return True diff --git a/copyparty/util.py b/copyparty/util.py index 4e5cbfbe..94e0f249 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -2253,6 +2253,10 @@ def find_prefix(ips: list[str], cidrs: list[str]) -> list[str]: return ret +def json_hesc(s: str) -> str: + return s.replace("<", "\\u003c").replace(">", "\\u003e").replace("&", "\\u0026") + + def html_escape(s: str, quot: bool = False, crlf: bool = False) -> str: """html.escape but also newlines""" s = s.replace("&", "&").replace("<", "<").replace(">", ">") From 13d5631b4897423c3f97dd02b24691f3b1e2c138 Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 21:26:27 +0000 Subject: [PATCH 017/174] more escapes in case --- copyparty/authsrv.py | 3 ++- copyparty/httpcli.py | 22 ++++++++++------------ copyparty/util.py | 7 +++++++ copyparty/web/browser.html | 4 ++-- 4 files changed, 21 insertions(+), 15 deletions(-) diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 89cab889..28441114 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -33,6 +33,7 @@ from .util import ( afsenc, get_df, humansize, + json_hesc, min_ex, odfusion, read_utf8, @@ -2785,7 +2786,7 @@ class AuthSrv(object): "lifetime": vn.js_ls["lifetime"], "u2sort": self.args.u2sort, } - vn.js_htm = json.dumps(js_htm) + vn.js_htm = json_hesc(json.dumps(js_htm)) vols = list(vfs.all_nodes.values()) if enshare: diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index fa6d7418..bdbaced2 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -79,6 +79,7 @@ from .util import ( hidedir, html_bescape, html_escape, + html_sh_esc, humansize, ipnorm, json_hesc, @@ -4912,11 +4913,8 @@ class HttpCli(object): else: rip = host - # safer than html_escape/quotep since this avoids both XSS and shell-stuff - pw = re.sub(r"[<>&$?`\"']", "_", self.pw or "hunter2") - vp = re.sub(r"[<>&$?`\"']", "_", self.uparam["hc"] or "").lstrip("/") - pw = pw.replace(" ", "%20") - vp = vp.replace(" ", "%20") + vp = (self.uparam["hc"] or "").lstrip("/") + pw = self.pw or "hunter2" if pw in self.asrv.sesa: pw = "hunter2" @@ -4925,14 +4923,14 @@ class HttpCli(object): args=self.args, accs=bool(self.asrv.acct), s="s" if self.is_https else "", - rip=rip, - ep=ep, - vp=vp, - rvp=vjoin(self.args.R, vp), - host=host, - hport=hport, + rip=html_sh_esc(rip), + ep=html_sh_esc(ep), + vp=html_sh_esc(vp), + rvp=html_sh_esc(vjoin(self.args.R, vp)), + host=html_sh_esc(host), + hport=html_sh_esc(hport), aname=aname, - pw=pw, + pw=html_sh_esc(pw), ) self.reply(html.encode("utf-8")) return True diff --git a/copyparty/util.py b/copyparty/util.py index 94e0f249..d081378a 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -244,6 +244,7 @@ except: RE_ANSI = re.compile("\033\\[[^mK]*[mK]") +RE_HTML_SH = re.compile(r"[<>&$?`\"';]") RE_CTYPE = re.compile(r"^content-type: *([^; ]+)", re.IGNORECASE) RE_CDISP = re.compile(r"^content-disposition: *([^; ]+)", re.IGNORECASE) RE_CDISP_FIELD = re.compile( @@ -2253,6 +2254,12 @@ def find_prefix(ips: list[str], cidrs: list[str]) -> list[str]: return ret +def html_sh_esc(s: str) -> str: + s = re.sub(RE_HTML_SH, "_", s).replace(" ", "%20") + s = s.replace("\r", "_").replace("\n", "_") + return s + + def json_hesc(s: str) -> str: return s.replace("<", "\\u003c").replace(">", "\\u003e").replace("&", "\\u0026") diff --git a/copyparty/web/browser.html b/copyparty/web/browser.html index a642e9a9..05bdacbc 100644 --- a/copyparty/web/browser.html +++ b/copyparty/web/browser.html @@ -109,8 +109,8 @@ {%- for f in files %} {{ f.lead }}{{ f.name|e }}{{ f.sz }} {%- if f.tags is defined %} - {%- for k in taglist %}{{ f.tags[k] }}{%- endfor %} -{%- endif %}{{ f.ext }}{{ f.dt }} + {%- for k in taglist %}{{ f.tags[k]|e }}{%- endfor %} +{%- endif %}{{ f.ext|e }}{{ f.dt }} {%- endfor %} From daa44be1a595374cddecde6c2de5f925fbe8fd92 Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 21:31:54 +0000 Subject: [PATCH 018/174] v1.18.7 --- copyparty/__main__.py | 2 +- copyparty/__version__.py | 4 ++-- copyparty/web/browser.js | 2 +- scripts/pyinstaller/build.sh | 1 - 4 files changed, 4 insertions(+), 5 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index f7e93b24..97944c13 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1557,7 +1557,7 @@ def add_ui(ap, retry): ap2.add_argument("--hsortn", metavar="N", type=int, default=2, help="number of sorting rules to include in media URLs by default (volflag=hsortn)") ap2.add_argument("--see-dots", action="store_true", help="default-enable seeing dotfiles; only takes effect if user has the necessary permissions") ap2.add_argument("--qdel", metavar="LVL", type=int, default=2, help="number of confirmations to show when deleting files (2/1/0)") - ap2.add_argument("--unlist", metavar="REGEX", type=u, default="", help="don't show files matching \033[33mREGEX\033[0m in file list. Purely cosmetic! Does not affect API calls, just the browser. Example: [\033[32m\\.(js|css)$\033[0m] (volflag=unlist)") + ap2.add_argument("--unlist", metavar="REGEX", type=u, default="", help="don't show files/folders matching \033[33mREGEX\033[0m in file list. WARNING: Purely cosmetic! Does not affect API calls, just the browser. Example: [\033[32m\\.(js|css)$\033[0m] (volflag=unlist)") ap2.add_argument("--favico", metavar="TXT", type=u, default="c 000 none" if retry else "🎉 000 none", help="\033[33mfavicon-text\033[0m [ \033[33mforeground\033[0m [ \033[33mbackground\033[0m ] ], set blank to disable") ap2.add_argument("--ext-th", metavar="E=VP", type=u, action="append", help="use thumbnail-image \033[33mVP\033[0m for file-extension \033[33mE\033[0m, example: [\033[32mexe=/.res/exe.png\033[0m] (volflag=ext_th)") ap2.add_argument("--mpmc", metavar="URL", type=u, default="", help="change the mediaplayer-toggle mouse cursor; URL to a folder with {2..5}.png inside (or disable with [\033[32m.\033[0m])") diff --git a/copyparty/__version__.py b/copyparty/__version__.py index 3cbb3b4a..2006cc7a 100644 --- a/copyparty/__version__.py +++ b/copyparty/__version__.py @@ -1,8 +1,8 @@ # coding: utf-8 -VERSION = (1, 18, 6) +VERSION = (1, 18, 7) CODENAME = "logtail" -BUILD_DT = (2025, 7, 28) +BUILD_DT = (2025, 7, 30) S_VERSION = ".".join(map(str, VERSION)) S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index fc63bf9f..59b1ab5f 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -2393,7 +2393,7 @@ var Ls = { "fu_xe2": "404: Datei nicht gefunden??", "fz_tar": "Unkomprimierte GNU TAR-Datei (Linux / Mac)", - "fz_pax": "Unkomprimierte pax-format TAR-Datei (slower)", + "fz_pax": "Unkomprimierte pax-format TAR-Datei (etwas langsamer)", //m "fz_targz": "GNU-TAR mit gzip Level 3 Kompression$N$Nüblicherweise recht langsam,$Nbenutze stattdessen ein unkomprimiertes TAR", "fz_tarxz": "GNU-TAR mit xz level 1 Kompression$N$Nüblicherweise recht langsam,$Nbenutze stattdessen ein unkomprimiertes TAR", "fz_zip8": "ZIP mit UTF8-Dateinamen (könnte kaputt gehen auf Windows 7 oder älter)", diff --git a/scripts/pyinstaller/build.sh b/scripts/pyinstaller/build.sh index bb1d477e..a444ff94 100644 --- a/scripts/pyinstaller/build.sh +++ b/scripts/pyinstaller/build.sh @@ -14,7 +14,6 @@ clean=--clean uname -s | grep WOW64 && m=64 || m=32 uname -s | grep NT-10 && w10=1 || w7=1 -[ $w7 ] && export PRTY_NO_MAGIC=1 [ $w7 ] && [ -e up2k.sh ] && [ ! "$1" ] && ./up2k.sh [ $w7 ] && pyv=37 || pyv=313 From 5df2cbe5d76db8aee7d37a92b5a2670ef1084044 Mon Sep 17 00:00:00 2001 From: ed Date: Wed, 30 Jul 2025 21:59:58 +0000 Subject: [PATCH 019/174] update pkgs to 1.18.7 --- contrib/package/arch/PKGBUILD | 4 ++-- contrib/package/nix/copyparty/pin.json | 6 +++--- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/contrib/package/arch/PKGBUILD b/contrib/package/arch/PKGBUILD index 959a9821..2fe085ff 100644 --- a/contrib/package/arch/PKGBUILD +++ b/contrib/package/arch/PKGBUILD @@ -1,6 +1,6 @@ # Maintainer: icxes pkgname=copyparty -pkgver="1.18.6" +pkgver="1.18.7" pkgrel=1 pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" arch=("any") @@ -22,7 +22,7 @@ optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tag ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") backup=("etc/${pkgname}.d/init" ) -sha256sums=("80762d91ac88815e73d0ca2806c6391dcf8ccd521bc402cc312349f3bc8e8b28") +sha256sums=("6738f623905276e8664bd8791f1497d0c61f8816c6608e76c4d9097a12849170") build() { cd "${srcdir}/${pkgname}-${pkgver}" diff --git a/contrib/package/nix/copyparty/pin.json b/contrib/package/nix/copyparty/pin.json index eae921c7..966def8e 100644 --- a/contrib/package/nix/copyparty/pin.json +++ b/contrib/package/nix/copyparty/pin.json @@ -1,5 +1,5 @@ { - "url": "https://github.com/9001/copyparty/releases/download/v1.18.6/copyparty-sfx.py", - "version": "1.18.6", - "hash": "sha256-No89mzKHHZZH19ws9dqfvQO0pnZw7jKDMGhNa4LOFlY=" + "url": "https://github.com/9001/copyparty/releases/download/v1.18.7/copyparty-sfx.py", + "version": "1.18.7", + "hash": "sha256-GRoiNnYRWNcq5ITywPv7bK4pdgTey6Or1cZqyE2XDf4=" } \ No newline at end of file From c17ce4892ecdb4e11437bc2785d132bd8100eaec Mon Sep 17 00:00:00 2001 From: ed Date: Thu, 31 Jul 2025 08:32:52 +0000 Subject: [PATCH 020/174] fix pkgres on older python3 versions --- copyparty/util.py | 2 -- 1 file changed, 2 deletions(-) diff --git a/copyparty/util.py b/copyparty/util.py index d081378a..73284bbb 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -4200,8 +4200,6 @@ def load_resource(E: EnvParams, name: str, mode="rb") -> IO[bytes]: ap = os.path.join(E.mod, name) if PY2: - import codecs - return codecs.open(ap, "r", encoding=enc) # type: ignore return open(ap, mode, encoding=enc) From dccef40f3dbd3caad45ce11fb6e86cc01675cc39 Mon Sep 17 00:00:00 2001 From: ed Date: Thu, 31 Jul 2025 08:33:34 +0000 Subject: [PATCH 021/174] v1.18.8 --- copyparty/__version__.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/copyparty/__version__.py b/copyparty/__version__.py index 2006cc7a..cd73581d 100644 --- a/copyparty/__version__.py +++ b/copyparty/__version__.py @@ -1,8 +1,8 @@ # coding: utf-8 -VERSION = (1, 18, 7) +VERSION = (1, 18, 8) CODENAME = "logtail" -BUILD_DT = (2025, 7, 30) +BUILD_DT = (2025, 7, 31) S_VERSION = ".".join(map(str, VERSION)) S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) From cd460902b0d5e55c059677767e3016437424d905 Mon Sep 17 00:00:00 2001 From: ed Date: Thu, 31 Jul 2025 08:45:18 +0000 Subject: [PATCH 022/174] update pkgs to 1.18.8 --- contrib/package/arch/PKGBUILD | 4 ++-- contrib/package/nix/copyparty/pin.json | 6 +++--- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/contrib/package/arch/PKGBUILD b/contrib/package/arch/PKGBUILD index 2fe085ff..92423af8 100644 --- a/contrib/package/arch/PKGBUILD +++ b/contrib/package/arch/PKGBUILD @@ -1,6 +1,6 @@ # Maintainer: icxes pkgname=copyparty -pkgver="1.18.7" +pkgver="1.18.8" pkgrel=1 pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" arch=("any") @@ -22,7 +22,7 @@ optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tag ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") backup=("etc/${pkgname}.d/init" ) -sha256sums=("6738f623905276e8664bd8791f1497d0c61f8816c6608e76c4d9097a12849170") +sha256sums=("d8af012f552a6f1dfc501d0ce94990b932d422df74f6afb66ca2a31985f5809b") build() { cd "${srcdir}/${pkgname}-${pkgver}" diff --git a/contrib/package/nix/copyparty/pin.json b/contrib/package/nix/copyparty/pin.json index 966def8e..1afe1c79 100644 --- a/contrib/package/nix/copyparty/pin.json +++ b/contrib/package/nix/copyparty/pin.json @@ -1,5 +1,5 @@ { - "url": "https://github.com/9001/copyparty/releases/download/v1.18.7/copyparty-sfx.py", - "version": "1.18.7", - "hash": "sha256-GRoiNnYRWNcq5ITywPv7bK4pdgTey6Or1cZqyE2XDf4=" + "url": "https://github.com/9001/copyparty/releases/download/v1.18.8/copyparty-sfx.py", + "version": "1.18.8", + "hash": "sha256-ZQtpGVcC6fmJCC65hXN4v3ERiB3jPz+oqvFHtbxR8+o=" } \ No newline at end of file From 0b05c726de914e52a9931fcba32c9a5da13f04ba Mon Sep 17 00:00:00 2001 From: Altair <27139654+A1Asriel@users.noreply.github.com> Date: Thu, 31 Jul 2025 20:01:41 +0300 Subject: [PATCH 023/174] Translate to Russian (#321) --- copyparty/web/browser.js | 628 ++++++++++++++++++++++++++++++++++++++- copyparty/web/splash.js | 42 +++ 2 files changed, 669 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 59b1ab5f..db9986f2 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -2510,9 +2510,635 @@ var Ls = { "lang_set": "Neuladen um Änderungen anzuwenden?", }, + "rus": { + "tt": "Русский", + + "cols": { + "c": "кнопки действий", + "dur": "длительность", + "q": "качество / битрейт", + "Ac": "аудио кодек", + "Vc": "видео кодек", + "Fmt": "формат / контейнер", + "Ahash": "контрольная сумма аудио", + "Vhash": "контрольная сумма видео", + "Res": "разрешение", + "T": "тип файла", + "aq": "качество аудио / битрейт", + "vq": "качество видео / битрейт", + "pixfmt": "сабсемплинг / пиксельный формат", + "resw": "горизонтальное разрешение", + "resh": "вертикальное разрешение", + "chs": "аудио каналы", + "hz": "частота дискретизации" + }, + + "hks": [ + [ + "разное", + ["ESC", "закрыть всякие штуки"], + + "файловый менеджер", + ["G", "переключиться между списком / плиткой"], + ["T", "переключиться между миниатюрами / иконками"], + ["⇧ A/D", "размер миниатюры"], + ["ctrl-K", "удалить выделенное"], + ["ctrl-X", "вырезать выделенное в буфер"], + ["ctrl-C", "копировать выделенное в буфер"], + ["ctrl-V", "вставить (переместить/копировать) сюда"], + ["Y", "скачать выделенное"], + ["F2", "переименовать выделенное"], + + "выделение файлов", + ["пробел", "выделить/снять выделение с текущего файла"], + ["↑/↓", "двигать курсор"], + ["ctrl ↑/↓", "двигать курсор и скроллить"], + ["⇧ ↑/↓", "выделить предыдущий/следующий файл"], + ["ctrl-A", "выделить все файлы и папки"], + ], [ + "навигация", + ["B", "показать/скрыть панель навигации"], + ["I/K", "предыдущая/следующая папка"], + ["M", "перейти на уровень выше (или свернуть текущую папку)"], + ["V", "переключиться между папками / текстовыми файлами в панели навигации"], + ["A/D", "уменьшить/увеличить панель навигации"], + ], [ + "аудиоплеер", + ["J/L", "предыдущий/следующий трек"], + ["U/O", "перемотать на 10 секунд назад/вперёд"], + ["0..9", "перемотать на 0%..90%"], + ["P", "играть/пауза (или подгрузить трек)"], + ["S", "выделить текущий трек"], + ["Y", "скачать трек"], + ], [ + "просмотрщик изображений", + ["J/L, ←/→", "предыдущее/следующее изображение"], + ["Home/End", "первое/последнее изображение"], + ["F", "развернуть на полный экран"], + ["R", "повернуть по часовой стрелке"], + ["⇧ R", "повернуть против часовой стрелки"], + ["S", "выделить изображение"], + ["Y", "скачать изображение"], + ], [ + "видеоплеер", + ["U/O", "перемотать на 10 секунд назад/вперёд"], + ["P/K/Space", "играть/пауза"], + ["C", "продолжить проигрывать следующее"], + ["V", "повтор"], + ["M", "выключить звук"], + ["[ and ]", "задать интервал повтора"], + ], [ + "просмотрщик текстовых файлов", + ["I/K", "предыдущий/следующий файл"], + ["M", "закрыть файл"], + ["E", "отредактировать файл"], + ["S", "выделить файл"], + ] + ], + + "m_ok": "OK", + "m_ng": "Отмена", + + "enable": "Включить", + "danger": "ВНИМАНИЕ", + "clipped": "скопировано в буфер обмена", + + "ht_s1": "секунда", + "ht_s2": "секунды", + "ht_m1": "минута", + "ht_m2": "минуты", + "ht_h1": "час", + "ht_h2": "часы", + "ht_d1": "день", + "ht_d2": "дни", + "ht_and": " и ", + + "goh": "панель управления", + "gop": 'предыдущая папка">пред', + "gou": 'родительская папка">вверх', + "gon": 'следующая папка">след', + "logout": "Выйти ", + "access": " доступ", + "ot_close": "закрыть подменю", + "ot_search": "искать файлы по атрибутам, пути / имени, музыкальным тегам или любой другой комбинации из следующих конструкций$N$N<code>foo bar</code> = обязано содержать «foo» И «bar»,$N<code>foo -bar</code> = обязано содержать «foo», но не «bar»,$N<code>^yana .opus$</code> = начинается с «yana» и имеет расширение «opus»$N<code>"try unite"</code> = содержит именно «try unite»$N$Nформат времени задаётся по стандарту iso-8601, например$N<code>2009-12-31</code> или <code>2020-09-12 23:30:00</code>", + "ot_unpost": "unpost: удалить ваши недавние загрузки и отменить незавершённые", + "ot_bup": "bup: легковесный загрузчик файлов, поддерживает даже netscape 4.0", + "ot_mkdir": "mkdir: создать новую папку", + "ot_md": "new-md: создать новый markdown-документ", + "ot_msg": "msg: отправить сообщение в лог сервера", + "ot_mp": "настройка медиаплеера", + "ot_cfg": "остальные настройки", + "ot_u2i": 'up2k: загрузить файлы (если имеется доступ к записи) или переключиться в режим поиска$N$Nзагрузки являются возобновляемыми и многопоточными, а даты изменения файлов сохраняются в процессе, но этот метод использует больше ресурсов процессора, чем [🎈]  (легковесный загрузчик)

во время загрузки эта иконка превращается в индикатор!', + "ot_u2w": 'up2k: загрузить файлы с поддержкой возобновления (закиньте те же файлы после перезапуска браузера)$N$Nподдерживается многопоточность, даты изменения файлов сохраняются в процессе, но этот метод использует больше ресурсов процессора, чем [🎈]  (легковесный загрузчик)

во время загрузки эта иконка превращается в индикатор!', + "ot_noie": 'Пожалуйста, используйте Chrome / Firefox / Edge', + + "ab_mkdir": "создать папку", + "ab_mkdoc": "создать markdown-документ", + "ab_msg": "отправить сообщение в лог сервера", + + "ay_path": "перейти к папкам", + "ay_files": "перейти к файлам", + + "wt_ren": "переименовать выделенные файлы$NГорячая клавиша: F2", + "wt_del": "удалить выделенные файлы$NГорячая клавиша: ctrl-K", + "wt_cut": "вырезать выделенные файлы <small>(затем вставить куда-то в другое место)</small>$NГорячая клавиша: ctrl-X", + "wt_cpy": "копировать выделенные файлы в буфер$N(чтобы вставить их куда-то ещё)$NГорячая клавиша: ctrl-C", + "wt_pst": "вставить ранее вырезанный / скопированный файл$NГорячая клавиша: ctrl-V", + "wt_selall": "выделить все файлы$NГорячая клавиша: ctrl-A (когда выделен хотя бы один файл)", + "wt_selinv": "инвертировать выделение", + "wt_zip1": "скачать эту папку как архив", + "wt_selzip": "скачать выделенные файлы как архив", + "wt_seldl": "скачать выделенные файлы по-отдельности$NГорячая клавиша: Y", + "wt_npirc": "копировать информацию о треке в формате irc", + "wt_nptxt": "копировать информацию о треке обычным текстом", + "wt_m3ua": "добавить в плейлист m3u (нажмите 📻коп. в конце)", + "wt_m3uc": "копировать плейлист m3u в буфер обмена", + "wt_grid": "переключить между сеткой / списком$NГорячая клавиша: G", + "wt_prev": "предыдущий трек$NГорячая клавиша: J", + "wt_play": "играть / пауза$NГорячая клавиша: P", + "wt_next": "следующий трек$NГорячая клавиша: L", + + "ul_par": "параллельные загрузки:", + "ut_rand": "случайные имена файлов", + "ut_u2ts": "копировать время последнего изменения$Nиз вашей файловой системы на сервер\">📅", + "ut_ow": "перезаписывать существующие файлы на сервере?$N🛡️: нет (для повторяющихся файлов будут создаваться новые имена)$N🕒: перезаписать файлы с датой изменения старее, чем у загружаемых$N♻️: всегда перезаписывать (если файлы различаются по содержанию)", + "ut_mt": "продолжать хешировать другие файлы во время загрузки$N$Nесть смысл отключить при медленном диске или процессоре", + "ut_ask": 'требовать подтверждения перед началом загрузки">💭', + "ut_pot": "улучшить скорость загрузки на слабых устройства$Nс помощью упрощения интерфейса", + "ut_srch": "не загружать, а проверять, существуют ли данные файлы $N на сервере (проверка всех доступных вам папок)", + "ut_par": "при 0 загрузка встанет на паузу$N$Nследует повысить, если ваше подключение медленное$N$Nоставьте 1, если используется локальная сеть или диск сервера медленный", + "ul_btn": "отпустите файлы / папки
здесь (или нажмите)", + "ul_btnu": "З А Г Р У З И Т Ь", + "ul_btns": "И С К А Т Ь", + + "ul_hash": "хеш", + "ul_send": "отправка", + "ul_done": "готово", + "ul_idle1": "нет загрузок в очереди", + "ut_etah": "средняя скорость <em>хеширования</em> и примерное время до завершения", + "ut_etau": "средняя скорость <em>загрузки</em> и примерное время до завершения", + "ut_etat": "средняя <em>общая</em> скорость и примерное время до завершения", + + "uct_ok": "успешно завершены", + "uct_ng": "ошибки / отказы / не найдены", + "uct_done": "готово (ok и ng вместе)", + "uct_bz": "хешируются или загружаются", + "uct_q": "в очереди", + + "utl_name": "имя файла", + "utl_ulist": "список", + "utl_ucopy": "копировать", + "utl_links": "ссылки", + "utl_stat": "статус", + "utl_prog": "прогресс", + + // keep short: + "utl_404": "404", + "utl_err": "ОШИБКА", + "utl_oserr": "ошибка ОС", + "utl_found": "найдено", + "utl_defer": "отложить", + "utl_yolo": "турбо", + "utl_done": "готово", + + "ul_flagblk": "файлы были добавлены в очередь,
однако в другой вкладке уже есть активная загрузка через up2k,
поэтому ожидаем её завершения", + "ul_btnlk": "настройки сервера запрещают изменение состояния этой опции", + + "udt_up": "Загрузить", + "udt_srch": "Поиск", + "udt_drop": "отпустите здесь", + + "u_nav_m": '
лады, что там у вас?
Enter = Файлы (один или больше)\nESC = Одна папка (с учётом подпапок)', + "u_nav_b": 'ФайлыОдна папка', + + "cl_opts": "переключатели", + "cl_themes": "тема", + "cl_langs": "язык", + "cl_ziptype": "архивация папок", + "cl_uopts": "опции up2k", + "cl_favico": "иконка", + "cl_bigdir": "бол. папки", + "cl_hsort": "#сорт.", + "cl_keytype": "схема горячих клавиш", + "cl_hiddenc": "скрытые столбцы", + "cl_hidec": "скрыть", + "cl_reset": "сбросить", + "cl_hpick": "нажмите на заголовки столбцов, чтобы скрыть их в таблице ниже", + "cl_hcancel": "скрытие столбца отменено", + + "ct_grid": '田 сетка', + "ct_ttips": '◔ ◡ ◔">ℹ️ подсказки', + "ct_thumb": 'переключение между иконками и миниатюрами в режиме сетки$NГорячая клавиша: T">🖼️ миниат.', + "ct_csel": 'держите CTRL или SHIFT для выделения файлов в режиме сетки">выбор', + "ct_ihop": 'показывать последний открытый файл после закрытия просмотрщика изображений">g⮯', + "ct_dots": 'показывать скрытые файлы (если есть доступ)">скрыт.', + "ct_qdel": 'спрашивать подтверждение только один раз перед удалением файлов">быстр. удал.', + "ct_dir1st": 'разместить папки над файлами">📁 сверху', + "ct_nsort": 'сортировка по числам$N(например, файл с 2 в начале названия идёт перед 11)">нат. сорт.', + "ct_readme": 'показывать содержимое README.md в описании папки">📜 ридми', + "ct_idxh": 'показывать страницу index.html в текущей папке вместо интерфейса">htm', + "ct_sbars": 'показывать полосы прокрутки">⟊', + + "cut_umod": "если файл уже существует на сервере, обновить время последнего изменения на сервере в соответствии с локальным файлом (требуются права write+delete)\">перенос 📅", + + "cut_turbo": "используйте эту функцию С ОСТОРОЖНОСТЬЮ:$N$Nпригодится в случае, если была прервана загрузка большого количества файлов, и вы хотите возобновить её как можно быстрее$N$Nпроверка файлов по хешу заменяется на простой алгоритм: "если размер файла отличается - тогда загрузить", но содержание файлов не сравнивается$N$Nследует отключить эту функцию после окончания загрузки, а затем "загрузить" те же файлы заново, чтобы валидировать их\">турбо", + + "cut_datechk": "работает только при включённой кнопке "турбо"$N$Nчуть-чуть повышает надёжность турбо-загрузок с помощью сверки дат изменений между файлами на сервере и вашими$N$Nв теории достаточно для проверки большинства незавершённых / повреждённых загрузок, но не является альтернативой валидации файлов после турбо-загрузки\">провер. дат", + + "cut_u2sz": "размер (в МиБ) каждой загружаемой части; большие значения показывают лучшие результаты для дальних соединений. Если подключение нестабильное, попробуйте значение пониже", + + "cut_flag": "разрешить одновременную загрузку только из одной вкладки за раз $N -- обязательно включить эту опцию в остальных вкладках $N -- работает только в пределах одного домена", + + "cut_az": "загружать файлы в алфавитном порядке вместо "от меньшего к большему"$N$Nэто позволит проще отследить проблемы во время загрузки, но скорость слегка ниже на очень быстрых соединениях (например, в локальной сети)", + + "cut_nag": "системное уведомление по завершении загрузки$N(только при неактивной вкладке браузера)", + "cut_sfx": "звуковое уведомление по завершении загрузки$N(только при неактивной вкладке браузера)", + + "cut_mt": "использовать многопоточность для ускорения хеширования$N$Nиспользует Web Worker'ы и требует больше памяти (до 512 МиБ)$N$Nускоряет https на 30%, http - в 4,5 раз\">мп", + + "cut_wasm": "использовать модуль WASM вместо встроенной в браузер функции хеширования; ускоряет процесс в браузерах на основе Chromium, но увеличивает нагрузку на процессор. Старые версии Chrome содержат баги, которые заполняют всю оперативную память и крашат браузер, когда включена эта опция\">wasm", + + "cft_text": "текст для иконки (очистите поле и перезагрузите страницу для применения)", + "cft_fg": "цвет текста", + "cft_bg": "цвет фона", + + "cdt_lim": "максимальное количество файлов для показа в папке", + "cdt_ask": "внизу страницы спрашивать о действии вместо автоматической загрузки следующих файлов", + "cdt_hsort": "сколько правил сортировки (<code>,sorthref</code>) включать в адрес страницы. Если значение равно 0, по нажатии на ссылки будут игнорироваться правила, включённые в них", + + "tt_entree": "показать панель навигации$NГорячая клавиша: B", + "tt_detree": "скрыть панель навигации$NГорячая клавиша: B", + "tt_visdir": "прокрутить до выделенной папки", + "tt_ftree": "переключить между иерархией и списком текстовых файлов$NГорячая клавиша: V", + "tt_pdock": "закрепить родительские папки сверху панели", + "tt_dynt": "автоматическое расширение панели", + "tt_wrap": "перенос слов", + "tt_hover": "раскрывать обрезанные строки при наведении$N( ломает скроллинг, если $N  курсор не в пустоте слева )", + + "ml_pmode": "в конце папки...", + "ml_btns": "команды", + "ml_tcode": "транскодировать", + "ml_tcode2": "транскод. в", + "ml_tint": "затемн.", + "ml_eq": "эквалайзер", + "ml_drc": "компрессор", + + "mt_loop": "повторять один трек\">🔁", + "mt_one": "остановить после этого трека\">1️⃣", + "mt_shuf": "перемешать треки во всех папках\">🔀", + "mt_aplay": "автоматически играть треки по нажатии на ссылки с их ID$N$Nпри отключении адрес сайта также перестанет обновляться в соответствии с текущим треком\">a▶", + "mt_preload": "подгружать следующий трек перед концом текущего для бесшовного переключения\">предзагр.", + "mt_prescan": "переходить в следующую папку перед окончанием последнего трека$Nне даёт браузеру прервать следующий плейлист\">нав.", + "mt_fullpre": "подгружать следующий трек целиком;$N✅ полезно при нестабильном подключении,$N❌ при медленной скорости лучше выключить\">цел.", + "mt_fau": "для телефонов: начинать следующий трек сразу, даже если он не успел подгрузиться целиком (может сломать отображение тегов)\">☕️", + "mt_waves": "визуализация:$Nпоказывать волну громкости на полосе воспроизведения\">~", + "mt_npclip": "показать кнопки копирования для текущего трека\">/np", + "mt_m3u_c": "показать кнопки копирования для выделенных треков$Nв формате плейлистов m3u8\">📻", + "mt_octl": "интеграция с ОС (поддержка медиа-клавиш и музыкальных виджетов)\">интегр.", + "mt_oseek": "позволить перематывать треки через системные виджеты$N$Nвнимание: на некоторых устройствах (iPhone)$Nэто заменит кнопку следующего трека\">перемотка", + "mt_oscv": "показывать картинки альбомов в виджетах\">арт", + "mt_follow": "держать фокус на играющем треке\">🎯", + "mt_compact": "компактный плеер\">⟎", + "mt_uncache": "очистить кеш  (если браузер кешировал повреждённый$Nтрек и отказывается его запускать)\">уд. кеш", + "mt_mloop": "повторять треки в папке\">🔁 цикл", + "mt_mnext": "загрузить следующую папку и продолжить в ней\">📂 след.", + "mt_mstop": "приостановить воспроизведение\">⏸ стоп", + "mt_cflac": "конвертировать flac / wav в opus\">flac", + "mt_caac": "конвертировать aac / m4a в opus\">aac", + "mt_coth": "конвертировать всё остальное (кроме mp3) в opus\">др.", + "mt_c2opus": "лучший вариант для компьютеров и устройств на Android\">opus", + "mt_c2owa": "opus-weba, для iOS 17.5 и выше\">owa", + "mt_c2caf": "opus-caf, для iOS 11-17\">caf", + "mt_c2mp3": "для очень старых устройств\">mp3", + "mt_c2ok": "хороший выбор", + "mt_c2nd": "это не рекомендованный вариант формата для вашего устройства, но сойдёт", + "mt_c2ng": "не похоже, что ваше устройство поддерживает этот формат, но давайте попробуем и узнаем наверняка", + "mt_xowa": "в iOS есть баги, препятствующие фоновому воспроизведению этого формата. Пожалуйста, используйте caf или mp3", + "mt_tint": "непрозрачность фона (0-100) на полосе воспроизведения$N$Nделает буферизацию менее отвлекающей", + "mt_eq": "включить эквалайзер$N$Nboost <code>0</code> = стандартная громкость$N$Nwidth <code>1  </code> = обычное стерео$Nwidth <code>0.5</code> = микширование левого и правого каналов на 50%$Nwidth <code>0  </code> = моно$N$Nboost <code>-0.8</code> и width <code>10</code> = удаление голоса :^)$N$Nвключённый эквалайзер полностью убирает задержку между треками, поэтому следует его включить со всеми значениями на 0 (кроме width = 1), если вам нужно бесшовное воспроизведение", + "mt_drc": "включить компрессор; также включит эквалайзер для баланса вселенной, так что выставьте всё на 0 кроме width, если он вам не нужен$N$Nпонижает громкость при волне выше значения dB в tresh; каждый dB в ratio равен одному dB на выходе, так что стандартные значения tresh = -24 и ratio = 12 сделают так, что звук никогда не будет громче -22 dB. При таком раскладе можно поставить boost в эквалайзере на 0.8 или даже 1.8 при значении atk = 0 и огромном rls вроде 90 (работает только в Firefox, rls не может быть выше 1 в других браузерах)$N$N(загляните в википедию, там всё объяснено подробнее)", + + "mb_play": "играть", + "mm_hashplay": "воспроизвести этот музыкальный файл?", + "mm_m3u": "нажмите Enter/OK, чтобы играть\nнажмите ESC/Отмена, чтобы редактировать", + "mp_breq": "требуется Firefox 82+, Chrome 73+ или iOS 15+", + "mm_bload": "загружаю...", + "mm_bconv": "конвертирую в {0}, подождите...", + "mm_opusen": "ваш браузер не может воспроизводить файлы aac / m4a;\nвключено транскодирование в opus", + "mm_playerr": "ошибка воспроизведения: ", + "mm_eabrt": "Попытка воспроизведения была отменена", + "mm_enet": "Ваше подключение нестабильно", + "mm_edec": "Этот файл, возможно, повреждён??", + "mm_esupp": "Ваш браузер не распознаёт этот аудио-формат", + "mm_eunk": "Неопознанная ошибка", + "mm_e404": "Не удалось воспроизвести аудио; ошибка 404: Файл не найден.", + "mm_e403": "Не удалось воспроизвести аудио; ошибка 403: Доступ запрещён.\n\nПопробуйте перезагрузить страницу, возможно, ваша сессия истекла", + "mm_e500": "Не удалось воспроизвести аудио; ошибка 500: Проверьте логи сервера.", + "mm_e5xx": "Не удалось воспроизвести аудио; ошибка сервера ", + "mm_nof": "больше аудио-файлов не найдено", + "mm_prescan": "Поиск музыки для воспроизведения дальше...", + "mm_scank": "Найден следующий трек:", + "mm_uncache": "кеш очищен; все треки будут загружены заново при воспроизведении", + "mm_hnf": "это трек больше не существует", + + "im_hnf": "это изображение больше не существует", + + "f_empty": 'эта папка пуста', + "f_chide": 'это скроет столбец «{0}»\n\nвы можете показать скрытые столбцы в настройках', + "f_bigtxt": "объём данного файла - {0} МиБ. точно открыть как текст?", + "f_bigtxt2": "просмотреть только конец файла? это также включит обновление в реальном времени, показывая новые строки сразу после их добавления", + "fbd_more": '
показано {0} из {1} файлов; показать {2} или показать всё
', + "fbd_all": '
показано {0} из {1} файлов; показать всё
', + "f_anota": "только {0} из {1} файлов было выделено;\nчтобы выделить всё папку, отмотайте до низа", + + "f_dls": 'ссылки на файлы в данной папке были\nзаменены ссылками на скачивание', + + "f_partial": "Чтобы безопасно скачать файл, который в текущий момент загружается, нажмите на файл с таким же названием, но без расширения .PARTIAL. Пожалуйста, нажмите Отмена или ESC, чтобы сделать это.\n\nПри нажатии OK / Enter, вы скачаете этот временный файл, который с огромной вероятностью содержит лишь неполные данные.", + + "ft_paste": "вставить {0} файлов$NГорячая клавиша: ctrl-V", + "fr_eperm": 'не удалось переименовать:\nу вас нет разрешения “move” в этой папке', + "fd_eperm": 'не удалось удалить:\nу вас нет разрешения “delete” в этой папке', + "fc_eperm": 'не удалось вырезать:\nу вас нет разрешения “move” в этой папке', + "fp_eperm": 'не удалось вставить:\nу вас нет разрешения “write” в этой папке', + "fr_emore": "выделите хотя бы один файл, чтобы переименовать", + "fd_emore": "выделите хотя бы один файл, чтобы удалить", + "fc_emore": "выделите хотя бы один файл, чтобы вырезать", + "fcp_emore": "выделите хотя бы один файл, чтобы скопировать в буфер", + + "fs_sc": "поделиться текущей папкой", + "fs_ss": "поделиться выделенными файлами", + "fs_just1d": "вы не можете выбрать больше одной папки\nили смешивать файлы с папками при выделении", + "fs_abrt": "❌ отменить", + "fs_rand": "🎲 случ. имя", + "fs_go": "✅ создать доступ", + "fs_name": "имя", + "fs_src": "путь", + "fs_pwd": "пароль", + "fs_exp": "срок", + "fs_tmin": "мин", + "fs_thrs": "часов", + "fs_tdays": "дней", + "fs_never": "вечно", + "fs_pname": "имя ссылки; генерируется случайно если не указано", + "fs_tsrc": "путь к файлу или папке, которыми нужно поделиться", + "fs_ppwd": "пароль (необязательно)", + "fs_w8": "создаю доступ...", + "fs_ok": "нажмите Enter/OK, чтобы скопировать\nнажмите ESC/Cancel, чтобы закрыть", + + "frt_dec": "может исправить некоторые случаи с некорректными именами файлов\">декодировать url", + "frt_rst": "сбросить изменённые имена обратно к оригинальным\">↺ сброс", + "frt_abrt": "отменить операцию и закрыть это окно\">❌ отмена", + "frb_apply": "ПЕРЕИМЕНОВАТЬ", + "fr_adv": "переименование массовое / метаданных / по шаблону\">эксперт", + "fr_case": "чувствительный к регистру regex\">РеГиСтР", + "fr_win": "совместимые с windows имена; заменяет <>:"\\|?* японскими полноширинными символами\">win", + "fr_slash": "заменяет / символом, который не создаёт новые папки\">без /", + "fr_re": "поиск по шаблону regex, применяемый к оригинальным именам; группы захвата могут применяться в поле форматирования с помощью <code>(1)</code> и <code>(2)</code> и так далее", + "fr_fmt": "вдохновлено foobar2000:$N<code>(title)</code> заменяется названием трека,$N<code>[(artist) - ](title)</code> пропускает [эту] часть, если композитор не указан$N<code>$lpad((tn),2,0)</code> добавляет ведущие нули до двух цифр", + "fr_pdel": "удалить", + "fr_pnew": "сохранить как", + "fr_pname": "предоставьте название для нового шаблона", + "fr_aborted": "прервано", + "fr_lold": "старое имя", + "fr_lnew": "новое имя", + "fr_tags": "теги для выделенных файлов (не редактируется, это для инструкции):", + "fr_busy": "переименовываю {0} файлов...\n\n{1}", + "fr_efail": "ошибка переименования:\n", + "fr_nchg": "{0} новых имён были модифицированы для соответствия опциям win и/или без /\n\nХотите использовать эти имена?", + + "fd_ok": "успешно удалено", + "fd_err": "ошибка удаления:\n", + "fd_none": "ничего не удалено; возможно, не позволяет конфигурация сервера (xbd)?", + "fd_busy": "удалено {0} файлов...\n\n{1}", + "fd_warn1": "УДАЛИТЬ эти {0} файлов?", + "fd_warn2": "Внимание! Это необратимый процесс. Удалить?", + + "fc_ok": "вырезано {0} файлов", + "fc_warn": 'вырезано {0} файлов\n\nно только эта вкладка браузера может их вставить\n(поскольку выделение оказалось настолько огромным)', + + "fcc_ok": "скопировано {0} файлов в буфер", + "fcc_warn": 'скопировано {0} файлов в буфер\n\nно только эта вкладка браузера может их вставить\n(поскольку выделение оказалось настолько огромным)', + + "fp_apply": "использовать эти имена", + "fp_ecut": "сначала вырезать или скопировать только некоторые файлы / папки\n\nучтите: вы можете вырезать / вставлять файлы между вкладками", + "fp_ename": "{0} файлов невозможно перенести сюда, потому что их имена уже заняты. Введите имена ниже, чтобы продолжить, или оставьте поля пустыми, чтобы пропустить:", + "fcp_ename": "{0} файлов невозможно скопировать сюда, потому что их имена уже заняты. Введите имена ниже, чтобы продолжить, или оставьте поля пустыми, чтобы пропустить:", + "fp_emore": "есть ещё коллизии имён, которые требуется исправить", + "fp_ok": "успешно перенесено", + "fcp_ok": "успешно скопировано", + "fp_busy": "перемещаю {0} файлов...\n\n{1}", + "fcp_busy": "копирую {0} файлов...\n\n{1}", + "fp_err": "ошибка перемещения:\n", + "fcp_err": "ошибка копирования:\n", + "fp_confirm": "переместить эти {0} файлов сюда?", + "fcp_confirm": "скопировать эти {0} файлов сюда?", + "fp_etab": 'ошибка чтения буфера обмена из другой вкладки браузера', + "fp_name": "загружаю файл с вашего устройства. Назовите его:", + "fp_both_m": '
выберите, что вставить
Enter = Перенести {0} файлов из «{1}»\nESC = Загрузить {2} файлов с вашего устройства', + "fcp_both_m": '
выберите, что вставить
Enter = Скопировать {0} файлов из «{1}»\nESC = Загрузить {2} файлов с вашего устройства', + "fp_both_b": 'ПереместитьЗагрузить', + "fcp_both_b": 'СкопироватьЗагрузить', + + "mk_noname": "введите имя в текстовое поле слева перед тем, как это делать :p", + + "tv_load": "Загружаю текстовый документ:\n\n{0}\n\n{1}% ({2} из {3} МиБ загружено)", + "tv_xe1": "не удалось загрузить текстовый файл:\n\nошибка ", + "tv_xe2": "404, файл не найден", + "tv_lst": "список текстовых файлов в", + "tvt_close": "вернуться в обзор папки$NГорячая клавиша: M (или Esc)\">❌ закрыть", + "tvt_dl": "скачать этот файл$NГорячая клавиша: Y\">💾 скачать", + "tvt_prev": "показать предыдущий документ$NГорячая клавиша: i\">⬆ пред", + "tvt_next": "показать следующий документ$NГорячая клавиша: K\">⬇ след", + "tvt_sel": "выбрать документ   ( для вырезания / копирования / удаления / ... )$NГорячая клавиша: S\">выд", + "tvt_edit": "открыть документ в текстовом редакторе$NГорячая клавиша: E\">✏️ изменить", + "tvt_tail": "проверять файл на изменения; показывать новые строки в реальном времени\">📡 обновлять", + "tvt_wrap": "перенос слов\">↵", + "tvt_atail": "прикрепить вид к низу страницы\">⚓", + "tvt_ctail": "декодировать цвета терминала (ansi escape codes)\">🌈", + "tvt_ntail": "лимит прокрутки (как много байт текста держать в памяти)", + + "m3u_add1": "трек добавлен в плейлист m3u", + "m3u_addn": "{0} треков добавлено в плейлист m3u", + "m3u_clip": "плейлист m3u скопирован в буфер\n\nсоздайте файл с расширением .m3u и вставьте текст туда, чтобы сделать из него плейлист", + + "gt_vau": "не показывать видео, только воспроизводить аудио\">🎧", + "gt_msel": "включить режим выделения; держите ctrl при нажатии для инвертации действия$N$N<em>когда активно: дважды кликните на файле / папке, чтобы открыть их</em>$N$NГорячая клавиша: S\">выделение", + "gt_crop": "обрезать миниатюры\">обрезка", + "gt_3x": "миниатюры высокого разрешения\">3x", + "gt_zoom": "размер", + "gt_chop": "длина имён", + "gt_sort": "сортировать по", + "gt_name": "имени", + "gt_sz": "размеру", + "gt_ts": "дате", + "gt_ext": "типу", + "gt_c1": "укоротить названия файлов", + "gt_c2": "удлинить названия файлов", + + "sm_w8": "ищем...", + "sm_prev": "результаты поиска ниже - из предыдущего запроса:\n ", + "sl_close": "закрыть результаты поиска", + "sl_hits": "показ {0} совпадений", + "sl_moar": "загрузить больше", + + "s_sz": "размер", + "s_dt": "дата", + "s_rd": "путь", + "s_fn": "имя", + "s_ta": "теги", + "s_ua": "дата⬆️", + "s_ad": "другое", + "s_s1": "минимум МиБ", + "s_s2": "максимум МиБ", + "s_d1": "мин. iso8601", + "s_d2": "макс. iso8601", + "s_u1": "загружено после", + "s_u2": "и/или до", + "s_r1": "путь содержит   (разделить пробелами)", + "s_f1": "имя содержит   (для исключения писать -nope)", + "s_t1": "теги содержат   (^=начало, конец=$)", + "s_a1": "свойства метаданных", + + "md_eshow": "не удалось показать ", + "md_off": "[📜ридми] отключён в [⚙️] -- документ скрыт", + + "badreply": "Ошибка обработки ответа сервера", + + "xhr403": "403: Доступ запрещён\n\nпопробуйте перезагрузить страницу, возможно, ваша сессия истекла", + "xhr0": "неизвестно (возможно, потеряно соединение с сервером, либо он отключён)", + "cf_ok": "просим прощения -- сработала защита от DD" + wah + "oS\n\nвсё должно вернуться в норму через 30 сек\n\nесли ничего не происходит - перезагрузите страницу", + "tl_xe1": "не удалось показать подпапки:\n\nошибка ", + "tl_xe2": "404: Папка не найдена", + "fl_xe1": "не удалось показать файлы:\n\nошибка ", + "fl_xe2": "404: Папка не найдена", + "fd_xe1": "не удалось создать подпапку:\n\nошибка ", + "fd_xe2": "404: Родительская папка не найдена", + "fsm_xe1": "не удалось отправить сообщение:\n\nошибка ", + "fsm_xe2": "404: Родительская папка не найдена", + "fu_xe1": "не удалось удалить список с сервера:\n\nошибка ", + "fu_xe2": "404: Файл не найден??", + + "fz_tar": "несжатый файл gnu-tar (linux / mac)", + "fz_pax": "несжатый pax-форматированный tar (медленнее)", + "fz_targz": "gnu-tar с 3 уровнем сжатия gzip$N$Nобычно это очень медленно,$Nлучше использовать несжатый tar", + "fz_tarxz": "gnu-tar с 1 уровнем сжатия xz$N$Nобычно это очень медленно,$Nлучше использовать несжатый tar", + "fz_zip8": "zip с именами по utf8 (может работать криво на windows 7 и ниже)", + "fz_zipd": "zip с именами по cp437, для очень старого софта", + "fz_zipc": "cp437 с предварительным вычислением crc32,$N для MS-DOS PKZIP v2.04g (октябрь 1993)$N(требует больше времени для обработки перед скачиванием)", + + "un_m1": "вы можете удалить ваши недавние загрузки (или отменить незавершённые) ниже", + "un_upd": "обновить", + "un_m4": "или поделиться файлами снизу:", + "un_ulist": "показать", + "un_ucopy": "копировать", + "un_flt": "опциональный фильтр:  адрес должен содержать", + "un_fclr": "очистить фильтр", + "un_derr": 'ошибка удаления:\n', + "un_f5": 'что-то сломалось, пожалуйста перезагрузите страницу', + "un_uf5": "извините, но вам нужно перезагрузить страницу (F5 или Ctrl+R) перед тем, как отменить эту загрузку", + "un_nou": 'внимание: сервер слишком нагружен, чтобы показать незавершённые загрузки; нажмите на ссылку "обновления" через пару секунд', + "un_noc": 'внимание: удаление уже загруженных файлов запрещено конфигурацией сервера', + "un_max": "показаны первые 2000 файлов (используйте фильтр)", + "un_avail": "{0} недавних загрузок может быть удалено
{1} незавершённых может быть отменено", + "un_m2": "отсортировано по времени загрузки; сначала самые последние:", + "un_no1": "ха, поверил! достаточно свежих загрузок ещё нет", + "un_no2": "ха, поверил! достаточно свежих загрузок, соответствующих фильтру, ещё нет", + "un_next": "удалить следующие {0} файлов ниже", + "un_abrt": "отменить", + "un_del": "удалить", + "un_m3": "загружаю ваши недавние загрузки...", + "un_busy": "удаляю {0} файлов...", + "un_clip": "{0} ссылок скопировано в буфер", + + "u_https1": "вам стоит", + "u_https2": "включить https", + "u_https3": "для лучшей производительности", + "u_ancient": 'у вас действительно антикварный браузер -- возможно, стоит использовать bup', + "u_nowork": "требуется firefox 53+, chrome 57+ или iOS 11+", + "tail_2old": "требуется firefox 105+, chrome 71+ или iOS 14.5+", + "u_nodrop": 'ваш браузер слишком старый для загрузки через перетаскивание', + "u_notdir": "это не папка!\n\nваш браузер слишком старый,\nиспользуйте перетаскивание", + "u_uri": "чтобы перетащить картинку из других окон браузера,\nотпустите её на большую кнопку загрузки", + "u_enpot": 'переключиться на простой интерфейс (может ускорить загрузку)', + "u_depot": 'переключиться на модный интерфейс (может замедлить загрузку)', + "u_gotpot": 'переключаюсь на простой интерфес для ускорения загрузки,\n\nможете переключиться обратно, если хотите!', + "u_pott": "

файлы:   {0} завершено,   {1} ошибок,   {2} загружаются,   {3} в очереди

", + "u_ever": "это упрощённый загрузчик; up2k требует хотя бы
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'это упрощённый загрузчик; up2k лучше', + "u_uput": 'увеличить скорость (пропуск подсчёта контрольных сумм)', + "u_ewrite": 'у вас нет прав на запись в эту папку', + "u_eread": 'у вас нет прав на чтение из этой папки', + "u_enoi": 'поиск файлов выключен настройками сервера', + "u_enoow": "перезапись здесь не работает; требуются права на удаление", + "u_badf": 'Эти {0} из {1} файлов были пропущены, вероятно, из-за настроек доступа в файловой системе:\n\n', + "u_blankf": 'Эти {0} из {1} файлов пустые; всё равно загрузить?\n\n', + "u_applef": 'Эти {0} из {1} файлов, вероятно, нежелательны;\nНажмите OK/Enter, чтобы ПРОПУСТИТЬ их,\nНажмите Отмена/ESC, чтобы проигнорировать сообщение и ЗАГРУЗИТЬ их:\n\n', + "u_just1": '\nВозможно, будет лучше, если вы выберете только один файл', + "u_ff_many": "если вы используете Linux / MacOS / Android, тогда такое количество файлов может крашнуть Firefox!\nв таком случае попробуйте снова (или используйте Chrome).", + "u_up_life": "Эта загрузка будет удалена с сервера\nчерез {0} после её завершения", + "u_asku": 'загрузить эти {0} файлов в {1}', + "u_unpt": "вы можете отменить / удалить эту загрузку с помощью 🧯 сверху слева", + "u_bigtab": 'будет показано {0} файлов\n\nэто может крашнуть браузер, вы уверены?', + "u_scan": 'Сканирую файлы...', + "u_dirstuck": 'сканер папки завис при попытке получить доступ к следующим {0} файлам; будет пропущено:', + "u_etadone": 'Готово ({0}, {1} файлов)', + "u_etaprep": '(подготавливаю загрузку)', + "u_hashdone": 'хеширование выполнено', + "u_hashing": 'хеш', + "u_hs": 'подготовка к хешированию...', + "u_started": "файлы загружаются; подробнее в [🚀]", + "u_dupdefer": "дубликат; будет обработан после всех остальных файлов", + "u_actx": "нажмите на этот текст, чтобы предотвратить
падение производительности при просмотре других вкладок / окон", + "u_fixed": "Окей!  Исправлено 👍", + "u_cuerr": "не удалось загрузить фрагмент {0} из {1};\nвероятно, не критично - продолжаю\n\nфайл: {2}", + "u_cuerr2": "отказ в загрузке (фрагмент {0} из {1});\nпопытаюсь повторить позже\n\nфайл: {2}\n\nошибка ", + "u_ehstmp": "попытаюсь повторить позже; подробнее снизу справа", + "u_ehsfin": "отказ в запросе завершения загрузки; повторяю...", + "u_ehssrch": "отказ в запросе поиска; повторяю...", + "u_ehsinit": "отказ в запросе начала загрузки; повторяю...", + "u_eneths": "ошибка подключения во время подготовки загрузки; повторяю...", + "u_enethd": "ошибка подключения во время проверки существования целевого файла; повторяю...", + "u_cbusy": "ожидаю возвращения доступа к серверу после ошибки подключения...", + "u_ehsdf": "на сервере закончилось место!\n\nбуду пытаться повторить, на случай если\nместо будет освобождено", + "u_emtleak1": "кажется, у вашего браузера может быть утечка памяти;\nпожалуйста,", + "u_emtleak2": ' перейдите на https (рекомендовано) или ', + "u_emtleak3": ' ', + "u_emtleakc": 'попробуйте сделать так:\n
  • нажмите F5 для обновления страницы
  • затем отключите опцию  мп  в  ⚙️ настройках
  • и повторите попытку загрузки
Она будет чуть медленнее, но что поделать.\nИзвините за неудобства!\n\nPS: в chrome v107 это исправили', + "u_emtleakf": 'попробуйте сделать так:\n
  • нажмите F5 для обновления страницы
  • затем включите опцию 🥔 (простой интерфейс) во вкладке загрузок
  • и повторите попытку
\nPS: firefox скоро должны починить в этом аспекте', + "u_s404": "не найдено на сервере", + "u_expl": "объяснить", + "u_maxconn": "в большинстве браузеров это нельзя поднять выше 6, но firefox позволяет увеличить лимит с помощью connections-per-server в about:config", + "u_tu": '

ВНИМАНИЕ: активен режим турбо,  клиент может игнорировать незавершённые загрузки; подробнее при наведении на кнопку турбо

', + "u_ts": '

ВНИМАНИЕ: активен режим турбо,  результаты поиска могут быть некорректными; подробнее при наведении на кнопку турбо

', + "u_turbo_c": "режим турбо отключён сервером", + "u_turbo_g": "отключаю турбо, поскольку у вас нет прав\nна просмотр папок в этом хранилище", + "u_life_cfg": 'автоудаление через мин (или часов)', + "u_life_est": 'загрузка будет удалена ---', + "u_life_max": 'эта папка требует\nавтоудаления файлов через {0}', + "u_unp_ok": 'удаление разрешено для {0}', + "u_unp_ng": 'удаление НЕ будет разрешено', + "ue_ro": 'ваш доступ к данной папке - только чтение\n\n', + "ue_nl": 'на данный момент вы не авторизованы', + "ue_la": 'вы авторизованы как "{0}"', + "ue_sr": 'сейчас вы в режиме поиска файлов\n\nперейдите в режим загрузки, нажав на лупу 🔎 (рядом с огромной кнопкой ИСКАТЬ), и попробуйте снова\n\nизвините', + "ue_ta": 'попробуйте загрузить снова, теперь должно сработать', + "ue_ab": "этот файл уже загружают в другую папку, та загрузка должна быть завершена перед тем, как загрузить этот же файл в другое место.\n\nВы можете отменить свою загрузку через 🧯 сверху слева", + "ur_1uo": "OK: Файл успешно загружен", + "ur_auo": "OK: Все {0} файлов успешно загружены", + "ur_1so": "OK: Файл найден на сервере", + "ur_aso": "OK: Все {0} файлов найдены на сервере", + "ur_1un": "Загрузка не удалась, извините", + "ur_aun": "Все {0} загрузок не удались, извините", + "ur_1sn": "Файл НЕ был найден на сервере", + "ur_asn": "Все {0} файлов НЕ было найдено на сервере", + "ur_um": "Завершено;\n{0} успешно,\n{1} ошибок, извините", + "ur_sm": "Завершено;\n{0} файлов найдено на сервере,\n{1} файлов НЕ найдено на сервере", + + "lang_set": "перезагрузить страницу, чтобы применить изменения?", + }, }; -var LANGS = ["eng", "nor", "chi", "deu"]; +var LANGS = ["eng", "nor", "chi", "deu", "rus"]; if (window.langmod) langmod(); diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index c2e60eb6..638c9ddc 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -135,6 +135,48 @@ var Ls = { "ae1": "Aktive Downloads:", "af1": "Zeige neue Uploads", }, + "eng": { + "a1": "обновить", + "b1": "приветик, незнакомец   (вы не авторизованы)", + "c1": "выйти", + "d1": "трассировка стека", // TLNote: "d2" is the tooltip for this button + "d2": "показывает состояние всех активных потоков", + "e1": "перезагрузить конфиг", + "e2": "перезагрузить файлы конфига (аккаунты/хранилища/флаги),$Nи пересканировать все хранилища с флагом e2ds$N$Nвнимание: изменения глобальных настроек$Nтребуют полного перезапуска сервера", + "f1": "вы можете видеть:", + "g1": "вы можете загружать файлы в:", + "cc1": "всякая всячина:", + "h1": "отключить k304", // TLNote: "j1" explains what k304 is + "i1": "включить k304", + "j1": "включённый k304 будет отключать вас при получении HTTP 304, что может помочь при работе с некоторыми глючными прокси (перестают загружаться страницы), но это также сделает работу клиента медленнее", + "k1": "сбросить локальные настройки", + "l1": "авторизуйтесь для других опций:", + "m1": "с возвращением,", // TLNote: "welcome back, USERNAME" + "n1": "404 не найдено  ┐( ´ -`)┌", + "o1": 'или у вас нет доступа -- попробуйте авторизоваться или вернуться на главную', + "p1": "403 доступ запрещён  ~┻━┻", + "q1": 'авторизуйтесь или вернитесь на главную', + "r1": "вернуться на главную", + ".s1": "пересканировать", + "t1": "действия", // TLNote: this is the header above the "rescan" buttons + "u2": "время с последней записи на сервер$N( загрузка / переименование / ... )$N$N17d = 17 дней$N1h23 = 1 час 23 минут$N4m56 = 4 минут 56 секунд", + "v1": "подключить", + "v2": "использовать сервер как локальный диск", + "w1": "перейти на https", + "x1": "поменять пароль", + "y1": "управление доступом", // TLNote: shows the list of folders that the user has decided to share + "z1": "разблокировать:", // TLNote: the password prompt to see a hidden share + "ta1": "сначала введите свой новый пароль", + "ta2": "повторите новый пароль:", + "ta3": "опечатка; попробуйте снова", + "aa1": "входящие файлы:", + "ab1": "отключить no304", + "ac1": "включить no304", + "ad1": "включённый no304 полностью отключит хеширование; используйте, если k304 не помог. Сильно увеличит объём трафика!", + "ae1": "активные скачивания:", + "af1": "показать недавние загрузки", + "ag1": "показать известных IdP-пользователей", + }, }; if (window.langmod) From fea45e451d609b3b89033429b527d08f6fc6ba22 Mon Sep 17 00:00:00 2001 From: Vlad Date: Thu, 31 Jul 2025 19:20:36 +0200 Subject: [PATCH 024/174] add Ukrainian translation (#350) --- copyparty/web/browser.js | 628 ++++++++++++++++++++++++++++++++++++++- 1 file changed, 627 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index db9986f2..b8b8b202 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3136,9 +3136,635 @@ var Ls = { "lang_set": "перезагрузить страницу, чтобы применить изменения?", }, + "ukr": { + "tt": "Українська", + + "cols": { + "c": "кнопки дій", + "dur": "тривалість", + "q": "якість / бітрейт", + "Ac": "аудіо кодек", + "Vc": "відео кодек", + "Fmt": "формат / контейнер", + "Ahash": "контрольна сума аудіо", + "Vhash": "контрольна сума відео", + "Res": "роздільність", + "T": "тип файлу", + "aq": "якість аудіо / бітрейт", + "vq": "якість відео / бітрейт", + "pixfmt": "підвибірка / структура пікселів", + "resw": "горизонтальна роздільність", + "resh": "вертикальна роздільність", + "chs": "аудіо канали", + "hz": "частота дискретизації" + }, + + "hks": [ + [ + "різне", + ["ESC", "закрити різні речі"], + + "файловий менеджер", + ["G", "перемкнути список / сітку"], + ["T", "перемкнути мініатюри / іконки"], + ["⇧ A/D", "розмір мініатюр"], + ["ctrl-K", "видалити вибране"], + ["ctrl-X", "вирізати до буфера"], + ["ctrl-C", "копіювати до буфера"], + ["ctrl-V", "вставити (перемістити/копіювати) сюди"], + ["Y", "завантажити вибране"], + ["F2", "перейменувати вибране"], + + "вибір файлів у списку", + ["space", "перемкнути вибір файлу"], + ["↑/↓", "перемістити курсор вибору"], + ["ctrl ↑/↓", "перемістити курсор і вікно"], + ["⇧ ↑/↓", "вибрати попередній/наступний файл"], + ["ctrl-A", "вибрати всі файли / папки"], + ], [ + "навігація", + ["B", "перемкнути хлібні крихти / панель навігації"], + ["I/K", "попередня/наступна папка"], + ["M", "батьківська папка (або згорнути поточну)"], + ["V", "перемкнути папки / текстові файли в панелі навігації"], + ["A/D", "розмір панелі навігації"], + ], [ + "аудіо плеєр", + ["J/L", "попередня/наступна пісня"], + ["U/O", "перемотати на 10сек назад/вперед"], + ["0..9", "перейти до 0%..90%"], + ["P", "відтворити/пауза (також запускає)"], + ["S", "вибрати поточну пісню"], + ["Y", "завантажити пісню"], + ], [ + "переглядач зображень", + ["J/L, ←/→", "попереднє/наступне зображення"], + ["Home/End", "перше/останнє зображення"], + ["F", "повний екран"], + ["R", "повернути за годинниковою стрілкою"], + ["⇧ R", "повернути проти годинникової стрілки"], + ["S", "вибрати зображення"], + ["Y", "завантажити зображення"], + ], [ + "відео плеєр", + ["U/O", "перемотати на 10сек назад/вперед"], + ["P/K/Space", "відтворити/пауза"], + ["C", "продовжити відтворення наступного"], + ["V", "повтор"], + ["M", "вимкнути звук"], + ["[ and ]", "встановити інтервал повтору"], + ], [ + "переглядач текстових файлів", + ["I/K", "попередній/наступний файл"], + ["M", "закрити текстовий файл"], + ["E", "редагувати текстовий файл"], + ["S", "вибрати файл (для вирізання/копіювання/перейменування)"], + ] + ], + + "m_ok": "Гаразд", + "m_ng": "Скасувати", + + "enable": "Увімкнути", + "danger": "НЕБЕЗПЕКА", + "clipped": "скопійовано до буфера обміну", + + "ht_s1": "секунда", + "ht_s2": "секунд", + "ht_m1": "хвилина", + "ht_m2": "хвилин", + "ht_h1": "година", + "ht_h2": "годин", + "ht_d1": "день", + "ht_d2": "днів", + "ht_and": " і ", + + "goh": "панель керування", + "gop": 'попередній сусід">назад', + "gou": 'батьківська папка">вгору', + "gon": 'наступна папка">далі', + "logout": "Вийти ", + "access": " доступ", + "ot_close": "закрити підменю", + "ot_search": "пошук файлів за атрибутами, шляхом / іменем, музичними тегами, або будь-якою комбінацією$N$N<code>foo bar</code> = має містити «foo» і «bar»,$N<code>foo -bar</code> = має містити «foo», але не «bar»,$N<code>^yana .opus$</code> = починатися з «yana» і бути файлом «opus»$N<code>"try unite"</code> = містити точно «try unite»$N$Nформат дати - iso-8601, наприклад$N<code>2009-12-31</code> або <code>2020-09-12 23:30:00</code>", + "ot_unpost": "скасувати: видалити недавні завантаження або перервати незавершені", + "ot_bup": "bup: основний завантажувач, підтримує навіть netscape 4.0", + "ot_mkdir": "mkdir: створити нову папку", + "ot_md": "new-md: створити новий markdown документ", + "ot_msg": "msg: надіслати повідомлення в лог сервера", + "ot_mp": "налаштування медіаплеєра", + "ot_cfg": "параметри конфігурації", + "ot_u2i": 'up2k: завантажити файли (якщо у вас є доступ для запису) або переключитися на режим пошуку, щоб побачити, чи існують вони десь на сервері$N$Nзавантаження можна поновлювати, багатопотокові, і часові мітки файлів зберігаються, але використовує більше CPU ніж [🎈]  (основний завантажувач)

під час завантаження ця іконка стає індикатором прогресу!', + "ot_u2w": 'up2k: завантажити файли з підтримкою поновлення (закрийте браузер і перетягніть ті самі файли пізніше)$N$Nбагатопотокові, і часові мітки файлів зберігаються, але використовує більше CPU ніж [🎈]  (основний завантажувач)

під час завантаження ця іконка стає індикатором прогресу!', + "ot_noie": 'Будь ласка, використовуйте Chrome / Firefox / Edge', + + "ab_mkdir": "створити папку", + "ab_mkdoc": "новий markdown документ", + "ab_msg": "надіслати повідомлення в лог сервера", + + "ay_path": "перейти до папок", + "ay_files": "перейти до файлів", + + "wt_ren": "перейменувати вибрані елементи$NГаряча клавіша: F2", + "wt_del": "видалити вибрані елементи$NГаряча клавіша: ctrl-K", + "wt_cut": "вирізати вибрані елементи <small>(потім вставити в іншому місці)</small>$NГаряча клавіша: ctrl-X", + "wt_cpy": "копіювати вибрані елементи до буфера$N(щоб вставити їх в іншому місці)$NГаряча клавіша: ctrl-C", + "wt_pst": "вставити раніше вирізане / скопійоване$NГаряча клавіша: ctrl-V", + "wt_selall": "вибрати всі файли$NГаряча клавіша: ctrl-A (коли фокус на файлі)", + "wt_selinv": "інвертувати вибір", + "wt_zip1": "завантажити цю папку як архів", + "wt_selzip": "завантажити вибір як архів", + "wt_seldl": "завантажити вибір як окремі файли$NГаряча клавіша: Y", + "wt_npirc": "копіювати інформацію треку у форматі irc", + "wt_nptxt": "копіювати інформацію треку у текстовому форматі", + "wt_m3ua": "додати до m3u плейлисту (потім клацніть 📻копіювати)", + "wt_m3uc": "копіювати m3u плейлист до буфера", + "wt_grid": "перемкнути сітку / список$NГаряча клавіша: G", + "wt_prev": "попередній трек$NГаряча клавіша: J", + "wt_play": "відтворити / пауза$NГаряча клавіша: P", + "wt_next": "наступний трек$NГаряча клавіша: L", + + "ul_par": "паралельні завантаження:", + "ut_rand": "випадкові імена файлів", + "ut_u2ts": "копіювати часову мітку останньої зміни$Nз вашої файлової системи на сервер\">📅", + "ut_ow": "перезаписати існуючі файли на сервері?$N🛡️: ніколи (замість цього створить нове ім'я файлу)$N🕒: перезаписати, якщо файл на сервері старіший за ваш$N♻️: завжди перезаписувати, якщо файли відрізняються", + "ut_mt": "продовжувати хешування інших файлів під час завантаження$N$Nможливо, вимкніть, якщо ваш CPU або HDD є вузьким місцем", + "ut_ask": 'запитати підтвердження перед початком завантаження">💭', + "ut_pot": "покращити швидкість завантаження на повільних пристроях$Nроблячи інтерфейс менш складним", + "ut_srch": "не завантажувати, а перевірити, чи файли вже $N існують на сервері (сканує всі папки, які ви можете читати)", + "ut_par": "призупинити завантаження, встановивши 0$N$Nзбільшіть, якщо ваше з'єднання повільне / висока затримка$N$Nзалишіть 1 в локальній мережі або якщо HDD сервера є вузьким місцем", + "ul_btn": "перетягніть файли / папки
(або клацніть сюди)", + "ul_btnu": "З А В А Н Т А Ж И Т И", + "ul_btns": "П О Ш У К", + + "ul_hash": "хеш", + "ul_send": "надіслати", + "ul_done": "готово", + "ul_idle1": "завантаження ще не поставлені в чергу", + "ut_etah": "середня швидкість <em>хешування</em> і орієнтовний час до завершення", + "ut_etau": "середня швидкість <em>завантаження</em> і орієнтовний час до завершення", + "ut_etat": "середня <em>загальна</em> швидкість і орієнтовний час до завершення", + + "uct_ok": "успішно завершено", + "uct_ng": "невдало: помилка / відхилено / не знайдено", + "uct_done": "ok і ng разом", + "uct_bz": "хешування або завантаження", + "uct_q": "очікує, в черзі", + + "utl_name": "ім'я файлу", + "utl_ulist": "список", + "utl_ucopy": "копіювати", + "utl_links": "посилання", + "utl_stat": "статус", + "utl_prog": "прогрес", + + // keep short: + "utl_404": "404", + "utl_err": "ПОМИЛКА", + "utl_oserr": "помилка ОС", + "utl_found": "знайдено", + "utl_defer": "відкласти", + "utl_yolo": "YOLO", + "utl_done": "готово", + + "ul_flagblk": "файли були додані до черги
однак є зайнятий up2k в іншій вкладці браузера,
тому чекаємо, поки він завершиться спочатку", + "ul_btnlk": "конфігурація сервера заблокувала цей перемикач у цьому стані", + + "udt_up": "Завантаження", + "udt_srch": "Пошук", + "udt_drop": "перетягніть сюди", + + "u_nav_m": '
гаразд, що у вас є?
Enter = Файли (один або більше)\nESC = Одна папка (включаючи підпапки)', + "u_nav_b": 'ФайлиОдна папка', + + "cl_opts": "перемикачі", + "cl_themes": "тема", + "cl_langs": "мова", + "cl_ziptype": "завантаження папки", + "cl_uopts": "перемикачі up2k", + "cl_favico": "favicon", + "cl_bigdir": "великі папки", + "cl_hsort": "#сортування", + "cl_keytype": "позначення клавіш", + "cl_hiddenc": "приховані стовпці", + "cl_hidec": "приховати", + "cl_reset": "скинути", + "cl_hpick": "натисніть на заголовки стовпців, щоб приховати їх у таблиці нижче", + "cl_hcancel": "приховання стовпців скасовано", + + "ct_grid": '田 сітка', + "ct_ttips": '◔ ◡ ◔">ℹ️ підказки', + "ct_thumb": 'у режимі сітки, перемкнути іконки або мініатюри$NГаряча клавіша: T">🖼️ мініатюри', + "ct_csel": 'використовувати CTRL і SHIFT для вибору файлів у режимі сітки">вибір', + "ct_ihop": 'коли переглядач зображень закрито, прокрутити вниз до останнього переглянутого файлу">g⮯', + "ct_dots": 'показати приховані файли (якщо сервер дозволяє)">приховані файли', + "ct_qdel": 'при видаленні файлів, запитати підтвердження лише один раз">швидке видалення', + "ct_dir1st": 'сортувати папки перед файлами">спочатку 📁', + "ct_nsort": 'природне сортування (для імен файлів з початковими цифрами)">природне сортування', + "ct_readme": 'показати README.md у списках папок">📜 readme', + "ct_idxh": 'показати index.html замість списку папки">htm', + "ct_sbars": 'показати смуги прокрутки">⟊', + + "cut_umod": "якщо файл вже існує на сервері, оновити часову мітку останньої зміни сервера відповідно до вашого локального файлу (потребує дозволів на запис+видалення)\">re📅", + + "cut_turbo": "кнопка yolo, ви, ймовірно, НЕ хочете її вмикати:$N$Nвикористовуйте це, якщо ви завантажували величезну кількість файлів і змушені були перезапустити з якоїсь причини, і хочете продовжити завантаження якнайшвидше$N$Nце замінює перевірку хешу простою "чи має цей файл той самий розмір на сервері?", тому якщо вміст файлу відрізняється, він НЕ буде завантажений$N$Nви повинні вимкнути це, коли завантаження буде завершено, а потім "завантажити" ті самі файли знову, щоб дозволити клієнту перевірити їх\">turbo", + + "cut_datechk": "не має ефекту, якщо кнопка turbo не ввімкнена$N$Nзменшує yolo фактор на крихту; перевіряє, чи відповідають часові мітки файлів на сервері вашим$N$Nтеоретично, повинно зловити більшість незавершених / пошкоджених завантажень, але не є заміною виконання перевірки з вимкненим turbo потім\">date-chk", + + "cut_u2sz": "розмір (у MiB) кожного фрагмента завантаження; великі значення краще летять через атлантичний океан. Спробуйте низькі значення на дуже ненадійних з'єднаннях", + + "cut_flag": "переконатися, що лише одна вкладка завантажує одночасно $N -- інші вкладки також повинні мати це ввімкнене $N -- впливає лише на вкладки на тому самому домені", + + "cut_az": "завантажувати файли в алфавітному порядку, а не від найменшого файлу$N$Nалфавітний порядок може полегшити огляд, якщо щось пішло не так на сервері, але робить завантаження трохи повільнішим на fiber / LAN", + + "cut_nag": "сповіщення ОС після завершення завантаження$N(тільки якщо браузер або вкладка не активні)", + "cut_sfx": "звуковий сигнал після завершення завантаження$N(тільки якщо браузер або вкладка не активні)", + + "cut_mt": "використовувати багатопотоковість для прискорення хешування файлів$N$Nце використовує веб-воркери і потребує$Nбільше пам'яті (до 512 MiB додатково)$N$Nробить https на 30% швидше, http у 4.5 рази швидше\">mt", + + "cut_wasm": "використовувати wasm замість вбудованого хешера браузера; покращує швидкість у браузерах на базі chrome, але збільшує навантаження CPU, плюс багато старих версій chrome мають баги, які змушують браузер споживати всю пам'ять і вилітати, якщо ця опція ввімкненаа\">wasm", + + "cft_text": "текст favicon (порожній і оновити для відключення)", + "cft_fg": "колір переднього плану", + "cft_bg": "колір фону", + + "cdt_lim": "максимальна кількість файлів для показу в папці", + "cdt_ask": "при прокрутці до низу,$Nзамість завантаження більше файлів,$Nзапитати, що робити", + "cdt_hsort": "скільки правил сортування (<code>,sorthref</code>) включати в медіа-URL. Встановлення цього в 0 також буде ігнорувати правила сортування, включені в медіа посилання при їх натисканні", + + "tt_entree": "показати панель навігації (бічна панель дерева каталогів)$NГаряча клавіша: B", + "tt_detree": "показати хлібні крихти$NГаряча клавіша: B", + "tt_visdir": "прокрутити до вибраної папки", + "tt_ftree": "перемкнути дерево папок / текстові файли$NГаряча клавіша: V", + "tt_pdock": "показати батьківські папки в закріпленій панелі зверху", + "tt_dynt": "автоматично збільшуватися при розширенні дерева", + "tt_wrap": "перенесення слів", + "tt_hover": "показувати переповнені рядки при наведенні$N( порушує прокрутку, якщо курсор $N  миші не знаходиться в лівому відступі )", + + "ml_pmode": "в кінці папки...", + "ml_btns": "команди", + "ml_tcode": "транскодувати", + "ml_tcode2": "транскодувати в", + "ml_tint": "відтінок", + "ml_eq": "аудіо еквалайзер", + "ml_drc": "компресор динамічного діапазону", + + "mt_loop": "зациклити/повторити одну пісню\">🔁", + "mt_one": "зупинити після однієї пісні\">1️⃣", + "mt_shuf": "перемішати пісні в кожній папці\">🔀", + "mt_aplay": "автовідтворення, якщо є ID пісні в посиланні, по якому ви клацнули для доступу до сервера$N$Nвідключення цього також зупинить оновлення URL сторінки з ID пісень під час відтворення музики, щоб запобігти автовідтворенню, якщо ці налаштування втрачені, але URL залишається\">a▶", + "mt_preload": "почати завантаження наступної пісні ближче до кінця для безперервного відтворення\">preload", + "mt_prescan": "перейти до наступної папки перед тим, як остання пісня$Nзакінчиться, підтримуючи веб-браузер у робочому стані$Nщоб він не зупинив відтворення\">nav", + "mt_fullpre": "спробувати попередньо завантажити всю пісню;$N✅ увімкніть на ненадійних з'єднаннях,$N❌ вимкніть на повільних з'єднаннях, ймовірно\">full", + "mt_fau": "на телефонах, запобігти зупинці музики, якщо наступна пісня не завантажується достатньо швидко (може зробити відображення тегів глючним)\">☕️", + "mt_waves": "смуга хвильової форми:$Nпоказати амплітуду аудіо в повзунку\">~s", + "mt_npclip": "показати кнопки для копіювання до буфера поточної пісні, що відтворюється\">/np", + "mt_m3u_c": "показати кнопки для копіювання до буфера$Nвибраних пісень як записи плейлисту m3u8\">📻", + "mt_octl": "інтеграція з ОС (медіа гарячі клавіші / osd)\">os-ctl", + "mt_oseek": "дозволити перемотування через інтеграцію з ОС$N$Nзауваження: на деяких пристроях (iPhone),$Nце замінює кнопку наступної пісні\">seek", + "mt_oscv": "показати обкладинку альбому в osd\">art", + "mt_follow": "тримати трек, що відтворюється, у полі зору\">🎯", + "mt_compact": "компактні елементи керування\">⟎", + "mt_uncache": "очистити кеш  (спробуйте це, якщо ваш браузер закешував$Nпошкоджену копію пісні, тому відмовляється її відтворювати)\">uncache", + "mt_mloop": "зациклити відкриту папку\">🔁 loop", + "mt_mnext": "завантажити наступну папку і продовжити\">📂 next", + "mt_mstop": "зупинити відтворення\">⏸ stop", + "mt_cflac": "конвертувати flac / wav в opus\">flac", + "mt_caac": "конвертувати aac / m4a в opus\">aac", + "mt_coth": "конвертувати всі інші (не mp3) в opus\">oth", + "mt_c2opus": "найкращий вибір для робочих столів, ноутбуків, android\">opus", + "mt_c2owa": "opus-weba, для iOS 17.5 і новіших\">owa", + "mt_c2caf": "opus-caf, для iOS 11 до 17\">caf", + "mt_c2mp3": "використовуйте це на дуже старих пристроях\">mp3", + "mt_c2ok": "гарно, хороший вибір", + "mt_c2nd": "це не рекомендований вихідний формат для вашого пристрою, але це нормально", + "mt_c2ng": "ваш пристрій, здається, не підтримує цей вихідний формат, але давайте все одно спробуємо", + "mt_xowa": "є баги в iOS, які запобігають фоновому відтворенню з використанням цього формату; будь ласка, використовуйте caf або mp3 замість цього", + "mt_tint": "рівень фону (0-100) на смузі перемотування$Nщоб зробити буферизацію менш відвертаючою", + "mt_eq": "вмикає еквалайзер і контроль посилення;$N$Nпосилення <code>0</code> = стандартна 100% гучність (немодифікована)$N$Nширина <code>1  </code> = стандартне стерео (немодифіковане)$Nширина <code>0.5</code> = 50% перехресне живлення ліво-право$Nширина <code>0  </code> = моно$N$Nпосилення <code>-0.8</code> & ширина <code>10</code> = видалення вокалу :^)$N$Nвключення еквалайзера робить безшовні альбоми повністю безшовними, тому залишайте його увімкненим з усіма значеннями в нулі (окрім ширини = 1), якщо вам це важливо", + "mt_drc": "вмикає компресор динамічного діапазону (вирівнювач гучності / цегловий вал); також увімкне EQ для балансування спагеті, тому встановіть всі поля EQ окрім 'width' в 0, якщо ви цього не хочете$N$Nзнижує гучність аудіо вище THRESHOLD дБ; для кожного RATIO дБ понад THRESHOLD є 1 дБ виходу, тому стандартні значення tresh -24 і ratio 12 означають, що він ніколи не повинен стати гучнішим за -22 дБ і безпечно збільшити посилення еквалайзера до 0.8, або навіть 1.8 з ATK 0 і величезним RLS як 90 (працює тільки в firefox; RLS максимум 1 в інших браузерах)$N$N(дивіться вікіпедію, вони пояснюють це набагато краще)", + + "mb_play": "відтворити", + "mm_hashplay": "відтворити цей аудіо файл?", + "mm_m3u": "натисніть Enter/OK для відтворення\nнатисніть ESC/Cancel для редагування", + "mp_breq": "потрібен firefox 82+ або chrome 73+ або iOS 15+", + "mm_bload": "зараз завантажується...", + "mm_bconv": "конвертується в {0}, будь ласка, зачекайте...", + "mm_opusen": "ваш браузер не може відтворювати aac / m4a файли;\nтранскодування в opus тепер увімкнено", + "mm_playerr": "відтворення невдале: ", + "mm_eabrt": "Спроба відтворення була скасована", + "mm_enet": "Ваше інтернет-з'єднання нестабільне", + "mm_edec": "Цей файл нібито пошкоджений??", + "mm_esupp": "Ваш браузер не розуміє цей аудіо формат", + "mm_eunk": "Невідома помилка", + "mm_e404": "Не вдалося відтворити аудіо; помилка 404: Файл не знайдено.", + "mm_e403": "Не вдалося відтворити аудіо; помилка 403: Доступ заборонено.\n\nСпробуйте натиснути F5 для перезавантаження, можливо, ви вийшли з системи", + "mm_e500": "Не вдалося відтворити аудіо; помилка 500: Перевірте логи сервера.", + "mm_e5xx": "Не вдалося відтворити аудіо; помилка сервера ", + "mm_nof": "не знаходжу більше аудіо файлів поблизу", + "mm_prescan": "Шукаю музику для наступного відтворення...", + "mm_scank": "Знайшов наступну пісню:", + "mm_uncache": "кеш очищено; всі пісні будуть перезавантажені при наступному відтворенні", + "mm_hnf": "ця пісня більше не існує", + + "im_hnf": "це зображення більше не існує", + + "f_empty": 'ця папка порожня', + "f_chide": 'це приховає стовпець «{0}»\n\nви можете показати стовпці в вкладці налаштувань', + "f_bigtxt": "цей файл розміром {0} MiB -- дійсно переглядати як текст?", + "f_bigtxt2": "переглянути лише кінець файлу замість цього? це також увімкне відслідковування/tailing, показуючи новододані рядки тексту в реальному часі", + "fbd_more": '
показано {0} з {1} файлів; показати {2} або показати всі
', + "fbd_all": '
показано {0} з {1} файлів; показати всі
', + "f_anota": "лише {0} з {1} елементів було вибрано;\nщоб вибрати всю папку, спочатку прокрутіть до низу", + + "f_dls": 'посилання на файли в поточній папці були\nзмінені на посилання для завантаження', + + "f_partial": "Щоб безпечно завантажити файл, який зараз завантажується, будь ласка, клацніть на файл, який має таке саме ім'я, але без розширення .PARTIAL. Будь ласка, натисніть CANCEL або Escape, щоб зробити це.\n\nНатиснення OK / Enter проігнорує це попередження і продовжить завантаження .PARTIAL робочого файлу замість цього, що майже напевно дасть вам пошкоджені дані.", + + "ft_paste": "вставити {0} елементів$NГаряча клавіша: ctrl-V", + "fr_eperm": 'не можу перейменувати:\nу вас немає дозволу "переміщення" в цій папці', + "fd_eperm": 'не можу видалити:\nу вас немає дозволу "видалення" в цій папці', + "fc_eperm": 'не можу вирізати:\nу вас немає дозволу "переміщення" в цій папці', + "fp_eperm": 'не можу вставити:\nу вас немає дозволу "запису" в цій папці', + "fr_emore": "виберіть принаймні один елемент для перейменування", + "fd_emore": "виберіть принаймні один елемент для видалення", + "fc_emore": "виберіть принаймні один елемент для вирізання", + "fcp_emore": "виберіть принаймні один елемент для копіювання до буфера", + + "fs_sc": "поділитися папкою, в якій ви знаходитесь", + "fs_ss": "поділитися вибраними файлами", + "fs_just1d": "ви не можете вибрати більше однієї папки,\nабо змішувати файли і папки в одному виборі", + "fs_abrt": "❌ скасувати", + "fs_rand": "🎲 випадк.ім'я", + "fs_go": "✅ створити спільний доступ", + "fs_name": "ім'я", + "fs_src": "джерело", + "fs_pwd": "пароль", + "fs_exp": "термін дії", + "fs_tmin": "хв", + "fs_thrs": "годин", + "fs_tdays": "днів", + "fs_never": "вічний", + "fs_pname": "необов'язкове ім'я посилання; буде випадковим, якщо порожнє", + "fs_tsrc": "файл або папка для спільного доступу", + "fs_ppwd": "необов'язковий пароль", + "fs_w8": "створення спільного доступу...", + "fs_ok": "натисніть Enter/OK для копіювання до буфера\nнатисніть ESC/Cancel для закриття", + + "frt_dec": "може виправити деякі випадки пошкоджених імен файлів\">url-decode", + "frt_rst": "скинути змінені імена файлів назад до оригінальних\">↺ reset", + "frt_abrt": "перервати і закрити це вікно\">❌ cancel", + "frb_apply": "ЗАСТОСУВАТИ ПЕРЕЙМЕНУВАННЯ", + "fr_adv": "пакетне / метадані / шаблонне перейменування\">розширене", + "fr_case": "регулярний вираз з урахуванням регістру\">регістр", + "fr_win": "безпечні для windows імена; замінити <>:"\\|?* на японські повноширинні символи\">win", + "fr_slash": "замінити / на символ, який не призводить до створення нових папок\">без /", + "fr_re": "шаблон пошуку регулярного виразу для застосування до оригінальних імен файлів; групи захоплення можна посилатися в полі формату нижче як <code>(1)</code> і <code>(2)</code> і так далі", + "fr_fmt": "натхненний foobar2000:$N<code>(title)</code> замінюється назвою пісні,$N<code>[(artist) - ](title)</code> пропускає [цю] частину, якщо виконавець порожній$N<code>$lpad((tn),2,0)</code> доповнює номер треку до 2 цифр", + "fr_pdel": "видалити", + "fr_pnew": "зберегти як", + "fr_pname": "надайте ім'я для вашого нового пресету", + "fr_aborted": "перервано", + "fr_lold": "старе ім'я", + "fr_lnew": "нове ім'я", + "fr_tags": "теги для вибраних файлів (тільки для читання, лише для довідки):", + "fr_busy": "перейменування {0} елементів...\n\n{1}", + "fr_efail": "перейменування невдале:\n", + "fr_nchg": "{0} з нових імен були змінені через win та/або no /\n\nOK продовжити з цими зміненими новими іменами?", + + "fd_ok": "видалення OK", + "fd_err": "видалення невдале:\n", + "fd_none": "нічого не було видалено; можливо, заблоковано конфігурацією сервера (xbd)?", + "fd_busy": "видалення {0} елементів...\n\n{1}", + "fd_warn1": "ВИДАЛИТИ ці {0} елементи?", + "fd_warn2": "Останній шанс! Неможливо скасувати. Видалити?", + + "fc_ok": "вирізано {0} елементів", + "fc_warn": 'вирізано {0} елементів\n\nале: тільки ця вкладка браузера може їх вставити\n(оскільки вибір настільки величезний)', + + "fcc_ok": "скопійовано {0} елементів до буфера", + "fcc_warn": 'скопійовано {0} елементів до буфера\n\nале: тільки ця вкладка браузера може їх вставити\n(оскільки вибір настільки величезний)', + + "fp_apply": "використовувати ці імена", + "fp_ecut": "спочатку вирізати або скопіювати деякі файли / папки для вставки / переміщення\n\nзауваження: ви можете вирізати / вставляти через різні вкладки браузера", + "fp_ename": "{0} елементів не можуть бути переміщені сюди, тому що імена вже зайняті. Дайте їм нові імена нижче для продовження, або залиште ім'я порожнім, щоб пропустити їх:", + "fcp_ename": "{0} елементів не можуть бути скопійовані сюди, тому що імена вже зайняті. Дайте їм нові імена нижче для продовження, або залиште ім'я порожнім, щоб пропустити їх:", + "fp_emore": "є ще деякі конфлікти імен файлів, які потрібно виправити", + "fp_ok": "переміщення OK", + "fcp_ok": "копіювання OK", + "fp_busy": "переміщення {0} елементів...\n\n{1}", + "fcp_busy": "копіювання {0} елементів...\n\n{1}", + "fp_err": "переміщення невдале:\n", + "fcp_err": "копіювання невдале:\n", + "fp_confirm": "перемістити ці {0} елементи сюди?", + "fcp_confirm": "скопіювати ці {0} елементи сюди?", + "fp_etab": 'не вдалося прочитати буфер з іншої вкладки браузера', + "fp_name": "завантаження файлу з вашого пристрою. Дайте йому ім'я:", + "fp_both_m": '
виберіть, що вставити
Enter = Перемістити {0} файлів з «{1}»\nESC = Завантажити {2} файлів з вашого пристрою', + "fcp_both_m": '
виберіть, що вставити
Enter = Скопіювати {0} файлів з «{1}»\nESC = Завантажити {2} файлів з вашого пристрою', + "fp_both_b": 'ПереміститиЗавантажити', + "fcp_both_b": 'СкопіюватиЗавантажити', + + "mk_noname": "введіть ім'я в текстове поле зліва перед тим, як робити це :p", + + "tv_load": "Завантаження текстового документа:\n\n{0}\n\n{1}% ({2} з {3} MiB завантажено)", + "tv_xe1": "не вдалося завантажити текстовий файл:\n\nпомилка ", + "tv_xe2": "404, файл не знайдено", + "tv_lst": "список текстових файлів в", + "tvt_close": "повернутися до перегляду папки$NГаряча клавіша: M (або Esc)\">❌ закрити", + "tvt_dl": "завантажити цей файл$NГаряча клавіша: Y\">💾 завантажити", + "tvt_prev": "показати попередній документ$NГаряча клавіша: i\">⬆ попер", + "tvt_next": "показати наступний документ$NГаряча клавіша: K\">⬇ наст", + "tvt_sel": "вибрати файл   ( для вирізання / копіювання / видалення / ... )$NГаряча клавіша: S\">вибр", + "tvt_edit": "відкрити файл в текстовому редакторі$NГаряча клавіша: E\">✏️ редагувати", + "tvt_tail": "моніторити файл на зміни; показувати нові рядки в реальному часі\">📡 слідкувати", + "tvt_wrap": "перенесення слів\">↵", + "tvt_atail": "заблокувати прокрутку до низу сторінки\">⚓", + "tvt_ctail": "декодувати кольори терміналу (ansi escape коди)\">🌈", + "tvt_ntail": "ліміт історії прокрутки (скільки байтів тексту тримати завантаженими)", + + "m3u_add1": "пісня додана до m3u плейлисту", + "m3u_addn": "{0} пісень додано до m3u плейлисту", + "m3u_clip": "m3u плейлист тепер скопійований до буфера\n\nви повинні створити новий текстовий файл з назвою щось.m3u і вставити плейлист в цей документ; це зробить його відтворюваним", + + "gt_vau": "не показувати відео, лише відтворювати аудіо\">🎧", + "gt_msel": "увімкнути вибір файлів; ctrl-клік по файлу для перевизначення$N$N<em>коли активний: подвійний клік по файлу / папці щоб відкрити</em>$N$NГаряча клавіша: S\">мультивибір", + "gt_crop": "обрізати мініатюри по центру\">обрізка", + "gt_3x": "мініатюри високої роздільності\">3x", + "gt_zoom": "масштаб", + "gt_chop": "обрізати", + "gt_sort": "сортувати за", + "gt_name": "ім'ям", + "gt_sz": "розміром", + "gt_ts": "датою", + "gt_ext": "типом", + "gt_c1": "обрізати імена файлів більше (показувати менше)", + "gt_c2": "обрізати імена файлів менше (показувати більше)", + + "sm_w8": "пошук...", + "sm_prev": "результати пошуку нижче з попереднього запиту:\n ", + "sl_close": "закрити результати пошуку", + "sl_hits": "показано {0} результатів", + "sl_moar": "завантажити більше", + + "s_sz": "розмір", + "s_dt": "дата", + "s_rd": "шлях", + "s_fn": "ім'я", + "s_ta": "теги", + "s_ua": "up@", + "s_ad": "розш.", + "s_s1": "мінімум MiB", + "s_s2": "максимум MiB", + "s_d1": "мін. iso8601", + "s_d2": "макс. iso8601", + "s_u1": "завантажено після", + "s_u2": "та/або до", + "s_r1": "шлях містить   (розділені пробілами)", + "s_f1": "ім'я містить   (заперечення з -ні)", + "s_t1": "теги містять   (^=початок, кінець=$)", + "s_a1": "специфічні властивості метаданих", + + "md_eshow": "не можу відобразити ", + "md_off": "[📜readme] відключено в [⚙️] -- документ прихований", + + "badreply": "Не вдалося обробити відповідь сервера", + + "xhr403": "403: Доступ заборонено\n\nспробуйте натиснути F5, можливо ви вийшли з системи", + "xhr0": "невідома (ймовірно втрачено з'єднання з сервером, або сервер офлайн)", + "cf_ok": "вибачте за це -- захист від DD" + "oS спрацював\n\nречі повинні відновитися приблизно через 30 сек\n\nякщо нічого не відбувається, натисніть F5 для перезавантаження сторінки", + "tl_xe1": "не вдалося перелічити підпапки:\n\nпомилка ", + "tl_xe2": "404: Папка не знайдена", + "fl_xe1": "не вдалося перелічити файли в папці:\n\nпомилка ", + "fl_xe2": "404: Папка не знайдена", + "fd_xe1": "не вдалося створити підпапку:\n\nпомилка ", + "fd_xe2": "404: Батьківська папка не знайдена", + "fsm_xe1": "не вдалося надіслати повідомлення:\n\nпомилка ", + "fsm_xe2": "404: Батьківська папка не знайдена", + "fu_xe1": "не вдалося завантажити список unpost з сервера:\n\nпомилка ", + "fu_xe2": "404: Файл не знайдено??", + + "fz_tar": "нестиснутий gnu-tar файл (linux / mac)", + "fz_pax": "нестиснутий tar в pax-форматі (повільніше)", + "fz_targz": "gnu-tar зі стисненням gzip рівня 3$N$Nце зазвичай дуже повільно, тому$Nвикористовуйте нестиснутий tar замість цього", + "fz_tarxz": "gnu-tar зі стисненням xz рівня 1$N$Nце зазвичай дуже повільно, тому$Nвикористовуйте нестиснутий tar замість цього", + "fz_zip8": "zip з utf8 іменами файлів (можливо нестабільний на windows 7 і старіших)", + "fz_zipd": "zip з традиційними cp437 іменами файлів, для дуже старого ПЗ", + "fz_zipc": "cp437 з crc32, обчисленим заздалегідь,$Nдля MS-DOS PKZIP v2.04g (жовтень 1993)$N(потребує більше часу для обробки перед початком завантаження)", + + "un_m1": "ви можете видалити ваші недавні завантаження (або перервати незавершені) нижче", + "un_upd": "оновити", + "un_m4": "або поділитися файлами, видимими нижче:", + "un_ulist": "показати", + "un_ucopy": "копіювати", + "un_flt": "необов'язковий фільтр:  URL повинен містити", + "un_fclr": "очистити фільтр", + "un_derr": 'unpost-видалення невдале:\n', + "un_f5": 'щось зламалося, будь ласка, спробуйте оновити або натисніть F5', + "un_uf5": "вибачте, але ви повинні оновити сторінку (наприклад, натиснувши F5 або CTRL-R) перед тим, як це завантаження можна буде перервати", + "un_nou": 'попередження: сервер занадто зайнятий, щоб показати незавершені завантаження; клацніть посилання "оновити" трохи пізніше', + "un_noc": 'попередження: unpost повністю завантажених файлів не увімкнено/дозволено в конфігурації сервера', + "un_max": "показано перші 2000 файлів (використовуйте фільтр)", + "un_avail": "{0} недавніх завантажень можуть бути видалені
{1} незавершених можуть бути перервані", + "un_m2": "відсортовано за часом завантаження; найновіші спочатку:", + "un_no1": "ха! немає завантажень, достатньо недавніх", + "un_no2": "ха! немає завантажень, що відповідають цьому фільтру, достатньо недавніх", + "un_next": "видалити наступні {0} файлів нижче", + "un_abrt": "перервати", + "un_del": "видалити", + "un_m3": "завантаження ваших недавніх завантажень...", + "un_busy": "видалення {0} файлів...", + "un_clip": "{0} посилань скопійовано до буфера", + + "u_https1": "вам слід", + "u_https2": "переключитися на https", + "u_https3": "для кращої продуктивності", + "u_ancient": 'ваш браузер вражаюче старий -- можливо, вам слід використовувати bup замість цього', + "u_nowork": "потрібен firefox 53+ або chrome 57+ або iOS 11+", + "tail_2old": "потрібен firefox 105+ або chrome 71+ або iOS 14.5+", + "u_nodrop": 'ваш браузер занадто старий для перетягування завантажень', + "u_notdir": "це не папка!\n\nваш браузер занадто старий,\nбудь ласка, спробуйте перетягування замість цього", + "u_uri": "щоб перетягнути зображення з інших вікон браузера,\nбудь ласка, перетягніть його на велику кнопку завантаження", + "u_enpot": 'переключитися на картоплинний UI (може покращити швидкість завантаження)', + "u_depot": 'переключитися на вишуканий UI (може зменшити швидкість завантаження)', + "u_gotpot": 'переключення на картоплинний UI для покращення швидкості завантаження,\n\nне соромтеся не погодитися і переключитися назад!', + "u_pott": "

файли:   {0} завершено,   {1} невдало,   {2} зайнято,   {3} в черзі

", + "u_ever": "це базовий завантажувач; up2k потребує принаймні
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'це базовий завантажувач; up2k кращий', + "u_uput": 'оптимізувати для швидкості (пропустити контрольну суму)', + "u_ewrite": 'у вас немає доступу для запису в цю папку', + "u_eread": 'у вас немає доступу для читання цієї папки', + "u_enoi": 'пошук файлів не увімкнено в конфігурації сервера', + "u_enoow": "перезапис не працюватиме тут; потрібен дозвіл на видалення", + "u_badf": 'Ці {0} файли (з {1} загальних) були пропущені, можливо, через дозволи файлової системи:\n\n', + "u_blankf": 'Ці {0} файли (з {1} загальних) порожні; все одно завантажити їх?\n\n', + "u_applef": 'Ці {0} файли (з {1} загальних), ймовірно, небажані;\nНатисніть OK/Enter щоб ПРОПУСТИТИ наступні файли,\nНатисніть Cancel/ESC щоб НЕ виключати, і ЗАВАНТАЖИТИ їх також:\n\n', + "u_just1": '\nМожливо, це спрацює краще, якщо ви виберете лише один файл', + "u_ff_many": "якщо ви використовуєте Linux / MacOS / Android, то така кількість файлів може завісити Firefox!\nякщо це станеться, будь ласка, спробуйте знову (або використовуйте Chrome).", + "u_up_life": "Це завантаження буде видалено з сервера\n{0} після його завершення", + "u_asku": 'завантажити ці {0} файлів до {1}', + "u_unpt": "ви можете скасувати / видалити це завантаження, використовуючи 🧯 зверху зліва", + "u_bigtab": 'збираюся показати {0} файлів\n\nце може завісити ваш браузер, ви впевнені?', + "u_scan": 'Сканування файлів...', + "u_dirstuck": 'ітератор каталогу застряг, намагаючись отримати доступ до наступних {0} елементів; пропущу:', + "u_etadone": 'Готово ({0}, {1} файлів)', + "u_etaprep": '(підготовка до завантаження)', + "u_hashdone": 'хешування завершено', + "u_hashing": 'хешування', + "u_hs": 'рукостискання...', + "u_started": "файли тепер завантажуються; дивіться [🚀]", + "u_dupdefer": "дублікат; буде оброблено після всіх інших файлів", + "u_actx": "клацніть цей текст, щоб запобігти втраті
продуктивності при переключенні на інші вікна/вкладки", + "u_fixed": "OK!  Виправлено 👍", + "u_cuerr": "не вдалося завантажити фрагмент {0} з {1};\nймовірно, нешкідливо, продовжую\n\nфайл: {2}", + "u_cuerr2": "сервер відхилив завантаження (фрагмент {0} з {1});\nспробую пізніше\n\nфайл: {2}\n\nпомилка ", + "u_ehstmp": "спробую знову; дивіться внизу справа", + "u_ehsfin": "сервер відхилив запит на завершення завантаження; повторюю...", + "u_ehssrch": "сервер відхилив запит на виконання пошуку; повторюю...", + "u_ehsinit": "сервер відхилив запит на ініціацію завантаження; повторюю...", + "u_eneths": "мережева помилка під час виконання рукостискання завантаження; повторюю...", + "u_enethd": "мережева помилка під час тестування існування цілі; повторюю...", + "u_cbusy": "чекаємо, поки сервер знову нам довірятиме після мережевого збою...", + "u_ehsdf": "на сервері закінчилося місце на диску!\n\nбуду продовжувати спроби, на випадок, якщо хтось\nзвільнить достатньо місця для продовження", + "u_emtleak1": "схоже, ваш веб-браузер може мати витік пам'яті;\nбудь ласка,", + "u_emtleak2": ' переключіться на https (рекомендується) або ', + "u_emtleak3": ' ', + "u_emtleakc": 'спробуйте наступне:\n
  • натисніть F5 для оновлення сторінки
  • потім відключіть кнопку  mt  в  ⚙️ налаштуваннях
  • і спробуйте це завантаження знову
Завантаження будуть трохи повільнішими, але що поробиш.\nВибачте за незручності !\n\nPS: chrome v107 має виправлення для цього', + "u_emtleakf": 'спробуйте наступне:\n
  • натисніть F5 для оновлення сторінки
  • потім увімкніть 🥔 (картопля) в UI завантаження
  • і спробуйте це завантаження знову
\nPS: firefox сподіваємося, матиме виправлення в якийсь момент', + "u_s404": "не знайдено на сервері", + "u_expl": "пояснити", + "u_maxconn": "більшість браузерів обмежують це до 6, але firefox дозволяє підвищити це з connections-per-server в about:config", + "u_tu": '

ПОПЕРЕДЖЕННЯ: turbo увімкнено,  клієнт може не виявити і поновити неповні завантаження; дивіться підказку turbo-кнопки

', + "u_ts": '

ПОПЕРЕДЖЕННЯ: turbo увімкнено,  результати пошуку можуть бути неправильними; дивіться підказку turbo-кнопки

', + "u_turbo_c": "turbo відключено в конфігурації сервера", + "u_turbo_g": "відключаю turbo, тому що у вас немає\nпривілеїв перегляду каталогів в цьому томі", + "u_life_cfg": 'автовидалення через хв (або годин)', + "u_life_est": 'завантаження буде видалено ---', + "u_life_max": 'ця папка забезпечує\nмакс. термін життя {0}', + "u_unp_ok": 'unpost дозволено для {0}', + "u_unp_ng": 'unpost НЕ буде дозволено', + "ue_ro": 'ваш доступ до цієї папки тільки для читання\n\n', + "ue_nl": 'ви зараз не увійшли в систему', + "ue_la": 'ви зараз увійшли як "{0}"', + "ue_sr": 'ви зараз в режимі пошуку файлів\n\nпереключіться на режим завантаження, клацнувши лупу 🔎 (поруч з великою кнопкою ПОШУК), і спробуйте завантажити знову\n\nвибачте', + "ue_ta": 'спробуйте завантажити знову, це повинно спрацювати зараз', + "ue_ab": "цей файл вже завантажується в іншу папку, і це завантаження повинно бути завершено перед тим, як файл можна буде завантажити в інше місце.\n\nВи можете перервати і забути початкове завантаження, використовуючи 🧯 зверху зліва", + "ur_1uo": "OK: Файл успішно завантажено", + "ur_auo": "OK: Всі {0} файлів успішно завантажено", + "ur_1so": "OK: Файл знайдено на сервері", + "ur_aso": "OK: Всі {0} файлів знайдено на сервері", + "ur_1un": "Завантаження невдале, вибачте", + "ur_aun": "Всі {0} завантажень невдалі, вибачте", + "ur_1sn": "Файл НЕ знайдено на сервері", + "ur_asn": "{0} файлів НЕ знайдено на сервері", + "ur_um": "Завершено;\n{0} завантажень OK,\n{1} завантажень невдалих, вибачте", + "ur_sm": "Завершено;\n{0} файлів знайдено на сервері,\n{1} файлів НЕ знайдено на сервері", + + "lang_set": "оновити сторінку, щоб зміни набули чинності?", + }, }; -var LANGS = ["eng", "nor", "chi", "deu", "rus"]; +var LANGS = ["eng", "nor", "chi", "deu", "rus", "ukr"]; if (window.langmod) langmod(); From b46b5c35e3c7b0f4698e0b214a6716851c4d6331 Mon Sep 17 00:00:00 2001 From: ed Date: Thu, 31 Jul 2025 18:19:47 +0000 Subject: [PATCH 025/174] tl cleanup --- copyparty/web/splash.js | 29 +++++++++++++++-------------- 1 file changed, 15 insertions(+), 14 deletions(-) diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 638c9ddc..56507169 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -98,33 +98,33 @@ var Ls = { "a1": "Neu laden", "b1": "Tach, wie geht's?   (Du bist nicht angemeldet)", "c1": "Abmelden", - "d1": "Zustand", // TLNote: "d2" is the tooltip for this button + "d1": "Zustand", "d2": "Zeigt den Zustand aller aktiven Threads", "e1": "Config neu laden", "e2": "Konfigurationsdatei neu laden (Accounts/Volumes/VolFlags)$Nund scannt alle e2ds-Volumes$N$NBeachte: Jegliche Änderung an globalen Einstellungen$Nbenötigt einen Neustart zum Anwenden", "f1": "Du kannst lesen:", "g1": "Du kannst hochladen nach:", "cc1": "Andere Dinge:", - "h1": "k304 deaktivieren", // TLNote: "j1" explains what k304 is + "h1": "k304 deaktivieren", "i1": "k304 aktivieren", "j1": "k304 trennt die Clientverbindung bei jedem HTTP 304, was Bugs mit problematischen Proxies vorbeugen kann (z.B. nicht ladenden Seiten), macht Dinge aber generell langsamer", "k1": "Client-Einstellungen zurücksetzen", "l1": "Melde dich an für mehr:", - "m1": "Willkommen zurück,", // TLNote: "welcome back, USERNAME" + "m1": "Willkommen zurück,", "n1": "404 Nicht gefunden  ┐( ´ -`)┌", "o1": 'or maybe you don\'t have access -- try a password or go home', "p1": "403 Verboten  ~┻━┻", "q1": 'Benutze ein Passwort oder gehe zur Homepage', "r1": "Gehe zur Homepage", ".s1": "Neu scannen", - "t1": "Aktion", // TLNote: this is the header above the "rescan" buttons + "t1": "Aktion", "u2": "time since the last server write$N( upload / rename / ... )$N$N17d = 17 days$N1h23 = 1 hour 23 minutes$N4m56 = 4 minutes 56 seconds", "v1": "Verbinden", "v2": "Benutze diesen Server als lokale Festplatte", "w1": "Zu HTTPS wechseln", "x1": "Passwort ändern", - "y1": "Shares bearbeiten", // TLNote: shows the list of folders that the user has decided to share - "z1": "Share entsperren:", // TLNote: the password prompt to see a hidden share + "y1": "Shares bearbeiten", + "z1": "Share entsperren:", "ta1": "Trage zuerst dein Passwort ein", "ta2": "Wiederhole dein Passwort zur Bestätigung:", "ta3": "Da stimmt etwas nicht; probier's nochmal", @@ -133,39 +133,40 @@ var Ls = { "ac1": "no304 aktivieren", "ad1": "Das Aktivieren von no304 deaktiviert jegliche Form von Caching; probier dies, wenn k304 nicht genug war. Dies verschwendet eine grosse Menge Netzwerk-Traffic!", "ae1": "Aktive Downloads:", - "af1": "Zeige neue Uploads", + "af1": "Zeige neue Uploads", }, - "eng": { + + "rus": { "a1": "обновить", "b1": "приветик, незнакомец   (вы не авторизованы)", "c1": "выйти", - "d1": "трассировка стека", // TLNote: "d2" is the tooltip for this button + "d1": "трассировка стека", "d2": "показывает состояние всех активных потоков", "e1": "перезагрузить конфиг", "e2": "перезагрузить файлы конфига (аккаунты/хранилища/флаги),$Nи пересканировать все хранилища с флагом e2ds$N$Nвнимание: изменения глобальных настроек$Nтребуют полного перезапуска сервера", "f1": "вы можете видеть:", "g1": "вы можете загружать файлы в:", "cc1": "всякая всячина:", - "h1": "отключить k304", // TLNote: "j1" explains what k304 is + "h1": "отключить k304", "i1": "включить k304", "j1": "включённый k304 будет отключать вас при получении HTTP 304, что может помочь при работе с некоторыми глючными прокси (перестают загружаться страницы), но это также сделает работу клиента медленнее", "k1": "сбросить локальные настройки", "l1": "авторизуйтесь для других опций:", - "m1": "с возвращением,", // TLNote: "welcome back, USERNAME" + "m1": "с возвращением,", "n1": "404 не найдено  ┐( ´ -`)┌", "o1": 'или у вас нет доступа -- попробуйте авторизоваться или вернуться на главную', "p1": "403 доступ запрещён  ~┻━┻", "q1": 'авторизуйтесь или вернитесь на главную', "r1": "вернуться на главную", ".s1": "пересканировать", - "t1": "действия", // TLNote: this is the header above the "rescan" buttons + "t1": "действия", "u2": "время с последней записи на сервер$N( загрузка / переименование / ... )$N$N17d = 17 дней$N1h23 = 1 час 23 минут$N4m56 = 4 минут 56 секунд", "v1": "подключить", "v2": "использовать сервер как локальный диск", "w1": "перейти на https", "x1": "поменять пароль", - "y1": "управление доступом", // TLNote: shows the list of folders that the user has decided to share - "z1": "разблокировать:", // TLNote: the password prompt to see a hidden share + "y1": "управление доступом", + "z1": "разблокировать:", "ta1": "сначала введите свой новый пароль", "ta2": "повторите новый пароль:", "ta3": "опечатка; попробуйте снова", From a2faf4e1e9db217976663049cf02414ea095acf0 Mon Sep 17 00:00:00 2001 From: crypt0rr <57799908+crypt0rr@users.noreply.github.com> Date: Thu, 31 Jul 2025 20:16:26 +0200 Subject: [PATCH 026/174] Added Dutch(NL) translation --- copyparty/web/browser.js | 559 +++++++++++++++++++++++++++++++++++++++ 1 file changed, 559 insertions(+) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index b8b8b202..6c39b78b 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -2510,6 +2510,565 @@ var Ls = { "lang_set": "Neuladen um Änderungen anzuwenden?", }, + "nl": { + "tt": "Dutch", + + "cols": { + "c": "actieknoppen", + "dur": "duur", + "q": "kwaliteit / bitrate", + "Ac": "audiocodec", + "Vc": "videocodec", + "Fmt": "formaat / container", + "Ahash": "audio-controlegetal", + "Vhash": "video-controlegetal", + "Res": "resolutie", + "T": "bestandstype", + "aq": "audiokwaliteit / bitrate", + "vq": "videokwaliteit / bitrate", + "pixfmt": "subsampling / pixelstructuur", + "resw": "horizontale resolutie", + "resh": "verticale resolutie", + "chs": "audiokanalen", + "hz": "samplefrequentie" + }, + + "hks": [ + [ + "misc", + ["ESC", "sluit verschillende vensters"], + + "file-manager", + ["G", "wissel tussen lijst- en rasterweergave"], + ["T", "wissel miniaturen / pictogrammen"], + ["⇧ A/D", "miniatuurgrootte aanpassen"], + ["ctrl-K", "verwijder selectie"], + ["ctrl-X", "knip selectie naar klembord"], + ["ctrl-C", "kopieer selectie naar klembord"], + ["ctrl-V", "plak (verplaats/kopieer) hier"], + ["Y", "download selectie"], + ["F2", "hernoem selectie"], + + "file-list-sel", + ["space", "selectie van bestand wisselen"], + ["↑/↓", "verplaats selectiecursor"], + ["ctrl ↑/↓", "verplaats cursor én scherm"], + ["⇧ ↑/↓", "selecteer vorig/volgend bestand"], + ["ctrl-A", "selecteer alle bestanden / mappen"] + ], [ + "navigation", + ["B", "wissel kruimelpad / navigatiepaneel"], + ["I/K", "vorige/volgende map"], + ["M", "bovenliggende map (of vouw huidige in)"], + ["V", "wissel mappen / tekstbestanden in navigatie"], + ["A/D", "formaat van navigatiepaneel"], + ], [ + "audio-player", + ["J/L", "vorig/volgend nummer"], + ["U/O", "10 sec terug/vooruit"], + ["0..9", "ga naar 0%..90%"], + ["P", "afspelen/pauzeren (start ook)"], + ["S", "selecteer afgespeeld nummer"], + ["Y", "download nummer"], + ], [ + "image-viewer", + ["J/L, ←/→", "vorige/volgende afbeelding"], + ["Home/End", "eerste/laatste afbeelding"], + ["F", "volledig scherm"], + ["R", "roteer met de klok mee"], + ["⇧ R", "roteer tegen de klok in"], + ["S", "selecteer afbeelding"], + ["Y", "download afbeelding"], + ], [ + "video-player", + ["U/O", "10 sec terug/vooruit"], + ["P/K/Space", "afspelen/pauzeren"], + ["C", "speel volgende automatisch af"], + ["V", "herhaal"], + ["M", "dempen"], + ["[ and ]", "stel herhaal-interval in"], + ], [ + "textfile-viewer", + ["I/K", "vorig/volgend bestand"], + ["M", "sluit tekstbestand"], + ["E", "bewerk tekstbestand"], + ["S", "selecteer bestand (voor knippen/kopiëren/hernoemen)"] + ] + ], + + "m_ok": "OK", + "m_ng": "Annuleren", + + "enable": "Inschakelen", + "danger": "GEVAAR", + "clipped": "gekopieerd naar klembord", + + "ht_s1": "seconde", + "ht_s2": "seconden", + "ht_m1": "minuut", + "ht_m2": "minuten", + "ht_h1": "uur", + "ht_h2": "uren", + "ht_d1": "dag", + "ht_d2": "dagen", + "ht_and": " en ", + + "goh": "configuratiescherm", + "gop": 'vorige map">vorige', + "gou": 'bovenliggende map">omhoog', + "gon": 'volgende map">volgende', + "logout": "Uitloggen ", + "access": " toegang", + "ot_close": "submenu sluiten", + "ot_search": "zoek naar bestanden op basis van attributen, pad / naam, muziektags of een combinatie hiervan$N$Nfoo bar = bevat zowel «foo» als «bar»,$Nfoo -bar = bevat «foo» maar niet «bar»,$N^yana .opus$ = begint met «yana» en is een «opus»-bestand$N\"try unite\" = bevat exact «try unite»$N$Nhet datumformaat is iso-8601, zoals$N2009-12-31 of 2020-09-12 23:30:00", + "ot_unpost": "unpost: verwijder je recente uploads of annuleer onafgeronde uploads", + "ot_bup": "bup: eenvoudige uploader, zelfs compatibel met Netscape 4.0", + "ot_mkdir": "mkdir: maak een nieuwe map", + "ot_md": "new-md: maak een nieuw markdown-document", + "ot_msg": "msg: stuur een bericht naar de serverlog", + "ot_mp": "media player opties", + "ot_cfg": "configuratie-opties", + "ot_u2i": "up2k: upload bestanden (als je schrijfrechten hebt) of schakel naar zoekmodus om te zien of ze al op de server bestaan$N$Nuploads zijn hervatbaar, multithreaded en behouden bestandstijdstempels, maar gebruiken meer CPU dan [🎈] (de eenvoudige uploader)

tijdens het uploaden wordt dit pictogram een voortgangsindicator!", + "ot_u2w": "up2k: upload bestanden met hervat-ondersteuning (sluit je browser en voeg later dezelfde bestanden weer toe)$N$Nmultithreaded en bestandstijdstempels worden behouden, maar gebruikt meer CPU dan [🎈] (de eenvoudige uploader)

tijdens het uploaden wordt dit pictogram een voortgangsindicator!", + "ot_noie": "Gebruik alsjeblieft Chrome / Firefox / Edge", + + "ab_mkdir": "maak map", + "ab_mkdoc": "nieuw markdown-document", + "ab_msg": "stuur bericht naar serverlog", + + "ay_path": "spring naar mappen", + "ay_files": "spring naar bestanden", + + "wt_ren": "hernoem geselecteerde items$NToets: F2", + "wt_del": "verwijder geselecteerde items$NToets: ctrl-K", + "wt_cut": "knip geselecteerde items (en plak elders)$NToets: ctrl-X", + "wt_cpy": "kopieer geselecteerde items naar klembord$N(om ergens anders te plakken)$NToets: ctrl-C", + "wt_pst": "plak eerder geknipte/gekopieerde selectie$NToets: ctrl-V", + "wt_selall": "selecteer alle bestanden$NToets: ctrl-A (wanneer bestand geselecteerd)", + "wt_selinv": "selectie omkeren", + "wt_zip1": "download deze map als archief", + "wt_selzip": "download selectie als archief", + "wt_seldl": "download selectie als losse bestanden$NToets: Y", + "wt_npirc": "kopieer trackinfo in irc-formaat", + "wt_nptxt": "kopieer trackinfo als platte tekst", + "wt_m3ua": "voeg toe aan m3u-afspeellijst (klik later op 📻kopieer)", + "wt_m3uc": "kopieer m3u-afspeellijst naar klembord", + "wt_grid": "wissel tussen raster- en lijstweergave$NToets: G", + "wt_prev": "vorig nummer$NToets: J", + "wt_play": "afspelen / pauzeren$NToets: P", + "wt_next": "volgend nummer$NToets: L", + + "ul_par": "parallelle uploads:", + "ut_rand": "bestandsnamen willekeurig maken", + "ut_u2ts": "kopieer laatste wijzigingstijdstip$Nvan je bestandssysteem naar de server\">📅", + "ut_ow": "bestaande bestanden op server overschrijven?$N🛡️: nooit (maakt een nieuwe bestandsnaam)$N🕒: overschrijven als serverbestand ouder is$N♻️: altijd overschrijven als de bestanden verschillen", + "ut_mt": "hash andere bestanden terwijl er geüpload wordt$N$NUitschakelen als CPU of HDD traag is", + "ut_ask": "vraag om bevestiging voor upload start\">💭", + "ut_pot": "verbeter uploadsnelheid op trage apparaten$Ndoor de interface eenvoudiger te maken", + "ut_srch": "upload niet echt, maar controleer of de bestanden al$Nop de server bestaan (scant alle mappen die je kunt lezen)", + "ut_par": "pauzeer uploads door deze op 0 te zetten$N$Nverhoog bij langzame of hoge-latentie verbinding$N$Nlaat op 1 in LAN of bij trage server HDD", + "ul_btn": "sleep bestanden / mappen
hierheen (of klik op mij)", + "ul_btnu": "U P L O A D", + "ul_btns": "Z O E K", + + "ul_hash": "hash", + "ul_send": "verzenden", + "ul_done": "klaar", + "ul_idle1": "er staan nog geen uploads in de wachtrij", + "ut_etah": "gemiddelde hash-snelheid en geschatte resterende tijd", + "ut_etau": "gemiddelde upload-snelheid en geschatte resterende tijd", + "ut_etat": "gemiddelde totale snelheid en geschatte resterende tijd", + + "uct_ok": "succesvol voltooid", + "uct_ng": "niet-goed: mislukt / geweigerd / niet gevonden", + "uct_done": "ok en ng gecombineerd", + "uct_bz": "aan het hashen of uploaden", + "uct_q": "inactief, in wachtrij", + + "utl_name": "bestandsnaam", + "utl_ulist": "lijst", + "utl_ucopy": "kopiëren", + "utl_links": "links", + "utl_stat": "status", + "utl_prog": "voortgang", + + "utl_404": "404", + "utl_err": "FOUT", + "utl_oserr": "OS-fout", + "utl_found": "gevonden", + "utl_defer": "uitstellen", + "utl_yolo": "YOLO", + "utl_done": "klaar", + + "ul_flagblk": "de bestanden zijn aan de wachtrij toegevoegd
maar er is een actieve up2k in een ander browsertabblad,
dus eerst wachten tot die klaar is", + "ul_btnlk": "de serverconfiguratie heeft deze schakelaar vergrendeld in deze status", + + "udt_up": "Uploaden", + "udt_srch": "Zoeken", + "udt_drop": "sleep hierheen", + + "u_nav_m": "
oké, wat heb je?
Enter = Bestanden (een of meer)\nESC = Eén map (inclusief submappen)", + "u_nav_b": "BestandenEén map", + + "cl_opts": "schakelaars", + "cl_themes": "thema", + "cl_langs": "taal", + "cl_ziptype": "map downloaden", + "cl_uopts": "up2k-opties", + "cl_favico": "favicon", + "cl_bigdir": "grote mappen", + "cl_hsort": "#sorteer", + "cl_keytype": "toetsnotatie", + "cl_hiddenc": "verborgen kolommen", + "cl_hidec": "verbergen", + "cl_reset": "resetten", + "cl_hpick": "tik op kolomkoppen om ze hieronder te verbergen", + "cl_hcancel": "verbergen van kolommen geannuleerd", + + "ct_grid": "田 het raster", + "ct_ttips": "ℹ️ tooltips", + "ct_thumb": "in rasterweergave, schakel pictogrammen of miniaturen in$NHotkey: T", + "ct_csel": "gebruik CTRL en SHIFT om bestanden te selecteren in rasterweergave", + "ct_ihop": "wanneer de afbeeldingsviewer is gesloten, scroll naar het laatst bekeken bestand", + "ct_dots": "toon verborgen bestanden (indien toegestaan door server)", + "ct_qdel": "bij verwijderen slechts éénmaal bevestiging vragen", + "ct_dir1st": "mappen eerst sorteren", + "ct_nsort": "natuurlijke sortering (voor bestandsnamen met cijfers vooraan)", + "ct_readme": "toon README.md in mapweergave", + "ct_idxh": "toon index.html in plaats van mapinhoud", + "ct_sbars": "toon schuifbalken", + + "cut_umod": "indien een bestand al bestaat op de server, werk de wijzigingsdatum bij om overeen te komen met je lokale bestand (vereist schrijf- en verwijderrechten)", + + "cut_turbo": "de yolo-knop, je wilt dit waarschijnlijk NIET aanzetten:$N$Ngebruik dit als je een grote hoeveelheid bestanden aan het uploaden was en je moest herstarten, en je zo snel mogelijk wilt doorgaan$N$Nvervangt de hash-controle door een simpele “heeft dit dezelfde bestandsgrootte?” controle, dus bij andere inhoud wordt het NIET geüpload$N$Nzet dit uit wanneer de upload klaar is en 'upload' dezelfde bestanden opnieuw om te verifiëren", + + "cut_datechk": "heeft geen effect tenzij turbo is ingeschakeld$N$Nverlaagt de yolo-factor iets; controleert of de bestandstijdstempels overeenkomen$N$Nzou theoretisch onvolledige/corrupte uploads moeten detecteren, maar is geen vervanging voor een verificatie achteraf met turbo uit", + + "cut_u2sz": "grootte (in MiB) van elk uploadblok; grote waarden werken beter over de Atlantische Oceaan. Gebruik lage waarden bij onstabiele verbindingen", + + "cut_flag": "zorg dat slechts één tabblad tegelijk uploadt$N -- andere tabbladen moeten dit ook aan hebben$N -- werkt alleen op hetzelfde domein", + + "cut_az": "upload bestanden in alfabetische volgorde in plaats van kleinste eerst$N$Nalfabetisch kan helpen om fouten op de server sneller te herkennen, maar is iets langzamer bij snelle verbindingen", + + "cut_nag": "OS-melding als upload klaar is$N(alleen als het tabblad niet actief is)", + "cut_sfx": "geluidssignaal als upload klaar is$N(alleen als het tabblad niet actief is)", + + "cut_mt": "gebruik multithreading voor snellere hashing$N$Nmaakt gebruik van web-workers en vraagt$Nmeer RAM (tot 512 MiB extra)$N$Nmaakt https 30% sneller, http 4,5x sneller", + + "cut_wasm": "gebruik wasm in plaats van ingebouwde browser-hasher; sneller op Chrome, maar zwaarder voor CPU en kan bugs veroorzaken in oudere Chrome-versies", + + "cft_text": "favicon-tekst (leeg laten en vernieuwen om uit te schakelen)", + "cft_fg": "voorgrondkleur", + "cft_bg": "achtergrondkleur", + + "cdt_lim": "maximaal aantal bestanden om in een map te tonen", + "cdt_ask": "bij scrollen naar beneden,$Nvraag wat te doen in plaats van meer bestanden laden", + "cdt_hsort": "hoeveel sorteerregels (<code>,sorthref</code>) opnemen in media-URLs. Waarde 0 negeert ook sorteringen in links", + + "tt_entree": "toon navigatievenster (mapstructuur)$NHotkey: B", + "tt_detree": "toon kruimelpad$NHotkey: B", + "tt_visdir": "scroll naar geselecteerde map", + "tt_ftree": "schakel boomstructuur / tekstbestanden$NHotkey: V", + "tt_pdock": "toon bovenliggende mappen bovenaan vastgezet", + "tt_dynt": "automatisch uitbreiden bij uitklappen boomstructuur", + "tt_wrap": "regelterugloop", + "tt_hover": "toon overlopende regels bij hover$N(breekt scrollen tenzij muis in linkerrand staat)", + + "im_hnf": "die afbeelding bestaat niet meer", + + "f_empty": "deze map is leeg", + "f_chide": "dit zal de kolom «{0}» verbergen\n\nje kunt kolommen weer zichtbaar maken in het instellingen-tabblad", + "f_bigtxt": "dit bestand is {0} MiB groot — echt als tekst weergeven?", + "f_bigtxt2": "alleen het einde van het bestand weergeven? dit schakelt ook volgen in, waarbij nieuwe regels in realtime verschijnen", + "fbd_more": "
toont {0} van {1} bestanden; toon {2} of toon alles
", + "fbd_all": "
toont {0} van {1} bestanden; toon alles
", + "f_anota": "slechts {0} van de {1} items zijn geselecteerd;\nscroll eerst helemaal naar beneden om alles te selecteren", + + "f_dls": "de bestandslinks in deze map zijn\nomgezet naar downloadlinks", + + "f_partial": "Om veilig een bestand te downloaden dat nog wordt geüpload, klik op het bestand met dezelfde naam maar zonder de extensie .PARTIAL. Klik op ANNULEREN of Escape om dit te doen.\n\nKlikken op OK / Enter negeert deze waarschuwing en downloadt het .PARTIAL-bestand, wat vrijwel zeker corrupte data oplevert.", + + "ft_paste": "plak {0} items$NSneltoets: ctrl-V", + "fr_eperm": "kan niet hernoemen:\nje hebt geen 'verplaats'-rechten in deze map", + "fd_eperm": "kan niet verwijderen:\nje hebt geen 'verwijder'-rechten in deze map", + "fc_eperm": "kan niet knippen:\nje hebt geen 'verplaats'-rechten in deze map", + "fp_eperm": "kan niet plakken:\nje hebt geen 'schrijf'-rechten in deze map", + "fr_emore": "selecteer ten minste één item om te hernoemen", + "fd_emore": "selecteer ten minste één item om te verwijderen", + "fc_emore": "selecteer ten minste één item om te knippen", + "fcp_emore": "selecteer ten minste één item om naar klembord te kopiëren", + + "fs_sc": "deel de map waarin je je bevindt", + "fs_ss": "deel de geselecteerde bestanden", + "fs_just1d": "je kunt niet meer dan één map selecteren,\nof bestanden en mappen door elkaar", + "fs_abrt": "❌ annuleren", + "fs_rand": "🎲 willek.naam", + "fs_go": "✅ deel aanmaken", + "fs_name": "naam", + "fs_src": "bron", + "fs_pwd": "wachtw", + "fs_exp": "verloopt", + "fs_tmin": "min", + "fs_thrs": "uur", + "fs_tdays": "dagen", + "fs_never": "eeuwig", + "fs_pname": "optionele linknaam; willekeurig indien leeg", + "fs_tsrc": "het bestand of de map om te delen", + "fs_ppwd": "optioneel wachtwoord", + "fs_w8": "deel aanmaken...", + "fs_ok": "druk op Enter/OK om te kopiëren\nof op ESC/Annuleer om te sluiten", + + "frt_dec": "kan sommige kapotte bestandsnamen herstellen\">url-decoderen", + "frt_rst": "herstel gewijzigde bestandsnamen naar origineel\">↺ reset", + "frt_abrt": "annuleren en venster sluiten\">❌ annuleren", + "frb_apply": "HERNOEMEN TOEPASSEN", + "fr_adv": "batch / metadata / patroon-hernoemen\">geavanceerd", + "fr_case": "hoofdlettergevoelige regex\">hoofdletter", + "fr_win": "Windows-veilige namen; vervang <>:\"\\|?* door Japanse tekens\">win", + "fr_slash": "vervang / door een teken dat geen nieuwe mappen maakt\">geen /", + "fr_re": "regex-zoekpatroon voor originele bestandsnamen; groepen kun je hieronder gebruiken met bijv. <code>(1)</code>", + "fr_fmt": "geïnspireerd op foobar2000:$N<code>(title)</code> wordt vervangen door titel,$N<code>[(artist) - ](title)</code> slaat [dit] over als artiest leeg is$N<code>$lpad((tn),2,0)</code> vult tracknummer aan tot 2 cijfers", + "fr_pdel": "verwijderen", + "fr_pnew": "opslaan als", + "fr_pname": "geef een naam voor je nieuwe preset", + "fr_aborted": "geannuleerd", + "fr_lold": "oude naam", + "fr_lnew": "nieuwe naam", + "fr_tags": "tags voor geselecteerde bestanden (alleen-lezen, ter referentie):", + "fr_busy": "bezig met hernoemen van {0} items...\n\n{1}", + "fr_efail": "hernoemen mislukt:\n", + "fr_nchg": "{0} van de nieuwe namen zijn aangepast door win en/of geen /\n\nOK om door te gaan met deze aangepaste namen?", + + "fd_ok": "verwijderen geslaagd", + "fd_err": "verwijderen mislukt:\n", + "fd_none": "niets verwijderd; mogelijk geblokkeerd door serverconfiguratie (xbd)?", + "fd_busy": "bezig met verwijderen van {0} items...\n\n{1}", + "fd_warn1": "WIL JE deze {0} items VERWIJDEREN?", + "fd_warn2": "Laatste kans! Dit kan niet ongedaan worden. Verwijderen?", + + "fc_ok": "geknipt: {0} items", + "fc_warn": "geknipt: {0} items\n\nmaar: alleen dit browser-tabblad kan ze plakken\n(aangezien de selectie enorm groot is)", + + "fcc_ok": "{0} items naar klembord gekopieerd", + "fcc_warn": "{0} items naar klembord gekopieerd\n\nmaar: alleen dit browser-tabblad kan ze plakken\n(aangezien de selectie enorm groot is)", + + "fp_apply": "gebruik deze namen", + "fp_ecut": "knip of kopieer eerst bestanden / mappen om te kunnen plakken / verplaatsen\n\nlet op: knippen / plakken werkt ook tussen verschillende tabs", + "fp_ename": "{0} items kunnen hier niet worden verplaatst omdat de namen al bestaan. Geef hieronder nieuwe namen op om verder te gaan, of laat het veld leeg om ze over te slaan:", + "fcp_ename": "{0} items kunnen hier niet worden gekopieerd omdat de namen al bestaan. Geef hieronder nieuwe namen op om verder te gaan, of laat het veld leeg om ze over te slaan:", + "fp_emore": "er zijn nog bestandsnaam-conflicten die opgelost moeten worden", + "fp_ok": "verplaatsen geslaagd", + "fcp_ok": "kopiëren geslaagd", + "fp_busy": "bezig met verplaatsen van {0} items...\n\n{1}", + "fcp_busy": "bezig met kopiëren van {0} items...\n\n{1}", + "fp_err": "verplaatsen mislukt:\n", + "fcp_err": "kopiëren mislukt:\n", + "fp_confirm": "wil je deze {0} items hier verplaatsen?", + "fcp_confirm": "wil je deze {0} items hier kopiëren?", + "fp_etab": "kon klembord van ander tabblad niet lezen", + "fp_name": "je uploadt een bestand vanaf je apparaat. Geef het een naam:", + "fp_both_m": "
kies wat je wilt plakken
Enter = Verplaats {0} bestanden van «{1}»\nESC = Upload {2} bestanden vanaf je apparaat", + "fcp_both_m": "
kies wat je wilt plakken
Enter = Kopieer {0} bestanden van «{1}»\nESC = Upload {2} bestanden vanaf je apparaat", + "fp_both_b": "VerplaatsenUploaden", + "fcp_both_b": "KopiërenUploaden", + + "mk_noname": "typ eerst een naam in het tekstveld links voordat je dat doet :p", + + "tv_load": "Tekstdocument laden:\n\n{0}\n\n{1}% ({2} van {3} MiB geladen)", + "tv_xe1": "kon tekstbestand niet laden:\n\nfout ", + "tv_xe2": "404, bestand niet gevonden", + "tv_lst": "lijst met tekstbestanden in", + "tvt_close": "terug naar mapweergave$NSneltoets: M (of Esc)\">❌ sluiten", + "tvt_dl": "download dit bestand$NSneltoets: Y\">💾 download", + "tvt_prev": "toon vorig document$NSneltoets: i\">⬆ vorige", + "tvt_next": "toon volgend document$NSneltoets: K\">⬇ volgende", + "tvt_sel": "selecteer bestand   ( om te knippen / kopiëren / verwijderen / ... )$NSneltoets: S\">selecteer", + "tvt_edit": "open bestand in teksteditor$NSneltoets: E\">✏️ bewerk", + "tvt_tail": "bewaking inschakelen; toon nieuwe regels in realtime\">📡 volgen", + "tvt_wrap": "regelafbreking\">↵", + "tvt_atail": "scroll vergrendelen naar onderkant van pagina\">⚓", + "tvt_ctail": "terminalkleuren decoderen (ansi-codes)\">🌈", + "tvt_ntail": "scrollback-limiet (hoeveel tekstbytes worden geladen)", + + "m3u_add1": "nummer toegevoegd aan m3u-afspeellijst", + "m3u_addn": "{0} nummers toegevoegd aan m3u-afspeellijst", + "m3u_clip": "m3u-afspeellijst gekopieerd naar klembord\n\nmaak een nieuw tekstbestand met naam iets.m3u en plak daar de afspeellijst in; dan kun je het afspelen", + + "gt_vau": "toon geen video, speel alleen het geluid af\">🎧", + "gt_msel": "bestandsselectie inschakelen; ctrl-klik om te forceren$N$Nactief: dubbelklik opent bestand / map$N$NSneltoets: S\">meerdere selecteren", + "gt_crop": "miniaturen bijsnijden naar midden\">bijsnijd", + "gt_3x": "hoge resolutie miniaturen\">3x", + "gt_zoom": "zoomen", + "gt_chop": "afsnijden", + "gt_sort": "sorteren op", + "gt_name": "naam", + "gt_sz": "grootte", + "gt_ts": "datum", + "gt_ext": "type", + "gt_c1": "kortere bestandsnamen tonen (minder weergeven)", + "gt_c2": "langere bestandsnamen tonen (meer weergeven)", + + "sm_w8": "bezig met zoeken...", + "sm_prev": "zoekresultaten hieronder komen van een vorige zoekopdracht:\n ", + "sl_close": "zoekresultaten sluiten", + "sl_hits": "{0} treffers gevonden", + "sl_moar": "meer laden", + + "s_sz": "grootte", + "s_dt": "datum", + "s_rd": "pad", + "s_fn": "naam", + "s_ta": "tags", + "s_ua": "upload@", + "s_ad": "geav.", + "s_s1": "minimaal MiB", + "s_s2": "maximaal MiB", + "s_d1": "min. iso8601", + "s_d2": "max. iso8601", + "s_u1": "geüpload na", + "s_u2": "en/of voor", + "s_r1": "pad bevat   (spatiegescheiden)", + "s_f1": "naam bevat   (negeren met -nope)", + "s_t1": "tags bevat   (^=begin, eind=$)", + "s_a1": "specifieke metagegevens", + + "md_eshow": "kan niet weergeven ", + "md_off": "[📜readme] uitgeschakeld in [⚙️] -- document verborgen", + + "badreply": "Antwoord van server kon niet worden verwerkt", + + "xhr403": "403: Toegang geweigerd\n\nprobeer F5, je bent misschien uitgelogd", + "xhr0": "onbekend (mogelijk verbinding met server verbroken of server offline)", + "cf_ok": "sorry daarvoor — DD" + wah + "oS-bescherming trad in werking\n\ndit zou binnen ~30 sec moeten herstellen\n\ngebeurt er niets, druk dan op F5 om de pagina opnieuw te laden", + "tl_xe1": "kon submappen niet ophalen:\n\nfout ", + "tl_xe2": "404: Map niet gevonden", + "fl_xe1": "kon bestanden in map niet ophalen:\n\nfout ", + "fl_xe2": "404: Map niet gevonden", + "fd_xe1": "kon submap niet aanmaken:\n\nfout ", + "fd_xe2": "404: Bovenliggende map niet gevonden", + "fsm_xe1": "kon bericht niet verzenden:\n\nfout ", + "fsm_xe2": "404: Bovenliggende map niet gevonden", + "fu_xe1": "kon onpost-lijst niet laden van server:\n\nfout ", + "fu_xe2": "404: Bestand niet gevonden??", + + "fz_tar": "niet-gecomprimeerd gnu-tar-bestand (linux / mac)", + "fz_pax": "niet-gecomprimeerde tar in pax-formaat (langzamer)", + "fz_targz": "gnu-tar met gzip-compressie niveau 3$N$Ndit is meestal erg traag, gebruik liever on-gecomprimeerde tar", + "fz_tarxz": "gnu-tar met xz-compressie niveau 1$N$Ndit is meestal erg traag, gebruik liever on-gecomprimeerde tar", + "fz_zip8": "zip met utf8-bestandsnamen (kan raar doen op windows 7 en ouder)", + "fz_zipd": "zip met klassieke cp437-bestandsnamen, voor erg oude software", + "fz_zipc": "cp437 met crc32 vooraf berekend,$Nvoor MS-DOS PKZIP v2.04g (oktober 1993)$N(duurt langer voor downloaden start)", + + "un_m1": "je kunt je recente uploads verwijderen (of onafgemaakte afbreken) hieronder", + "un_upd": "verversen", + "un_m4": "of deel de bestanden hieronder zichtbaar:", + "un_ulist": "toon", + "un_ucopy": "kopieer", + "un_flt": "optionele filter:  URL moet bevatten", + "un_fclr": "filter wissen", + "un_derr": "verwijderen mislukt:\n", + "un_f5": "er ging iets mis, probeer te verversen of druk op F5", + "un_uf5": "sorry, maar je moet de pagina verversen (bijv. met F5 of CTRL-R) voordat deze upload kan worden afgebroken", + "un_nou": "waarschuwing: server te druk om onafgemaakte uploads te tonen; klik later op de \"ververs\"-link", + "un_noc": "waarschuwing: unpost van voltooide bestanden is niet ingeschakeld/toegestaan in serverconfiguratie", + "un_max": "alleen eerste 2000 bestanden worden getoond (gebruik de filter)", + "un_avail": "{0} recente uploads kunnen worden verwijderd
{1} onafgemaakte kunnen worden afgebroken", + "un_m2": "gesorteerd op uploadtijd; meest recent eerst:", + "un_no1": "haha! geen uploads zijn recent genoeg", + "un_no2": "haha! geen uploads met die filter zijn recent genoeg", + "un_next": "verwijder de volgende {0} bestanden hieronder", + "un_abrt": "afbreken", + "un_del": "verwijderen", + "un_m3": "laden van je recente uploads...", + "un_busy": "bezig met verwijderen van {0} bestanden...", + "un_clip": "{0} links gekopieerd naar klembord", + + "u_https1": "je zou", + "u_https2": "moeten overschakelen naar https", + "u_https3": "voor betere prestaties", + "u_ancient": "je browser is indrukwekkend oud -- misschien kun je beter bup gebruiken", + "u_nowork": "vereist firefox 53+ of chrome 57+ of iOS 11+", + "tail_2old": "vereist firefox 105+ of chrome 71+ of iOS 14.5+", + "u_nodrop": "je browser is te oud voor drag-and-drop uploaden", + "u_notdir": "dat is geen map!\nje browser is te oud,\nprobeer dragdrop in plaats daarvan", + "u_uri": "om afbeeldingen te slepen vanuit andere browservensters,\nsleep ze naar de grote uploadknop", + "u_enpot": "overschakelen naar aardappel-UI (kan uploadsnelheid verbeteren)", + "u_depot": "overschakelen naar mooie UI (kan uploadsnelheid verminderen)", + "u_gotpot": "overschakelen naar aardappel-UI voor snellere upload,\n\nvoel je vrij om terug te schakelen!", + "u_pott": "

bestanden:   {0} voltooid,   {1} mislukt,   {2} bezig,   {3} in wachtrij

", + "u_ever": "dit is de basis-uploader; up2k vereist minimaal
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": "dit is de basis-uploader; up2k is beter", + "u_uput": "optimaliseer voor snelheid (sla checksum over)", + "u_ewrite": "je hebt geen schrijfrechten op deze map", + "u_eread": "je hebt geen leesrechten op deze map", + "u_enoi": "bestandszoekfunctie is niet ingeschakeld in serverconfig", + "u_enoow": "overschrijven werkt hier niet; verwijderrechten nodig", + "u_badf": "Deze {0} bestanden (van {1} totaal) zijn overgeslagen, mogelijk door bestandsrechten:\n\n", + "u_blankf": "Deze {0} bestanden (van {1} totaal) zijn leeg; toch uploaden?\n\n", + "u_applef": "Deze {0} bestanden (van {1} totaal) zijn waarschijnlijk ongewenst;\nDruk op OK/Enter om deze bestanden OVER TE SLAAN,\nDruk op Annuleer/ESC om ze WEL te uploaden:\n\n", + "u_just1": "\nMisschien werkt het beter als je maar één bestand selecteert", + "u_ff_many": "als je Linux / MacOS / Android gebruikt, kan deze hoeveelheid bestanden Firefox doen crashen!\nals dat gebeurt, probeer opnieuw (of gebruik Chrome).", + "u_up_life": "Deze upload wordt verwijderd van de server\n{0} na voltooiing", + "u_asku": "deze {0} bestanden uploaden naar {1}", + "u_unpt": "je kunt deze upload ongedaan maken/verwijderen via de 🧯 linksboven", + "u_bigtab": "op het punt om {0} bestanden te tonen\n\nkan je browser doen crashen, zeker weten?", + "u_scan": "Bestanden scannen...", + "u_dirstuck": "mapscanner bleef hangen bij toegang tot de volgende {0} items; worden overgeslagen:", + "u_etadone": "Klaar ({0}, {1} bestanden)", + "u_etaprep": "(bezig met voorbereiden upload)", + "u_hashdone": "hashing voltooid", + "u_hashing": "hashen", + "u_hs": "handshake...", + "u_started": "de bestanden worden nu geüpload; zie [🚀]", + "u_dupdefer": "duplicaat; wordt verwerkt na alle andere bestanden", + "u_actx": "klik hier om verlies van prestaties te voorkomen\nbij wisselen tussen vensters/tabs", + "u_fixed": "OK!  Opgelost 👍", + "u_cuerr": "uploaden van blok {0} van {1} mislukt;\nwaarschijnlijk onschuldig, ga verder\n\nbestand: {2}", + "u_cuerr2": "server weigerde upload (blok {0} van {1});\nprobeert later opnieuw\n\nbestand: {2}\n\nfout ", + "u_ehstmp": "probeert opnieuw; zie rechtsonder", + "u_ehsfin": "server weigerde het afronden van de upload; probeert opnieuw...", + "u_ehssrch": "server weigerde zoekverzoek; probeert opnieuw...", + "u_ehsinit": "server weigerde het starten van de upload; probeert opnieuw...", + "u_eneths": "netwerkfout tijdens upload-handshake; probeert opnieuw...", + "u_enethd": "netwerkfout tijdens test op doelbestand; probeert opnieuw...", + "u_cbusy": "wacht tot de server ons weer vertrouwt na een netwerkprobleem...", + "u_ehsdf": "server heeft geen schijfruimte meer!\n\nblijft proberen in de hoop dat iemand\nruimte vrijmaakt", + "u_emtleak1": "het lijkt erop dat je browser een geheugenlek heeft;\nprobeer", + "u_emtleak2": ' over te schakelen naar https (aanbevolen) of ', + "u_emtleak3": ' ', + "u_emtleakc": 'probeer het volgende:\n
  • druk op F5 om de pagina te vernieuwen
  • schakel dan de knop  mt  uit bij  ⚙️ instellingen
  • probeer de upload opnieuw
Uploads zullen wat trager zijn, maar oké.\nSorry voor het ongemak!\n\nPS: chrome v107 heeft hier een bugfix voor', + "u_emtleakf": 'probeer het volgende:\n
  • druk op F5 om de pagina te vernieuwen
  • schakel daarna 🥔 (aardappelmodus) in de upload-UI in
  • probeer de upload opnieuw
\nPS: firefox heeft hopelijk binnenkort een bugfix', + "u_s404": "niet gevonden op server", + "u_expl": "uitleg", + "u_maxconn": "de meeste browsers beperken dit tot 6, maar firefox laat je dit verhogen met connections-per-server in about:config", + "u_tu": "

WAARSCHUWING: turbo ingeschakeld,  client detecteert mogelijk geen onvoltooide uploads; zie tooltip bij turbo-knop

", + "u_ts": "

WAARSCHUWING: turbo ingeschakeld,  zoekresultaten kunnen onjuist zijn; zie tooltip bij turbo-knop

", + "u_turbo_c": "turbo is uitgeschakeld in serverconfiguratie", + "u_turbo_g": "turbo uitgeschakeld omdat je geen mapinhoud mag bekijken binnen dit volume", + "u_life_cfg": "automatisch verwijderen na min (of uur)", + "u_life_est": "upload wordt verwijderd op ---", + "u_life_max": "deze map heeft een\nmaximale bewaartijd van {0}", + "u_unp_ok": "verwijderen is toegestaan voor {0}", + "u_unp_ng": "verwijderen is NIET toegestaan", + "ue_ro": "je hebt alleen-lezen toegang tot deze map\n\n", + "ue_nl": "je bent momenteel niet ingelogd", + "ue_la": "je bent momenteel ingelogd als \"{0}\"", + "ue_sr": "je bent momenteel in bestandszoekmodus\n\nschakel naar uploadmodus door op het vergrootglas 🔎 te klikken (naast de grote ZOEK-knop), en probeer opnieuw te uploaden\n\nsorry", + "ue_ta": "probeer opnieuw te uploaden, het zou nu moeten werken", + "ue_ab": "dit bestand wordt al geüpload in een andere map, en die upload moet eerst worden voltooid.\n\nJe kunt die upload afbreken via de 🧯 linksboven", + "ur_1uo": "OK: Bestand succesvol geüpload", + "ur_auo": "OK: Alle {0} bestanden succesvol geüpload", + "ur_1so": "OK: Bestand gevonden op server", + "ur_aso": "OK: Alle {0} bestanden gevonden op server", + "ur_1un": "Upload mislukt, sorry", + "ur_aun": "Alle {0} uploads zijn mislukt, sorry", + "ur_1sn": "Bestand is NIET gevonden op server", + "ur_asn": "De {0} bestanden zijn NIET gevonden op server", + "ur_um": "Voltooid;\n{0} uploads OK,\n{1} uploads mislukt, sorry", + "ur_sm": "Voltooid;\n{0} bestanden gevonden op server,\n{1} bestanden NIET gevonden op server", + + "lang_set": "verversen om de wijziging door te voeren?", + }, "rus": { "tt": "Русский", From 1c86b64a4ee5c577baa208f3682fb86e723ca82f Mon Sep 17 00:00:00 2001 From: Bart <57799908+crypt0rr@users.noreply.github.com> Date: Thu, 31 Jul 2025 20:43:55 +0200 Subject: [PATCH 027/174] Added Dutch(NL) Translation to `splash.js` (#373) --- copyparty/web/splash.js | 42 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 42 insertions(+) diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 56507169..aa956bb3 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -136,6 +136,48 @@ var Ls = { "af1": "Zeige neue Uploads", }, + "nl": { + "a1": "verversen", + "b1": "hallo onbekende   (je bent niet ingelogd)", + "c1": "uitloggen", + "d1": "stapel dumpen", + "d2": "toont de status van alle actieve threads", + "e1": "herlaad config", + "e2": "herlaad configuratiebestanden (accounts/volumes/volflags),$Nen scan alle e2ds-volumes opnieuw$N$Nlet op: wijzigingen in globale instellingen$Nvereisen een volledige herstart", + "f1": "je kunt bladeren:", + "g1": "je kunt uploaden naar:", + "cc1": "overige zaken:", + "h1": "k304 uitschakelen", + "i1": "k304 inschakelen", + "j1": "het inschakelen van k304 verbreekt je verbinding bij elke HTTP 304, wat sommige buggy proxies kan helpen (zoals pagina’s die plots niet laden), maar maakt alles over het algemeen trager", + "k1": "clientinstellingen resetten", + "l1": "log in voor meer:", + "m1": "welkom terug,", + "n1": "404 niet gevonden  ┐( ´ -`)┌", + "o1": "of misschien heb je geen toegang -- probeer een wachtwoord of ga naar home", + "p1": "403 verboden  ~┻━┻", + "q1": "gebruik een wachtwoord of ga naar home", + "r1": "ga naar home", + ".s1": "opnieuw scannen", + "t1": "actie", + "u2": "tijd sinds laatste schrijfactie op de server$N( upload / hernoemen / ... )$N$N17d = 17 dagen$N1h23 = 1 uur 23 minuten$N4m56 = 4 minuten 56 seconden", + "v1": "verbinden", + "v2": "gebruik deze server als lokale harde schijf", + "w1": "schakel over naar https", + "x1": "wachtwoord wijzigen", + "y1": "gedeelde mappen bewerken", + "z1": "ontgrendel deze share:", + "ta1": "vul eerst je nieuwe wachtwoord in", + "ta2": "herhaal om te bevestigen:", + "ta3": "tikfout gevonden; probeer opnieuw", + "aa1": "inkomende bestanden:", + "ab1": "no304 uitschakelen", + "ac1": "no304 inschakelen", + "ad1": "no304 schakelt alle caching uit; probeer dit als k304 niet genoeg was. Dit verbruikt enorm veel netwerkverkeer!", + "ae1": "actieve downloads:", + "af1": "toon recente uploads" + }, + "rus": { "a1": "обновить", "b1": "приветик, незнакомец   (вы не авторизованы)", From 714744f73e74996dc2d057d3231d3ad03679d1d4 Mon Sep 17 00:00:00 2001 From: ed Date: Thu, 31 Jul 2025 21:55:38 +0000 Subject: [PATCH 028/174] Revert "Added Dutch(NL) Translation to `splash.js` (#373)" (#379) This reverts commit 1c86b64a4ee5c577baa208f3682fb86e723ca82f. --- copyparty/web/splash.js | 42 ----------------------------------------- 1 file changed, 42 deletions(-) diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index aa956bb3..56507169 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -136,48 +136,6 @@ var Ls = { "af1": "Zeige neue Uploads", }, - "nl": { - "a1": "verversen", - "b1": "hallo onbekende   (je bent niet ingelogd)", - "c1": "uitloggen", - "d1": "stapel dumpen", - "d2": "toont de status van alle actieve threads", - "e1": "herlaad config", - "e2": "herlaad configuratiebestanden (accounts/volumes/volflags),$Nen scan alle e2ds-volumes opnieuw$N$Nlet op: wijzigingen in globale instellingen$Nvereisen een volledige herstart", - "f1": "je kunt bladeren:", - "g1": "je kunt uploaden naar:", - "cc1": "overige zaken:", - "h1": "k304 uitschakelen", - "i1": "k304 inschakelen", - "j1": "het inschakelen van k304 verbreekt je verbinding bij elke HTTP 304, wat sommige buggy proxies kan helpen (zoals pagina’s die plots niet laden), maar maakt alles over het algemeen trager", - "k1": "clientinstellingen resetten", - "l1": "log in voor meer:", - "m1": "welkom terug,", - "n1": "404 niet gevonden  ┐( ´ -`)┌", - "o1": "of misschien heb je geen toegang -- probeer een wachtwoord of ga naar home", - "p1": "403 verboden  ~┻━┻", - "q1": "gebruik een wachtwoord of ga naar home", - "r1": "ga naar home", - ".s1": "opnieuw scannen", - "t1": "actie", - "u2": "tijd sinds laatste schrijfactie op de server$N( upload / hernoemen / ... )$N$N17d = 17 dagen$N1h23 = 1 uur 23 minuten$N4m56 = 4 minuten 56 seconden", - "v1": "verbinden", - "v2": "gebruik deze server als lokale harde schijf", - "w1": "schakel over naar https", - "x1": "wachtwoord wijzigen", - "y1": "gedeelde mappen bewerken", - "z1": "ontgrendel deze share:", - "ta1": "vul eerst je nieuwe wachtwoord in", - "ta2": "herhaal om te bevestigen:", - "ta3": "tikfout gevonden; probeer opnieuw", - "aa1": "inkomende bestanden:", - "ab1": "no304 uitschakelen", - "ac1": "no304 inschakelen", - "ad1": "no304 schakelt alle caching uit; probeer dit als k304 niet genoeg was. Dit verbruikt enorm veel netwerkverkeer!", - "ae1": "actieve downloads:", - "af1": "toon recente uploads" - }, - "rus": { "a1": "обновить", "b1": "приветик, незнакомец   (вы не авторизованы)", From edb5c2bdce1afa830f603138ff461c0d466ab4af Mon Sep 17 00:00:00 2001 From: ed Date: Thu, 31 Jul 2025 21:56:06 +0000 Subject: [PATCH 029/174] Revert "Added Dutch(NL) translation" (#380) This reverts commit a2faf4e1e9db217976663049cf02414ea095acf0. --- copyparty/web/browser.js | 559 --------------------------------------- 1 file changed, 559 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 6c39b78b..b8b8b202 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -2510,565 +2510,6 @@ var Ls = { "lang_set": "Neuladen um Änderungen anzuwenden?", }, - "nl": { - "tt": "Dutch", - - "cols": { - "c": "actieknoppen", - "dur": "duur", - "q": "kwaliteit / bitrate", - "Ac": "audiocodec", - "Vc": "videocodec", - "Fmt": "formaat / container", - "Ahash": "audio-controlegetal", - "Vhash": "video-controlegetal", - "Res": "resolutie", - "T": "bestandstype", - "aq": "audiokwaliteit / bitrate", - "vq": "videokwaliteit / bitrate", - "pixfmt": "subsampling / pixelstructuur", - "resw": "horizontale resolutie", - "resh": "verticale resolutie", - "chs": "audiokanalen", - "hz": "samplefrequentie" - }, - - "hks": [ - [ - "misc", - ["ESC", "sluit verschillende vensters"], - - "file-manager", - ["G", "wissel tussen lijst- en rasterweergave"], - ["T", "wissel miniaturen / pictogrammen"], - ["⇧ A/D", "miniatuurgrootte aanpassen"], - ["ctrl-K", "verwijder selectie"], - ["ctrl-X", "knip selectie naar klembord"], - ["ctrl-C", "kopieer selectie naar klembord"], - ["ctrl-V", "plak (verplaats/kopieer) hier"], - ["Y", "download selectie"], - ["F2", "hernoem selectie"], - - "file-list-sel", - ["space", "selectie van bestand wisselen"], - ["↑/↓", "verplaats selectiecursor"], - ["ctrl ↑/↓", "verplaats cursor én scherm"], - ["⇧ ↑/↓", "selecteer vorig/volgend bestand"], - ["ctrl-A", "selecteer alle bestanden / mappen"] - ], [ - "navigation", - ["B", "wissel kruimelpad / navigatiepaneel"], - ["I/K", "vorige/volgende map"], - ["M", "bovenliggende map (of vouw huidige in)"], - ["V", "wissel mappen / tekstbestanden in navigatie"], - ["A/D", "formaat van navigatiepaneel"], - ], [ - "audio-player", - ["J/L", "vorig/volgend nummer"], - ["U/O", "10 sec terug/vooruit"], - ["0..9", "ga naar 0%..90%"], - ["P", "afspelen/pauzeren (start ook)"], - ["S", "selecteer afgespeeld nummer"], - ["Y", "download nummer"], - ], [ - "image-viewer", - ["J/L, ←/→", "vorige/volgende afbeelding"], - ["Home/End", "eerste/laatste afbeelding"], - ["F", "volledig scherm"], - ["R", "roteer met de klok mee"], - ["⇧ R", "roteer tegen de klok in"], - ["S", "selecteer afbeelding"], - ["Y", "download afbeelding"], - ], [ - "video-player", - ["U/O", "10 sec terug/vooruit"], - ["P/K/Space", "afspelen/pauzeren"], - ["C", "speel volgende automatisch af"], - ["V", "herhaal"], - ["M", "dempen"], - ["[ and ]", "stel herhaal-interval in"], - ], [ - "textfile-viewer", - ["I/K", "vorig/volgend bestand"], - ["M", "sluit tekstbestand"], - ["E", "bewerk tekstbestand"], - ["S", "selecteer bestand (voor knippen/kopiëren/hernoemen)"] - ] - ], - - "m_ok": "OK", - "m_ng": "Annuleren", - - "enable": "Inschakelen", - "danger": "GEVAAR", - "clipped": "gekopieerd naar klembord", - - "ht_s1": "seconde", - "ht_s2": "seconden", - "ht_m1": "minuut", - "ht_m2": "minuten", - "ht_h1": "uur", - "ht_h2": "uren", - "ht_d1": "dag", - "ht_d2": "dagen", - "ht_and": " en ", - - "goh": "configuratiescherm", - "gop": 'vorige map">vorige', - "gou": 'bovenliggende map">omhoog', - "gon": 'volgende map">volgende', - "logout": "Uitloggen ", - "access": " toegang", - "ot_close": "submenu sluiten", - "ot_search": "zoek naar bestanden op basis van attributen, pad / naam, muziektags of een combinatie hiervan$N$Nfoo bar = bevat zowel «foo» als «bar»,$Nfoo -bar = bevat «foo» maar niet «bar»,$N^yana .opus$ = begint met «yana» en is een «opus»-bestand$N\"try unite\" = bevat exact «try unite»$N$Nhet datumformaat is iso-8601, zoals$N2009-12-31 of 2020-09-12 23:30:00", - "ot_unpost": "unpost: verwijder je recente uploads of annuleer onafgeronde uploads", - "ot_bup": "bup: eenvoudige uploader, zelfs compatibel met Netscape 4.0", - "ot_mkdir": "mkdir: maak een nieuwe map", - "ot_md": "new-md: maak een nieuw markdown-document", - "ot_msg": "msg: stuur een bericht naar de serverlog", - "ot_mp": "media player opties", - "ot_cfg": "configuratie-opties", - "ot_u2i": "up2k: upload bestanden (als je schrijfrechten hebt) of schakel naar zoekmodus om te zien of ze al op de server bestaan$N$Nuploads zijn hervatbaar, multithreaded en behouden bestandstijdstempels, maar gebruiken meer CPU dan [🎈] (de eenvoudige uploader)

tijdens het uploaden wordt dit pictogram een voortgangsindicator!", - "ot_u2w": "up2k: upload bestanden met hervat-ondersteuning (sluit je browser en voeg later dezelfde bestanden weer toe)$N$Nmultithreaded en bestandstijdstempels worden behouden, maar gebruikt meer CPU dan [🎈] (de eenvoudige uploader)

tijdens het uploaden wordt dit pictogram een voortgangsindicator!", - "ot_noie": "Gebruik alsjeblieft Chrome / Firefox / Edge", - - "ab_mkdir": "maak map", - "ab_mkdoc": "nieuw markdown-document", - "ab_msg": "stuur bericht naar serverlog", - - "ay_path": "spring naar mappen", - "ay_files": "spring naar bestanden", - - "wt_ren": "hernoem geselecteerde items$NToets: F2", - "wt_del": "verwijder geselecteerde items$NToets: ctrl-K", - "wt_cut": "knip geselecteerde items (en plak elders)$NToets: ctrl-X", - "wt_cpy": "kopieer geselecteerde items naar klembord$N(om ergens anders te plakken)$NToets: ctrl-C", - "wt_pst": "plak eerder geknipte/gekopieerde selectie$NToets: ctrl-V", - "wt_selall": "selecteer alle bestanden$NToets: ctrl-A (wanneer bestand geselecteerd)", - "wt_selinv": "selectie omkeren", - "wt_zip1": "download deze map als archief", - "wt_selzip": "download selectie als archief", - "wt_seldl": "download selectie als losse bestanden$NToets: Y", - "wt_npirc": "kopieer trackinfo in irc-formaat", - "wt_nptxt": "kopieer trackinfo als platte tekst", - "wt_m3ua": "voeg toe aan m3u-afspeellijst (klik later op 📻kopieer)", - "wt_m3uc": "kopieer m3u-afspeellijst naar klembord", - "wt_grid": "wissel tussen raster- en lijstweergave$NToets: G", - "wt_prev": "vorig nummer$NToets: J", - "wt_play": "afspelen / pauzeren$NToets: P", - "wt_next": "volgend nummer$NToets: L", - - "ul_par": "parallelle uploads:", - "ut_rand": "bestandsnamen willekeurig maken", - "ut_u2ts": "kopieer laatste wijzigingstijdstip$Nvan je bestandssysteem naar de server\">📅", - "ut_ow": "bestaande bestanden op server overschrijven?$N🛡️: nooit (maakt een nieuwe bestandsnaam)$N🕒: overschrijven als serverbestand ouder is$N♻️: altijd overschrijven als de bestanden verschillen", - "ut_mt": "hash andere bestanden terwijl er geüpload wordt$N$NUitschakelen als CPU of HDD traag is", - "ut_ask": "vraag om bevestiging voor upload start\">💭", - "ut_pot": "verbeter uploadsnelheid op trage apparaten$Ndoor de interface eenvoudiger te maken", - "ut_srch": "upload niet echt, maar controleer of de bestanden al$Nop de server bestaan (scant alle mappen die je kunt lezen)", - "ut_par": "pauzeer uploads door deze op 0 te zetten$N$Nverhoog bij langzame of hoge-latentie verbinding$N$Nlaat op 1 in LAN of bij trage server HDD", - "ul_btn": "sleep bestanden / mappen
hierheen (of klik op mij)", - "ul_btnu": "U P L O A D", - "ul_btns": "Z O E K", - - "ul_hash": "hash", - "ul_send": "verzenden", - "ul_done": "klaar", - "ul_idle1": "er staan nog geen uploads in de wachtrij", - "ut_etah": "gemiddelde hash-snelheid en geschatte resterende tijd", - "ut_etau": "gemiddelde upload-snelheid en geschatte resterende tijd", - "ut_etat": "gemiddelde totale snelheid en geschatte resterende tijd", - - "uct_ok": "succesvol voltooid", - "uct_ng": "niet-goed: mislukt / geweigerd / niet gevonden", - "uct_done": "ok en ng gecombineerd", - "uct_bz": "aan het hashen of uploaden", - "uct_q": "inactief, in wachtrij", - - "utl_name": "bestandsnaam", - "utl_ulist": "lijst", - "utl_ucopy": "kopiëren", - "utl_links": "links", - "utl_stat": "status", - "utl_prog": "voortgang", - - "utl_404": "404", - "utl_err": "FOUT", - "utl_oserr": "OS-fout", - "utl_found": "gevonden", - "utl_defer": "uitstellen", - "utl_yolo": "YOLO", - "utl_done": "klaar", - - "ul_flagblk": "de bestanden zijn aan de wachtrij toegevoegd
maar er is een actieve up2k in een ander browsertabblad,
dus eerst wachten tot die klaar is", - "ul_btnlk": "de serverconfiguratie heeft deze schakelaar vergrendeld in deze status", - - "udt_up": "Uploaden", - "udt_srch": "Zoeken", - "udt_drop": "sleep hierheen", - - "u_nav_m": "
oké, wat heb je?
Enter = Bestanden (een of meer)\nESC = Eén map (inclusief submappen)", - "u_nav_b": "BestandenEén map", - - "cl_opts": "schakelaars", - "cl_themes": "thema", - "cl_langs": "taal", - "cl_ziptype": "map downloaden", - "cl_uopts": "up2k-opties", - "cl_favico": "favicon", - "cl_bigdir": "grote mappen", - "cl_hsort": "#sorteer", - "cl_keytype": "toetsnotatie", - "cl_hiddenc": "verborgen kolommen", - "cl_hidec": "verbergen", - "cl_reset": "resetten", - "cl_hpick": "tik op kolomkoppen om ze hieronder te verbergen", - "cl_hcancel": "verbergen van kolommen geannuleerd", - - "ct_grid": "田 het raster", - "ct_ttips": "ℹ️ tooltips", - "ct_thumb": "in rasterweergave, schakel pictogrammen of miniaturen in$NHotkey: T", - "ct_csel": "gebruik CTRL en SHIFT om bestanden te selecteren in rasterweergave", - "ct_ihop": "wanneer de afbeeldingsviewer is gesloten, scroll naar het laatst bekeken bestand", - "ct_dots": "toon verborgen bestanden (indien toegestaan door server)", - "ct_qdel": "bij verwijderen slechts éénmaal bevestiging vragen", - "ct_dir1st": "mappen eerst sorteren", - "ct_nsort": "natuurlijke sortering (voor bestandsnamen met cijfers vooraan)", - "ct_readme": "toon README.md in mapweergave", - "ct_idxh": "toon index.html in plaats van mapinhoud", - "ct_sbars": "toon schuifbalken", - - "cut_umod": "indien een bestand al bestaat op de server, werk de wijzigingsdatum bij om overeen te komen met je lokale bestand (vereist schrijf- en verwijderrechten)", - - "cut_turbo": "de yolo-knop, je wilt dit waarschijnlijk NIET aanzetten:$N$Ngebruik dit als je een grote hoeveelheid bestanden aan het uploaden was en je moest herstarten, en je zo snel mogelijk wilt doorgaan$N$Nvervangt de hash-controle door een simpele “heeft dit dezelfde bestandsgrootte?” controle, dus bij andere inhoud wordt het NIET geüpload$N$Nzet dit uit wanneer de upload klaar is en 'upload' dezelfde bestanden opnieuw om te verifiëren", - - "cut_datechk": "heeft geen effect tenzij turbo is ingeschakeld$N$Nverlaagt de yolo-factor iets; controleert of de bestandstijdstempels overeenkomen$N$Nzou theoretisch onvolledige/corrupte uploads moeten detecteren, maar is geen vervanging voor een verificatie achteraf met turbo uit", - - "cut_u2sz": "grootte (in MiB) van elk uploadblok; grote waarden werken beter over de Atlantische Oceaan. Gebruik lage waarden bij onstabiele verbindingen", - - "cut_flag": "zorg dat slechts één tabblad tegelijk uploadt$N -- andere tabbladen moeten dit ook aan hebben$N -- werkt alleen op hetzelfde domein", - - "cut_az": "upload bestanden in alfabetische volgorde in plaats van kleinste eerst$N$Nalfabetisch kan helpen om fouten op de server sneller te herkennen, maar is iets langzamer bij snelle verbindingen", - - "cut_nag": "OS-melding als upload klaar is$N(alleen als het tabblad niet actief is)", - "cut_sfx": "geluidssignaal als upload klaar is$N(alleen als het tabblad niet actief is)", - - "cut_mt": "gebruik multithreading voor snellere hashing$N$Nmaakt gebruik van web-workers en vraagt$Nmeer RAM (tot 512 MiB extra)$N$Nmaakt https 30% sneller, http 4,5x sneller", - - "cut_wasm": "gebruik wasm in plaats van ingebouwde browser-hasher; sneller op Chrome, maar zwaarder voor CPU en kan bugs veroorzaken in oudere Chrome-versies", - - "cft_text": "favicon-tekst (leeg laten en vernieuwen om uit te schakelen)", - "cft_fg": "voorgrondkleur", - "cft_bg": "achtergrondkleur", - - "cdt_lim": "maximaal aantal bestanden om in een map te tonen", - "cdt_ask": "bij scrollen naar beneden,$Nvraag wat te doen in plaats van meer bestanden laden", - "cdt_hsort": "hoeveel sorteerregels (<code>,sorthref</code>) opnemen in media-URLs. Waarde 0 negeert ook sorteringen in links", - - "tt_entree": "toon navigatievenster (mapstructuur)$NHotkey: B", - "tt_detree": "toon kruimelpad$NHotkey: B", - "tt_visdir": "scroll naar geselecteerde map", - "tt_ftree": "schakel boomstructuur / tekstbestanden$NHotkey: V", - "tt_pdock": "toon bovenliggende mappen bovenaan vastgezet", - "tt_dynt": "automatisch uitbreiden bij uitklappen boomstructuur", - "tt_wrap": "regelterugloop", - "tt_hover": "toon overlopende regels bij hover$N(breekt scrollen tenzij muis in linkerrand staat)", - - "im_hnf": "die afbeelding bestaat niet meer", - - "f_empty": "deze map is leeg", - "f_chide": "dit zal de kolom «{0}» verbergen\n\nje kunt kolommen weer zichtbaar maken in het instellingen-tabblad", - "f_bigtxt": "dit bestand is {0} MiB groot — echt als tekst weergeven?", - "f_bigtxt2": "alleen het einde van het bestand weergeven? dit schakelt ook volgen in, waarbij nieuwe regels in realtime verschijnen", - "fbd_more": "
toont {0} van {1} bestanden; toon {2} of toon alles
", - "fbd_all": "
toont {0} van {1} bestanden; toon alles
", - "f_anota": "slechts {0} van de {1} items zijn geselecteerd;\nscroll eerst helemaal naar beneden om alles te selecteren", - - "f_dls": "de bestandslinks in deze map zijn\nomgezet naar downloadlinks", - - "f_partial": "Om veilig een bestand te downloaden dat nog wordt geüpload, klik op het bestand met dezelfde naam maar zonder de extensie .PARTIAL. Klik op ANNULEREN of Escape om dit te doen.\n\nKlikken op OK / Enter negeert deze waarschuwing en downloadt het .PARTIAL-bestand, wat vrijwel zeker corrupte data oplevert.", - - "ft_paste": "plak {0} items$NSneltoets: ctrl-V", - "fr_eperm": "kan niet hernoemen:\nje hebt geen 'verplaats'-rechten in deze map", - "fd_eperm": "kan niet verwijderen:\nje hebt geen 'verwijder'-rechten in deze map", - "fc_eperm": "kan niet knippen:\nje hebt geen 'verplaats'-rechten in deze map", - "fp_eperm": "kan niet plakken:\nje hebt geen 'schrijf'-rechten in deze map", - "fr_emore": "selecteer ten minste één item om te hernoemen", - "fd_emore": "selecteer ten minste één item om te verwijderen", - "fc_emore": "selecteer ten minste één item om te knippen", - "fcp_emore": "selecteer ten minste één item om naar klembord te kopiëren", - - "fs_sc": "deel de map waarin je je bevindt", - "fs_ss": "deel de geselecteerde bestanden", - "fs_just1d": "je kunt niet meer dan één map selecteren,\nof bestanden en mappen door elkaar", - "fs_abrt": "❌ annuleren", - "fs_rand": "🎲 willek.naam", - "fs_go": "✅ deel aanmaken", - "fs_name": "naam", - "fs_src": "bron", - "fs_pwd": "wachtw", - "fs_exp": "verloopt", - "fs_tmin": "min", - "fs_thrs": "uur", - "fs_tdays": "dagen", - "fs_never": "eeuwig", - "fs_pname": "optionele linknaam; willekeurig indien leeg", - "fs_tsrc": "het bestand of de map om te delen", - "fs_ppwd": "optioneel wachtwoord", - "fs_w8": "deel aanmaken...", - "fs_ok": "druk op Enter/OK om te kopiëren\nof op ESC/Annuleer om te sluiten", - - "frt_dec": "kan sommige kapotte bestandsnamen herstellen\">url-decoderen", - "frt_rst": "herstel gewijzigde bestandsnamen naar origineel\">↺ reset", - "frt_abrt": "annuleren en venster sluiten\">❌ annuleren", - "frb_apply": "HERNOEMEN TOEPASSEN", - "fr_adv": "batch / metadata / patroon-hernoemen\">geavanceerd", - "fr_case": "hoofdlettergevoelige regex\">hoofdletter", - "fr_win": "Windows-veilige namen; vervang <>:\"\\|?* door Japanse tekens\">win", - "fr_slash": "vervang / door een teken dat geen nieuwe mappen maakt\">geen /", - "fr_re": "regex-zoekpatroon voor originele bestandsnamen; groepen kun je hieronder gebruiken met bijv. <code>(1)</code>", - "fr_fmt": "geïnspireerd op foobar2000:$N<code>(title)</code> wordt vervangen door titel,$N<code>[(artist) - ](title)</code> slaat [dit] over als artiest leeg is$N<code>$lpad((tn),2,0)</code> vult tracknummer aan tot 2 cijfers", - "fr_pdel": "verwijderen", - "fr_pnew": "opslaan als", - "fr_pname": "geef een naam voor je nieuwe preset", - "fr_aborted": "geannuleerd", - "fr_lold": "oude naam", - "fr_lnew": "nieuwe naam", - "fr_tags": "tags voor geselecteerde bestanden (alleen-lezen, ter referentie):", - "fr_busy": "bezig met hernoemen van {0} items...\n\n{1}", - "fr_efail": "hernoemen mislukt:\n", - "fr_nchg": "{0} van de nieuwe namen zijn aangepast door win en/of geen /\n\nOK om door te gaan met deze aangepaste namen?", - - "fd_ok": "verwijderen geslaagd", - "fd_err": "verwijderen mislukt:\n", - "fd_none": "niets verwijderd; mogelijk geblokkeerd door serverconfiguratie (xbd)?", - "fd_busy": "bezig met verwijderen van {0} items...\n\n{1}", - "fd_warn1": "WIL JE deze {0} items VERWIJDEREN?", - "fd_warn2": "Laatste kans! Dit kan niet ongedaan worden. Verwijderen?", - - "fc_ok": "geknipt: {0} items", - "fc_warn": "geknipt: {0} items\n\nmaar: alleen dit browser-tabblad kan ze plakken\n(aangezien de selectie enorm groot is)", - - "fcc_ok": "{0} items naar klembord gekopieerd", - "fcc_warn": "{0} items naar klembord gekopieerd\n\nmaar: alleen dit browser-tabblad kan ze plakken\n(aangezien de selectie enorm groot is)", - - "fp_apply": "gebruik deze namen", - "fp_ecut": "knip of kopieer eerst bestanden / mappen om te kunnen plakken / verplaatsen\n\nlet op: knippen / plakken werkt ook tussen verschillende tabs", - "fp_ename": "{0} items kunnen hier niet worden verplaatst omdat de namen al bestaan. Geef hieronder nieuwe namen op om verder te gaan, of laat het veld leeg om ze over te slaan:", - "fcp_ename": "{0} items kunnen hier niet worden gekopieerd omdat de namen al bestaan. Geef hieronder nieuwe namen op om verder te gaan, of laat het veld leeg om ze over te slaan:", - "fp_emore": "er zijn nog bestandsnaam-conflicten die opgelost moeten worden", - "fp_ok": "verplaatsen geslaagd", - "fcp_ok": "kopiëren geslaagd", - "fp_busy": "bezig met verplaatsen van {0} items...\n\n{1}", - "fcp_busy": "bezig met kopiëren van {0} items...\n\n{1}", - "fp_err": "verplaatsen mislukt:\n", - "fcp_err": "kopiëren mislukt:\n", - "fp_confirm": "wil je deze {0} items hier verplaatsen?", - "fcp_confirm": "wil je deze {0} items hier kopiëren?", - "fp_etab": "kon klembord van ander tabblad niet lezen", - "fp_name": "je uploadt een bestand vanaf je apparaat. Geef het een naam:", - "fp_both_m": "
kies wat je wilt plakken
Enter = Verplaats {0} bestanden van «{1}»\nESC = Upload {2} bestanden vanaf je apparaat", - "fcp_both_m": "
kies wat je wilt plakken
Enter = Kopieer {0} bestanden van «{1}»\nESC = Upload {2} bestanden vanaf je apparaat", - "fp_both_b": "VerplaatsenUploaden", - "fcp_both_b": "KopiërenUploaden", - - "mk_noname": "typ eerst een naam in het tekstveld links voordat je dat doet :p", - - "tv_load": "Tekstdocument laden:\n\n{0}\n\n{1}% ({2} van {3} MiB geladen)", - "tv_xe1": "kon tekstbestand niet laden:\n\nfout ", - "tv_xe2": "404, bestand niet gevonden", - "tv_lst": "lijst met tekstbestanden in", - "tvt_close": "terug naar mapweergave$NSneltoets: M (of Esc)\">❌ sluiten", - "tvt_dl": "download dit bestand$NSneltoets: Y\">💾 download", - "tvt_prev": "toon vorig document$NSneltoets: i\">⬆ vorige", - "tvt_next": "toon volgend document$NSneltoets: K\">⬇ volgende", - "tvt_sel": "selecteer bestand   ( om te knippen / kopiëren / verwijderen / ... )$NSneltoets: S\">selecteer", - "tvt_edit": "open bestand in teksteditor$NSneltoets: E\">✏️ bewerk", - "tvt_tail": "bewaking inschakelen; toon nieuwe regels in realtime\">📡 volgen", - "tvt_wrap": "regelafbreking\">↵", - "tvt_atail": "scroll vergrendelen naar onderkant van pagina\">⚓", - "tvt_ctail": "terminalkleuren decoderen (ansi-codes)\">🌈", - "tvt_ntail": "scrollback-limiet (hoeveel tekstbytes worden geladen)", - - "m3u_add1": "nummer toegevoegd aan m3u-afspeellijst", - "m3u_addn": "{0} nummers toegevoegd aan m3u-afspeellijst", - "m3u_clip": "m3u-afspeellijst gekopieerd naar klembord\n\nmaak een nieuw tekstbestand met naam iets.m3u en plak daar de afspeellijst in; dan kun je het afspelen", - - "gt_vau": "toon geen video, speel alleen het geluid af\">🎧", - "gt_msel": "bestandsselectie inschakelen; ctrl-klik om te forceren$N$Nactief: dubbelklik opent bestand / map$N$NSneltoets: S\">meerdere selecteren", - "gt_crop": "miniaturen bijsnijden naar midden\">bijsnijd", - "gt_3x": "hoge resolutie miniaturen\">3x", - "gt_zoom": "zoomen", - "gt_chop": "afsnijden", - "gt_sort": "sorteren op", - "gt_name": "naam", - "gt_sz": "grootte", - "gt_ts": "datum", - "gt_ext": "type", - "gt_c1": "kortere bestandsnamen tonen (minder weergeven)", - "gt_c2": "langere bestandsnamen tonen (meer weergeven)", - - "sm_w8": "bezig met zoeken...", - "sm_prev": "zoekresultaten hieronder komen van een vorige zoekopdracht:\n ", - "sl_close": "zoekresultaten sluiten", - "sl_hits": "{0} treffers gevonden", - "sl_moar": "meer laden", - - "s_sz": "grootte", - "s_dt": "datum", - "s_rd": "pad", - "s_fn": "naam", - "s_ta": "tags", - "s_ua": "upload@", - "s_ad": "geav.", - "s_s1": "minimaal MiB", - "s_s2": "maximaal MiB", - "s_d1": "min. iso8601", - "s_d2": "max. iso8601", - "s_u1": "geüpload na", - "s_u2": "en/of voor", - "s_r1": "pad bevat   (spatiegescheiden)", - "s_f1": "naam bevat   (negeren met -nope)", - "s_t1": "tags bevat   (^=begin, eind=$)", - "s_a1": "specifieke metagegevens", - - "md_eshow": "kan niet weergeven ", - "md_off": "[📜readme] uitgeschakeld in [⚙️] -- document verborgen", - - "badreply": "Antwoord van server kon niet worden verwerkt", - - "xhr403": "403: Toegang geweigerd\n\nprobeer F5, je bent misschien uitgelogd", - "xhr0": "onbekend (mogelijk verbinding met server verbroken of server offline)", - "cf_ok": "sorry daarvoor — DD" + wah + "oS-bescherming trad in werking\n\ndit zou binnen ~30 sec moeten herstellen\n\ngebeurt er niets, druk dan op F5 om de pagina opnieuw te laden", - "tl_xe1": "kon submappen niet ophalen:\n\nfout ", - "tl_xe2": "404: Map niet gevonden", - "fl_xe1": "kon bestanden in map niet ophalen:\n\nfout ", - "fl_xe2": "404: Map niet gevonden", - "fd_xe1": "kon submap niet aanmaken:\n\nfout ", - "fd_xe2": "404: Bovenliggende map niet gevonden", - "fsm_xe1": "kon bericht niet verzenden:\n\nfout ", - "fsm_xe2": "404: Bovenliggende map niet gevonden", - "fu_xe1": "kon onpost-lijst niet laden van server:\n\nfout ", - "fu_xe2": "404: Bestand niet gevonden??", - - "fz_tar": "niet-gecomprimeerd gnu-tar-bestand (linux / mac)", - "fz_pax": "niet-gecomprimeerde tar in pax-formaat (langzamer)", - "fz_targz": "gnu-tar met gzip-compressie niveau 3$N$Ndit is meestal erg traag, gebruik liever on-gecomprimeerde tar", - "fz_tarxz": "gnu-tar met xz-compressie niveau 1$N$Ndit is meestal erg traag, gebruik liever on-gecomprimeerde tar", - "fz_zip8": "zip met utf8-bestandsnamen (kan raar doen op windows 7 en ouder)", - "fz_zipd": "zip met klassieke cp437-bestandsnamen, voor erg oude software", - "fz_zipc": "cp437 met crc32 vooraf berekend,$Nvoor MS-DOS PKZIP v2.04g (oktober 1993)$N(duurt langer voor downloaden start)", - - "un_m1": "je kunt je recente uploads verwijderen (of onafgemaakte afbreken) hieronder", - "un_upd": "verversen", - "un_m4": "of deel de bestanden hieronder zichtbaar:", - "un_ulist": "toon", - "un_ucopy": "kopieer", - "un_flt": "optionele filter:  URL moet bevatten", - "un_fclr": "filter wissen", - "un_derr": "verwijderen mislukt:\n", - "un_f5": "er ging iets mis, probeer te verversen of druk op F5", - "un_uf5": "sorry, maar je moet de pagina verversen (bijv. met F5 of CTRL-R) voordat deze upload kan worden afgebroken", - "un_nou": "waarschuwing: server te druk om onafgemaakte uploads te tonen; klik later op de \"ververs\"-link", - "un_noc": "waarschuwing: unpost van voltooide bestanden is niet ingeschakeld/toegestaan in serverconfiguratie", - "un_max": "alleen eerste 2000 bestanden worden getoond (gebruik de filter)", - "un_avail": "{0} recente uploads kunnen worden verwijderd
{1} onafgemaakte kunnen worden afgebroken", - "un_m2": "gesorteerd op uploadtijd; meest recent eerst:", - "un_no1": "haha! geen uploads zijn recent genoeg", - "un_no2": "haha! geen uploads met die filter zijn recent genoeg", - "un_next": "verwijder de volgende {0} bestanden hieronder", - "un_abrt": "afbreken", - "un_del": "verwijderen", - "un_m3": "laden van je recente uploads...", - "un_busy": "bezig met verwijderen van {0} bestanden...", - "un_clip": "{0} links gekopieerd naar klembord", - - "u_https1": "je zou", - "u_https2": "moeten overschakelen naar https", - "u_https3": "voor betere prestaties", - "u_ancient": "je browser is indrukwekkend oud -- misschien kun je beter bup gebruiken", - "u_nowork": "vereist firefox 53+ of chrome 57+ of iOS 11+", - "tail_2old": "vereist firefox 105+ of chrome 71+ of iOS 14.5+", - "u_nodrop": "je browser is te oud voor drag-and-drop uploaden", - "u_notdir": "dat is geen map!\nje browser is te oud,\nprobeer dragdrop in plaats daarvan", - "u_uri": "om afbeeldingen te slepen vanuit andere browservensters,\nsleep ze naar de grote uploadknop", - "u_enpot": "overschakelen naar aardappel-UI (kan uploadsnelheid verbeteren)", - "u_depot": "overschakelen naar mooie UI (kan uploadsnelheid verminderen)", - "u_gotpot": "overschakelen naar aardappel-UI voor snellere upload,\n\nvoel je vrij om terug te schakelen!", - "u_pott": "

bestanden:   {0} voltooid,   {1} mislukt,   {2} bezig,   {3} in wachtrij

", - "u_ever": "dit is de basis-uploader; up2k vereist minimaal
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", - "u_su2k": "dit is de basis-uploader; up2k is beter", - "u_uput": "optimaliseer voor snelheid (sla checksum over)", - "u_ewrite": "je hebt geen schrijfrechten op deze map", - "u_eread": "je hebt geen leesrechten op deze map", - "u_enoi": "bestandszoekfunctie is niet ingeschakeld in serverconfig", - "u_enoow": "overschrijven werkt hier niet; verwijderrechten nodig", - "u_badf": "Deze {0} bestanden (van {1} totaal) zijn overgeslagen, mogelijk door bestandsrechten:\n\n", - "u_blankf": "Deze {0} bestanden (van {1} totaal) zijn leeg; toch uploaden?\n\n", - "u_applef": "Deze {0} bestanden (van {1} totaal) zijn waarschijnlijk ongewenst;\nDruk op OK/Enter om deze bestanden OVER TE SLAAN,\nDruk op Annuleer/ESC om ze WEL te uploaden:\n\n", - "u_just1": "\nMisschien werkt het beter als je maar één bestand selecteert", - "u_ff_many": "als je Linux / MacOS / Android gebruikt, kan deze hoeveelheid bestanden Firefox doen crashen!\nals dat gebeurt, probeer opnieuw (of gebruik Chrome).", - "u_up_life": "Deze upload wordt verwijderd van de server\n{0} na voltooiing", - "u_asku": "deze {0} bestanden uploaden naar {1}", - "u_unpt": "je kunt deze upload ongedaan maken/verwijderen via de 🧯 linksboven", - "u_bigtab": "op het punt om {0} bestanden te tonen\n\nkan je browser doen crashen, zeker weten?", - "u_scan": "Bestanden scannen...", - "u_dirstuck": "mapscanner bleef hangen bij toegang tot de volgende {0} items; worden overgeslagen:", - "u_etadone": "Klaar ({0}, {1} bestanden)", - "u_etaprep": "(bezig met voorbereiden upload)", - "u_hashdone": "hashing voltooid", - "u_hashing": "hashen", - "u_hs": "handshake...", - "u_started": "de bestanden worden nu geüpload; zie [🚀]", - "u_dupdefer": "duplicaat; wordt verwerkt na alle andere bestanden", - "u_actx": "klik hier om verlies van prestaties te voorkomen\nbij wisselen tussen vensters/tabs", - "u_fixed": "OK!  Opgelost 👍", - "u_cuerr": "uploaden van blok {0} van {1} mislukt;\nwaarschijnlijk onschuldig, ga verder\n\nbestand: {2}", - "u_cuerr2": "server weigerde upload (blok {0} van {1});\nprobeert later opnieuw\n\nbestand: {2}\n\nfout ", - "u_ehstmp": "probeert opnieuw; zie rechtsonder", - "u_ehsfin": "server weigerde het afronden van de upload; probeert opnieuw...", - "u_ehssrch": "server weigerde zoekverzoek; probeert opnieuw...", - "u_ehsinit": "server weigerde het starten van de upload; probeert opnieuw...", - "u_eneths": "netwerkfout tijdens upload-handshake; probeert opnieuw...", - "u_enethd": "netwerkfout tijdens test op doelbestand; probeert opnieuw...", - "u_cbusy": "wacht tot de server ons weer vertrouwt na een netwerkprobleem...", - "u_ehsdf": "server heeft geen schijfruimte meer!\n\nblijft proberen in de hoop dat iemand\nruimte vrijmaakt", - "u_emtleak1": "het lijkt erop dat je browser een geheugenlek heeft;\nprobeer", - "u_emtleak2": ' over te schakelen naar https (aanbevolen) of ', - "u_emtleak3": ' ', - "u_emtleakc": 'probeer het volgende:\n
  • druk op F5 om de pagina te vernieuwen
  • schakel dan de knop  mt  uit bij  ⚙️ instellingen
  • probeer de upload opnieuw
Uploads zullen wat trager zijn, maar oké.\nSorry voor het ongemak!\n\nPS: chrome v107 heeft hier een bugfix voor', - "u_emtleakf": 'probeer het volgende:\n
  • druk op F5 om de pagina te vernieuwen
  • schakel daarna 🥔 (aardappelmodus) in de upload-UI in
  • probeer de upload opnieuw
\nPS: firefox heeft hopelijk binnenkort een bugfix', - "u_s404": "niet gevonden op server", - "u_expl": "uitleg", - "u_maxconn": "de meeste browsers beperken dit tot 6, maar firefox laat je dit verhogen met connections-per-server in about:config", - "u_tu": "

WAARSCHUWING: turbo ingeschakeld,  client detecteert mogelijk geen onvoltooide uploads; zie tooltip bij turbo-knop

", - "u_ts": "

WAARSCHUWING: turbo ingeschakeld,  zoekresultaten kunnen onjuist zijn; zie tooltip bij turbo-knop

", - "u_turbo_c": "turbo is uitgeschakeld in serverconfiguratie", - "u_turbo_g": "turbo uitgeschakeld omdat je geen mapinhoud mag bekijken binnen dit volume", - "u_life_cfg": "automatisch verwijderen na min (of uur)", - "u_life_est": "upload wordt verwijderd op ---", - "u_life_max": "deze map heeft een\nmaximale bewaartijd van {0}", - "u_unp_ok": "verwijderen is toegestaan voor {0}", - "u_unp_ng": "verwijderen is NIET toegestaan", - "ue_ro": "je hebt alleen-lezen toegang tot deze map\n\n", - "ue_nl": "je bent momenteel niet ingelogd", - "ue_la": "je bent momenteel ingelogd als \"{0}\"", - "ue_sr": "je bent momenteel in bestandszoekmodus\n\nschakel naar uploadmodus door op het vergrootglas 🔎 te klikken (naast de grote ZOEK-knop), en probeer opnieuw te uploaden\n\nsorry", - "ue_ta": "probeer opnieuw te uploaden, het zou nu moeten werken", - "ue_ab": "dit bestand wordt al geüpload in een andere map, en die upload moet eerst worden voltooid.\n\nJe kunt die upload afbreken via de 🧯 linksboven", - "ur_1uo": "OK: Bestand succesvol geüpload", - "ur_auo": "OK: Alle {0} bestanden succesvol geüpload", - "ur_1so": "OK: Bestand gevonden op server", - "ur_aso": "OK: Alle {0} bestanden gevonden op server", - "ur_1un": "Upload mislukt, sorry", - "ur_aun": "Alle {0} uploads zijn mislukt, sorry", - "ur_1sn": "Bestand is NIET gevonden op server", - "ur_asn": "De {0} bestanden zijn NIET gevonden op server", - "ur_um": "Voltooid;\n{0} uploads OK,\n{1} uploads mislukt, sorry", - "ur_sm": "Voltooid;\n{0} bestanden gevonden op server,\n{1} bestanden NIET gevonden op server", - - "lang_set": "verversen om de wijziging door te voeren?", - }, "rus": { "tt": "Русский", From 0e35f37638cba9d08403c7b2f405ab3e11ee3d5d Mon Sep 17 00:00:00 2001 From: ed Date: Thu, 31 Jul 2025 22:11:49 +0000 Subject: [PATCH 030/174] tl cleanup --- copyparty/web/browser.js | 17 +++++++++-------- copyparty/web/splash.js | 1 + 2 files changed, 10 insertions(+), 8 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index b8b8b202..028bb2a5 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3,6 +3,7 @@ var XHR = XMLHttpRequest, img_re = /\.(a?png|avif|bmp|gif|heif|jpe?g|jfif|svg|webp|webm|mkv|mp4|m4v|mov)(\?|$)/i; +// please add translations in alphabetic order, but keep "eng" and "nor" first var Ls = { "eng": { "tt": "English", @@ -2049,9 +2050,9 @@ var Ls = { "ul_send": "senden", "ul_done": "fertig", "ul_idle1": "keine Uploads in der Warteschlange", - "ut_etah": "durchschnittl. <em>hashing</em> Geschw. & gesch. Restzeit", - "ut_etau": "durchschnittl. <em>upload</em> Geschw. & gesch. Restzeit", - "ut_etat": "durchschnittl. <em>total</em> Geschw. & gesch. Restzeit", + "ut_etah": "durchschnittl. <em>hashing</em> Geschw. & gesch. Restzeit", + "ut_etau": "durchschnittl. <em>upload</em> Geschw. & gesch. Restzeit", + "ut_etat": "durchschnittl. <em>total</em> Geschw. & gesch. Restzeit", "uct_ok": "Erfolgreich abgeschlossen", "uct_ng": "no-good: fehlgeschlagen / abgelehnt / nicht gefunden", @@ -2734,7 +2735,7 @@ var Ls = { "ct_dots": 'показывать скрытые файлы (если есть доступ)">скрыт.', "ct_qdel": 'спрашивать подтверждение только один раз перед удалением файлов">быстр. удал.', "ct_dir1st": 'разместить папки над файлами">📁 сверху', - "ct_nsort": 'сортировка по числам$N(например, файл с 2 в начале названия идёт перед 11)">нат. сорт.', + "ct_nsort": 'сортировка по числам$N(например, файл с >code<2>/code< в начале названия идёт перед >code<11>/code<)">нат. сорт.', "ct_readme": 'показывать содержимое README.md в описании папки">📜 ридми', "ct_idxh": 'показывать страницу index.html в текущей папке вместо интерфейса">htm', "ct_sbars": 'показывать полосы прокрутки">⟊', @@ -3482,10 +3483,10 @@ var Ls = { "f_partial": "Щоб безпечно завантажити файл, який зараз завантажується, будь ласка, клацніть на файл, який має таке саме ім'я, але без розширення .PARTIAL. Будь ласка, натисніть CANCEL або Escape, щоб зробити це.\n\nНатиснення OK / Enter проігнорує це попередження і продовжить завантаження .PARTIAL робочого файлу замість цього, що майже напевно дасть вам пошкоджені дані.", "ft_paste": "вставити {0} елементів$NГаряча клавіша: ctrl-V", - "fr_eperm": 'не можу перейменувати:\nу вас немає дозволу "переміщення" в цій папці', - "fd_eperm": 'не можу видалити:\nу вас немає дозволу "видалення" в цій папці', - "fc_eperm": 'не можу вирізати:\nу вас немає дозволу "переміщення" в цій папці', - "fp_eperm": 'не можу вставити:\nу вас немає дозволу "запису" в цій папці', + "fr_eperm": 'не можу перейменувати:\nу вас немає дозволу “переміщення“ в цій папці', + "fd_eperm": 'не можу видалити:\nу вас немає дозволу “видалення“ в цій папці', + "fc_eperm": 'не можу вирізати:\nу вас немає дозволу “переміщення“ в цій папці', + "fp_eperm": 'не можу вставити:\nу вас немає дозволу “запису“ в цій папці', "fr_emore": "виберіть принаймні один елемент для перейменування", "fd_emore": "виберіть принаймні один елемент для видалення", "fc_emore": "виберіть принаймні один елемент для вирізання", diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 56507169..16a9d517 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -1,3 +1,4 @@ +// please add translations in alphabetic order, but keep "nor" and "eng" first var Ls = { "nor": { "a1": "oppdater", From 3222ba3acdb6b10f4f4c8cb5e539b8fe605e8370 Mon Sep 17 00:00:00 2001 From: ed Date: Thu, 31 Jul 2025 22:56:45 +0000 Subject: [PATCH 031/174] man --- copyparty/up2k.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/up2k.py b/copyparty/up2k.py index 49bf21b7..f544d0e3 100644 --- a/copyparty/up2k.py +++ b/copyparty/up2k.py @@ -2827,7 +2827,7 @@ class Up2k(object): # v5a -> v5b # store rd+fn rather than warks to support nohash vols try: - cur.execute("select ws, rd, fn from iu limit 1").fetchone() + cur.execute("select c, w, rd, fn from iu limit 1").fetchone() return except: pass From 1f966bb9d5a9d6ad6adfec5f950b4828bb38aac3 Mon Sep 17 00:00:00 2001 From: ed Date: Thu, 31 Jul 2025 22:57:14 +0000 Subject: [PATCH 032/174] devnotes: add tgz to build-from-scratch procedure --- docs/devnotes.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/docs/devnotes.md b/docs/devnotes.md index 3db6b201..229cb398 100644 --- a/docs/devnotes.md +++ b/docs/devnotes.md @@ -415,6 +415,8 @@ to get started, first `cd` into the `scripts` folder * if you want to build the `.pyz` standalone "binary", now run `./make-pyz.sh` +* if you want to build the `tar.gz` for use in a linux-distro package, now run `./make-tgz-release.sh theVersionNumber` + * if you want to build a pypi package, now run `./make-pypi-release.sh d` * if you want to build a docker-image, you have two options: From b69d590176e3bb49c8e039a2f17c4ba21b5f55c2 Mon Sep 17 00:00:00 2001 From: Benjamin Bock Date: Thu, 31 Jul 2025 20:52:58 +0200 Subject: [PATCH 033/174] Improve Python 2 compatibility --- copyparty/httpcli.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index bdbaced2..f698255a 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -911,7 +911,7 @@ class HttpCli(object): if status == 304: self.out_headers.pop("Content-Length", None) self.out_headers.pop("Content-Type", None) - self.out_headerlist.clear() + self.out_headerlist[:] = [] if self.k304(): self.keepalive = False else: From f401fa7f6c4b5be949be7d774225045d374d2043 Mon Sep 17 00:00:00 2001 From: Toast <39011842+toast003@users.noreply.github.com> Date: Thu, 31 Jul 2025 02:19:22 +0200 Subject: [PATCH 034/174] nix: remove space from list separator The extra speces made copyparty freak out if you tried listening to an ip address and a unix socket at the same time --- contrib/nixos/modules/copyparty.nix | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/contrib/nixos/modules/copyparty.nix b/contrib/nixos/modules/copyparty.nix index cd4fef40..994b2575 100644 --- a/contrib/nixos/modules/copyparty.nix +++ b/contrib/nixos/modules/copyparty.nix @@ -22,7 +22,7 @@ let mkValueString = value: if isList value then - (concatStringsSep ", " (map mkValueString value)) + (concatStringsSep "," (map mkValueString value)) else if isAttrs value then "\n" + (mkAttrsString value) else From e965dc9c74652f40ccef1ccc25e1c1b6a971e884 Mon Sep 17 00:00:00 2001 From: Kevin Leutzinger <6435727+kleutzinger@users.noreply.github.com> Date: Tue, 29 Jul 2025 15:54:13 -0400 Subject: [PATCH 035/174] Add `uvx copyparty` Signed-off-by: Kevin Leutzinger <6435727+kleutzinger@users.noreply.github.com> --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 83c47875..ba843777 100644 --- a/README.md +++ b/README.md @@ -152,6 +152,7 @@ just run **[copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/ * or install [on arch](#arch-package) ╱ [on NixOS](#nixos-module) ╱ [through nix](#nix-package) * or if you are on android, [install copyparty in termux](#install-on-android) * or maybe you have a [synology nas / dsm](./docs/synology-dsm.md) +* or if you have [uv](https://docs.astral.sh/uv/) installed, run `uvx copyparty` * or if your computer is messed up and nothing else works, [try the pyz](#zipapp) * or if your OS is dead, give the [bootable flashdrive / cd-rom](https://a.ocv.me/pub/stuff/edcd001/enterprise-edition/) a spin * or if you don't trust copyparty yet and want to isolate it a little, then... From d357ff0d164c1790ec9ae77d52b9c098de5e419a Mon Sep 17 00:00:00 2001 From: Kevin Leutzinger <6435727+kleutzinger@users.noreply.github.com> Date: Tue, 29 Jul 2025 15:57:05 -0400 Subject: [PATCH 036/174] Update README.md Signed-off-by: Kevin Leutzinger <6435727+kleutzinger@users.noreply.github.com> --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index ba843777..889ea8b8 100644 --- a/README.md +++ b/README.md @@ -152,7 +152,7 @@ just run **[copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/ * or install [on arch](#arch-package) ╱ [on NixOS](#nixos-module) ╱ [through nix](#nix-package) * or if you are on android, [install copyparty in termux](#install-on-android) * or maybe you have a [synology nas / dsm](./docs/synology-dsm.md) -* or if you have [uv](https://docs.astral.sh/uv/) installed, run `uvx copyparty` +* or if you have [uv](https://docs.astral.sh/uv/) installed, run `uv tool run copyparty` * or if your computer is messed up and nothing else works, [try the pyz](#zipapp) * or if your OS is dead, give the [bootable flashdrive / cd-rom](https://a.ocv.me/pub/stuff/edcd001/enterprise-edition/) a spin * or if you don't trust copyparty yet and want to isolate it a little, then... From a1dfd0be33428e5e31a6186db8017b54eed5759e Mon Sep 17 00:00:00 2001 From: Juan Herruzo Date: Wed, 30 Jul 2025 11:22:07 +0200 Subject: [PATCH 037/174] added spanish draft --- copyparty/web/browser.js | 627 ++++++++++++++++++++++++++++++++++++++- 1 file changed, 626 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 028bb2a5..6afa9b26 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3763,9 +3763,634 @@ var Ls = { "lang_set": "оновити сторінку, щоб зміни набули чинності?", }, + "esp": { + "tt": "Español", + + "cols": { + "c": "acciones", + "dur": "duración", + "q": "calidad / bitrate", + "Ac": "códec de audio", + "Vc": "códec de vídeo", + "Fmt": "formato / contenedor", + "Ahash": "checksum de audio", + "Vhash": "checksum de vídeo", + "Res": "resolución", + "T": "tipo de archivo", + "aq": "calidad de audio / bitrate", + "vq": "calidad de vídeo / bitrate", + "pixfmt": "submuestreo / estructura de píxel", + "resw": "resolución horizontal", + "resh": "resolución vertical", + "chs": "canales de audio", + "hz": "frecuencia de muestreo" + }, + + "hks": [ + [ + "varios", + ["ESC", "cerrar varias cosas"], + + "gestor de archivos", + ["G", "alternar vista de lista / cuadrícula"], + ["T", "alternar miniaturas / iconos"], + ["⇧ A/D", "tamaño de miniatura"], + ["ctrl-K", "eliminar seleccionados"], + ["ctrl-X", "cortar selección al portapapeles"], + ["ctrl-C", "copiar selección al portapapeles"], + ["ctrl-V", "pegar (mover/copiar) aquí"], + ["Y", "descargar seleccionados"], + ["F2", "renombrar seleccionados"], + + "selección en lista de archivos", + ["space", "alternar selección de archivo"], + ["↑/↓", "mover cursor de selección"], + ["ctrl ↑/↓", "mover cursor y vista"], + ["⇧ ↑/↓", "seleccionar anterior/siguiente archivo"], + ["ctrl-A", "seleccionar todos los archivos / carpetas"] + ], [ + "navegación", + ["B", "alternar breadcrumbs / panel de navegación"], + ["I/K", "anterior/siguiente carpeta"], + ["M", "carpeta de nivel superior (o contraer actual)"], + ["V", "alternar carpetas / archivos en panel de navegación"], + ["A/D", "tamaño del panel de navegación"] + ], [ + "reproductor de audio", + ["J/L", "anterior/siguiente canción"], + ["U/O", "saltar 10s atrás/adelante"], + ["0..9", "saltar a 0%..90%"], + ["P", "reproducir/pausar (también inicia)"], + ["S", "seleccionar canción en reproducción"], + ["Y", "descargar canción"] + ], [ + "visor de imágenes", + ["J/L, ←/→", "anterior/siguiente imagen"], + ["Home/End", "primera/última imagen"], + ["F", "pantalla completa"], + ["R", "rotar en sentido horario"], + ["⇧ R", "rotar en sentido antihorario"], + ["S", "seleccionar imagen"], + ["Y", "descargar imagen"] + ], [ + "reproductor de vídeo", + ["U/O", "saltar 10s atrás/adelante"], + ["P/K/Space", "reproducir/pausar"], + ["C", "continuar con el siguiente"], + ["V", "bucle"], + ["M", "silenciar"], + ["[ y ]", "establecer intervalo de bucle"] + ], [ + "visor de texto", + ["I/K", "anterior/siguiente archivo"], + ["M", "cerrar archivo"], + ["E", "editar archivo"], + ["S", "seleccionar archivo (para cortar/copiar/renombrar)"] + ] + ], + + "m_ok": "Aceptar", + "m_ng": "Cancelar", + + "enable": "Activar", + "danger": "PELIGRO", + "clipped": "copiado al portapapeles", + + "ht_s1": "segundo", + "ht_s2": "segundos", + "ht_m1": "minuto", + "ht_m2": "minutos", + "ht_h1": "hora", + "ht_h2": "horas", + "ht_d1": "día", + "ht_d2": "días", + "ht_and": " y ", + + "goh": "panel-de-control", + "gop": 'hermano anterior">anterior', + "gou": 'carpeta de nivel superior">subir', + "gon": 'siguiente carpeta">siguiente', + "logout": "Cerrar sesión ", + "access": " acceso", + "ot_close": "cerrar submenú", + "ot_search": "buscar archivos por atributos, ruta / nombre, etiquetas de música, o cualquier combinación$N$N<code>foo bar</code> = debe contener «foo» y «bar»,$N<code>foo -bar</code> = debe contener «foo» pero no «bar»,$N<code>^yana .opus$</code> = empieza con «yana» y es un archivo «opus»$N<code>"try unite"</code> = contiene exactamente «try unite»$N$Nel formato de fecha es iso-8601, como$N<code>2009-12-31</code> o <code>2020-09-12 23:30:00</code>", + "ot_unpost": "dessubir: elimina tus subidas recientes, o aborta las inacabadas", + "ot_bup": "bup: subidor básico, soporta hasta netscape 4.0", + "ot_mkdir": "mkdir: crear un nuevo directorio", + "ot_md": "new-md: crear un nuevo documento markdown", + "ot_msg": "msg: enviar un mensaje al registro del servidor", + "ot_mp": "opciones del reproductor multimedia", + "ot_cfg": "opciones de configuración", + "ot_u2i": "up2k: subir archivos (si tienes acceso de escritura) o cambiar a modo de búsqueda para ver si existen en el servidor$N$Nlas subidas se pueden reanudar, usan múltiples hilos y conservan la fecha de los archivos, pero consume más CPU que [🎈]  (el subidor básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", + "ot_u2w": "up2k: subir archivos con soporte para reanudación (cierra tu navegador y arrastra los mismos archivos más tarde)$N$NMultihilo y conserva las fechas de los archivos, pero usa más CPU que [🎈]  (el subidor básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", + "ot_noie": "Por favor, usa Chrome / Firefox / Edge", + + "ab_mkdir": "crear directorio", + "ab_mkdoc": "nuevo documento markdown", + "ab_msg": "enviar msg al registro del servidor", + + "ay_path": "saltar a carpetas", + "ay_files": "saltar a archivos", + + "wt_ren": "renombrar elementos seleccionados$NAtajo: F2", + "wt_del": "eliminar elementos seleccionados$NAtajo: ctrl-K", + "wt_cut": "cortar elementos seleccionados <small>(luego pegar en otro lugar)</small>$NAtajo: ctrl-X", + "wt_cpy": "copiar elementos seleccionados al portapapeles$N(para pegarlos en otro lugar)$NAtajo: ctrl-C", + "wt_pst": "pegar una selección previamente cortada / copiada$NAtajo: ctrl-V", + "wt_selall": "seleccionar todos los archivos$NAtajo: ctrl-A (con un archivo con foco)", + "wt_selinv": "invertir selección", + "wt_zip1": "descargar esta carpeta como un archivo comprimido", + "wt_selzip": "descargar selección como archivo comprimido", + "wt_seldl": "descargar selección como archivos separados$NAtajo: Y", + "wt_npirc": "copiar información de pista en formato IRC", + "wt_nptxt": "copiar información de pista en texto plano", + "wt_m3ua": "añadir a lista m3u (haz clic en 📻copiar después)", + "wt_m3uc": "copiar lista m3u al portapapeles", + "wt_grid": "alternar vista de cuadrícula / lista$NAtajo: G", + "wt_prev": "pista anterior$NAtajo: J", + "wt_play": "reproducir / pausar$NAtajo: P", + "wt_next": "siguiente pista$NAtajo: L", + + "ul_par": "subidas paralelas:", + "ut_rand": "aleatorizar nombres de archivo", + "ut_u2ts": 'copiar la fecha de última modificación$Nde tu sistema de archivos al servidor">📅', + "ut_ow": "sobrescribir archivos existentes en el servidor?$N🛡️: nunca (generará un nuevo nombre de archivo en su lugar)$N🕒: sobrescribir si el archivo del servidor es más antiguo que el tuyo$N♻️: siempre sobrescribir si los archivos son diferentes", + "ut_mt": "continuar generando hashes de otros archivos mientras se sube$N$Nquizás desactivar si tu CPU o HDD es un cuello de botella", + "ut_ask": 'pedir confirmación antes de iniciar la subida">💭', + "ut_pot": "mejorar la velocidad de subida en dispositivos lentos$Nsimplificando la interfaz de usuario", + "ut_srch": "no subir, en su lugar comprobar si los archivos ya $N existen en el servidor (escaneará todas las carpetas que puedas leer)", + "ut_par": "pausar subidas poniéndolo a 0$N$Naumentar si tu conexión es lenta / de alta latencia$N$Nmantener en 1 en LAN o si el HDD del servidor es un cuello de botella", + "ul_btn": "arrastra archivos / carpetas
aquí (o haz clic)", + "ul_btnu": "S U B I R", + "ul_btns": "B U S C A R", + + "ul_hash": "hash", + "ul_send": "envio", + "ul_done": "hecho", + "ul_idle1": "aún no hay subidas en cola", + "ut_etah": "velocidad media de <em>hashing</em>, y tiempo estimado para finalizar", + "ut_etau": "velocidad media de <em>subida</em> y tiempo estimado para finalizar", + "ut_etat": "velocidad media <em>total</em> y tiempo estimado para finalizar", + + "uct_ok": "completado con éxito", + "uct_ng": "fallido: error / rechazado / no encontrado", + "uct_done": "éxitos y fallos combinados", + "uct_bz": "generando hash o subiendo", + "uct_q": "inactivo, pendiente", + + "utl_name": "nombre de archivo", + "utl_ulist": "lista", + "utl_ucopy": "copiar", + "utl_links": "enlaces", + "utl_stat": "estado", + "utl_prog": "progreso", + + "utl_404": "404", + "utl_err": "ERROR", + "utl_oserr": "Error-SO", + "utl_found": "encontrado", + "utl_defer": "posponer", + "utl_yolo": "YOLO", + "utl_done": "hecho", + + "ul_flagblk": "los archivos se añadieron a la cola
sin embargo, hay un up2k ocupado en otra pestaña del navegador,
esperando a que termine primero", + "ul_btnlk": "la configuración del servidor ha bloqueado esta opción en este estado", + + "udt_up": "Subir", + "udt_srch": "Buscar", + "udt_drop": "suéltalo aquí", + + "u_nav_m": "
vale, ¿qué tienes?
Intro = Archivos (uno o más)\nESC = Una carpeta (incluyendo subcarpetas)", + "u_nav_b": "ArchivosUna carpeta", + + "cl_opts": "opciones", + "cl_themes": "tema", + "cl_langs": "idioma", + "cl_ziptype": "descarga de carpeta", + "cl_uopts": "opciones up2k", + "cl_favico": "favicon", + "cl_bigdir": "directorios grandes", + "cl_hsort": "#ordenar", + "cl_keytype": "notación de teclas", + "cl_hiddenc": "columnas ocultas", + "cl_hidec": "ocultar", + "cl_reset": "restablecer", + "cl_hpick": "toca en las cabeceras de columna para ocultarlas en la tabla de abajo", + "cl_hcancel": "ocultación de columna cancelada", + + "ct_grid": '田 la cuadrícula', + "ct_ttips": '◔ ◡ ◔">ℹ️ tooltips', + "ct_thumb": 'en vista de cuadrícula, alternar iconos o miniaturas$NAtajo: T">🖼️ miniaturas', + "ct_csel": 'usa CTRL y SHIFT para seleccionar archivos en la vista de cuadrícula">sel', + "ct_ihop": 'al cerrar el visor de imágenes, desplazarse hasta el último archivo visto">g⮯', + "ct_dots": 'mostrar archivos ocultos (si el servidor lo permite)">archivos ocultos', + "ct_qdel": 'al eliminar archivos, pedir confirmación solo una vez">elim. rápida', + "ct_dir1st": 'ordenar carpetas antes que archivos">📁 primero', + "ct_nsort": 'orden natural (para nombres de archivo con dígitos iniciales)">ord. natural', + "ct_readme": 'mostrar README.md en los listados de carpetas">📜 léeme', + "ct_idxh": 'mostrar index.html en lugar del listado de carpetas">htm', + "ct_sbars": 'mostrar barra lateral">⟊', + + "cut_umod": 'si un archivo ya existe en el servidor, actualiza la fecha de última modificación del servidor para que coincida con tu archivo local (requiere permisos de escritura+eliminación)">re📅', + + "cut_turbo": 'el botón yolo, probablemente NO quieras activarlo:$N$Núsalo si estabas subiendo una gran cantidad de archivos y tuviste que reiniciar por alguna razón, y quieres continuar la subida lo antes posible$N$Nesto reemplaza la comprobación de hash por un simple "¿tiene este el mismo tamaño de archivo en el servidor?" así que si el contenido del archivo es diferente, NO se subirá$N$Ndeberías desactivar esto cuando la subida termine, y luego "subir" los mismos archivos de nuevo para que el cliente los verifique">turbo', + + "cut_datechk": 'no tiene efecto a menos que el botón turbo esté activado$N$Nreduce el factor yolo en una pequeña cantidad; comprueba si las fechas de los archivos en el servidor coinciden con las tuyas$N$Nteóricamente debería detectar la mayoría de las subidas inacabadas / corruptas, pero no es un sustituto de hacer una pasada de verificación con el turbo desactivado después">verif. fecha', + + "cut_u2sz": "tamaño (en MiB) de cada trozo de subida; los valores grandes vuelan mejor a través del atlántico. Prueba valores bajos en conexiones muy poco fiables", + + "cut_flag": "asegura que solo una pestaña esté subiendo a la vez $N -- otras pestañas también deben tener esto activado $N -- solo afecta a pestañas en el mismo dominio", + + "cut_az": "subir archivos en orden alfabético, en lugar de los más pequeños primero$N$Nel orden alfabético puede facilitar la detección visual de si algo salió mal en el servidor, pero hace la subida ligeramente más lenta en fibra / LAN", + + "cut_nag": "notificación del SO cuando la subida se complete$N(solo si el navegador o la pestaña no están activos)", + "cut_sfx": "alerta sonora cuando la subida se complete$N(solo si el navegador o la pestaña no están activos)", + + "cut_mt": 'usar multithreading para acelerar el hashing de archivos$N$Nesto usa web-workers y requiere$Nmás RAM (hasta 512 MiB extra)$N$Nhace https un 30% más rápido, http 4.5x más rápido">mt', + + "cut_wasm": 'usar wasm en lugar del hasher incorporado del navegador; mejora la velocidad en navegadores basados en chrome pero aumenta la carga de la CPU, y muchas versiones antiguas de chrome tienen errores que hacen que el navegador consuma toda la RAM y se bloquee si esto está activado">wasm', + + "cft_text": "texto del favicon (dejar en blanco y refrescar para desactivar)", + "cft_fg": "color de primer plano", + "cft_bg": "color de fondo", + + "cdt_lim": "número máximo de archivos a mostrar en una carpeta", + "cdt_ask": "al llegar al final,$Nen lugar de cargar más archivos,$Npreguntar qué hacer", + "cdt_hsort": "cuántas reglas de ordenación (<code>,sorthref</code>) incluir en las URLs de medios. Ponerlo a 0 también ignorará las reglas de ordenación incluidas en los enlaces de medios al hacer clic en ellos", + + "tt_entree": "mostrar panel de navegación (barra lateral con árbol de directorios)$NAtajo: B", + "tt_detree": "mostrar breadcrumbs$NAtajo: B", + "tt_visdir": "desplazarse a la carpeta seleccionada", + "tt_ftree": "alternar árbol de carpetas / archivos de texto$NAtajo: V", + "tt_pdock": "mostrar carpetas de niveles superiores en un panel acoplado en la parte superior", + "tt_dynt": "crecimiento automático a medida que el árbol se expande", + "tt_wrap": "ajuste de línea", + "tt_hover": "revelar líneas que se desbordan al pasar el ratón$N( rompe el desplazamiento a menos que el $N  cursor esté en el margen izquierdo )", + + "ml_pmode": "al final de la carpeta...", + "ml_btns": "cmds", + "ml_tcode": "transcodificar", + "ml_tcode2": "transcodificar a", + "ml_tint": "tinte", + "ml_eq": "ecualizador de audio", + "ml_drc": "compresor de rango dinámico", + + "mt_loop": 'poner en bucle/repetir una canción">🔁', + "mt_one": 'parar después de una canción">1️⃣', + "mt_shuf": 'reproducir aleatoriamente las canciones en cada carpeta">🔀', + "mt_aplay": 'reproducción automática si hay un ID de canción en el enlace en el que hiciste clic para acceder al servidor$N$Ndesactivar esto también evitará que la URL de la página se actualice con IDs de canción al reproducir música, para prevenir la reproducción automática si se pierden estos ajustes pero la URL permanece">a▶', + "mt_preload": 'empezar a cargar la siguiente canción cerca del final para una reproducción sin pausas">precarga', + "mt_prescan": 'ir a la siguiente carpeta antes de que la última canción$Ntermine, manteniendo contento al navegador$Npara que no detenga la reproducción">nav', + "mt_fullpre": 'intentar precargar la canción entera;$N✅ activar en conexiones inestables,$N❌ desactivar probablemente en conexiones lentas">completa', + "mt_fau": 'en teléfonos, evitar que la música se detenga si la siguiente canción no se precarga lo suficientemente rápido (puede causar fallos en la visualización de etiquetas)">☕️', + "mt_waves": 'barra de búsqueda con forma de onda:$Nmostrar la amplitud del audio en la barra de progreso">~s', + "mt_npclip": 'mostrar botones para copiar al portapapeles la canción actual">/np', + "mt_m3u_c": 'mostrar botones para copiar al portapapeles las$Ncanciones seleccionadas como entradas de lista m3u8">📻', + "mt_octl": 'integración con SO (teclas multimedia / OSD)">ctl-so', + "mt_oseek": 'permitir buscar a través de la integración con el SO$N$Nnota: en algunos dispositivos (iPhones),$Nesto reemplaza el botón de siguiente canción">búsqueda', + "mt_oscv": 'mostrar carátula del álbum en OSD">arte', + "mt_follow": 'mantener la pista en reproducción visible en pantalla">🎯', + "mt_compact": 'controles compactos">⟎', + "mt_uncache": 'limpiar caché  (prueba esto si tu navegador guardó en caché$Nuna copia rota de una canción que se niega a reproducir)">limpiar caché', + "mt_mloop": 'repetir la carpeta actual">🔁 bucle', + "mt_mnext": 'cargar la siguiente carpeta y continuar">📂 sig', + "mt_mstop": 'detener reproducción">⏸ parar', + "mt_cflac": 'convertir flac / wav a opus">flac', + "mt_caac": 'convertir aac / m4a a opus">aac', + "mt_coth": 'convertir todos los demás (no mp3) a opus">oth', + "mt_c2opus": 'la mejor opción para ordenadores, portátiles, android">opus', + "mt_c2owa": 'opus-weba, para iOS 17.5 y superior">owa', + "mt_c2caf": 'opus-caf, para iOS 11 a 17">caf', + "mt_c2mp3": 'usar en dispositivos muy antiguos">mp3', + "mt_c2ok": "bien, buena elección", + "mt_c2nd": "ese no es el formato de salida recomendado para tu dispositivo, pero está bien", + "mt_c2ng": "tu dispositivo no parece soportar este formato de salida, pero intentémoslo de todas formas", + "mt_xowa": "hay errores en iOS que impiden la reproducción en segundo plano con este formato; por favor, usa caf o mp3 en su lugar", + "mt_tint": "nivel de fondo (0-100) en la barra de búsqueda$Npara hacer el buffering menos molesto", + "mt_eq": "activa el ecualizador y el control de ganancia;$N$Nganancia <code>0</code> = volumen estándar 100% (sin modificar)$N$Nancho <code>1  </code> = estéreo estándar (sin modificar)$Nancho <code>0.5</code> = 50% de crossfeed izq-der$Nancho <code>0  </code> = mono$N$Nganancia <code>-0.8</code> y ancho <code>10</code> = eliminación de voz :^)$N$Nactivar el ecualizador hace que los álbumes sin pausas sean completamente sin pausas, así que déjalo activado con todos los valores a cero (excepto ancho = 1) si eso te importa", + "mt_drc": "activa el compresor de rango dinámico (aplanador de volumen / brickwaller); también activará el EQ para equilibrar el espagueti, así que pon todos los campos de EQ excepto 'ancho' a 0 si no lo quieres$N$Nbaja el volumen del audio por encima de THRESHOLD dB; por cada RATIO dB pasado THRESHOLD hay 1 dB de salida, así que los valores por defecto de umbral -24 y ratio 12 significan que nunca debería sonar más fuerte de -22 dB y es seguro aumentar la ganancia del ecualizador a 0.8, o incluso 1.8 con ATK 0 y un RLS enorme como 90 (solo funciona en firefox; RLS es máx. 1 en otros navegadores)$N$N(ver wikipedia, lo explican mucho mejor)", + + "mb_play": "reproducir", + "mm_hashplay": "¿reproducir este archivo de audio?", + "mm_m3u": "pulsa Intro/Aceptar para Reproducir\npulsa ESC/Cancelar para Editar", + "mp_breq": "se necesita firefox 82+ o chrome 73+ o iOS 15+", + "mm_bload": "cargando...", + "mm_bconv": "convirtiendo a {0}, por favor espera...", + "mm_opusen": "tu navegador no puede reproducir archivos aac / m4a;\nse ha activado la transcodificación a opus", + "mm_playerr": "fallo de reproducción: ", + "mm_eabrt": "El intento de reproducción fue cancelado", + "mm_enet": "Tu conexión a internet es inestable", + "mm_edec": "¿Este archivo está supuestamente corrupto?", + "mm_esupp": "Tu navegador no entiende este formato de audio", + "mm_eunk": "Error desconocido", + "mm_e404": "No se pudo reproducir el audio; error 404: Archivo no encontrado.", + "mm_e403": "No se pudo reproducir el audio; error 403: Acceso denegado.\n\nIntenta pulsar F5 para recargar, quizás se cerró tu sesión", + "mm_e500": "No se pudo reproducir el audio; error 500: Revisa los registros del servidor.", + "mm_e5xx": "No se pudo reproducir el audio; error del servidor ", + "mm_nof": "no se encuentran más archivos de audio cerca", + "mm_prescan": "Buscando música para reproducir a continuación...", + "mm_scank": "Encontrada la siguiente canción:", + "mm_uncache": "caché limpiada; todas las canciones se volverán a descargar en la próxima reproducción", + "mm_hnf": "esa canción ya no existe", + + "im_hnf": "esa imagen ya no existe", + + "f_empty": "esta carpeta está vacía", + "f_chide": "esto ocultará la columna «{0}»\n\npuedes volver a mostrar las columnas en la pestaña de configuración", + "f_bigtxt": "este archivo pesa {0} MiB -- ¿realmente verlo como texto?", + "f_bigtxt2": "¿ver solo el final del archivo en su lugar? esto también activará el seguimiento, mostrando las líneas de texto recién añadidas en tiempo real", + "fbd_more": '
mostrando {0} de {1} archivos; mostrar {2} o mostrar todos
', + "fbd_all": '
mostrando {0} de {1} archivos; mostrar todos
', + "f_anota": "solo {0} de los {1} elementos fueron seleccionados;\npara seleccionar la carpeta completa, primero desplázate hasta el final", + + "f_dls": "los enlaces a archivos en la carpeta actual se han\nconvertido en enlaces de descarga", + + "f_partial": "Para descargar de forma segura un archivo que se está subiendo actualmente, por favor haz clic en el archivo con el mismo nombre, pero sin la extensión .PARTIAL. Por favor, pulsa CANCELAR o Escape para hacer esto.\n\nPulsar ACEPTAR o Intro ignorará esta advertencia y continuará descargando el archivo temporal .PARTIAL, lo que casi con toda seguridad te dará datos corruptos.", + + "ft_paste": "pegar {0} elementos$NAtajo: ctrl-V", + "fr_eperm": "no se puede renombrar:\nno tienes permiso de “mover” en esta carpeta", + "fd_eperm": "no se puede eliminar:\nno tienes permiso de “eliminar” en esta carpeta", + "fc_eperm": "no se puede cortar:\nno tienes permiso de “mover” en esta carpeta", + "fp_eperm": "no se puede pegar:\nno tienes permiso de “escribir” en esta carpeta", + "fr_emore": "selecciona al menos un elemento para renombrar", + "fd_emore": "selecciona al menos un elemento para eliminar", + "fc_emore": "selecciona al menos un elemento para cortar", + "fcp_emore": "selecciona al menos un elemento para copiar al portapapeles", + + "fs_sc": "compartir la carpeta en la que estás", + "fs_ss": "compartir los archivos seleccionados", + "fs_just1d": "no puedes seleccionar más de una carpeta,\no mezclar archivos y carpetas en una selección", + "fs_abrt": "❌ abortar", + "fs_rand": "🎲 nombre aleatorio", + "fs_go": "✅ crear enlace", + "fs_name": "nombre", + "fs_src": "origen", + "fs_pwd": "contraseña", + "fs_exp": "caducidad", + "fs_tmin": "minutos", + "fs_thrs": "horas", + "fs_tdays": "días", + "fs_never": "eterno", + "fs_pname": "nombre opcional del enlace; será aleatorio si se deja en blanco", + "fs_tsrc": "el archivo o carpeta a compartir", + "fs_ppwd": "contraseña opcional", + "fs_w8": "creando enlace...", + "fs_ok": "pulsa Intro/Aceptar para Copiar al Portapapeles\npulsa ESC/Cancelar para Cerrar", + + "frt_dec": "puede arreglar algunos casos de nombres de archivo rotos\">url-decode", + "frt_rst": "restaurar los nombres de archivo modificados a los originales\">↺ restablecer", + "frt_abrt": "abortar y cerrar esta ventana\">❌ cancelar", + "frb_apply": "APLICAR RENOMBRADO", + "fr_adv": "renombrado por lotes / metadatos / patrones\">avanzado", + "fr_case": "regex sensible a mayúsculas\">mayús", + "fr_win": "nombres seguros para windows; reemplaza <>:"\\|?* con caracteres japoneses de ancho completo\">win", + "fr_slash": "reemplaza / con un carácter que no cree nuevas carpetas\">sin /", + "fr_re": "patrón de búsqueda regex para aplicar a los nombres de archivo originales; los grupos de captura se pueden referenciar en el campo de formato de abajo como <code>(1)</code> y <code>(2)</code> y así sucesivamente", + "fr_fmt": "inspirado en foobar2000:$N<code>(title)</code> se reemplaza por el título de la canción,$N<code>[(artist) - ](title)</code> omite la parte [entre corchetes] si el artista está en blanco$N<code>$lpad((tn),2,0)</code> rellena el número de pista a 2 dígitos", + "fr_pdel": "eliminar", + "fr_pnew": "guardar como", + "fr_pname": "proporciona un nombre para tu nuevo preajuste", + "fr_aborted": "abortado", + "fr_lold": "nombre antiguo", + "fr_lnew": "nombre nuevo", + "fr_tags": "etiquetas para los archivos seleccionados (solo lectura, como referencia):", + "fr_busy": "renombrando {0} elementos...\n\n{1}", + "fr_efail": "fallo al renombrar:\n", + "fr_nchg": "{0} de los nuevos nombres fueron alterados debido a win y/o sin /\n\n¿Aceptar para continuar con estos nuevos nombres alterados?", + + "fd_ok": "eliminación correcta", + "fd_err": "fallo al eliminar:\n", + "fd_none": "no se eliminó nada; quizás bloqueado por la configuración del servidor (xbd)?", + "fd_busy": "eliminando {0} elementos...\n\n{1}", + "fd_warn1": "¿ELIMINAR estos {0} elementos?", + "fd_warn2": "¡Última oportunidad! No se puede deshacer. ¿Eliminar?", + + "fc_ok": "cortados {0} elementos", + "fc_warn": "cortados {0} elementos\n\npero: solo esta pestaña del navegador puede pegarlos\n(dado que la selección es absolutamente masiva)", + + "fcc_ok": "copiados {0} elementos al portapapeles", + "fcc_warn": "copiados {0} elementos al portapapeles\n\npero: solo esta pestaña del navegador puede pegarlos\n(dado que la selección es absolutamente masiva)", + + "fp_apply": "usar estos nombres", + "fp_ecut": "primero corta o copia algunos archivos / carpetas para pegar / mover\n\nnota: puedes cortar / pegar entre diferentes pestañas del navegador", + "fp_ename": "{0} elementos no se pueden mover aquí porque los nombres ya existen. Dales nuevos nombres abajo para continuar, o deja el nombre en blanco para omitirlos:", + "fcp_ename": "{0} elementos no se pueden copiar aquí porque los nombres ya existen. Dales nuevos nombres abajo para continuar, o deja el nombre en blanco para omitirlos:", + "fp_emore": "todavía quedan algunas colisiones de nombres por resolver", + "fp_ok": "movimiento correcto", + "fcp_ok": "copia correcta", + "fp_busy": "moviendo {0} elementos...\n\n{1}", + "fcp_busy": "copiando {0} elementos...\n\n{1}", + "fp_err": "fallo al mover:\n", + "fcp_err": "fallo al copiar:\n", + "fp_confirm": "¿mover estos {0} elementos aquí?", + "fcp_confirm": "¿copiar estos {0} elementos aquí?", + "fp_etab": "fallo al leer el portapapeles de otra pestaña del navegador", + "fp_name": "subiendo un archivo desde tu dispositivo. Dale un nombre:", + "fp_both_m": "
elige qué pegar
Intro = Mover {0} archivos desde «{1}»\nESC = Subir {2} archivos desde tu dispositivo", + "fcp_both_m": "
elige qué pegar
Intro = Copiar {0} archivos desde «{1}»\nESC = Subir {2} archivos desde tu dispositivo", + "fp_both_b": "MoverSubir", + "fcp_both_b": "CopiarSubir", + + "mk_noname": "escribe un nombre en el campo de texto de la izquierda antes de hacer eso :p", + + "tv_load": "Cargando documento de texto:\n\n{0}\n\n{1}% ({2} de {3} MiB cargados)", + "tv_xe1": "no se pudo cargar el archivo de texto:\n\nerror ", + "tv_xe2": "404, archivo no encontrado", + "tv_lst": "lista de archivos de texto en", + "tvt_close": "volver a la vista de carpetas$NAtajo: M (o Esc)\">❌ cerrar", + "tvt_dl": "descargar este archivo$NAtajo: Y\">💾 descargar", + "tvt_prev": "mostrar documento anterior$NAtajo: i\">⬆ ant", + "tvt_next": "mostrar siguiente documento$NAtajo: K\">⬇ sig", + "tvt_sel": "seleccionar archivo   ( para cortar / copiar / eliminar / ... )$NAtajo: S\">sel", + "tvt_edit": "abrir archivo en editor de texto$NAtajo: E\">✏️ editar", + "tvt_tail": "monitorizar cambios en el archivo; mostrar nuevas líneas en tiempo real\">📡 seguir", + "tvt_wrap": "ajuste de línea\">↵", + "tvt_atail": "bloquear el desplazamiento al final de la página\">⚓", + "tvt_ctail": "decodificar colores de terminal (códigos de escape ansi)\">🌈", + "tvt_ntail": "límite de historial (cuántos bytes de texto mantener cargados)", + + "m3u_add1": "canción añadida a la lista m3u", + "m3u_addn": "{0} canciones añadidas a la lista m3u", + "m3u_clip": "lista m3u copiada al portapapeles\n\ndebes crear un nuevo archivo de texto llamado algo.m3u y pegar la lista en ese documento; esto lo hará reproducible", + + "gt_vau": "no mostrar vídeos, solo reproducir el audio\">🎧", + "gt_msel": "activar selección de archivos; ctrl-clic en un archivo para anular$N$N<em>cuando está activo: doble clic en un archivo / carpeta para abrirlo</em>$N$NAtajo: S\">multiselección", + "gt_crop": "recortar miniaturas\">recortar", + "gt_3x": "miniaturas de alta resolución\">3x", + "gt_zoom": "zoom", + "gt_chop": "recortar", + "gt_sort": "ordenar por", + "gt_name": "nombre", + "gt_sz": "tamaño", + "gt_ts": "fecha", + "gt_ext": "tipo", + "gt_c1": "truncar más los nombres de archivo (mostrar menos)", + "gt_c2": "truncar menos los nombres de archivo (mostrar más)", + + "sm_w8": "buscando...", + "sm_prev": "los resultados de búsqueda a continuación son de una consulta anterior:\n ", + "sl_close": "cerrar resultados de búsqueda", + "sl_hits": "mostrando {0} resultados", + "sl_moar": "cargar más", + + "s_sz": "tamaño", + "s_dt": "fecha", + "s_rd": "ruta", + "s_fn": "nombre", + "s_ta": "etiquetas", + "s_ua": "subido@", + "s_ad": "avanzado", + "s_s1": "MiB mínimo", + "s_s2": "MiB máximo", + "s_d1": "mín. iso8601", + "s_d2": "máx. iso8601", + "s_u1": "subido después de", + "s_u2": "y/o antes de", + "s_r1": "la ruta contiene   (separado por espacios)", + "s_f1": "el nombre contiene   (negar con -no)", + "s_t1": "las etiquetas contienen   (^=inicio, fin=$)", + "s_a1": "propiedades de metadatos específicas", + + "md_eshow": "no se puede renderizar ", + "md_off": "[📜léeme] desactivado en [⚙️] -- documento oculto", + + "badreply": "Fallo al procesar la respuesta del servidor", + + "xhr403": "403: Acceso denegado\n\nintenta pulsar F5, quizás se cerró tu sesión", + "xhr0": "desconocido (probablemente se perdió la conexión con el servidor, o el servidor está desconectado)", + "cf_ok": "perdón por eso -- la protección DD" + wah + "oS se activó\n\nlas cosas deberían reanudarse en unos 30 segundos\n\nsi no pasa nada, pulsa F5 para recargar la página", + "tl_xe1": "no se pudieron listar las subcarpetas:\n\nerror ", + "tl_xe2": "404: Carpeta no encontrada", + "fl_xe1": "no se pudieron listar los archivos en la carpeta:\n\nerror ", + "fl_xe2": "404: Carpeta no encontrada", + "fd_xe1": "no se pudo crear la subcarpeta:\n\nerror ", + "fd_xe2": "404: Carpeta de nivel superior no encontrada", + "fsm_xe1": "no se pudo enviar el mensaje:\n\nerror ", + "fsm_xe2": "404: Carpeta de nivel superior no encontrada", + "fu_xe1": "fallo al cargar la lista de deshacer del servidor:\n\nerror ", + "fu_xe2": "404: ¿Archivo no encontrado?", + + "fz_tar": "archivo gnu-tar sin comprimir (linux / mac)", + "fz_pax": "tar formato pax sin comprimir (más lento)", + "fz_targz": "gnu-tar con compresión gzip nivel 3$N$Nesto suele ser muy lento, así que$Nusa tar sin comprimir en su lugar", + "fz_tarxz": "gnu-tar con compresión xz nivel 1$N$Nesto suele ser muy lento, así que$Nusa tar sin comprimir en su lugar", + "fz_zip8": "zip con nombres de archivo utf8 (puede dar problemas en windows 7 y anteriores)", + "fz_zipd": "zip con nombres de archivo cp437 tradicionales, para software muy antiguo", + "fz_zipc": "cp437 con crc32 calculado tempranamente,$Npara MS-DOS PKZIP v2.04g (octubre 1993)$N(tarda más en procesar antes de que la descarga pueda empezar)", + + "un_m1": "puedes eliminar tus subidas recientes (o abortar las inacabadas) a continuación", + "un_upd": "actualizar", + "un_m4": "o compartir los archivos visibles a continuación:", + "un_ulist": "mostrar", + "un_ucopy": "copiar", + "un_flt": "filtro opcional:  la URL debe contener", + "un_fclr": "limpiar filtro", + "un_derr": "fallo al deshacer-eliminar:\n", + "un_f5": "algo se rompió, por favor intenta actualizar o pulsa F5", + "un_uf5": "lo siento pero tienes que refrescar la página (por ejemplo pulsando F5 o CTRL-R) antes de que esta subida pueda ser abortada", + "un_nou": "aviso: servidor demasiado ocupado para mostrar subidas inacabadas; haz clic en el enlace \"actualizar\" en un momento", + "un_noc": "aviso: la opción de deshacer subidas completadas no está activada/permitida en la configuración del servidor", + "un_max": "mostrando los primeros 2000 archivos (usa el filtro)", + "un_avail": "{0} subidas recientes se pueden eliminar
{1} inacabadas se pueden abortar", + "un_m2": "ordenado por tiempo de subida; más recientes primero:", + "un_no1": "¡pues no! ninguna subida es suficientemente reciente", + "un_no2": "¡pues no! ninguna subida que coincida con ese filtro es suficientemente reciente", + "un_next": "eliminar los siguientes {0} archivos a continuación", + "un_abrt": "abortar", + "un_del": "eliminar", + "un_m3": "cargando tus subidas recientes...", + "un_busy": "eliminando {0} archivos...", + "un_clip": "{0} enlaces copiados al portapapeles", + + "u_https1": "deberías", + "u_https2": "cambiar a https", + "u_https3": "para un mejor rendimiento", + "u_ancient": "tu navegador es impresionantemente antiguo -- quizás deberías usar bup en su lugar", + "u_nowork": "se necesita firefox 53+ o chrome 57+ o iOS 11+", + "tail_2old": "se necesita firefox 105+ o chrome 71+ o iOS 14.5+", + "u_nodrop": "tu navegador es demasiado antiguo para subir arrastrando y soltando", + "u_notdir": "¡eso no es una carpeta!\n\ntu navegador es demasiado antiguo,\npor favor intenta arrastrar y soltar en su lugar", + "u_uri": "para arrastrar y soltar imágenes desde otras ventanas del navegador,\npor favor suéltalas sobre el gran botón de subida", + "u_enpot": "cambiar a UI ligera (puede mejorar la velocidad de subida)", + "u_depot": "cambiar a UI elegante (puede reducir la velocidad de subida)", + "u_gotpot": "cambiando a la UI ligera para mejorar la velocidad de subida,\n\n¡siéntete libre de no estar de acuerdo y volver a cambiar!", + "u_pott": "

archivos:   {0} finalizados,   {1} fallidos,   {2} ocupados,   {3} en cola

", + "u_ever": "este es el subidor básico; up2k necesita al menos
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": "este es el subidor básico; up2k es mejor", + "u_uput": "optimizar para velocidad (omitir checksum)", + "u_ewrite": "no tienes acceso de escritura a esta carpeta", + "u_eread": "no tienes acceso de lectura a esta carpeta", + "u_enoi": "la búsqueda de archivos no está activada en la configuración del servidor", + "u_enoow": "sobrescribir no funcionará aquí; se necesita permiso de eliminación", + "u_badf": "Estos {0} archivos (de un total de {1}) se omitieron, posiblemente debido a permisos del sistema de archivos:\n\n", + "u_blankf": "Estos {0} archivos (de un total de {1}) están en blanco / vacíos; ¿subirlos de todos modos?\n\n", + "u_applef": "Estos {0} archivos (de un total de {1}) probablemente no son deseables;\nPulsa Aceptar/Intro para OMITIR los siguientes archivos,\nPulsa Cancelar/ESC para NO excluir, y SUBIR esos también:\n\n", + "u_just1": "\nQuizás funcione mejor si seleccionas solo un archivo", + "u_ff_many": "si usas Linux / MacOS / Android, esta cantidad de archivos podría bloquear Firefox!\nsi eso ocurre, por favor inténtalo de nuevo (o usa Chrome).", + "u_up_life": "Esta subida será eliminada del servidor\n{0} después de que se complete", + "u_asku": "subir estos {0} archivos a {1}", + "u_unpt": "puedes deshacer / eliminar esta subida usando el 🧯 de arriba a la izquierda", + "u_bigtab": "a punto de mostrar {0} archivos\n\nesto podría bloquear tu navegador, ¿estás seguro?", + "u_scan": "Escaneando archivos...", + "u_dirstuck": "el iterador de directorios se atascó intentando acceder a los siguientes {0} elementos; se omitirán:", + "u_etadone": "Hecho ({0}, {1} archivos)", + "u_etaprep": "(preparando para subir)", + "u_hashdone": "hashing completado", + "u_hashing": "hash", + "u_hs": "negociando...", + "u_started": "los archivos se están subiendo ahora; mira en [🚀]", + "u_dupdefer": "duplicado; se procesará después de todos los demás archivos", + "u_actx": "haz clic en este texto para evitar la pérdida de
rendimiento al cambiar a otras ventanas/pestañas", + "u_fixed": "¡OK!  Arreglado 👍", + "u_cuerr": "fallo al subir el trozo {0} de {1};\nprobablemente inofensivo, continuando\n\narchivo: {2}", + "u_cuerr2": "el servidor rechazó la subida (trozo {0} de {1});\nse reintentará más tarde\n\narchivo: {2}\n\nerror ", + "u_ehstmp": "se reintentará; mira abajo a la derecha", + "u_ehsfin": "el servidor rechazó la solicitud para finalizar la subida; reintentando...", + "u_ehssrch": "el servidor rechazó la solicitud para realizar la búsqueda; reintentando...", + "u_ehsinit": "el servidor rechazó la solicitud para iniciar la subida; reintentando...", + "u_eneths": "error de red al realizar la negociación de subida; reintentando...", + "u_enethd": "error de red al comprobar la existencia del destino; reintentando...", + "u_cbusy": "esperando a que el servidor vuelva a confiar en nosotros después de un fallo de red...", + "u_ehsdf": "¡el servidor se quedó sin espacio en disco!\n\nse seguirá reintentando, por si alguien\nlibera suficiente espacio para continuar", + "u_emtleak1": "parece que tu navegador podría tener una fuga de memoria;\npor favor", + "u_emtleak2": " cambia a https (recomendado) o ", + "u_emtleak3": " ", + "u_emtleakc": "prueba lo siguiente:\n
  • pulsa F5 para refrescar la página
  • luego desactiva el botón  mt  en los  ⚙️ ajustes
  • e intenta esa subida de nuevo
Las subidas serán un poco más lentas, pero bueno.\n¡Perdón por las molestias!\n\nPD: chrome v107 tiene una solución para esto", + "u_emtleakf": "prueba lo siguiente:\n
  • pulsa F5 para refrescar la página
  • luego activa 🥔 (ligera) en la interfaz de subida
  • e intenta esa subida de nuevo
\nPD: firefox con suerte tendrá una solución en algún momento", + "u_s404": "no encontrado en el servidor", + "u_expl": "explicar", + "u_maxconn": "la mayoría de los navegadores limitan esto a 6, pero firefox te permite aumentarlo con connections-per-server en about:config", + "u_tu": '

AVISO: turbo activado,  el cliente puede no detectar y reanudar subidas incompletas; ver tooltip del botón turbo

', + "u_ts": '

AVISO: turbo activado,  los resultados de búsqueda pueden ser incorrectos; ver tooltip del botón turbo

', + "u_turbo_c": "turbo está desactivado en la configuración del servidor", + "u_turbo_g": "desactivando turbo porque no tienes\nprivilegios para listar directorios en este volumen", + "u_life_cfg": 'autoeliminar después de min (o horas)', + "u_life_est": 'la subida se eliminará ---', + "u_life_max": "esta carpeta impone una\nvida máxima de {0}", + "u_unp_ok": "se permite deshacer la subida durante {0}", + "u_unp_ng": "NO se permitirá deshacer la subida", + "ue_ro": "tu acceso a esta carpeta es de solo lectura\n\n", + "ue_nl": "actualmente no has iniciado sesión", + "ue_la": "actualmente has iniciado sesión como \"{0}\"", + "ue_sr": "actualmente estás en modo de búsqueda de archivos\n\ncambia a modo de subida haciendo clic en la lupa 🔎 (junto al gran botón BUSCAR), e intenta subir de nuevo\n\nlo siento", + "ue_ta": "intenta subir de nuevo, ahora debería funcionar", + "ue_ab": "este archivo ya se está subiendo a otra carpeta, y esa subida debe completarse antes de que el archivo pueda ser subido a otro lugar.\n\nPuedes abortar y olvidar la subida inicial usando el 🧯 de arriba a la izquierda", + "ur_1uo": "OK: Archivo subido con éxito", + "ur_auo": "OK: Todos los {0} archivos subidos con éxito", + "ur_1so": "OK: Archivo encontrado en el servidor", + "ur_aso": "OK: Todos los {0} archivos encontrados en el servidor", + "ur_1un": "Subida fallida, lo siento", + "ur_aun": "Todas las {0} subidas fallaron, lo siento", + "ur_1sn": "El archivo NO se encontró en el servidor", + "ur_asn": "Los {0} archivos NO se encontraron en el servidor", + "ur_um": "Finalizado;\n{0} subidas OK,\n{1} subidas fallidas, lo siento", + "ur_sm": "Finalizado;\n{0} archivos encontrados en el servidor,\n{1} archivos NO encontrados en el servidor", + + "lang_set": "¿refrescar para que el cambio surta efecto?" + } }; -var LANGS = ["eng", "nor", "chi", "deu", "rus", "ukr"]; +var LANGS = ["eng", "nor", "chi", "deu", "rus", "ukr", "spa"]; if (window.langmod) langmod(); From be729fe5572567d1ca55c90bb4ec19fb63016421 Mon Sep 17 00:00:00 2001 From: Juan Herruzo Date: Wed, 30 Jul 2025 11:44:17 +0200 Subject: [PATCH 038/174] small tweaks in the localization when looking a it in the gui --- copyparty/web/browser.js | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 6afa9b26..e090515d 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3866,7 +3866,7 @@ var Ls = { "ht_d2": "días", "ht_and": " y ", - "goh": "panel-de-control", + "goh": "panel de control", "gop": 'hermano anterior">anterior', "gou": 'carpeta de nivel superior">subir', "gon": 'siguiente carpeta">siguiente', @@ -3875,14 +3875,14 @@ var Ls = { "ot_close": "cerrar submenú", "ot_search": "buscar archivos por atributos, ruta / nombre, etiquetas de música, o cualquier combinación$N$N<code>foo bar</code> = debe contener «foo» y «bar»,$N<code>foo -bar</code> = debe contener «foo» pero no «bar»,$N<code>^yana .opus$</code> = empieza con «yana» y es un archivo «opus»$N<code>"try unite"</code> = contiene exactamente «try unite»$N$Nel formato de fecha es iso-8601, como$N<code>2009-12-31</code> o <code>2020-09-12 23:30:00</code>", "ot_unpost": "dessubir: elimina tus subidas recientes, o aborta las inacabadas", - "ot_bup": "bup: subidor básico, soporta hasta netscape 4.0", + "ot_bup": "bup: uploader básico, soporta hasta netscape 4.0", "ot_mkdir": "mkdir: crear un nuevo directorio", "ot_md": "new-md: crear un nuevo documento markdown", "ot_msg": "msg: enviar un mensaje al registro del servidor", "ot_mp": "opciones del reproductor multimedia", "ot_cfg": "opciones de configuración", - "ot_u2i": "up2k: subir archivos (si tienes acceso de escritura) o cambiar a modo de búsqueda para ver si existen en el servidor$N$Nlas subidas se pueden reanudar, usan múltiples hilos y conservan la fecha de los archivos, pero consume más CPU que [🎈]  (el subidor básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", - "ot_u2w": "up2k: subir archivos con soporte para reanudación (cierra tu navegador y arrastra los mismos archivos más tarde)$N$NMultihilo y conserva las fechas de los archivos, pero usa más CPU que [🎈]  (el subidor básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", + "ot_u2i": "up2k: subir archivos (si tienes acceso de escritura) o cambiar a modo de búsqueda para ver si existen en el servidor$N$Nlas subidas se pueden reanudar, usan múltiples hilos y conservan la fecha de los archivos, pero consume más CPU que [🎈]  (el uploader básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", + "ot_u2w": "up2k: subir archivos con soporte para reanudación (cierra tu navegador y arrastra los mismos archivos más tarde)$N$NMultihilo y conserva las fechas de los archivos, pero usa más CPU que [🎈]  (el uploader básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", "ot_noie": "Por favor, usa Chrome / Firefox / Edge", "ab_mkdir": "crear directorio", @@ -3978,7 +3978,7 @@ var Ls = { "cl_hpick": "toca en las cabeceras de columna para ocultarlas en la tabla de abajo", "cl_hcancel": "ocultación de columna cancelada", - "ct_grid": '田 la cuadrícula', + "ct_grid": '田 cuadrícula', "ct_ttips": '◔ ◡ ◔">ℹ️ tooltips', "ct_thumb": 'en vista de cuadrícula, alternar iconos o miniaturas$NAtajo: T">🖼️ miniaturas', "ct_csel": 'usa CTRL y SHIFT para seleccionar archivos en la vista de cuadrícula">sel', @@ -4315,8 +4315,8 @@ var Ls = { "u_depot": "cambiar a UI elegante (puede reducir la velocidad de subida)", "u_gotpot": "cambiando a la UI ligera para mejorar la velocidad de subida,\n\n¡siéntete libre de no estar de acuerdo y volver a cambiar!", "u_pott": "

archivos:   {0} finalizados,   {1} fallidos,   {2} ocupados,   {3} en cola

", - "u_ever": "este es el subidor básico; up2k necesita al menos
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", - "u_su2k": "este es el subidor básico; up2k es mejor", + "u_ever": "este es el uploader básico; up2k necesita al menos
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": "este es el uploader básico; up2k es mejor", "u_uput": "optimizar para velocidad (omitir checksum)", "u_ewrite": "no tienes acceso de escritura a esta carpeta", "u_eread": "no tienes acceso de lectura a esta carpeta", From af34fbf1a4e9526258bc4ccac66db126f3d8e6fa Mon Sep 17 00:00:00 2001 From: Juan Herruzo Date: Wed, 30 Jul 2025 19:11:36 +0200 Subject: [PATCH 039/174] fix badly translated keys --- copyparty/web/browser.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index e090515d..268bc76c 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3971,7 +3971,7 @@ var Ls = { "cl_favico": "favicon", "cl_bigdir": "directorios grandes", "cl_hsort": "#ordenar", - "cl_keytype": "notación de teclas", + "cl_keytype": "notación musical", "cl_hiddenc": "columnas ocultas", "cl_hidec": "ocultar", "cl_reset": "restablecer", @@ -4028,7 +4028,7 @@ var Ls = { "tt_hover": "revelar líneas que se desbordan al pasar el ratón$N( rompe el desplazamiento a menos que el $N  cursor esté en el margen izquierdo )", "ml_pmode": "al final de la carpeta...", - "ml_btns": "cmds", + "ml_btns": "acciones", "ml_tcode": "transcodificar", "ml_tcode2": "transcodificar a", "ml_tint": "tinte", @@ -4038,7 +4038,7 @@ var Ls = { "mt_loop": 'poner en bucle/repetir una canción">🔁', "mt_one": 'parar después de una canción">1️⃣', "mt_shuf": 'reproducir aleatoriamente las canciones en cada carpeta">🔀', - "mt_aplay": 'reproducción automática si hay un ID de canción en el enlace en el que hiciste clic para acceder al servidor$N$Ndesactivar esto también evitará que la URL de la página se actualice con IDs de canción al reproducir música, para prevenir la reproducción automática si se pierden estos ajustes pero la URL permanece">a▶', + "mt_aplay": 'reproducir automaticamente si hay un ID de canción en el enlace en el que hiciste clic para acceder al servidor$N$Ndesactivar esto también evitará que la URL de la página se actualice con IDs de canción al reproducir música, para prevenir la reproducción automática si se pierden estos ajustes pero la URL permanece">a▶', "mt_preload": 'empezar a cargar la siguiente canción cerca del final para una reproducción sin pausas">precarga', "mt_prescan": 'ir a la siguiente carpeta antes de que la última canción$Ntermine, manteniendo contento al navegador$Npara que no detenga la reproducción">nav', "mt_fullpre": 'intentar precargar la canción entera;$N✅ activar en conexiones inestables,$N❌ desactivar probablemente en conexiones lentas">completa', @@ -4051,7 +4051,7 @@ var Ls = { "mt_oscv": 'mostrar carátula del álbum en OSD">arte', "mt_follow": 'mantener la pista en reproducción visible en pantalla">🎯', "mt_compact": 'controles compactos">⟎', - "mt_uncache": 'limpiar caché  (prueba esto si tu navegador guardó en caché$Nuna copia rota de una canción que se niega a reproducir)">limpiar caché', + "mt_uncache": 'limpiar caché  (prueba esto si tu navegador guardó en caché$Nuna copia rota de una canción que se niega a reproducir)">limpiar caché', "mt_mloop": 'repetir la carpeta actual">🔁 bucle', "mt_mnext": 'cargar la siguiente carpeta y continuar">📂 sig', "mt_mstop": 'detener reproducción">⏸ parar', From 6e35171c88770e04ff82d95f95266285c75b8ac5 Mon Sep 17 00:00:00 2001 From: Juan Herruzo Date: Thu, 31 Jul 2025 09:59:53 +0200 Subject: [PATCH 040/174] added splash.js translation --- copyparty/web/splash.js | 43 ++++++++++++++++++++++++++++++++++++++++- 1 file changed, 42 insertions(+), 1 deletion(-) diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 16a9d517..6baf75d2 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -94,7 +94,6 @@ var Ls = { "af1": "显示最近上传的文件", //m "ag1": "查看已知 IdP 用户", //m }, - "deu": { "a1": "Neu laden", "b1": "Tach, wie geht's?   (Du bist nicht angemeldet)", @@ -179,6 +178,48 @@ var Ls = { "af1": "показать недавние загрузки", "ag1": "показать известных IdP-пользователей", }, + "esp": { + "a1": "actualizar", + "b1": "hola   (no has iniciado sesión)", + "c1": "cerrar sesión", + "d1": "volcar estado de la pila", + "d2": "muestra el estado de todos los hilos activos", + "e1": "recargar configuración", + "e2": "recargar archivos de configuración (cuentas/volúmenes/indicadores de vol.),$Ny reescanear todos los volúmenes e2ds$N$Nnota: cualquier cambio en la configuración global$Nrequiere un reinicio completo para surtir efecto", + "f1": "puedes explorar:", + "g1": "puedes subir a:", + "cc1": "otras cosas:", + "h1": "desactivar k304", + "i1": "activar k304", + "j1": "activar k304 desconectará tu cliente en cada HTTP 304, lo que puede evitar que algunos proxies con errores se atasquen (dejando de cargar páginas de repente), pero también ralentizará las cosas en general", + "k1": "restablecer config. de cliente", + "l1": "inicia sesión para más:", + "m1": "bienvenido de nuevo,", + "n1": "404 no encontrado  ┐( ´ -`)┌", + "o1": '¿o quizás no tienes acceso? -- prueba con una contraseña o vuelve al inicio', + "p1": "403 prohibido  ~┻━┻", + "q1": 'usa una contraseña o vuelve al inicio', + "r1": "ir al inicio", + ".s1": "reescanear", + "t1": "acción", + "u2": "tiempo desde la última escritura en el servidor$N( subida / renombrar / ... )$N$N17d = 17 días$N1h23 = 1 hora 23 minutos$N4m56 = 4 minutos 56 segundos", + "v1": "conectar", + "v2": "usar este servidor como un disco duro local", + "w1": "cambiar a https", + "x1": "cambiar contraseña", + "y1": "editar recursos compartidos", + "z1": "desbloquear este recurso compartido:", + "ta1": "primero escribe tu nueva contraseña", + "ta2": "repite para confirmar la nueva contraseña:", + "ta3": "hay un error; por favor, inténtalo de nuevo", + "aa1": "archivos entrantes:", + "ab1": "desactivar no304", + "ac1": "activar no304", + "ad1": "activar no304 desactivará todo el almacenamiento en caché; prueba esto si k304 no fue suficiente. ¡Esto desperdiciará una gran cantidad de tráfico de red!", + "ae1": "descargas activas:", + "af1": "mostrar subidas recientes", + "ag1": "mostrar usuarios IdP conocidos" + }, }; if (window.langmod) From 1a5b7d40a82ad456594e77d268b914936addfca9 Mon Sep 17 00:00:00 2001 From: Juan Herruzo Date: Fri, 1 Aug 2025 11:30:20 +0200 Subject: [PATCH 041/174] changed esp to spa --- copyparty/web/browser.js | 2 +- copyparty/web/splash.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 268bc76c..5a36ce79 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3763,7 +3763,7 @@ var Ls = { "lang_set": "оновити сторінку, щоб зміни набули чинності?", }, - "esp": { + "spa": { "tt": "Español", "cols": { diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 6baf75d2..89661f28 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -178,7 +178,7 @@ var Ls = { "af1": "показать недавние загрузки", "ag1": "показать известных IdP-пользователей", }, - "esp": { + "spa": { "a1": "actualizar", "b1": "hola   (no has iniciado sesión)", "c1": "cerrar sesión", From 9dcb45133b86b923d4bdb44048df16fc2a375c4d Mon Sep 17 00:00:00 2001 From: Juan Herruzo Date: Fri, 1 Aug 2025 11:32:58 +0200 Subject: [PATCH 042/174] order spa alphabetically --- copyparty/web/browser.js | 1250 +++++++++++++++++++------------------- copyparty/web/splash.js | 85 ++- 2 files changed, 667 insertions(+), 668 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 5a36ce79..569d6ab3 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -2511,6 +2511,631 @@ var Ls = { "lang_set": "Neuladen um Änderungen anzuwenden?", }, + "spa": { + "tt": "Español", + + "cols": { + "c": "acciones", + "dur": "duración", + "q": "calidad / bitrate", + "Ac": "códec de audio", + "Vc": "códec de vídeo", + "Fmt": "formato / contenedor", + "Ahash": "checksum de audio", + "Vhash": "checksum de vídeo", + "Res": "resolución", + "T": "tipo de archivo", + "aq": "calidad de audio / bitrate", + "vq": "calidad de vídeo / bitrate", + "pixfmt": "submuestreo / estructura de píxel", + "resw": "resolución horizontal", + "resh": "resolución vertical", + "chs": "canales de audio", + "hz": "frecuencia de muestreo" + }, + + "hks": [ + [ + "varios", + ["ESC", "cerrar varias cosas"], + + "gestor de archivos", + ["G", "alternar vista de lista / cuadrícula"], + ["T", "alternar miniaturas / iconos"], + ["⇧ A/D", "tamaño de miniatura"], + ["ctrl-K", "eliminar seleccionados"], + ["ctrl-X", "cortar selección al portapapeles"], + ["ctrl-C", "copiar selección al portapapeles"], + ["ctrl-V", "pegar (mover/copiar) aquí"], + ["Y", "descargar seleccionados"], + ["F2", "renombrar seleccionados"], + + "selección en lista de archivos", + ["space", "alternar selección de archivo"], + ["↑/↓", "mover cursor de selección"], + ["ctrl ↑/↓", "mover cursor y vista"], + ["⇧ ↑/↓", "seleccionar anterior/siguiente archivo"], + ["ctrl-A", "seleccionar todos los archivos / carpetas"] + ], [ + "navegación", + ["B", "alternar breadcrumbs / panel de navegación"], + ["I/K", "anterior/siguiente carpeta"], + ["M", "carpeta de nivel superior (o contraer actual)"], + ["V", "alternar carpetas / archivos en panel de navegación"], + ["A/D", "tamaño del panel de navegación"] + ], [ + "reproductor de audio", + ["J/L", "anterior/siguiente canción"], + ["U/O", "saltar 10s atrás/adelante"], + ["0..9", "saltar a 0%..90%"], + ["P", "reproducir/pausar (también inicia)"], + ["S", "seleccionar canción en reproducción"], + ["Y", "descargar canción"] + ], [ + "visor de imágenes", + ["J/L, ←/→", "anterior/siguiente imagen"], + ["Home/End", "primera/última imagen"], + ["F", "pantalla completa"], + ["R", "rotar en sentido horario"], + ["⇧ R", "rotar en sentido antihorario"], + ["S", "seleccionar imagen"], + ["Y", "descargar imagen"] + ], [ + "reproductor de vídeo", + ["U/O", "saltar 10s atrás/adelante"], + ["P/K/Space", "reproducir/pausar"], + ["C", "continuar con el siguiente"], + ["V", "bucle"], + ["M", "silenciar"], + ["[ y ]", "establecer intervalo de bucle"] + ], [ + "visor de texto", + ["I/K", "anterior/siguiente archivo"], + ["M", "cerrar archivo"], + ["E", "editar archivo"], + ["S", "seleccionar archivo (para cortar/copiar/renombrar)"] + ] + ], + + "m_ok": "Aceptar", + "m_ng": "Cancelar", + + "enable": "Activar", + "danger": "PELIGRO", + "clipped": "copiado al portapapeles", + + "ht_s1": "segundo", + "ht_s2": "segundos", + "ht_m1": "minuto", + "ht_m2": "minutos", + "ht_h1": "hora", + "ht_h2": "horas", + "ht_d1": "día", + "ht_d2": "días", + "ht_and": " y ", + + "goh": "panel de control", + "gop": 'hermano anterior">anterior', + "gou": 'carpeta de nivel superior">subir', + "gon": 'siguiente carpeta">siguiente', + "logout": "Cerrar sesión ", + "access": " acceso", + "ot_close": "cerrar submenú", + "ot_search": "buscar archivos por atributos, ruta / nombre, etiquetas de música, o cualquier combinación$N$N<code>foo bar</code> = debe contener «foo» y «bar»,$N<code>foo -bar</code> = debe contener «foo» pero no «bar»,$N<code>^yana .opus$</code> = empieza con «yana» y es un archivo «opus»$N<code>"try unite"</code> = contiene exactamente «try unite»$N$Nel formato de fecha es iso-8601, como$N<code>2009-12-31</code> o <code>2020-09-12 23:30:00</code>", + "ot_unpost": "dessubir: elimina tus subidas recientes, o aborta las inacabadas", + "ot_bup": "bup: uploader básico, soporta hasta netscape 4.0", + "ot_mkdir": "mkdir: crear un nuevo directorio", + "ot_md": "new-md: crear un nuevo documento markdown", + "ot_msg": "msg: enviar un mensaje al registro del servidor", + "ot_mp": "opciones del reproductor multimedia", + "ot_cfg": "opciones de configuración", + "ot_u2i": "up2k: subir archivos (si tienes acceso de escritura) o cambiar a modo de búsqueda para ver si existen en el servidor$N$Nlas subidas se pueden reanudar, usan múltiples hilos y conservan la fecha de los archivos, pero consume más CPU que [🎈]  (el uploader básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", + "ot_u2w": "up2k: subir archivos con soporte para reanudación (cierra tu navegador y arrastra los mismos archivos más tarde)$N$NMultihilo y conserva las fechas de los archivos, pero usa más CPU que [🎈]  (el uploader básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", + "ot_noie": "Por favor, usa Chrome / Firefox / Edge", + + "ab_mkdir": "crear directorio", + "ab_mkdoc": "nuevo documento markdown", + "ab_msg": "enviar msg al registro del servidor", + + "ay_path": "saltar a carpetas", + "ay_files": "saltar a archivos", + + "wt_ren": "renombrar elementos seleccionados$NAtajo: F2", + "wt_del": "eliminar elementos seleccionados$NAtajo: ctrl-K", + "wt_cut": "cortar elementos seleccionados <small>(luego pegar en otro lugar)</small>$NAtajo: ctrl-X", + "wt_cpy": "copiar elementos seleccionados al portapapeles$N(para pegarlos en otro lugar)$NAtajo: ctrl-C", + "wt_pst": "pegar una selección previamente cortada / copiada$NAtajo: ctrl-V", + "wt_selall": "seleccionar todos los archivos$NAtajo: ctrl-A (con un archivo con foco)", + "wt_selinv": "invertir selección", + "wt_zip1": "descargar esta carpeta como un archivo comprimido", + "wt_selzip": "descargar selección como archivo comprimido", + "wt_seldl": "descargar selección como archivos separados$NAtajo: Y", + "wt_npirc": "copiar información de pista en formato IRC", + "wt_nptxt": "copiar información de pista en texto plano", + "wt_m3ua": "añadir a lista m3u (haz clic en 📻copiar después)", + "wt_m3uc": "copiar lista m3u al portapapeles", + "wt_grid": "alternar vista de cuadrícula / lista$NAtajo: G", + "wt_prev": "pista anterior$NAtajo: J", + "wt_play": "reproducir / pausar$NAtajo: P", + "wt_next": "siguiente pista$NAtajo: L", + + "ul_par": "subidas paralelas:", + "ut_rand": "aleatorizar nombres de archivo", + "ut_u2ts": 'copiar la fecha de última modificación$Nde tu sistema de archivos al servidor">📅', + "ut_ow": "sobrescribir archivos existentes en el servidor?$N🛡️: nunca (generará un nuevo nombre de archivo en su lugar)$N🕒: sobrescribir si el archivo del servidor es más antiguo que el tuyo$N♻️: siempre sobrescribir si los archivos son diferentes", + "ut_mt": "continuar generando hashes de otros archivos mientras se sube$N$Nquizás desactivar si tu CPU o HDD es un cuello de botella", + "ut_ask": 'pedir confirmación antes de iniciar la subida">💭', + "ut_pot": "mejorar la velocidad de subida en dispositivos lentos$Nsimplificando la interfaz de usuario", + "ut_srch": "no subir, en su lugar comprobar si los archivos ya $N existen en el servidor (escaneará todas las carpetas que puedas leer)", + "ut_par": "pausar subidas poniéndolo a 0$N$Naumentar si tu conexión es lenta / de alta latencia$N$Nmantener en 1 en LAN o si el HDD del servidor es un cuello de botella", + "ul_btn": "arrastra archivos / carpetas
aquí (o haz clic)", + "ul_btnu": "S U B I R", + "ul_btns": "B U S C A R", + + "ul_hash": "hash", + "ul_send": "envio", + "ul_done": "hecho", + "ul_idle1": "aún no hay subidas en cola", + "ut_etah": "velocidad media de <em>hashing</em>, y tiempo estimado para finalizar", + "ut_etau": "velocidad media de <em>subida</em> y tiempo estimado para finalizar", + "ut_etat": "velocidad media <em>total</em> y tiempo estimado para finalizar", + + "uct_ok": "completado con éxito", + "uct_ng": "fallido: error / rechazado / no encontrado", + "uct_done": "éxitos y fallos combinados", + "uct_bz": "generando hash o subiendo", + "uct_q": "inactivo, pendiente", + + "utl_name": "nombre de archivo", + "utl_ulist": "lista", + "utl_ucopy": "copiar", + "utl_links": "enlaces", + "utl_stat": "estado", + "utl_prog": "progreso", + + "utl_404": "404", + "utl_err": "ERROR", + "utl_oserr": "Error-SO", + "utl_found": "encontrado", + "utl_defer": "posponer", + "utl_yolo": "YOLO", + "utl_done": "hecho", + + "ul_flagblk": "los archivos se añadieron a la cola
sin embargo, hay un up2k ocupado en otra pestaña del navegador,
esperando a que termine primero", + "ul_btnlk": "la configuración del servidor ha bloqueado esta opción en este estado", + + "udt_up": "Subir", + "udt_srch": "Buscar", + "udt_drop": "suéltalo aquí", + + "u_nav_m": "
vale, ¿qué tienes?
Intro = Archivos (uno o más)\nESC = Una carpeta (incluyendo subcarpetas)", + "u_nav_b": "ArchivosUna carpeta", + + "cl_opts": "opciones", + "cl_themes": "tema", + "cl_langs": "idioma", + "cl_ziptype": "descarga de carpeta", + "cl_uopts": "opciones up2k", + "cl_favico": "favicon", + "cl_bigdir": "directorios grandes", + "cl_hsort": "#ordenar", + "cl_keytype": "notación musical", + "cl_hiddenc": "columnas ocultas", + "cl_hidec": "ocultar", + "cl_reset": "restablecer", + "cl_hpick": "toca en las cabeceras de columna para ocultarlas en la tabla de abajo", + "cl_hcancel": "ocultación de columna cancelada", + + "ct_grid": '田 cuadrícula', + "ct_ttips": '◔ ◡ ◔">ℹ️ tooltips', + "ct_thumb": 'en vista de cuadrícula, alternar iconos o miniaturas$NAtajo: T">🖼️ miniaturas', + "ct_csel": 'usa CTRL y SHIFT para seleccionar archivos en la vista de cuadrícula">sel', + "ct_ihop": 'al cerrar el visor de imágenes, desplazarse hasta el último archivo visto">g⮯', + "ct_dots": 'mostrar archivos ocultos (si el servidor lo permite)">archivos ocultos', + "ct_qdel": 'al eliminar archivos, pedir confirmación solo una vez">elim. rápida', + "ct_dir1st": 'ordenar carpetas antes que archivos">📁 primero', + "ct_nsort": 'orden natural (para nombres de archivo con dígitos iniciales)">ord. natural', + "ct_readme": 'mostrar README.md en los listados de carpetas">📜 léeme', + "ct_idxh": 'mostrar index.html en lugar del listado de carpetas">htm', + "ct_sbars": 'mostrar barra lateral">⟊', + + "cut_umod": 'si un archivo ya existe en el servidor, actualiza la fecha de última modificación del servidor para que coincida con tu archivo local (requiere permisos de escritura+eliminación)">re📅', + + "cut_turbo": 'el botón yolo, probablemente NO quieras activarlo:$N$Núsalo si estabas subiendo una gran cantidad de archivos y tuviste que reiniciar por alguna razón, y quieres continuar la subida lo antes posible$N$Nesto reemplaza la comprobación de hash por un simple "¿tiene este el mismo tamaño de archivo en el servidor?" así que si el contenido del archivo es diferente, NO se subirá$N$Ndeberías desactivar esto cuando la subida termine, y luego "subir" los mismos archivos de nuevo para que el cliente los verifique">turbo', + + "cut_datechk": 'no tiene efecto a menos que el botón turbo esté activado$N$Nreduce el factor yolo en una pequeña cantidad; comprueba si las fechas de los archivos en el servidor coinciden con las tuyas$N$Nteóricamente debería detectar la mayoría de las subidas inacabadas / corruptas, pero no es un sustituto de hacer una pasada de verificación con el turbo desactivado después">verif. fecha', + + "cut_u2sz": "tamaño (en MiB) de cada trozo de subida; los valores grandes vuelan mejor a través del atlántico. Prueba valores bajos en conexiones muy poco fiables", + + "cut_flag": "asegura que solo una pestaña esté subiendo a la vez $N -- otras pestañas también deben tener esto activado $N -- solo afecta a pestañas en el mismo dominio", + + "cut_az": "subir archivos en orden alfabético, en lugar de los más pequeños primero$N$Nel orden alfabético puede facilitar la detección visual de si algo salió mal en el servidor, pero hace la subida ligeramente más lenta en fibra / LAN", + + "cut_nag": "notificación del SO cuando la subida se complete$N(solo si el navegador o la pestaña no están activos)", + "cut_sfx": "alerta sonora cuando la subida se complete$N(solo si el navegador o la pestaña no están activos)", + + "cut_mt": 'usar multithreading para acelerar el hashing de archivos$N$Nesto usa web-workers y requiere$Nmás RAM (hasta 512 MiB extra)$N$Nhace https un 30% más rápido, http 4.5x más rápido">mt', + + "cut_wasm": 'usar wasm en lugar del hasher incorporado del navegador; mejora la velocidad en navegadores basados en chrome pero aumenta la carga de la CPU, y muchas versiones antiguas de chrome tienen errores que hacen que el navegador consuma toda la RAM y se bloquee si esto está activado">wasm', + + "cft_text": "texto del favicon (dejar en blanco y refrescar para desactivar)", + "cft_fg": "color de primer plano", + "cft_bg": "color de fondo", + + "cdt_lim": "número máximo de archivos a mostrar en una carpeta", + "cdt_ask": "al llegar al final,$Nen lugar de cargar más archivos,$Npreguntar qué hacer", + "cdt_hsort": "cuántas reglas de ordenación (<code>,sorthref</code>) incluir en las URLs de medios. Ponerlo a 0 también ignorará las reglas de ordenación incluidas en los enlaces de medios al hacer clic en ellos", + + "tt_entree": "mostrar panel de navegación (barra lateral con árbol de directorios)$NAtajo: B", + "tt_detree": "mostrar breadcrumbs$NAtajo: B", + "tt_visdir": "desplazarse a la carpeta seleccionada", + "tt_ftree": "alternar árbol de carpetas / archivos de texto$NAtajo: V", + "tt_pdock": "mostrar carpetas de niveles superiores en un panel acoplado en la parte superior", + "tt_dynt": "crecimiento automático a medida que el árbol se expande", + "tt_wrap": "ajuste de línea", + "tt_hover": "revelar líneas que se desbordan al pasar el ratón$N( rompe el desplazamiento a menos que el $N  cursor esté en el margen izquierdo )", + + "ml_pmode": "al final de la carpeta...", + "ml_btns": "acciones", + "ml_tcode": "transcodificar", + "ml_tcode2": "transcodificar a", + "ml_tint": "tinte", + "ml_eq": "ecualizador de audio", + "ml_drc": "compresor de rango dinámico", + + "mt_loop": 'poner en bucle/repetir una canción">🔁', + "mt_one": 'parar después de una canción">1️⃣', + "mt_shuf": 'reproducir aleatoriamente las canciones en cada carpeta">🔀', + "mt_aplay": 'reproducir automaticamente si hay un ID de canción en el enlace en el que hiciste clic para acceder al servidor$N$Ndesactivar esto también evitará que la URL de la página se actualice con IDs de canción al reproducir música, para prevenir la reproducción automática si se pierden estos ajustes pero la URL permanece">a▶', + "mt_preload": 'empezar a cargar la siguiente canción cerca del final para una reproducción sin pausas">precarga', + "mt_prescan": 'ir a la siguiente carpeta antes de que la última canción$Ntermine, manteniendo contento al navegador$Npara que no detenga la reproducción">nav', + "mt_fullpre": 'intentar precargar la canción entera;$N✅ activar en conexiones inestables,$N❌ desactivar probablemente en conexiones lentas">completa', + "mt_fau": 'en teléfonos, evitar que la música se detenga si la siguiente canción no se precarga lo suficientemente rápido (puede causar fallos en la visualización de etiquetas)">☕️', + "mt_waves": 'barra de búsqueda con forma de onda:$Nmostrar la amplitud del audio en la barra de progreso">~s', + "mt_npclip": 'mostrar botones para copiar al portapapeles la canción actual">/np', + "mt_m3u_c": 'mostrar botones para copiar al portapapeles las$Ncanciones seleccionadas como entradas de lista m3u8">📻', + "mt_octl": 'integración con SO (teclas multimedia / OSD)">ctl-so', + "mt_oseek": 'permitir buscar a través de la integración con el SO$N$Nnota: en algunos dispositivos (iPhones),$Nesto reemplaza el botón de siguiente canción">búsqueda', + "mt_oscv": 'mostrar carátula del álbum en OSD">arte', + "mt_follow": 'mantener la pista en reproducción visible en pantalla">🎯', + "mt_compact": 'controles compactos">⟎', + "mt_uncache": 'limpiar caché  (prueba esto si tu navegador guardó en caché$Nuna copia rota de una canción que se niega a reproducir)">limpiar caché', + "mt_mloop": 'repetir la carpeta actual">🔁 bucle', + "mt_mnext": 'cargar la siguiente carpeta y continuar">📂 sig', + "mt_mstop": 'detener reproducción">⏸ parar', + "mt_cflac": 'convertir flac / wav a opus">flac', + "mt_caac": 'convertir aac / m4a a opus">aac', + "mt_coth": 'convertir todos los demás (no mp3) a opus">oth', + "mt_c2opus": 'la mejor opción para ordenadores, portátiles, android">opus', + "mt_c2owa": 'opus-weba, para iOS 17.5 y superior">owa', + "mt_c2caf": 'opus-caf, para iOS 11 a 17">caf', + "mt_c2mp3": 'usar en dispositivos muy antiguos">mp3', + "mt_c2ok": "bien, buena elección", + "mt_c2nd": "ese no es el formato de salida recomendado para tu dispositivo, pero está bien", + "mt_c2ng": "tu dispositivo no parece soportar este formato de salida, pero intentémoslo de todas formas", + "mt_xowa": "hay errores en iOS que impiden la reproducción en segundo plano con este formato; por favor, usa caf o mp3 en su lugar", + "mt_tint": "nivel de fondo (0-100) en la barra de búsqueda$Npara hacer el buffering menos molesto", + "mt_eq": "activa el ecualizador y el control de ganancia;$N$Nganancia <code>0</code> = volumen estándar 100% (sin modificar)$N$Nancho <code>1  </code> = estéreo estándar (sin modificar)$Nancho <code>0.5</code> = 50% de crossfeed izq-der$Nancho <code>0  </code> = mono$N$Nganancia <code>-0.8</code> y ancho <code>10</code> = eliminación de voz :^)$N$Nactivar el ecualizador hace que los álbumes sin pausas sean completamente sin pausas, así que déjalo activado con todos los valores a cero (excepto ancho = 1) si eso te importa", + "mt_drc": "activa el compresor de rango dinámico (aplanador de volumen / brickwaller); también activará el EQ para equilibrar el espagueti, así que pon todos los campos de EQ excepto 'ancho' a 0 si no lo quieres$N$Nbaja el volumen del audio por encima de THRESHOLD dB; por cada RATIO dB pasado THRESHOLD hay 1 dB de salida, así que los valores por defecto de umbral -24 y ratio 12 significan que nunca debería sonar más fuerte de -22 dB y es seguro aumentar la ganancia del ecualizador a 0.8, o incluso 1.8 con ATK 0 y un RLS enorme como 90 (solo funciona en firefox; RLS es máx. 1 en otros navegadores)$N$N(ver wikipedia, lo explican mucho mejor)", + + "mb_play": "reproducir", + "mm_hashplay": "¿reproducir este archivo de audio?", + "mm_m3u": "pulsa Intro/Aceptar para Reproducir\npulsa ESC/Cancelar para Editar", + "mp_breq": "se necesita firefox 82+ o chrome 73+ o iOS 15+", + "mm_bload": "cargando...", + "mm_bconv": "convirtiendo a {0}, por favor espera...", + "mm_opusen": "tu navegador no puede reproducir archivos aac / m4a;\nse ha activado la transcodificación a opus", + "mm_playerr": "fallo de reproducción: ", + "mm_eabrt": "El intento de reproducción fue cancelado", + "mm_enet": "Tu conexión a internet es inestable", + "mm_edec": "¿Este archivo está supuestamente corrupto?", + "mm_esupp": "Tu navegador no entiende este formato de audio", + "mm_eunk": "Error desconocido", + "mm_e404": "No se pudo reproducir el audio; error 404: Archivo no encontrado.", + "mm_e403": "No se pudo reproducir el audio; error 403: Acceso denegado.\n\nIntenta pulsar F5 para recargar, quizás se cerró tu sesión", + "mm_e500": "No se pudo reproducir el audio; error 500: Revisa los registros del servidor.", + "mm_e5xx": "No se pudo reproducir el audio; error del servidor ", + "mm_nof": "no se encuentran más archivos de audio cerca", + "mm_prescan": "Buscando música para reproducir a continuación...", + "mm_scank": "Encontrada la siguiente canción:", + "mm_uncache": "caché limpiada; todas las canciones se volverán a descargar en la próxima reproducción", + "mm_hnf": "esa canción ya no existe", + + "im_hnf": "esa imagen ya no existe", + + "f_empty": "esta carpeta está vacía", + "f_chide": "esto ocultará la columna «{0}»\n\npuedes volver a mostrar las columnas en la pestaña de configuración", + "f_bigtxt": "este archivo pesa {0} MiB -- ¿realmente verlo como texto?", + "f_bigtxt2": "¿ver solo el final del archivo en su lugar? esto también activará el seguimiento, mostrando las líneas de texto recién añadidas en tiempo real", + "fbd_more": '
mostrando {0} de {1} archivos; mostrar {2} o mostrar todos
', + "fbd_all": '
mostrando {0} de {1} archivos; mostrar todos
', + "f_anota": "solo {0} de los {1} elementos fueron seleccionados;\npara seleccionar la carpeta completa, primero desplázate hasta el final", + + "f_dls": "los enlaces a archivos en la carpeta actual se han\nconvertido en enlaces de descarga", + + "f_partial": "Para descargar de forma segura un archivo que se está subiendo actualmente, por favor haz clic en el archivo con el mismo nombre, pero sin la extensión .PARTIAL. Por favor, pulsa CANCELAR o Escape para hacer esto.\n\nPulsar ACEPTAR o Intro ignorará esta advertencia y continuará descargando el archivo temporal .PARTIAL, lo que casi con toda seguridad te dará datos corruptos.", + + "ft_paste": "pegar {0} elementos$NAtajo: ctrl-V", + "fr_eperm": "no se puede renombrar:\nno tienes permiso de “mover” en esta carpeta", + "fd_eperm": "no se puede eliminar:\nno tienes permiso de “eliminar” en esta carpeta", + "fc_eperm": "no se puede cortar:\nno tienes permiso de “mover” en esta carpeta", + "fp_eperm": "no se puede pegar:\nno tienes permiso de “escribir” en esta carpeta", + "fr_emore": "selecciona al menos un elemento para renombrar", + "fd_emore": "selecciona al menos un elemento para eliminar", + "fc_emore": "selecciona al menos un elemento para cortar", + "fcp_emore": "selecciona al menos un elemento para copiar al portapapeles", + + "fs_sc": "compartir la carpeta en la que estás", + "fs_ss": "compartir los archivos seleccionados", + "fs_just1d": "no puedes seleccionar más de una carpeta,\no mezclar archivos y carpetas en una selección", + "fs_abrt": "❌ abortar", + "fs_rand": "🎲 nombre aleatorio", + "fs_go": "✅ crear enlace", + "fs_name": "nombre", + "fs_src": "origen", + "fs_pwd": "contraseña", + "fs_exp": "caducidad", + "fs_tmin": "minutos", + "fs_thrs": "horas", + "fs_tdays": "días", + "fs_never": "eterno", + "fs_pname": "nombre opcional del enlace; será aleatorio si se deja en blanco", + "fs_tsrc": "el archivo o carpeta a compartir", + "fs_ppwd": "contraseña opcional", + "fs_w8": "creando enlace...", + "fs_ok": "pulsa Intro/Aceptar para Copiar al Portapapeles\npulsa ESC/Cancelar para Cerrar", + + "frt_dec": "puede arreglar algunos casos de nombres de archivo rotos\">url-decode", + "frt_rst": "restaurar los nombres de archivo modificados a los originales\">↺ restablecer", + "frt_abrt": "abortar y cerrar esta ventana\">❌ cancelar", + "frb_apply": "APLICAR RENOMBRADO", + "fr_adv": "renombrado por lotes / metadatos / patrones\">avanzado", + "fr_case": "regex sensible a mayúsculas\">mayús", + "fr_win": "nombres seguros para windows; reemplaza <>:"\\|?* con caracteres japoneses de ancho completo\">win", + "fr_slash": "reemplaza / con un carácter que no cree nuevas carpetas\">sin /", + "fr_re": "patrón de búsqueda regex para aplicar a los nombres de archivo originales; los grupos de captura se pueden referenciar en el campo de formato de abajo como <code>(1)</code> y <code>(2)</code> y así sucesivamente", + "fr_fmt": "inspirado en foobar2000:$N<code>(title)</code> se reemplaza por el título de la canción,$N<code>[(artist) - ](title)</code> omite la parte [entre corchetes] si el artista está en blanco$N<code>$lpad((tn),2,0)</code> rellena el número de pista a 2 dígitos", + "fr_pdel": "eliminar", + "fr_pnew": "guardar como", + "fr_pname": "proporciona un nombre para tu nuevo preajuste", + "fr_aborted": "abortado", + "fr_lold": "nombre antiguo", + "fr_lnew": "nombre nuevo", + "fr_tags": "etiquetas para los archivos seleccionados (solo lectura, como referencia):", + "fr_busy": "renombrando {0} elementos...\n\n{1}", + "fr_efail": "fallo al renombrar:\n", + "fr_nchg": "{0} de los nuevos nombres fueron alterados debido a win y/o sin /\n\n¿Aceptar para continuar con estos nuevos nombres alterados?", + + "fd_ok": "eliminación correcta", + "fd_err": "fallo al eliminar:\n", + "fd_none": "no se eliminó nada; quizás bloqueado por la configuración del servidor (xbd)?", + "fd_busy": "eliminando {0} elementos...\n\n{1}", + "fd_warn1": "¿ELIMINAR estos {0} elementos?", + "fd_warn2": "¡Última oportunidad! No se puede deshacer. ¿Eliminar?", + + "fc_ok": "cortados {0} elementos", + "fc_warn": "cortados {0} elementos\n\npero: solo esta pestaña del navegador puede pegarlos\n(dado que la selección es absolutamente masiva)", + + "fcc_ok": "copiados {0} elementos al portapapeles", + "fcc_warn": "copiados {0} elementos al portapapeles\n\npero: solo esta pestaña del navegador puede pegarlos\n(dado que la selección es absolutamente masiva)", + + "fp_apply": "usar estos nombres", + "fp_ecut": "primero corta o copia algunos archivos / carpetas para pegar / mover\n\nnota: puedes cortar / pegar entre diferentes pestañas del navegador", + "fp_ename": "{0} elementos no se pueden mover aquí porque los nombres ya existen. Dales nuevos nombres abajo para continuar, o deja el nombre en blanco para omitirlos:", + "fcp_ename": "{0} elementos no se pueden copiar aquí porque los nombres ya existen. Dales nuevos nombres abajo para continuar, o deja el nombre en blanco para omitirlos:", + "fp_emore": "todavía quedan algunas colisiones de nombres por resolver", + "fp_ok": "movimiento correcto", + "fcp_ok": "copia correcta", + "fp_busy": "moviendo {0} elementos...\n\n{1}", + "fcp_busy": "copiando {0} elementos...\n\n{1}", + "fp_err": "fallo al mover:\n", + "fcp_err": "fallo al copiar:\n", + "fp_confirm": "¿mover estos {0} elementos aquí?", + "fcp_confirm": "¿copiar estos {0} elementos aquí?", + "fp_etab": "fallo al leer el portapapeles de otra pestaña del navegador", + "fp_name": "subiendo un archivo desde tu dispositivo. Dale un nombre:", + "fp_both_m": "
elige qué pegar
Intro = Mover {0} archivos desde «{1}»\nESC = Subir {2} archivos desde tu dispositivo", + "fcp_both_m": "
elige qué pegar
Intro = Copiar {0} archivos desde «{1}»\nESC = Subir {2} archivos desde tu dispositivo", + "fp_both_b": "MoverSubir", + "fcp_both_b": "CopiarSubir", + + "mk_noname": "escribe un nombre en el campo de texto de la izquierda antes de hacer eso :p", + + "tv_load": "Cargando documento de texto:\n\n{0}\n\n{1}% ({2} de {3} MiB cargados)", + "tv_xe1": "no se pudo cargar el archivo de texto:\n\nerror ", + "tv_xe2": "404, archivo no encontrado", + "tv_lst": "lista de archivos de texto en", + "tvt_close": "volver a la vista de carpetas$NAtajo: M (o Esc)\">❌ cerrar", + "tvt_dl": "descargar este archivo$NAtajo: Y\">💾 descargar", + "tvt_prev": "mostrar documento anterior$NAtajo: i\">⬆ ant", + "tvt_next": "mostrar siguiente documento$NAtajo: K\">⬇ sig", + "tvt_sel": "seleccionar archivo   ( para cortar / copiar / eliminar / ... )$NAtajo: S\">sel", + "tvt_edit": "abrir archivo en editor de texto$NAtajo: E\">✏️ editar", + "tvt_tail": "monitorizar cambios en el archivo; mostrar nuevas líneas en tiempo real\">📡 seguir", + "tvt_wrap": "ajuste de línea\">↵", + "tvt_atail": "bloquear el desplazamiento al final de la página\">⚓", + "tvt_ctail": "decodificar colores de terminal (códigos de escape ansi)\">🌈", + "tvt_ntail": "límite de historial (cuántos bytes de texto mantener cargados)", + + "m3u_add1": "canción añadida a la lista m3u", + "m3u_addn": "{0} canciones añadidas a la lista m3u", + "m3u_clip": "lista m3u copiada al portapapeles\n\ndebes crear un nuevo archivo de texto llamado algo.m3u y pegar la lista en ese documento; esto lo hará reproducible", + + "gt_vau": "no mostrar vídeos, solo reproducir el audio\">🎧", + "gt_msel": "activar selección de archivos; ctrl-clic en un archivo para anular$N$N<em>cuando está activo: doble clic en un archivo / carpeta para abrirlo</em>$N$NAtajo: S\">multiselección", + "gt_crop": "recortar miniaturas\">recortar", + "gt_3x": "miniaturas de alta resolución\">3x", + "gt_zoom": "zoom", + "gt_chop": "recortar", + "gt_sort": "ordenar por", + "gt_name": "nombre", + "gt_sz": "tamaño", + "gt_ts": "fecha", + "gt_ext": "tipo", + "gt_c1": "truncar más los nombres de archivo (mostrar menos)", + "gt_c2": "truncar menos los nombres de archivo (mostrar más)", + + "sm_w8": "buscando...", + "sm_prev": "los resultados de búsqueda a continuación son de una consulta anterior:\n ", + "sl_close": "cerrar resultados de búsqueda", + "sl_hits": "mostrando {0} resultados", + "sl_moar": "cargar más", + + "s_sz": "tamaño", + "s_dt": "fecha", + "s_rd": "ruta", + "s_fn": "nombre", + "s_ta": "etiquetas", + "s_ua": "subido@", + "s_ad": "avanzado", + "s_s1": "MiB mínimo", + "s_s2": "MiB máximo", + "s_d1": "mín. iso8601", + "s_d2": "máx. iso8601", + "s_u1": "subido después de", + "s_u2": "y/o antes de", + "s_r1": "la ruta contiene   (separado por espacios)", + "s_f1": "el nombre contiene   (negar con -no)", + "s_t1": "las etiquetas contienen   (^=inicio, fin=$)", + "s_a1": "propiedades de metadatos específicas", + + "md_eshow": "no se puede renderizar ", + "md_off": "[📜léeme] desactivado en [⚙️] -- documento oculto", + + "badreply": "Fallo al procesar la respuesta del servidor", + + "xhr403": "403: Acceso denegado\n\nintenta pulsar F5, quizás se cerró tu sesión", + "xhr0": "desconocido (probablemente se perdió la conexión con el servidor, o el servidor está desconectado)", + "cf_ok": "perdón por eso -- la protección DD" + wah + "oS se activó\n\nlas cosas deberían reanudarse en unos 30 segundos\n\nsi no pasa nada, pulsa F5 para recargar la página", + "tl_xe1": "no se pudieron listar las subcarpetas:\n\nerror ", + "tl_xe2": "404: Carpeta no encontrada", + "fl_xe1": "no se pudieron listar los archivos en la carpeta:\n\nerror ", + "fl_xe2": "404: Carpeta no encontrada", + "fd_xe1": "no se pudo crear la subcarpeta:\n\nerror ", + "fd_xe2": "404: Carpeta de nivel superior no encontrada", + "fsm_xe1": "no se pudo enviar el mensaje:\n\nerror ", + "fsm_xe2": "404: Carpeta de nivel superior no encontrada", + "fu_xe1": "fallo al cargar la lista de deshacer del servidor:\n\nerror ", + "fu_xe2": "404: ¿Archivo no encontrado?", + + "fz_tar": "archivo gnu-tar sin comprimir (linux / mac)", + "fz_pax": "tar formato pax sin comprimir (más lento)", + "fz_targz": "gnu-tar con compresión gzip nivel 3$N$Nesto suele ser muy lento, así que$Nusa tar sin comprimir en su lugar", + "fz_tarxz": "gnu-tar con compresión xz nivel 1$N$Nesto suele ser muy lento, así que$Nusa tar sin comprimir en su lugar", + "fz_zip8": "zip con nombres de archivo utf8 (puede dar problemas en windows 7 y anteriores)", + "fz_zipd": "zip con nombres de archivo cp437 tradicionales, para software muy antiguo", + "fz_zipc": "cp437 con crc32 calculado tempranamente,$Npara MS-DOS PKZIP v2.04g (octubre 1993)$N(tarda más en procesar antes de que la descarga pueda empezar)", + + "un_m1": "puedes eliminar tus subidas recientes (o abortar las inacabadas) a continuación", + "un_upd": "actualizar", + "un_m4": "o compartir los archivos visibles a continuación:", + "un_ulist": "mostrar", + "un_ucopy": "copiar", + "un_flt": "filtro opcional:  la URL debe contener", + "un_fclr": "limpiar filtro", + "un_derr": "fallo al deshacer-eliminar:\n", + "un_f5": "algo se rompió, por favor intenta actualizar o pulsa F5", + "un_uf5": "lo siento pero tienes que refrescar la página (por ejemplo pulsando F5 o CTRL-R) antes de que esta subida pueda ser abortada", + "un_nou": "aviso: servidor demasiado ocupado para mostrar subidas inacabadas; haz clic en el enlace \"actualizar\" en un momento", + "un_noc": "aviso: la opción de deshacer subidas completadas no está activada/permitida en la configuración del servidor", + "un_max": "mostrando los primeros 2000 archivos (usa el filtro)", + "un_avail": "{0} subidas recientes se pueden eliminar
{1} inacabadas se pueden abortar", + "un_m2": "ordenado por tiempo de subida; más recientes primero:", + "un_no1": "¡pues no! ninguna subida es suficientemente reciente", + "un_no2": "¡pues no! ninguna subida que coincida con ese filtro es suficientemente reciente", + "un_next": "eliminar los siguientes {0} archivos a continuación", + "un_abrt": "abortar", + "un_del": "eliminar", + "un_m3": "cargando tus subidas recientes...", + "un_busy": "eliminando {0} archivos...", + "un_clip": "{0} enlaces copiados al portapapeles", + + "u_https1": "deberías", + "u_https2": "cambiar a https", + "u_https3": "para un mejor rendimiento", + "u_ancient": "tu navegador es impresionantemente antiguo -- quizás deberías usar bup en su lugar", + "u_nowork": "se necesita firefox 53+ o chrome 57+ o iOS 11+", + "tail_2old": "se necesita firefox 105+ o chrome 71+ o iOS 14.5+", + "u_nodrop": "tu navegador es demasiado antiguo para subir arrastrando y soltando", + "u_notdir": "¡eso no es una carpeta!\n\ntu navegador es demasiado antiguo,\npor favor intenta arrastrar y soltar en su lugar", + "u_uri": "para arrastrar y soltar imágenes desde otras ventanas del navegador,\npor favor suéltalas sobre el gran botón de subida", + "u_enpot": "cambiar a UI ligera (puede mejorar la velocidad de subida)", + "u_depot": "cambiar a UI elegante (puede reducir la velocidad de subida)", + "u_gotpot": "cambiando a la UI ligera para mejorar la velocidad de subida,\n\n¡siéntete libre de no estar de acuerdo y volver a cambiar!", + "u_pott": "

archivos:   {0} finalizados,   {1} fallidos,   {2} ocupados,   {3} en cola

", + "u_ever": "este es el uploader básico; up2k necesita al menos
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": "este es el uploader básico; up2k es mejor", + "u_uput": "optimizar para velocidad (omitir checksum)", + "u_ewrite": "no tienes acceso de escritura a esta carpeta", + "u_eread": "no tienes acceso de lectura a esta carpeta", + "u_enoi": "la búsqueda de archivos no está activada en la configuración del servidor", + "u_enoow": "sobrescribir no funcionará aquí; se necesita permiso de eliminación", + "u_badf": "Estos {0} archivos (de un total de {1}) se omitieron, posiblemente debido a permisos del sistema de archivos:\n\n", + "u_blankf": "Estos {0} archivos (de un total de {1}) están en blanco / vacíos; ¿subirlos de todos modos?\n\n", + "u_applef": "Estos {0} archivos (de un total de {1}) probablemente no son deseables;\nPulsa Aceptar/Intro para OMITIR los siguientes archivos,\nPulsa Cancelar/ESC para NO excluir, y SUBIR esos también:\n\n", + "u_just1": "\nQuizás funcione mejor si seleccionas solo un archivo", + "u_ff_many": "si usas Linux / MacOS / Android, esta cantidad de archivos podría bloquear Firefox!\nsi eso ocurre, por favor inténtalo de nuevo (o usa Chrome).", + "u_up_life": "Esta subida será eliminada del servidor\n{0} después de que se complete", + "u_asku": "subir estos {0} archivos a {1}", + "u_unpt": "puedes deshacer / eliminar esta subida usando el 🧯 de arriba a la izquierda", + "u_bigtab": "a punto de mostrar {0} archivos\n\nesto podría bloquear tu navegador, ¿estás seguro?", + "u_scan": "Escaneando archivos...", + "u_dirstuck": "el iterador de directorios se atascó intentando acceder a los siguientes {0} elementos; se omitirán:", + "u_etadone": "Hecho ({0}, {1} archivos)", + "u_etaprep": "(preparando para subir)", + "u_hashdone": "hashing completado", + "u_hashing": "hash", + "u_hs": "negociando...", + "u_started": "los archivos se están subiendo ahora; mira en [🚀]", + "u_dupdefer": "duplicado; se procesará después de todos los demás archivos", + "u_actx": "haz clic en este texto para evitar la pérdida de
rendimiento al cambiar a otras ventanas/pestañas", + "u_fixed": "¡OK!  Arreglado 👍", + "u_cuerr": "fallo al subir el trozo {0} de {1};\nprobablemente inofensivo, continuando\n\narchivo: {2}", + "u_cuerr2": "el servidor rechazó la subida (trozo {0} de {1});\nse reintentará más tarde\n\narchivo: {2}\n\nerror ", + "u_ehstmp": "se reintentará; mira abajo a la derecha", + "u_ehsfin": "el servidor rechazó la solicitud para finalizar la subida; reintentando...", + "u_ehssrch": "el servidor rechazó la solicitud para realizar la búsqueda; reintentando...", + "u_ehsinit": "el servidor rechazó la solicitud para iniciar la subida; reintentando...", + "u_eneths": "error de red al realizar la negociación de subida; reintentando...", + "u_enethd": "error de red al comprobar la existencia del destino; reintentando...", + "u_cbusy": "esperando a que el servidor vuelva a confiar en nosotros después de un fallo de red...", + "u_ehsdf": "¡el servidor se quedó sin espacio en disco!\n\nse seguirá reintentando, por si alguien\nlibera suficiente espacio para continuar", + "u_emtleak1": "parece que tu navegador podría tener una fuga de memoria;\npor favor", + "u_emtleak2": " cambia a https (recomendado) o ", + "u_emtleak3": " ", + "u_emtleakc": "prueba lo siguiente:\n
  • pulsa F5 para refrescar la página
  • luego desactiva el botón  mt  en los  ⚙️ ajustes
  • e intenta esa subida de nuevo
Las subidas serán un poco más lentas, pero bueno.\n¡Perdón por las molestias!\n\nPD: chrome v107 tiene una solución para esto", + "u_emtleakf": "prueba lo siguiente:\n
  • pulsa F5 para refrescar la página
  • luego activa 🥔 (ligera) en la interfaz de subida
  • e intenta esa subida de nuevo
\nPD: firefox con suerte tendrá una solución en algún momento", + "u_s404": "no encontrado en el servidor", + "u_expl": "explicar", + "u_maxconn": "la mayoría de los navegadores limitan esto a 6, pero firefox te permite aumentarlo con connections-per-server en about:config", + "u_tu": '

AVISO: turbo activado,  el cliente puede no detectar y reanudar subidas incompletas; ver tooltip del botón turbo

', + "u_ts": '

AVISO: turbo activado,  los resultados de búsqueda pueden ser incorrectos; ver tooltip del botón turbo

', + "u_turbo_c": "turbo está desactivado en la configuración del servidor", + "u_turbo_g": "desactivando turbo porque no tienes\nprivilegios para listar directorios en este volumen", + "u_life_cfg": 'autoeliminar después de min (o horas)', + "u_life_est": 'la subida se eliminará ---', + "u_life_max": "esta carpeta impone una\nvida máxima de {0}", + "u_unp_ok": "se permite deshacer la subida durante {0}", + "u_unp_ng": "NO se permitirá deshacer la subida", + "ue_ro": "tu acceso a esta carpeta es de solo lectura\n\n", + "ue_nl": "actualmente no has iniciado sesión", + "ue_la": "actualmente has iniciado sesión como \"{0}\"", + "ue_sr": "actualmente estás en modo de búsqueda de archivos\n\ncambia a modo de subida haciendo clic en la lupa 🔎 (junto al gran botón BUSCAR), e intenta subir de nuevo\n\nlo siento", + "ue_ta": "intenta subir de nuevo, ahora debería funcionar", + "ue_ab": "este archivo ya se está subiendo a otra carpeta, y esa subida debe completarse antes de que el archivo pueda ser subido a otro lugar.\n\nPuedes abortar y olvidar la subida inicial usando el 🧯 de arriba a la izquierda", + "ur_1uo": "OK: Archivo subido con éxito", + "ur_auo": "OK: Todos los {0} archivos subidos con éxito", + "ur_1so": "OK: Archivo encontrado en el servidor", + "ur_aso": "OK: Todos los {0} archivos encontrados en el servidor", + "ur_1un": "Subida fallida, lo siento", + "ur_aun": "Todas las {0} subidas fallaron, lo siento", + "ur_1sn": "El archivo NO se encontró en el servidor", + "ur_asn": "Los {0} archivos NO se encontraron en el servidor", + "ur_um": "Finalizado;\n{0} subidas OK,\n{1} subidas fallidas, lo siento", + "ur_sm": "Finalizado;\n{0} archivos encontrados en el servidor,\n{1} archivos NO encontrados en el servidor", + + "lang_set": "¿refrescar para que el cambio surta efecto?" + }, "rus": { "tt": "Русский", @@ -3762,631 +4387,6 @@ var Ls = { "ur_sm": "Завершено;\n{0} файлів знайдено на сервері,\n{1} файлів НЕ знайдено на сервері", "lang_set": "оновити сторінку, щоб зміни набули чинності?", - }, - "spa": { - "tt": "Español", - - "cols": { - "c": "acciones", - "dur": "duración", - "q": "calidad / bitrate", - "Ac": "códec de audio", - "Vc": "códec de vídeo", - "Fmt": "formato / contenedor", - "Ahash": "checksum de audio", - "Vhash": "checksum de vídeo", - "Res": "resolución", - "T": "tipo de archivo", - "aq": "calidad de audio / bitrate", - "vq": "calidad de vídeo / bitrate", - "pixfmt": "submuestreo / estructura de píxel", - "resw": "resolución horizontal", - "resh": "resolución vertical", - "chs": "canales de audio", - "hz": "frecuencia de muestreo" - }, - - "hks": [ - [ - "varios", - ["ESC", "cerrar varias cosas"], - - "gestor de archivos", - ["G", "alternar vista de lista / cuadrícula"], - ["T", "alternar miniaturas / iconos"], - ["⇧ A/D", "tamaño de miniatura"], - ["ctrl-K", "eliminar seleccionados"], - ["ctrl-X", "cortar selección al portapapeles"], - ["ctrl-C", "copiar selección al portapapeles"], - ["ctrl-V", "pegar (mover/copiar) aquí"], - ["Y", "descargar seleccionados"], - ["F2", "renombrar seleccionados"], - - "selección en lista de archivos", - ["space", "alternar selección de archivo"], - ["↑/↓", "mover cursor de selección"], - ["ctrl ↑/↓", "mover cursor y vista"], - ["⇧ ↑/↓", "seleccionar anterior/siguiente archivo"], - ["ctrl-A", "seleccionar todos los archivos / carpetas"] - ], [ - "navegación", - ["B", "alternar breadcrumbs / panel de navegación"], - ["I/K", "anterior/siguiente carpeta"], - ["M", "carpeta de nivel superior (o contraer actual)"], - ["V", "alternar carpetas / archivos en panel de navegación"], - ["A/D", "tamaño del panel de navegación"] - ], [ - "reproductor de audio", - ["J/L", "anterior/siguiente canción"], - ["U/O", "saltar 10s atrás/adelante"], - ["0..9", "saltar a 0%..90%"], - ["P", "reproducir/pausar (también inicia)"], - ["S", "seleccionar canción en reproducción"], - ["Y", "descargar canción"] - ], [ - "visor de imágenes", - ["J/L, ←/→", "anterior/siguiente imagen"], - ["Home/End", "primera/última imagen"], - ["F", "pantalla completa"], - ["R", "rotar en sentido horario"], - ["⇧ R", "rotar en sentido antihorario"], - ["S", "seleccionar imagen"], - ["Y", "descargar imagen"] - ], [ - "reproductor de vídeo", - ["U/O", "saltar 10s atrás/adelante"], - ["P/K/Space", "reproducir/pausar"], - ["C", "continuar con el siguiente"], - ["V", "bucle"], - ["M", "silenciar"], - ["[ y ]", "establecer intervalo de bucle"] - ], [ - "visor de texto", - ["I/K", "anterior/siguiente archivo"], - ["M", "cerrar archivo"], - ["E", "editar archivo"], - ["S", "seleccionar archivo (para cortar/copiar/renombrar)"] - ] - ], - - "m_ok": "Aceptar", - "m_ng": "Cancelar", - - "enable": "Activar", - "danger": "PELIGRO", - "clipped": "copiado al portapapeles", - - "ht_s1": "segundo", - "ht_s2": "segundos", - "ht_m1": "minuto", - "ht_m2": "minutos", - "ht_h1": "hora", - "ht_h2": "horas", - "ht_d1": "día", - "ht_d2": "días", - "ht_and": " y ", - - "goh": "panel de control", - "gop": 'hermano anterior">anterior', - "gou": 'carpeta de nivel superior">subir', - "gon": 'siguiente carpeta">siguiente', - "logout": "Cerrar sesión ", - "access": " acceso", - "ot_close": "cerrar submenú", - "ot_search": "buscar archivos por atributos, ruta / nombre, etiquetas de música, o cualquier combinación$N$N<code>foo bar</code> = debe contener «foo» y «bar»,$N<code>foo -bar</code> = debe contener «foo» pero no «bar»,$N<code>^yana .opus$</code> = empieza con «yana» y es un archivo «opus»$N<code>"try unite"</code> = contiene exactamente «try unite»$N$Nel formato de fecha es iso-8601, como$N<code>2009-12-31</code> o <code>2020-09-12 23:30:00</code>", - "ot_unpost": "dessubir: elimina tus subidas recientes, o aborta las inacabadas", - "ot_bup": "bup: uploader básico, soporta hasta netscape 4.0", - "ot_mkdir": "mkdir: crear un nuevo directorio", - "ot_md": "new-md: crear un nuevo documento markdown", - "ot_msg": "msg: enviar un mensaje al registro del servidor", - "ot_mp": "opciones del reproductor multimedia", - "ot_cfg": "opciones de configuración", - "ot_u2i": "up2k: subir archivos (si tienes acceso de escritura) o cambiar a modo de búsqueda para ver si existen en el servidor$N$Nlas subidas se pueden reanudar, usan múltiples hilos y conservan la fecha de los archivos, pero consume más CPU que [🎈]  (el uploader básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", - "ot_u2w": "up2k: subir archivos con soporte para reanudación (cierra tu navegador y arrastra los mismos archivos más tarde)$N$NMultihilo y conserva las fechas de los archivos, pero usa más CPU que [🎈]  (el uploader básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", - "ot_noie": "Por favor, usa Chrome / Firefox / Edge", - - "ab_mkdir": "crear directorio", - "ab_mkdoc": "nuevo documento markdown", - "ab_msg": "enviar msg al registro del servidor", - - "ay_path": "saltar a carpetas", - "ay_files": "saltar a archivos", - - "wt_ren": "renombrar elementos seleccionados$NAtajo: F2", - "wt_del": "eliminar elementos seleccionados$NAtajo: ctrl-K", - "wt_cut": "cortar elementos seleccionados <small>(luego pegar en otro lugar)</small>$NAtajo: ctrl-X", - "wt_cpy": "copiar elementos seleccionados al portapapeles$N(para pegarlos en otro lugar)$NAtajo: ctrl-C", - "wt_pst": "pegar una selección previamente cortada / copiada$NAtajo: ctrl-V", - "wt_selall": "seleccionar todos los archivos$NAtajo: ctrl-A (con un archivo con foco)", - "wt_selinv": "invertir selección", - "wt_zip1": "descargar esta carpeta como un archivo comprimido", - "wt_selzip": "descargar selección como archivo comprimido", - "wt_seldl": "descargar selección como archivos separados$NAtajo: Y", - "wt_npirc": "copiar información de pista en formato IRC", - "wt_nptxt": "copiar información de pista en texto plano", - "wt_m3ua": "añadir a lista m3u (haz clic en 📻copiar después)", - "wt_m3uc": "copiar lista m3u al portapapeles", - "wt_grid": "alternar vista de cuadrícula / lista$NAtajo: G", - "wt_prev": "pista anterior$NAtajo: J", - "wt_play": "reproducir / pausar$NAtajo: P", - "wt_next": "siguiente pista$NAtajo: L", - - "ul_par": "subidas paralelas:", - "ut_rand": "aleatorizar nombres de archivo", - "ut_u2ts": 'copiar la fecha de última modificación$Nde tu sistema de archivos al servidor">📅', - "ut_ow": "sobrescribir archivos existentes en el servidor?$N🛡️: nunca (generará un nuevo nombre de archivo en su lugar)$N🕒: sobrescribir si el archivo del servidor es más antiguo que el tuyo$N♻️: siempre sobrescribir si los archivos son diferentes", - "ut_mt": "continuar generando hashes de otros archivos mientras se sube$N$Nquizás desactivar si tu CPU o HDD es un cuello de botella", - "ut_ask": 'pedir confirmación antes de iniciar la subida">💭', - "ut_pot": "mejorar la velocidad de subida en dispositivos lentos$Nsimplificando la interfaz de usuario", - "ut_srch": "no subir, en su lugar comprobar si los archivos ya $N existen en el servidor (escaneará todas las carpetas que puedas leer)", - "ut_par": "pausar subidas poniéndolo a 0$N$Naumentar si tu conexión es lenta / de alta latencia$N$Nmantener en 1 en LAN o si el HDD del servidor es un cuello de botella", - "ul_btn": "arrastra archivos / carpetas
aquí (o haz clic)", - "ul_btnu": "S U B I R", - "ul_btns": "B U S C A R", - - "ul_hash": "hash", - "ul_send": "envio", - "ul_done": "hecho", - "ul_idle1": "aún no hay subidas en cola", - "ut_etah": "velocidad media de <em>hashing</em>, y tiempo estimado para finalizar", - "ut_etau": "velocidad media de <em>subida</em> y tiempo estimado para finalizar", - "ut_etat": "velocidad media <em>total</em> y tiempo estimado para finalizar", - - "uct_ok": "completado con éxito", - "uct_ng": "fallido: error / rechazado / no encontrado", - "uct_done": "éxitos y fallos combinados", - "uct_bz": "generando hash o subiendo", - "uct_q": "inactivo, pendiente", - - "utl_name": "nombre de archivo", - "utl_ulist": "lista", - "utl_ucopy": "copiar", - "utl_links": "enlaces", - "utl_stat": "estado", - "utl_prog": "progreso", - - "utl_404": "404", - "utl_err": "ERROR", - "utl_oserr": "Error-SO", - "utl_found": "encontrado", - "utl_defer": "posponer", - "utl_yolo": "YOLO", - "utl_done": "hecho", - - "ul_flagblk": "los archivos se añadieron a la cola
sin embargo, hay un up2k ocupado en otra pestaña del navegador,
esperando a que termine primero", - "ul_btnlk": "la configuración del servidor ha bloqueado esta opción en este estado", - - "udt_up": "Subir", - "udt_srch": "Buscar", - "udt_drop": "suéltalo aquí", - - "u_nav_m": "
vale, ¿qué tienes?
Intro = Archivos (uno o más)\nESC = Una carpeta (incluyendo subcarpetas)", - "u_nav_b": "ArchivosUna carpeta", - - "cl_opts": "opciones", - "cl_themes": "tema", - "cl_langs": "idioma", - "cl_ziptype": "descarga de carpeta", - "cl_uopts": "opciones up2k", - "cl_favico": "favicon", - "cl_bigdir": "directorios grandes", - "cl_hsort": "#ordenar", - "cl_keytype": "notación musical", - "cl_hiddenc": "columnas ocultas", - "cl_hidec": "ocultar", - "cl_reset": "restablecer", - "cl_hpick": "toca en las cabeceras de columna para ocultarlas en la tabla de abajo", - "cl_hcancel": "ocultación de columna cancelada", - - "ct_grid": '田 cuadrícula', - "ct_ttips": '◔ ◡ ◔">ℹ️ tooltips', - "ct_thumb": 'en vista de cuadrícula, alternar iconos o miniaturas$NAtajo: T">🖼️ miniaturas', - "ct_csel": 'usa CTRL y SHIFT para seleccionar archivos en la vista de cuadrícula">sel', - "ct_ihop": 'al cerrar el visor de imágenes, desplazarse hasta el último archivo visto">g⮯', - "ct_dots": 'mostrar archivos ocultos (si el servidor lo permite)">archivos ocultos', - "ct_qdel": 'al eliminar archivos, pedir confirmación solo una vez">elim. rápida', - "ct_dir1st": 'ordenar carpetas antes que archivos">📁 primero', - "ct_nsort": 'orden natural (para nombres de archivo con dígitos iniciales)">ord. natural', - "ct_readme": 'mostrar README.md en los listados de carpetas">📜 léeme', - "ct_idxh": 'mostrar index.html en lugar del listado de carpetas">htm', - "ct_sbars": 'mostrar barra lateral">⟊', - - "cut_umod": 'si un archivo ya existe en el servidor, actualiza la fecha de última modificación del servidor para que coincida con tu archivo local (requiere permisos de escritura+eliminación)">re📅', - - "cut_turbo": 'el botón yolo, probablemente NO quieras activarlo:$N$Núsalo si estabas subiendo una gran cantidad de archivos y tuviste que reiniciar por alguna razón, y quieres continuar la subida lo antes posible$N$Nesto reemplaza la comprobación de hash por un simple "¿tiene este el mismo tamaño de archivo en el servidor?" así que si el contenido del archivo es diferente, NO se subirá$N$Ndeberías desactivar esto cuando la subida termine, y luego "subir" los mismos archivos de nuevo para que el cliente los verifique">turbo', - - "cut_datechk": 'no tiene efecto a menos que el botón turbo esté activado$N$Nreduce el factor yolo en una pequeña cantidad; comprueba si las fechas de los archivos en el servidor coinciden con las tuyas$N$Nteóricamente debería detectar la mayoría de las subidas inacabadas / corruptas, pero no es un sustituto de hacer una pasada de verificación con el turbo desactivado después">verif. fecha', - - "cut_u2sz": "tamaño (en MiB) de cada trozo de subida; los valores grandes vuelan mejor a través del atlántico. Prueba valores bajos en conexiones muy poco fiables", - - "cut_flag": "asegura que solo una pestaña esté subiendo a la vez $N -- otras pestañas también deben tener esto activado $N -- solo afecta a pestañas en el mismo dominio", - - "cut_az": "subir archivos en orden alfabético, en lugar de los más pequeños primero$N$Nel orden alfabético puede facilitar la detección visual de si algo salió mal en el servidor, pero hace la subida ligeramente más lenta en fibra / LAN", - - "cut_nag": "notificación del SO cuando la subida se complete$N(solo si el navegador o la pestaña no están activos)", - "cut_sfx": "alerta sonora cuando la subida se complete$N(solo si el navegador o la pestaña no están activos)", - - "cut_mt": 'usar multithreading para acelerar el hashing de archivos$N$Nesto usa web-workers y requiere$Nmás RAM (hasta 512 MiB extra)$N$Nhace https un 30% más rápido, http 4.5x más rápido">mt', - - "cut_wasm": 'usar wasm en lugar del hasher incorporado del navegador; mejora la velocidad en navegadores basados en chrome pero aumenta la carga de la CPU, y muchas versiones antiguas de chrome tienen errores que hacen que el navegador consuma toda la RAM y se bloquee si esto está activado">wasm', - - "cft_text": "texto del favicon (dejar en blanco y refrescar para desactivar)", - "cft_fg": "color de primer plano", - "cft_bg": "color de fondo", - - "cdt_lim": "número máximo de archivos a mostrar en una carpeta", - "cdt_ask": "al llegar al final,$Nen lugar de cargar más archivos,$Npreguntar qué hacer", - "cdt_hsort": "cuántas reglas de ordenación (<code>,sorthref</code>) incluir en las URLs de medios. Ponerlo a 0 también ignorará las reglas de ordenación incluidas en los enlaces de medios al hacer clic en ellos", - - "tt_entree": "mostrar panel de navegación (barra lateral con árbol de directorios)$NAtajo: B", - "tt_detree": "mostrar breadcrumbs$NAtajo: B", - "tt_visdir": "desplazarse a la carpeta seleccionada", - "tt_ftree": "alternar árbol de carpetas / archivos de texto$NAtajo: V", - "tt_pdock": "mostrar carpetas de niveles superiores en un panel acoplado en la parte superior", - "tt_dynt": "crecimiento automático a medida que el árbol se expande", - "tt_wrap": "ajuste de línea", - "tt_hover": "revelar líneas que se desbordan al pasar el ratón$N( rompe el desplazamiento a menos que el $N  cursor esté en el margen izquierdo )", - - "ml_pmode": "al final de la carpeta...", - "ml_btns": "acciones", - "ml_tcode": "transcodificar", - "ml_tcode2": "transcodificar a", - "ml_tint": "tinte", - "ml_eq": "ecualizador de audio", - "ml_drc": "compresor de rango dinámico", - - "mt_loop": 'poner en bucle/repetir una canción">🔁', - "mt_one": 'parar después de una canción">1️⃣', - "mt_shuf": 'reproducir aleatoriamente las canciones en cada carpeta">🔀', - "mt_aplay": 'reproducir automaticamente si hay un ID de canción en el enlace en el que hiciste clic para acceder al servidor$N$Ndesactivar esto también evitará que la URL de la página se actualice con IDs de canción al reproducir música, para prevenir la reproducción automática si se pierden estos ajustes pero la URL permanece">a▶', - "mt_preload": 'empezar a cargar la siguiente canción cerca del final para una reproducción sin pausas">precarga', - "mt_prescan": 'ir a la siguiente carpeta antes de que la última canción$Ntermine, manteniendo contento al navegador$Npara que no detenga la reproducción">nav', - "mt_fullpre": 'intentar precargar la canción entera;$N✅ activar en conexiones inestables,$N❌ desactivar probablemente en conexiones lentas">completa', - "mt_fau": 'en teléfonos, evitar que la música se detenga si la siguiente canción no se precarga lo suficientemente rápido (puede causar fallos en la visualización de etiquetas)">☕️', - "mt_waves": 'barra de búsqueda con forma de onda:$Nmostrar la amplitud del audio en la barra de progreso">~s', - "mt_npclip": 'mostrar botones para copiar al portapapeles la canción actual">/np', - "mt_m3u_c": 'mostrar botones para copiar al portapapeles las$Ncanciones seleccionadas como entradas de lista m3u8">📻', - "mt_octl": 'integración con SO (teclas multimedia / OSD)">ctl-so', - "mt_oseek": 'permitir buscar a través de la integración con el SO$N$Nnota: en algunos dispositivos (iPhones),$Nesto reemplaza el botón de siguiente canción">búsqueda', - "mt_oscv": 'mostrar carátula del álbum en OSD">arte', - "mt_follow": 'mantener la pista en reproducción visible en pantalla">🎯', - "mt_compact": 'controles compactos">⟎', - "mt_uncache": 'limpiar caché  (prueba esto si tu navegador guardó en caché$Nuna copia rota de una canción que se niega a reproducir)">limpiar caché', - "mt_mloop": 'repetir la carpeta actual">🔁 bucle', - "mt_mnext": 'cargar la siguiente carpeta y continuar">📂 sig', - "mt_mstop": 'detener reproducción">⏸ parar', - "mt_cflac": 'convertir flac / wav a opus">flac', - "mt_caac": 'convertir aac / m4a a opus">aac', - "mt_coth": 'convertir todos los demás (no mp3) a opus">oth', - "mt_c2opus": 'la mejor opción para ordenadores, portátiles, android">opus', - "mt_c2owa": 'opus-weba, para iOS 17.5 y superior">owa', - "mt_c2caf": 'opus-caf, para iOS 11 a 17">caf', - "mt_c2mp3": 'usar en dispositivos muy antiguos">mp3', - "mt_c2ok": "bien, buena elección", - "mt_c2nd": "ese no es el formato de salida recomendado para tu dispositivo, pero está bien", - "mt_c2ng": "tu dispositivo no parece soportar este formato de salida, pero intentémoslo de todas formas", - "mt_xowa": "hay errores en iOS que impiden la reproducción en segundo plano con este formato; por favor, usa caf o mp3 en su lugar", - "mt_tint": "nivel de fondo (0-100) en la barra de búsqueda$Npara hacer el buffering menos molesto", - "mt_eq": "activa el ecualizador y el control de ganancia;$N$Nganancia <code>0</code> = volumen estándar 100% (sin modificar)$N$Nancho <code>1  </code> = estéreo estándar (sin modificar)$Nancho <code>0.5</code> = 50% de crossfeed izq-der$Nancho <code>0  </code> = mono$N$Nganancia <code>-0.8</code> y ancho <code>10</code> = eliminación de voz :^)$N$Nactivar el ecualizador hace que los álbumes sin pausas sean completamente sin pausas, así que déjalo activado con todos los valores a cero (excepto ancho = 1) si eso te importa", - "mt_drc": "activa el compresor de rango dinámico (aplanador de volumen / brickwaller); también activará el EQ para equilibrar el espagueti, así que pon todos los campos de EQ excepto 'ancho' a 0 si no lo quieres$N$Nbaja el volumen del audio por encima de THRESHOLD dB; por cada RATIO dB pasado THRESHOLD hay 1 dB de salida, así que los valores por defecto de umbral -24 y ratio 12 significan que nunca debería sonar más fuerte de -22 dB y es seguro aumentar la ganancia del ecualizador a 0.8, o incluso 1.8 con ATK 0 y un RLS enorme como 90 (solo funciona en firefox; RLS es máx. 1 en otros navegadores)$N$N(ver wikipedia, lo explican mucho mejor)", - - "mb_play": "reproducir", - "mm_hashplay": "¿reproducir este archivo de audio?", - "mm_m3u": "pulsa Intro/Aceptar para Reproducir\npulsa ESC/Cancelar para Editar", - "mp_breq": "se necesita firefox 82+ o chrome 73+ o iOS 15+", - "mm_bload": "cargando...", - "mm_bconv": "convirtiendo a {0}, por favor espera...", - "mm_opusen": "tu navegador no puede reproducir archivos aac / m4a;\nse ha activado la transcodificación a opus", - "mm_playerr": "fallo de reproducción: ", - "mm_eabrt": "El intento de reproducción fue cancelado", - "mm_enet": "Tu conexión a internet es inestable", - "mm_edec": "¿Este archivo está supuestamente corrupto?", - "mm_esupp": "Tu navegador no entiende este formato de audio", - "mm_eunk": "Error desconocido", - "mm_e404": "No se pudo reproducir el audio; error 404: Archivo no encontrado.", - "mm_e403": "No se pudo reproducir el audio; error 403: Acceso denegado.\n\nIntenta pulsar F5 para recargar, quizás se cerró tu sesión", - "mm_e500": "No se pudo reproducir el audio; error 500: Revisa los registros del servidor.", - "mm_e5xx": "No se pudo reproducir el audio; error del servidor ", - "mm_nof": "no se encuentran más archivos de audio cerca", - "mm_prescan": "Buscando música para reproducir a continuación...", - "mm_scank": "Encontrada la siguiente canción:", - "mm_uncache": "caché limpiada; todas las canciones se volverán a descargar en la próxima reproducción", - "mm_hnf": "esa canción ya no existe", - - "im_hnf": "esa imagen ya no existe", - - "f_empty": "esta carpeta está vacía", - "f_chide": "esto ocultará la columna «{0}»\n\npuedes volver a mostrar las columnas en la pestaña de configuración", - "f_bigtxt": "este archivo pesa {0} MiB -- ¿realmente verlo como texto?", - "f_bigtxt2": "¿ver solo el final del archivo en su lugar? esto también activará el seguimiento, mostrando las líneas de texto recién añadidas en tiempo real", - "fbd_more": '
mostrando {0} de {1} archivos; mostrar {2} o mostrar todos
', - "fbd_all": '
mostrando {0} de {1} archivos; mostrar todos
', - "f_anota": "solo {0} de los {1} elementos fueron seleccionados;\npara seleccionar la carpeta completa, primero desplázate hasta el final", - - "f_dls": "los enlaces a archivos en la carpeta actual se han\nconvertido en enlaces de descarga", - - "f_partial": "Para descargar de forma segura un archivo que se está subiendo actualmente, por favor haz clic en el archivo con el mismo nombre, pero sin la extensión .PARTIAL. Por favor, pulsa CANCELAR o Escape para hacer esto.\n\nPulsar ACEPTAR o Intro ignorará esta advertencia y continuará descargando el archivo temporal .PARTIAL, lo que casi con toda seguridad te dará datos corruptos.", - - "ft_paste": "pegar {0} elementos$NAtajo: ctrl-V", - "fr_eperm": "no se puede renombrar:\nno tienes permiso de “mover” en esta carpeta", - "fd_eperm": "no se puede eliminar:\nno tienes permiso de “eliminar” en esta carpeta", - "fc_eperm": "no se puede cortar:\nno tienes permiso de “mover” en esta carpeta", - "fp_eperm": "no se puede pegar:\nno tienes permiso de “escribir” en esta carpeta", - "fr_emore": "selecciona al menos un elemento para renombrar", - "fd_emore": "selecciona al menos un elemento para eliminar", - "fc_emore": "selecciona al menos un elemento para cortar", - "fcp_emore": "selecciona al menos un elemento para copiar al portapapeles", - - "fs_sc": "compartir la carpeta en la que estás", - "fs_ss": "compartir los archivos seleccionados", - "fs_just1d": "no puedes seleccionar más de una carpeta,\no mezclar archivos y carpetas en una selección", - "fs_abrt": "❌ abortar", - "fs_rand": "🎲 nombre aleatorio", - "fs_go": "✅ crear enlace", - "fs_name": "nombre", - "fs_src": "origen", - "fs_pwd": "contraseña", - "fs_exp": "caducidad", - "fs_tmin": "minutos", - "fs_thrs": "horas", - "fs_tdays": "días", - "fs_never": "eterno", - "fs_pname": "nombre opcional del enlace; será aleatorio si se deja en blanco", - "fs_tsrc": "el archivo o carpeta a compartir", - "fs_ppwd": "contraseña opcional", - "fs_w8": "creando enlace...", - "fs_ok": "pulsa Intro/Aceptar para Copiar al Portapapeles\npulsa ESC/Cancelar para Cerrar", - - "frt_dec": "puede arreglar algunos casos de nombres de archivo rotos\">url-decode", - "frt_rst": "restaurar los nombres de archivo modificados a los originales\">↺ restablecer", - "frt_abrt": "abortar y cerrar esta ventana\">❌ cancelar", - "frb_apply": "APLICAR RENOMBRADO", - "fr_adv": "renombrado por lotes / metadatos / patrones\">avanzado", - "fr_case": "regex sensible a mayúsculas\">mayús", - "fr_win": "nombres seguros para windows; reemplaza <>:"\\|?* con caracteres japoneses de ancho completo\">win", - "fr_slash": "reemplaza / con un carácter que no cree nuevas carpetas\">sin /", - "fr_re": "patrón de búsqueda regex para aplicar a los nombres de archivo originales; los grupos de captura se pueden referenciar en el campo de formato de abajo como <code>(1)</code> y <code>(2)</code> y así sucesivamente", - "fr_fmt": "inspirado en foobar2000:$N<code>(title)</code> se reemplaza por el título de la canción,$N<code>[(artist) - ](title)</code> omite la parte [entre corchetes] si el artista está en blanco$N<code>$lpad((tn),2,0)</code> rellena el número de pista a 2 dígitos", - "fr_pdel": "eliminar", - "fr_pnew": "guardar como", - "fr_pname": "proporciona un nombre para tu nuevo preajuste", - "fr_aborted": "abortado", - "fr_lold": "nombre antiguo", - "fr_lnew": "nombre nuevo", - "fr_tags": "etiquetas para los archivos seleccionados (solo lectura, como referencia):", - "fr_busy": "renombrando {0} elementos...\n\n{1}", - "fr_efail": "fallo al renombrar:\n", - "fr_nchg": "{0} de los nuevos nombres fueron alterados debido a win y/o sin /\n\n¿Aceptar para continuar con estos nuevos nombres alterados?", - - "fd_ok": "eliminación correcta", - "fd_err": "fallo al eliminar:\n", - "fd_none": "no se eliminó nada; quizás bloqueado por la configuración del servidor (xbd)?", - "fd_busy": "eliminando {0} elementos...\n\n{1}", - "fd_warn1": "¿ELIMINAR estos {0} elementos?", - "fd_warn2": "¡Última oportunidad! No se puede deshacer. ¿Eliminar?", - - "fc_ok": "cortados {0} elementos", - "fc_warn": "cortados {0} elementos\n\npero: solo esta pestaña del navegador puede pegarlos\n(dado que la selección es absolutamente masiva)", - - "fcc_ok": "copiados {0} elementos al portapapeles", - "fcc_warn": "copiados {0} elementos al portapapeles\n\npero: solo esta pestaña del navegador puede pegarlos\n(dado que la selección es absolutamente masiva)", - - "fp_apply": "usar estos nombres", - "fp_ecut": "primero corta o copia algunos archivos / carpetas para pegar / mover\n\nnota: puedes cortar / pegar entre diferentes pestañas del navegador", - "fp_ename": "{0} elementos no se pueden mover aquí porque los nombres ya existen. Dales nuevos nombres abajo para continuar, o deja el nombre en blanco para omitirlos:", - "fcp_ename": "{0} elementos no se pueden copiar aquí porque los nombres ya existen. Dales nuevos nombres abajo para continuar, o deja el nombre en blanco para omitirlos:", - "fp_emore": "todavía quedan algunas colisiones de nombres por resolver", - "fp_ok": "movimiento correcto", - "fcp_ok": "copia correcta", - "fp_busy": "moviendo {0} elementos...\n\n{1}", - "fcp_busy": "copiando {0} elementos...\n\n{1}", - "fp_err": "fallo al mover:\n", - "fcp_err": "fallo al copiar:\n", - "fp_confirm": "¿mover estos {0} elementos aquí?", - "fcp_confirm": "¿copiar estos {0} elementos aquí?", - "fp_etab": "fallo al leer el portapapeles de otra pestaña del navegador", - "fp_name": "subiendo un archivo desde tu dispositivo. Dale un nombre:", - "fp_both_m": "
elige qué pegar
Intro = Mover {0} archivos desde «{1}»\nESC = Subir {2} archivos desde tu dispositivo", - "fcp_both_m": "
elige qué pegar
Intro = Copiar {0} archivos desde «{1}»\nESC = Subir {2} archivos desde tu dispositivo", - "fp_both_b": "MoverSubir", - "fcp_both_b": "CopiarSubir", - - "mk_noname": "escribe un nombre en el campo de texto de la izquierda antes de hacer eso :p", - - "tv_load": "Cargando documento de texto:\n\n{0}\n\n{1}% ({2} de {3} MiB cargados)", - "tv_xe1": "no se pudo cargar el archivo de texto:\n\nerror ", - "tv_xe2": "404, archivo no encontrado", - "tv_lst": "lista de archivos de texto en", - "tvt_close": "volver a la vista de carpetas$NAtajo: M (o Esc)\">❌ cerrar", - "tvt_dl": "descargar este archivo$NAtajo: Y\">💾 descargar", - "tvt_prev": "mostrar documento anterior$NAtajo: i\">⬆ ant", - "tvt_next": "mostrar siguiente documento$NAtajo: K\">⬇ sig", - "tvt_sel": "seleccionar archivo   ( para cortar / copiar / eliminar / ... )$NAtajo: S\">sel", - "tvt_edit": "abrir archivo en editor de texto$NAtajo: E\">✏️ editar", - "tvt_tail": "monitorizar cambios en el archivo; mostrar nuevas líneas en tiempo real\">📡 seguir", - "tvt_wrap": "ajuste de línea\">↵", - "tvt_atail": "bloquear el desplazamiento al final de la página\">⚓", - "tvt_ctail": "decodificar colores de terminal (códigos de escape ansi)\">🌈", - "tvt_ntail": "límite de historial (cuántos bytes de texto mantener cargados)", - - "m3u_add1": "canción añadida a la lista m3u", - "m3u_addn": "{0} canciones añadidas a la lista m3u", - "m3u_clip": "lista m3u copiada al portapapeles\n\ndebes crear un nuevo archivo de texto llamado algo.m3u y pegar la lista en ese documento; esto lo hará reproducible", - - "gt_vau": "no mostrar vídeos, solo reproducir el audio\">🎧", - "gt_msel": "activar selección de archivos; ctrl-clic en un archivo para anular$N$N<em>cuando está activo: doble clic en un archivo / carpeta para abrirlo</em>$N$NAtajo: S\">multiselección", - "gt_crop": "recortar miniaturas\">recortar", - "gt_3x": "miniaturas de alta resolución\">3x", - "gt_zoom": "zoom", - "gt_chop": "recortar", - "gt_sort": "ordenar por", - "gt_name": "nombre", - "gt_sz": "tamaño", - "gt_ts": "fecha", - "gt_ext": "tipo", - "gt_c1": "truncar más los nombres de archivo (mostrar menos)", - "gt_c2": "truncar menos los nombres de archivo (mostrar más)", - - "sm_w8": "buscando...", - "sm_prev": "los resultados de búsqueda a continuación son de una consulta anterior:\n ", - "sl_close": "cerrar resultados de búsqueda", - "sl_hits": "mostrando {0} resultados", - "sl_moar": "cargar más", - - "s_sz": "tamaño", - "s_dt": "fecha", - "s_rd": "ruta", - "s_fn": "nombre", - "s_ta": "etiquetas", - "s_ua": "subido@", - "s_ad": "avanzado", - "s_s1": "MiB mínimo", - "s_s2": "MiB máximo", - "s_d1": "mín. iso8601", - "s_d2": "máx. iso8601", - "s_u1": "subido después de", - "s_u2": "y/o antes de", - "s_r1": "la ruta contiene   (separado por espacios)", - "s_f1": "el nombre contiene   (negar con -no)", - "s_t1": "las etiquetas contienen   (^=inicio, fin=$)", - "s_a1": "propiedades de metadatos específicas", - - "md_eshow": "no se puede renderizar ", - "md_off": "[📜léeme] desactivado en [⚙️] -- documento oculto", - - "badreply": "Fallo al procesar la respuesta del servidor", - - "xhr403": "403: Acceso denegado\n\nintenta pulsar F5, quizás se cerró tu sesión", - "xhr0": "desconocido (probablemente se perdió la conexión con el servidor, o el servidor está desconectado)", - "cf_ok": "perdón por eso -- la protección DD" + wah + "oS se activó\n\nlas cosas deberían reanudarse en unos 30 segundos\n\nsi no pasa nada, pulsa F5 para recargar la página", - "tl_xe1": "no se pudieron listar las subcarpetas:\n\nerror ", - "tl_xe2": "404: Carpeta no encontrada", - "fl_xe1": "no se pudieron listar los archivos en la carpeta:\n\nerror ", - "fl_xe2": "404: Carpeta no encontrada", - "fd_xe1": "no se pudo crear la subcarpeta:\n\nerror ", - "fd_xe2": "404: Carpeta de nivel superior no encontrada", - "fsm_xe1": "no se pudo enviar el mensaje:\n\nerror ", - "fsm_xe2": "404: Carpeta de nivel superior no encontrada", - "fu_xe1": "fallo al cargar la lista de deshacer del servidor:\n\nerror ", - "fu_xe2": "404: ¿Archivo no encontrado?", - - "fz_tar": "archivo gnu-tar sin comprimir (linux / mac)", - "fz_pax": "tar formato pax sin comprimir (más lento)", - "fz_targz": "gnu-tar con compresión gzip nivel 3$N$Nesto suele ser muy lento, así que$Nusa tar sin comprimir en su lugar", - "fz_tarxz": "gnu-tar con compresión xz nivel 1$N$Nesto suele ser muy lento, así que$Nusa tar sin comprimir en su lugar", - "fz_zip8": "zip con nombres de archivo utf8 (puede dar problemas en windows 7 y anteriores)", - "fz_zipd": "zip con nombres de archivo cp437 tradicionales, para software muy antiguo", - "fz_zipc": "cp437 con crc32 calculado tempranamente,$Npara MS-DOS PKZIP v2.04g (octubre 1993)$N(tarda más en procesar antes de que la descarga pueda empezar)", - - "un_m1": "puedes eliminar tus subidas recientes (o abortar las inacabadas) a continuación", - "un_upd": "actualizar", - "un_m4": "o compartir los archivos visibles a continuación:", - "un_ulist": "mostrar", - "un_ucopy": "copiar", - "un_flt": "filtro opcional:  la URL debe contener", - "un_fclr": "limpiar filtro", - "un_derr": "fallo al deshacer-eliminar:\n", - "un_f5": "algo se rompió, por favor intenta actualizar o pulsa F5", - "un_uf5": "lo siento pero tienes que refrescar la página (por ejemplo pulsando F5 o CTRL-R) antes de que esta subida pueda ser abortada", - "un_nou": "aviso: servidor demasiado ocupado para mostrar subidas inacabadas; haz clic en el enlace \"actualizar\" en un momento", - "un_noc": "aviso: la opción de deshacer subidas completadas no está activada/permitida en la configuración del servidor", - "un_max": "mostrando los primeros 2000 archivos (usa el filtro)", - "un_avail": "{0} subidas recientes se pueden eliminar
{1} inacabadas se pueden abortar", - "un_m2": "ordenado por tiempo de subida; más recientes primero:", - "un_no1": "¡pues no! ninguna subida es suficientemente reciente", - "un_no2": "¡pues no! ninguna subida que coincida con ese filtro es suficientemente reciente", - "un_next": "eliminar los siguientes {0} archivos a continuación", - "un_abrt": "abortar", - "un_del": "eliminar", - "un_m3": "cargando tus subidas recientes...", - "un_busy": "eliminando {0} archivos...", - "un_clip": "{0} enlaces copiados al portapapeles", - - "u_https1": "deberías", - "u_https2": "cambiar a https", - "u_https3": "para un mejor rendimiento", - "u_ancient": "tu navegador es impresionantemente antiguo -- quizás deberías usar bup en su lugar", - "u_nowork": "se necesita firefox 53+ o chrome 57+ o iOS 11+", - "tail_2old": "se necesita firefox 105+ o chrome 71+ o iOS 14.5+", - "u_nodrop": "tu navegador es demasiado antiguo para subir arrastrando y soltando", - "u_notdir": "¡eso no es una carpeta!\n\ntu navegador es demasiado antiguo,\npor favor intenta arrastrar y soltar en su lugar", - "u_uri": "para arrastrar y soltar imágenes desde otras ventanas del navegador,\npor favor suéltalas sobre el gran botón de subida", - "u_enpot": "cambiar a UI ligera (puede mejorar la velocidad de subida)", - "u_depot": "cambiar a UI elegante (puede reducir la velocidad de subida)", - "u_gotpot": "cambiando a la UI ligera para mejorar la velocidad de subida,\n\n¡siéntete libre de no estar de acuerdo y volver a cambiar!", - "u_pott": "

archivos:   {0} finalizados,   {1} fallidos,   {2} ocupados,   {3} en cola

", - "u_ever": "este es el uploader básico; up2k necesita al menos
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", - "u_su2k": "este es el uploader básico; up2k es mejor", - "u_uput": "optimizar para velocidad (omitir checksum)", - "u_ewrite": "no tienes acceso de escritura a esta carpeta", - "u_eread": "no tienes acceso de lectura a esta carpeta", - "u_enoi": "la búsqueda de archivos no está activada en la configuración del servidor", - "u_enoow": "sobrescribir no funcionará aquí; se necesita permiso de eliminación", - "u_badf": "Estos {0} archivos (de un total de {1}) se omitieron, posiblemente debido a permisos del sistema de archivos:\n\n", - "u_blankf": "Estos {0} archivos (de un total de {1}) están en blanco / vacíos; ¿subirlos de todos modos?\n\n", - "u_applef": "Estos {0} archivos (de un total de {1}) probablemente no son deseables;\nPulsa Aceptar/Intro para OMITIR los siguientes archivos,\nPulsa Cancelar/ESC para NO excluir, y SUBIR esos también:\n\n", - "u_just1": "\nQuizás funcione mejor si seleccionas solo un archivo", - "u_ff_many": "si usas Linux / MacOS / Android, esta cantidad de archivos podría bloquear Firefox!\nsi eso ocurre, por favor inténtalo de nuevo (o usa Chrome).", - "u_up_life": "Esta subida será eliminada del servidor\n{0} después de que se complete", - "u_asku": "subir estos {0} archivos a {1}", - "u_unpt": "puedes deshacer / eliminar esta subida usando el 🧯 de arriba a la izquierda", - "u_bigtab": "a punto de mostrar {0} archivos\n\nesto podría bloquear tu navegador, ¿estás seguro?", - "u_scan": "Escaneando archivos...", - "u_dirstuck": "el iterador de directorios se atascó intentando acceder a los siguientes {0} elementos; se omitirán:", - "u_etadone": "Hecho ({0}, {1} archivos)", - "u_etaprep": "(preparando para subir)", - "u_hashdone": "hashing completado", - "u_hashing": "hash", - "u_hs": "negociando...", - "u_started": "los archivos se están subiendo ahora; mira en [🚀]", - "u_dupdefer": "duplicado; se procesará después de todos los demás archivos", - "u_actx": "haz clic en este texto para evitar la pérdida de
rendimiento al cambiar a otras ventanas/pestañas", - "u_fixed": "¡OK!  Arreglado 👍", - "u_cuerr": "fallo al subir el trozo {0} de {1};\nprobablemente inofensivo, continuando\n\narchivo: {2}", - "u_cuerr2": "el servidor rechazó la subida (trozo {0} de {1});\nse reintentará más tarde\n\narchivo: {2}\n\nerror ", - "u_ehstmp": "se reintentará; mira abajo a la derecha", - "u_ehsfin": "el servidor rechazó la solicitud para finalizar la subida; reintentando...", - "u_ehssrch": "el servidor rechazó la solicitud para realizar la búsqueda; reintentando...", - "u_ehsinit": "el servidor rechazó la solicitud para iniciar la subida; reintentando...", - "u_eneths": "error de red al realizar la negociación de subida; reintentando...", - "u_enethd": "error de red al comprobar la existencia del destino; reintentando...", - "u_cbusy": "esperando a que el servidor vuelva a confiar en nosotros después de un fallo de red...", - "u_ehsdf": "¡el servidor se quedó sin espacio en disco!\n\nse seguirá reintentando, por si alguien\nlibera suficiente espacio para continuar", - "u_emtleak1": "parece que tu navegador podría tener una fuga de memoria;\npor favor", - "u_emtleak2": " cambia a https (recomendado) o ", - "u_emtleak3": " ", - "u_emtleakc": "prueba lo siguiente:\n
  • pulsa F5 para refrescar la página
  • luego desactiva el botón  mt  en los  ⚙️ ajustes
  • e intenta esa subida de nuevo
Las subidas serán un poco más lentas, pero bueno.\n¡Perdón por las molestias!\n\nPD: chrome v107 tiene una solución para esto", - "u_emtleakf": "prueba lo siguiente:\n
  • pulsa F5 para refrescar la página
  • luego activa 🥔 (ligera) en la interfaz de subida
  • e intenta esa subida de nuevo
\nPD: firefox con suerte tendrá una solución en algún momento", - "u_s404": "no encontrado en el servidor", - "u_expl": "explicar", - "u_maxconn": "la mayoría de los navegadores limitan esto a 6, pero firefox te permite aumentarlo con connections-per-server en about:config", - "u_tu": '

AVISO: turbo activado,  el cliente puede no detectar y reanudar subidas incompletas; ver tooltip del botón turbo

', - "u_ts": '

AVISO: turbo activado,  los resultados de búsqueda pueden ser incorrectos; ver tooltip del botón turbo

', - "u_turbo_c": "turbo está desactivado en la configuración del servidor", - "u_turbo_g": "desactivando turbo porque no tienes\nprivilegios para listar directorios en este volumen", - "u_life_cfg": 'autoeliminar después de min (o horas)', - "u_life_est": 'la subida se eliminará ---', - "u_life_max": "esta carpeta impone una\nvida máxima de {0}", - "u_unp_ok": "se permite deshacer la subida durante {0}", - "u_unp_ng": "NO se permitirá deshacer la subida", - "ue_ro": "tu acceso a esta carpeta es de solo lectura\n\n", - "ue_nl": "actualmente no has iniciado sesión", - "ue_la": "actualmente has iniciado sesión como \"{0}\"", - "ue_sr": "actualmente estás en modo de búsqueda de archivos\n\ncambia a modo de subida haciendo clic en la lupa 🔎 (junto al gran botón BUSCAR), e intenta subir de nuevo\n\nlo siento", - "ue_ta": "intenta subir de nuevo, ahora debería funcionar", - "ue_ab": "este archivo ya se está subiendo a otra carpeta, y esa subida debe completarse antes de que el archivo pueda ser subido a otro lugar.\n\nPuedes abortar y olvidar la subida inicial usando el 🧯 de arriba a la izquierda", - "ur_1uo": "OK: Archivo subido con éxito", - "ur_auo": "OK: Todos los {0} archivos subidos con éxito", - "ur_1so": "OK: Archivo encontrado en el servidor", - "ur_aso": "OK: Todos los {0} archivos encontrados en el servidor", - "ur_1un": "Subida fallida, lo siento", - "ur_aun": "Todas las {0} subidas fallaron, lo siento", - "ur_1sn": "El archivo NO se encontró en el servidor", - "ur_asn": "Los {0} archivos NO se encontraron en el servidor", - "ur_um": "Finalizado;\n{0} subidas OK,\n{1} subidas fallidas, lo siento", - "ur_sm": "Finalizado;\n{0} archivos encontrados en el servidor,\n{1} archivos NO encontrados en el servidor", - - "lang_set": "¿refrescar para que el cambio surta efecto?" } }; diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 89661f28..049ed0c5 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -135,49 +135,6 @@ var Ls = { "ae1": "Aktive Downloads:", "af1": "Zeige neue Uploads", }, - - "rus": { - "a1": "обновить", - "b1": "приветик, незнакомец   (вы не авторизованы)", - "c1": "выйти", - "d1": "трассировка стека", - "d2": "показывает состояние всех активных потоков", - "e1": "перезагрузить конфиг", - "e2": "перезагрузить файлы конфига (аккаунты/хранилища/флаги),$Nи пересканировать все хранилища с флагом e2ds$N$Nвнимание: изменения глобальных настроек$Nтребуют полного перезапуска сервера", - "f1": "вы можете видеть:", - "g1": "вы можете загружать файлы в:", - "cc1": "всякая всячина:", - "h1": "отключить k304", - "i1": "включить k304", - "j1": "включённый k304 будет отключать вас при получении HTTP 304, что может помочь при работе с некоторыми глючными прокси (перестают загружаться страницы), но это также сделает работу клиента медленнее", - "k1": "сбросить локальные настройки", - "l1": "авторизуйтесь для других опций:", - "m1": "с возвращением,", - "n1": "404 не найдено  ┐( ´ -`)┌", - "o1": 'или у вас нет доступа -- попробуйте авторизоваться или вернуться на главную', - "p1": "403 доступ запрещён  ~┻━┻", - "q1": 'авторизуйтесь или вернитесь на главную', - "r1": "вернуться на главную", - ".s1": "пересканировать", - "t1": "действия", - "u2": "время с последней записи на сервер$N( загрузка / переименование / ... )$N$N17d = 17 дней$N1h23 = 1 час 23 минут$N4m56 = 4 минут 56 секунд", - "v1": "подключить", - "v2": "использовать сервер как локальный диск", - "w1": "перейти на https", - "x1": "поменять пароль", - "y1": "управление доступом", - "z1": "разблокировать:", - "ta1": "сначала введите свой новый пароль", - "ta2": "повторите новый пароль:", - "ta3": "опечатка; попробуйте снова", - "aa1": "входящие файлы:", - "ab1": "отключить no304", - "ac1": "включить no304", - "ad1": "включённый no304 полностью отключит хеширование; используйте, если k304 не помог. Сильно увеличит объём трафика!", - "ae1": "активные скачивания:", - "af1": "показать недавние загрузки", - "ag1": "показать известных IdP-пользователей", - }, "spa": { "a1": "actualizar", "b1": "hola   (no has iniciado sesión)", @@ -220,6 +177,48 @@ var Ls = { "af1": "mostrar subidas recientes", "ag1": "mostrar usuarios IdP conocidos" }, + "rus": { + "a1": "обновить", + "b1": "приветик, незнакомец   (вы не авторизованы)", + "c1": "выйти", + "d1": "трассировка стека", + "d2": "показывает состояние всех активных потоков", + "e1": "перезагрузить конфиг", + "e2": "перезагрузить файлы конфига (аккаунты/хранилища/флаги),$Nи пересканировать все хранилища с флагом e2ds$N$Nвнимание: изменения глобальных настроек$Nтребуют полного перезапуска сервера", + "f1": "вы можете видеть:", + "g1": "вы можете загружать файлы в:", + "cc1": "всякая всячина:", + "h1": "отключить k304", + "i1": "включить k304", + "j1": "включённый k304 будет отключать вас при получении HTTP 304, что может помочь при работе с некоторыми глючными прокси (перестают загружаться страницы), но это также сделает работу клиента медленнее", + "k1": "сбросить локальные настройки", + "l1": "авторизуйтесь для других опций:", + "m1": "с возвращением,", + "n1": "404 не найдено  ┐( ´ -`)┌", + "o1": 'или у вас нет доступа -- попробуйте авторизоваться или вернуться на главную', + "p1": "403 доступ запрещён  ~┻━┻", + "q1": 'авторизуйтесь или вернитесь на главную', + "r1": "вернуться на главную", + ".s1": "пересканировать", + "t1": "действия", + "u2": "время с последней записи на сервер$N( загрузка / переименование / ... )$N$N17d = 17 дней$N1h23 = 1 час 23 минут$N4m56 = 4 минут 56 секунд", + "v1": "подключить", + "v2": "использовать сервер как локальный диск", + "w1": "перейти на https", + "x1": "поменять пароль", + "y1": "управление доступом", + "z1": "разблокировать:", + "ta1": "сначала введите свой новый пароль", + "ta2": "повторите новый пароль:", + "ta3": "опечатка; попробуйте снова", + "aa1": "входящие файлы:", + "ab1": "отключить no304", + "ac1": "включить no304", + "ad1": "включённый no304 полностью отключит хеширование; используйте, если k304 не помог. Сильно увеличит объём трафика!", + "ae1": "активные скачивания:", + "af1": "показать недавние загрузки", + "ag1": "показать известных IdP-пользователей", + }, }; if (window.langmod) From 1ee89ec21dddcb0146df832db911061d48c2b83b Mon Sep 17 00:00:00 2001 From: Juan Herruzo Date: Fri, 1 Aug 2025 11:35:03 +0200 Subject: [PATCH 043/174] order spa alphabetically in language selector --- copyparty/web/browser.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 569d6ab3..2b69425e 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -4390,7 +4390,7 @@ var Ls = { } }; -var LANGS = ["eng", "nor", "chi", "deu", "rus", "ukr", "spa"]; +var LANGS = ["eng", "nor", "chi", "deu", "spa", "rus", "ukr"]; if (window.langmod) langmod(); From 4e8b88d8f68931fcf884bdf2305a0f601b576eb4 Mon Sep 17 00:00:00 2001 From: Juan Herruzo Date: Fri, 1 Aug 2025 11:39:02 +0200 Subject: [PATCH 044/174] fixed newline structure --- copyparty/web/browser.js | 2 +- copyparty/web/splash.js | 2 ++ 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 2b69425e..affcbaa7 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -4387,7 +4387,7 @@ var Ls = { "ur_sm": "Завершено;\n{0} файлів знайдено на сервері,\n{1} файлів НЕ знайдено на сервері", "lang_set": "оновити сторінку, щоб зміни набули чинності?", - } + }, }; var LANGS = ["eng", "nor", "chi", "deu", "spa", "rus", "ukr"]; diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 049ed0c5..22e32381 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -94,6 +94,7 @@ var Ls = { "af1": "显示最近上传的文件", //m "ag1": "查看已知 IdP 用户", //m }, + "deu": { "a1": "Neu laden", "b1": "Tach, wie geht's?   (Du bist nicht angemeldet)", @@ -135,6 +136,7 @@ var Ls = { "ae1": "Aktive Downloads:", "af1": "Zeige neue Uploads", }, + "spa": { "a1": "actualizar", "b1": "hola   (no has iniciado sesión)", From 9c197535462c5ade79fc9aae2bfba1c57dbf1a06 Mon Sep 17 00:00:00 2001 From: KevinXuxuxu Date: Thu, 31 Jul 2025 00:07:52 -0700 Subject: [PATCH 045/174] [u2c.py] Fix unicode files argument handling for py2.7 --- bin/u2c.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/bin/u2c.py b/bin/u2c.py index 99c426c4..85298730 100755 --- a/bin/u2c.py +++ b/bin/u2c.py @@ -52,6 +52,7 @@ if PY2: sys.dont_write_bytecode = True bytes = str + files_decoder = lambda s: unicode(s, 'utf8') else: from urllib.parse import quote_from_bytes as quote from urllib.parse import unquote_to_bytes as unquote @@ -61,6 +62,7 @@ else: from queue import Queue unicode = str + files_decoder = unicode WTF8 = "replace" if PY2 else "surrogateescape" @@ -1532,7 +1534,7 @@ source file/folder selection uses rsync syntax, meaning that: """) ap.add_argument("url", type=unicode, help="server url, including destination folder") - ap.add_argument("files", type=unicode, nargs="+", help="files and/or folders to process") + ap.add_argument("files", type=files_decoder, nargs="+", help="files and/or folders to process") ap.add_argument("-v", action="store_true", help="verbose") ap.add_argument("-a", metavar="PASSWD", help="password or $filepath") ap.add_argument("-s", action="store_true", help="file-search (disables upload)") From 6d6d79fcbc92f9f07690939d8b8b745694f880c4 Mon Sep 17 00:00:00 2001 From: ed Date: Fri, 1 Aug 2025 15:08:55 +0000 Subject: [PATCH 046/174] fix upload-abort in shares; closes #347 --- copyparty/httpcli.py | 29 ++++++++++++++++++++++------- copyparty/up2k.py | 12 ++++++++---- copyparty/web/browser.js | 22 ++++++---------------- 3 files changed, 36 insertions(+), 27 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index f698255a..16f94a11 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -5363,6 +5363,7 @@ class HttpCli(object): else: shr_dbv = None + wret: dict[str, Any] = {} ret: list[dict[str, Any]] = [] t0 = time.time() lim = time.time() - self.args.unpost @@ -5384,7 +5385,13 @@ class HttpCli(object): x = self.conn.hsrv.broker.ask( "up2k.get_unfinished_by_user", self.uname, "" if bad_xff else self.ip ) - uret = x.get() + zdsa: dict[str, Any] = x.get() + uret: list[dict[str, Any]] = [] + if "timeout" in zdsa: + wret["nou"] = 1 + else: + uret = zdsa["f"] + nu = len(uret) if not self.args.unpost: allvols = [] @@ -5430,6 +5437,8 @@ class HttpCli(object): if len(ret) > 2000: ret = ret[:2000] + if len(ret) >= 2000: + wret["oc"] = 1 for rv in ret: rv["vp"] = quotep(rv["vp"]) @@ -5449,6 +5458,13 @@ class HttpCli(object): ) rv["vp"] += "?k=" + fk[:nfk] + if not allvols: + wret["noc"] = 1 + ret = [] + + nc = len(ret) + ret = uret + ret + if shr_dbv: # translate vpaths from share-target to share-url # to satisfy access checks @@ -5463,12 +5479,11 @@ class HttpCli(object): for v in ret: v["vp"] = self.args.SR + v["vp"] - if not allvols: - ret = [{"kinshi": 1}] - - jtxt = '{"u":%s,"c":%s}' % (uret, json.dumps(ret, separators=(",\n", ": "))) - zi = len(uret.split('\n"pd":')) - 1 - self.log("%s #%d+%d %.2fsec" % (lm, zi, len(ret), time.time() - t0)) + wret["f"] = ret + wret["nu"] = nu + wret["nc"] = nc + jtxt = json.dumps(wret, separators=(",\n", ": ")) + self.log("%s #%d+%d %.2fsec" % (lm, nu, nc, time.time() - t0)) self.reply(jtxt.encode("utf-8", "replace"), mime="application/json") return True diff --git a/copyparty/up2k.py b/copyparty/up2k.py index f544d0e3..ea2bbb57 100644 --- a/copyparty/up2k.py +++ b/copyparty/up2k.py @@ -399,12 +399,14 @@ class Up2k(object): return "{}" - def get_unfinished_by_user(self, uname, ip) -> str: + def get_unfinished_by_user(self, uname, ip) -> dict[str, Any]: + # returns dict due to ExceptionalQueue if PY2 or not self.reg_mutex.acquire(timeout=2): - return '[{"timeout":1}]' + return {"timeout":1} ret: list[tuple[int, str, int, int, int]] = [] userset = set([(uname or "\n"), "*"]) + n = 1000 try: for ptop, tab2 in self.registry.items(): cfg = self.flags.get(ptop, {}).get("u2abort", 1) @@ -419,7 +421,6 @@ class Up2k(object): or (addr and addr != job["addr"]) ): continue - zt5 = ( int(job["t0"]), djoin(job["vtop"], job["prel"], job["name"]), @@ -428,6 +429,9 @@ class Up2k(object): len(job["hash"]), ) ret.append(zt5) + n -= 1 + if not n: + break finally: self.reg_mutex.release() @@ -444,7 +448,7 @@ class Up2k(object): } for (at, vp, sz, nn, nh) in ret ] - return json.dumps(ret2, separators=(",\n", ": ")) + return {"f": ret2} def get_unfinished(self) -> str: if PY2 or not self.reg_mutex.acquire(timeout=0.5): diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index affcbaa7..239800db 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -12526,29 +12526,19 @@ var unpost = (function () { return ebi('op_unpost').innerHTML = '

' + L.badreply + ':

' + unpre(this.responseText); } - if (ores.u.length == 1 && ores.u[0].timeout) { + if (ores.nou) html.push('

' + L.un_nou + '

'); - ores.u = []; - } - if (ores.c.length == 1 && ores.c[0].kinshi) { + if (ores.noc) html.push('

' + L.un_noc + '

'); - ores.c = []; - } - for (var a = 0; a < ores.u.length; a++) - ores.u[a].k = 'u'; - - for (var a = 0; a < ores.c.length; a++) - ores.c[a].k = 'c'; - - var res = ores.u.concat(ores.c); + var res = ores.f; if (res.length) { - if (res.length == 2000) + if (ores.of) html.push("

" + L.un_max); else - html.push("

" + L.un_avail.format(ores.c.length, ores.u.length)); + html.push("

" + L.un_avail.format(ores.nc, ores.nu)); html.push("
" + L.un_m2 + "

"); html.push(""); @@ -12565,7 +12555,7 @@ var unpost = (function () { '' + L.un_next.format(Math.min(mods[b], res.length - a)) + ''); - var done = res[a].k == 'c'; + var done = res[a].pd === undefined; html.push( '' + '' + From fee1416cbc0a535ad92b0c9eb28ff622dacfb188 Mon Sep 17 00:00:00 2001 From: ed Date: Fri, 1 Aug 2025 15:09:10 +0000 Subject: [PATCH 047/174] redundant --- copyparty/web/browser.html | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/web/browser.html b/copyparty/web/browser.html index 05bdacbc..ebf41f8e 100644 --- a/copyparty/web/browser.html +++ b/copyparty/web/browser.html @@ -110,7 +110,7 @@ {%- if f.tags is defined %} {%- for k in taglist %}{%- endfor %} -{%- endif %} +{%- endif %} {%- endfor %} From 7ecedb2ce22a89359aed1353e17826a8f80a8614 Mon Sep 17 00:00:00 2001 From: exci <76759714+icxes@users.noreply.github.com> Date: Fri, 1 Aug 2025 20:46:46 +0300 Subject: [PATCH 048/174] add finnish translation (#381) * reorder translations alphabetically * R comes before S * add initial finnish translation * add splash.js for finnish translation * add ct_utc translation (fin) * fix finnish translation problems pointed out in review --------- Co-authored-by: icxes --- copyparty/web/browser.js | 1728 ++++++++++++++++++++++++++------------ copyparty/web/splash.js | 45 +- 2 files changed, 1219 insertions(+), 554 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 239800db..17fe489a 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -1258,7 +1258,6 @@ var Ls = { "lang_set": "passer det å laste siden på nytt?", }, - "chi": { // 以 //m 结尾的行是未经验证的机器翻译 "tt": "中文", @@ -2511,630 +2510,632 @@ var Ls = { "lang_set": "Neuladen um Änderungen anzuwenden?", }, - "spa": { - "tt": "Español", + "fin": { + "tt": "Suomi", "cols": { - "c": "acciones", - "dur": "duración", - "q": "calidad / bitrate", - "Ac": "códec de audio", - "Vc": "códec de vídeo", - "Fmt": "formato / contenedor", - "Ahash": "checksum de audio", - "Vhash": "checksum de vídeo", - "Res": "resolución", - "T": "tipo de archivo", - "aq": "calidad de audio / bitrate", - "vq": "calidad de vídeo / bitrate", - "pixfmt": "submuestreo / estructura de píxel", - "resw": "resolución horizontal", - "resh": "resolución vertical", - "chs": "canales de audio", - "hz": "frecuencia de muestreo" + "c": "toimintopainikkeet", + "dur": "kesto", + "q": "laatu / bittinopeus", + "Ac": "äänikoodekki", + "Vc": "videokoodekki", + "Fmt": "formaatti / säiliö", + "Ahash": "äänen tarkistussumma", + "Vhash": "videon tarkistussumma", + "Res": "resoluutio", + "T": "tiedostotyyppi", + "aq": "äänenlaatu / bittinopeus", + "vq": "kuvalaatu / bittinopeus", + "pixfmt": "alinäytteistys / pikselirakenne", + "resw": "horisontaalinen resoluutio", + "resh": "vertikaalinen resoluutio", + "chs": "äänikanavat", + "hz": "näytteenottotaajuus" }, "hks": [ [ - "varios", - ["ESC", "cerrar varias cosas"], + "misc", + ["ESC", "sulje asioita"], - "gestor de archivos", - ["G", "alternar vista de lista / cuadrícula"], - ["T", "alternar miniaturas / iconos"], - ["⇧ A/D", "tamaño de miniatura"], - ["ctrl-K", "eliminar seleccionados"], - ["ctrl-X", "cortar selección al portapapeles"], - ["ctrl-C", "copiar selección al portapapeles"], - ["ctrl-V", "pegar (mover/copiar) aquí"], - ["Y", "descargar seleccionados"], - ["F2", "renombrar seleccionados"], + "file-manager", + ["G", "vaihda lista/ruudukkonäkymään"], + ["T", "vaihda pienoiskuviin/kuvakkeisiin"], + ["⇧ A/D", "pienoiskuvien koko"], + ["ctrl-K", "poista valitut"], + ["ctrl-X", "siirrä valitut leikepöydälle"], + ["ctrl-C", "kopioi valitut leikepöydälle"], + ["ctrl-V", "siirrä tai kopioi tähän"], + ["Y", "lataa valitut"], + ["F2", "uudelleennimeä valitut"], - "selección en lista de archivos", - ["space", "alternar selección de archivo"], - ["↑/↓", "mover cursor de selección"], - ["ctrl ↑/↓", "mover cursor y vista"], - ["⇧ ↑/↓", "seleccionar anterior/siguiente archivo"], - ["ctrl-A", "seleccionar todos los archivos / carpetas"] + "file-list-sel", + ["space", "vaihda tiedostonvalintatilaan"], + ["↑/↓", "siirrä valintaosoitinta"], + ["ctrl ↑/↓", "siirrä osoitinta ja näkymää"], + ["⇧ ↑/↓", "valitse edellinen/seuraava tiedosto"], + ["ctrl-A", "valitse kaikki tiedostot / kansiot"], ], [ - "navegación", - ["B", "alternar breadcrumbs / panel de navegación"], - ["I/K", "anterior/siguiente carpeta"], - ["M", "carpeta de nivel superior (o contraer actual)"], - ["V", "alternar carpetas / archivos en panel de navegación"], - ["A/D", "tamaño del panel de navegación"] + "navigation", + ["B", "näytä linkkipolku"], + ["I/K", "siirry edelliseen/seuraavaan hakemistoon"], + ["M", "siirry ylähakemistoon/supista nykyinen hakemisto"], + ["V", "näytä kansiot/tekstitiedostot navigointipaneelissa"], + ["A/D", "navigointipaneelin koko"], ], [ - "reproductor de audio", - ["J/L", "anterior/siguiente canción"], - ["U/O", "saltar 10s atrás/adelante"], - ["0..9", "saltar a 0%..90%"], - ["P", "reproducir/pausar (también inicia)"], - ["S", "seleccionar canción en reproducción"], - ["Y", "descargar canción"] + "audio-player", + ["J/L", "edellinen/seuraava kappale"], + ["U/O", "kelaa 10s taaksepäin/eteenpäin"], + ["0..9", "siirry 0%..90%"], + ["P", "toista/pysäytä kappale"], + ["S", "valitse toistossa oleva kappale"], + ["Y", "lataa kappale"], ], [ - "visor de imágenes", - ["J/L, ←/→", "anterior/siguiente imagen"], - ["Home/End", "primera/última imagen"], - ["F", "pantalla completa"], - ["R", "rotar en sentido horario"], - ["⇧ R", "rotar en sentido antihorario"], - ["S", "seleccionar imagen"], - ["Y", "descargar imagen"] + "image-viewer", + ["J/L, ←/→", "edellinen/seuraava kuva"], + ["Home/End", "ensimmäinen/viimeinen kuva"], + ["F", "siirry koko näytön tilaan"], + ["R", "kierrä myötäpäivään"], + ["⇧ R", "kierrä vastapäivään"], + ["S", "valitse kuva"], + ["Y", "lataa kuva"], ], [ - "reproductor de vídeo", - ["U/O", "saltar 10s atrás/adelante"], - ["P/K/Space", "reproducir/pausar"], - ["C", "continuar con el siguiente"], - ["V", "bucle"], - ["M", "silenciar"], - ["[ y ]", "establecer intervalo de bucle"] + "video-player", + ["U/O", "kelaa 10s taaksepäin/eteenpäin"], + ["P/K/Space", "toista/pysäytä video"], + ["C", "jatka toistoa seuraavaan videoon"], + ["V", "toista uudelleen"], + ["M", "vaimenna"], + ["[ ja ]", "aseta videon uudelleentoistoväli"], ], [ - "visor de texto", - ["I/K", "anterior/siguiente archivo"], - ["M", "cerrar archivo"], - ["E", "editar archivo"], - ["S", "seleccionar archivo (para cortar/copiar/renombrar)"] + "textfile-viewer", + ["I/K", "edellinen/seuraava tiedosto"], + ["M", "sulje tekstitiedosto"], + ["E", "muokkaa tekstitiedostoa"], + ["S", "valitse tiedosto (leikkausta/kopiointia/uudelleennimeämistä varten)"], ] ], - "m_ok": "Aceptar", - "m_ng": "Cancelar", + "m_ok": "OK", + "m_ng": "Peruuta", - "enable": "Activar", - "danger": "PELIGRO", - "clipped": "copiado al portapapeles", + "enable": "Aktivoi", + "danger": "HUOMIO!", + "clipped": "kopioitu leikepöydälle", - "ht_s1": "segundo", - "ht_s2": "segundos", - "ht_m1": "minuto", - "ht_m2": "minutos", - "ht_h1": "hora", - "ht_h2": "horas", - "ht_d1": "día", - "ht_d2": "días", - "ht_and": " y ", + "ht_s1": "sekunti", + "ht_s2": "sekuntia", + "ht_m1": "minuutti", + "ht_m2": "minuuttia", + "ht_h1": "tunti", + "ht_h2": "tuntia", + "ht_d1": "päivä", + "ht_d2": "päivää", + "ht_and": " ja ", - "goh": "panel de control", - "gop": 'hermano anterior">anterior', - "gou": 'carpeta de nivel superior">subir', - "gon": 'siguiente carpeta">siguiente', - "logout": "Cerrar sesión ", - "access": " acceso", - "ot_close": "cerrar submenú", - "ot_search": "buscar archivos por atributos, ruta / nombre, etiquetas de música, o cualquier combinación$N$N<code>foo bar</code> = debe contener «foo» y «bar»,$N<code>foo -bar</code> = debe contener «foo» pero no «bar»,$N<code>^yana .opus$</code> = empieza con «yana» y es un archivo «opus»$N<code>"try unite"</code> = contiene exactamente «try unite»$N$Nel formato de fecha es iso-8601, como$N<code>2009-12-31</code> o <code>2020-09-12 23:30:00</code>", - "ot_unpost": "dessubir: elimina tus subidas recientes, o aborta las inacabadas", - "ot_bup": "bup: uploader básico, soporta hasta netscape 4.0", - "ot_mkdir": "mkdir: crear un nuevo directorio", - "ot_md": "new-md: crear un nuevo documento markdown", - "ot_msg": "msg: enviar un mensaje al registro del servidor", - "ot_mp": "opciones del reproductor multimedia", - "ot_cfg": "opciones de configuración", - "ot_u2i": "up2k: subir archivos (si tienes acceso de escritura) o cambiar a modo de búsqueda para ver si existen en el servidor$N$Nlas subidas se pueden reanudar, usan múltiples hilos y conservan la fecha de los archivos, pero consume más CPU que [🎈]  (el uploader básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", - "ot_u2w": "up2k: subir archivos con soporte para reanudación (cierra tu navegador y arrastra los mismos archivos más tarde)$N$NMultihilo y conserva las fechas de los archivos, pero usa más CPU que [🎈]  (el uploader básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", - "ot_noie": "Por favor, usa Chrome / Firefox / Edge", + "goh": "ohjauspaneeli", + "gop": 'viereinen hakemisto">edell', + "gou": 'ylempi hakemisto">ylös', + "gon": 'seuraava hakemisto">seur', + "logout": "Kirjaudu ulos ", + "access": " -oikeudet", + "ot_close": "sulje alavalikko", + "ot_search": "etsi tiedostoja ominaisuuksien, tiedostopolun tai -nimen, musiikkitägien tai näiden yhdistelmän perusteella$N$N<code>foo bar</code> = täytyy sisältää sekä «foo» että «bar»,$N<code>foo -bar</code> = täytyy sisältää «foo» mutta ei «bar»,$N<code>^yana .opus$</code> = alkaa «yana» ja on «opus»-tiedosto$N<code>"try unite"</code> = sisältää täsmälleen «try unite»$N$Npäivämäärän muoto on iso-8601, kuten$N<code>2009-12-31</code> tai <code>2020-09-12 23:30:00</code>", + "ot_unpost": "unpost: poista viimeaikaiset tai keskeytä keskeneräiset lataukset", + "ot_bup": "bup: tiedostojen 'perus'lähetysohjelma, tukee jopa netscape 4.0:aa", + "ot_mkdir": "mkdir: luo uusi hakemisto", + "ot_md": "new-md: luo uusi markdown-dokumentti", + "ot_msg": "msg: lähetä viesti palvelinlokiin", + "ot_mp": "mediasoittimen asetukset", + "ot_cfg": "asetukset", + "ot_u2i": 'up2k: lähetä tiedostoja (vaatii write-oikeudet) tai vaihda hakutilaan nähdäksesi, ovatko tiedostot jo olemassa jossain päin palvelinta$N$Nlatauksia voi jatkaa, ne ovat monisäikeistettyjä, ja tiedostojen aikaleimat säilytetään; nuijii prosessoria enemmän kuin [🎈]  (peruslatausohjelma)

tiedostojen lähetyksen aikana tämä kuvake muuttuu kertoo lähetyksen edistymisestilanteen!', + "ot_u2w": 'up2k: lähetä tiedostoja jatkamistoiminnolla (voit sulkea selaimen ja vetää samat tiedostot selainikkunaan myöhemmin)$N$monisäikeistetty, ja tiedostojen aikaleimat säilyvät; nuijii prosessoria enemmän kuin [🎈]  (peruslatausohjelma)

tiedostojen lähetyksen aikana tämä kuvake muuttuu kertoo lähetyksen edistymisestilanteen!', + "ot_noie": 'Suosittelemme käyttämään uudempaa selainta.', - "ab_mkdir": "crear directorio", - "ab_mkdoc": "nuevo documento markdown", - "ab_msg": "enviar msg al registro del servidor", + "ab_mkdir": "luo hakemisto", + "ab_mkdoc": "luo markdown-tiedosto", + "ab_msg": "lähetä viesti palvelinlokiin", - "ay_path": "saltar a carpetas", - "ay_files": "saltar a archivos", + "ay_path": "siirry kansioihin", + "ay_files": "siirry tiedostoihin", - "wt_ren": "renombrar elementos seleccionados$NAtajo: F2", - "wt_del": "eliminar elementos seleccionados$NAtajo: ctrl-K", - "wt_cut": "cortar elementos seleccionados <small>(luego pegar en otro lugar)</small>$NAtajo: ctrl-X", - "wt_cpy": "copiar elementos seleccionados al portapapeles$N(para pegarlos en otro lugar)$NAtajo: ctrl-C", - "wt_pst": "pegar una selección previamente cortada / copiada$NAtajo: ctrl-V", - "wt_selall": "seleccionar todos los archivos$NAtajo: ctrl-A (con un archivo con foco)", - "wt_selinv": "invertir selección", - "wt_zip1": "descargar esta carpeta como un archivo comprimido", - "wt_selzip": "descargar selección como archivo comprimido", - "wt_seldl": "descargar selección como archivos separados$NAtajo: Y", - "wt_npirc": "copiar información de pista en formato IRC", - "wt_nptxt": "copiar información de pista en texto plano", - "wt_m3ua": "añadir a lista m3u (haz clic en 📻copiar después)", - "wt_m3uc": "copiar lista m3u al portapapeles", - "wt_grid": "alternar vista de cuadrícula / lista$NAtajo: G", - "wt_prev": "pista anterior$NAtajo: J", - "wt_play": "reproducir / pausar$NAtajo: P", - "wt_next": "siguiente pista$NAtajo: L", + "wt_ren": "uudelleennimeä valitut kohteet$NPikanäppäin: F2", + "wt_del": "poista valitut kohteet$NPikanäppäin: ctrl-K", + "wt_cut": "siirrä valitut kohteet leikepöydälle <small>(siirtääksesi ne muualle)</small>$NPikanäppäin: ctrl-X", + "wt_cpy": "kopioi valitut kohteet leikepöydälle$N(liittääksesi ne muualle)$NPikanäppäin: ctrl-C", + "wt_pst": "liitä aiemmin leikatut / kopioidut valinnat$NPikanäppäin: ctrl-V", + "wt_selall": "valitse kaikki tiedostot$NPikanäppäin: ctrl-A (kun tiedosto on kohdistettu)", + "wt_selinv": "valitse vastakkaiset tiedostot", + "wt_zip1": "lataa tämä kansio pakattuna", + "wt_selzip": "lataa valitut kohteet pakattuna", + "wt_seldl": "lataa valitut kohteet paketoimatta$NPikanäppäin: Y", + "wt_npirc": "kopioi kappaletiedot IRC-muotoilulla", + "wt_nptxt": "kopioi kappaletiedot ilman muotoilua", + "wt_m3ua": "lisää m3u-soittolistaan (klikkaa 📻kopioi myöhemmin)", + "wt_m3uc": "kopioi m3u-soittolista leikepöydälle", + "wt_grid": "vaihda ruudukko- ja listanäkymän välillä$NPikanäppäin: G", + "wt_prev": "edellinen kappale$NPikanäppäin: J", + "wt_play": "toista / pysäytä$NPikanäppäin: P", + "wt_next": "seuraava kappale$NPikanäppäin: L", - "ul_par": "subidas paralelas:", - "ut_rand": "aleatorizar nombres de archivo", - "ut_u2ts": 'copiar la fecha de última modificación$Nde tu sistema de archivos al servidor">📅', - "ut_ow": "sobrescribir archivos existentes en el servidor?$N🛡️: nunca (generará un nuevo nombre de archivo en su lugar)$N🕒: sobrescribir si el archivo del servidor es más antiguo que el tuyo$N♻️: siempre sobrescribir si los archivos son diferentes", - "ut_mt": "continuar generando hashes de otros archivos mientras se sube$N$Nquizás desactivar si tu CPU o HDD es un cuello de botella", - "ut_ask": 'pedir confirmación antes de iniciar la subida">💭', - "ut_pot": "mejorar la velocidad de subida en dispositivos lentos$Nsimplificando la interfaz de usuario", - "ut_srch": "no subir, en su lugar comprobar si los archivos ya $N existen en el servidor (escaneará todas las carpetas que puedas leer)", - "ut_par": "pausar subidas poniéndolo a 0$N$Naumentar si tu conexión es lenta / de alta latencia$N$Nmantener en 1 en LAN o si el HDD del servidor es un cuello de botella", - "ul_btn": "arrastra archivos / carpetas
aquí (o haz clic)", - "ul_btnu": "S U B I R", - "ul_btns": "B U S C A R", + "ul_par": "rinnakkaisten latausten lkm:", + "ut_rand": "satunnaisgeneroidut tiedostonimet", + "ut_u2ts": "kopioi viimeksi muokattu aikaleima$Ntiedostojärjestelmästäsi palvelimelle\">📅", + "ut_ow": "korvaa olemassa olevat tiedostot palvelimella?$N🛡️: ei koskaan (luo sen sijaan uuden tiedostonimen)$N🕒: korvaa jos palvelintiedosto on vanhempi kuin omasi$N♻️: korvaa aina jos tiedostot ovat erilaisia", + "ut_mt": "jatka muiden tiedostojen tiivisteiden laskemista latauksen aikana$N$kannattanee poistaa käytöstä, mikäli prosessori tai kovalevy on vanhempaa mallia", + "ut_ask": 'kysy vahvistusta ennen latauksen aloittamista">💭', + "ut_pot": "paranna latausnopeutta hitailla laitteilla$Ntekemällä käyttöliittymäästä vähemmän monimutkaisen", + "ut_srch": "lataamisen sijaan tarkista, ovatko tiedostot jo $N olemassa palvelimella (käy läpi kaikki hakemistot, joihin sinulla on read-oikeudet)", + "ut_par": "keskeytä lataukset asettamalla se 0:aan$N$Nnosta, jos yhteytesi on hidas tai viive on suuri$N$Npidä se 1:ssä lähiverkossa tai jos palvelimen kovalevy on pullonkaula", + "ul_btn": "vedä tiedostoja / kansioita
tähän (tai napsauta tätä)", + "ul_btnu": "L A T A A", + "ul_btns": "E T S I", - "ul_hash": "hash", - "ul_send": "envio", - "ul_done": "hecho", - "ul_idle1": "aún no hay subidas en cola", - "ut_etah": "velocidad media de <em>hashing</em>, y tiempo estimado para finalizar", - "ut_etau": "velocidad media de <em>subida</em> y tiempo estimado para finalizar", - "ut_etat": "velocidad media <em>total</em> y tiempo estimado para finalizar", + "ul_hash": "tiiviste", + "ul_send": "lähetä", + "ul_done": "valmis", + "ul_idle1": "ei latauksia jonossa", + "ut_etah": "keskimääräinen <em>tiivisteiden lasku</em>nopeus ja arvioitu aika valmistumiseen", + "ut_etau": "keskimääräinen <em>lataus</em>nopeus ja arvioitu aika valmistumiseen", + "ut_etat": "keskimääräinen <em>kokonais</em>nopeus ja arvioitu aika valmistumiseen", - "uct_ok": "completado con éxito", - "uct_ng": "fallido: error / rechazado / no encontrado", - "uct_done": "éxitos y fallos combinados", - "uct_bz": "generando hash o subiendo", - "uct_q": "inactivo, pendiente", + "uct_ok": "onnistui", + "uct_ng": "ei-hyvä: epäonnistui / hylätty / ei löydy", + "uct_done": "ok ja ng yhdistettynä", + "uct_bz": "laskee tiivisteitä tai lataa", + "uct_q": "tyhjäkäynnillä, odottaa", - "utl_name": "nombre de archivo", + "utl_name": "tiedostonimi", "utl_ulist": "lista", - "utl_ucopy": "copiar", - "utl_links": "enlaces", - "utl_stat": "estado", - "utl_prog": "progreso", + "utl_ucopy": "kopioi", + "utl_links": "linkit", + "utl_stat": "tila", + "utl_prog": "edistyminen", + // keep short: "utl_404": "404", - "utl_err": "ERROR", - "utl_oserr": "Error-SO", - "utl_found": "encontrado", - "utl_defer": "posponer", + "utl_err": "VIRHE", + "utl_oserr": "Käyttöjärjestelmävirhe", + "utl_found": "löytyi", + "utl_defer": "lykkää", "utl_yolo": "YOLO", - "utl_done": "hecho", + "utl_done": "valmis", - "ul_flagblk": "los archivos se añadieron a la cola
sin embargo, hay un up2k ocupado en otra pestaña del navegador,
esperando a que termine primero", - "ul_btnlk": "la configuración del servidor ha bloqueado esta opción en este estado", + "ul_flagblk": "tiedostot lisättiin jonoon
mutta toisen selainvälilehden up2k on kiireinen,
joten odotetaan sen valmistumista ensin", + "ul_btnlk": "palvelinkonfiguraatio on lukinnut tämän kytkimen tähän tilaan", - "udt_up": "Subir", - "udt_srch": "Buscar", - "udt_drop": "suéltalo aquí", + "udt_up": "Lataa", + "udt_srch": "Etsi", + "udt_drop": "pudota se tähän", - "u_nav_m": "
vale, ¿qué tienes?
Intro = Archivos (uno o más)\nESC = Una carpeta (incluyendo subcarpetas)", - "u_nav_b": "ArchivosUna carpeta", + "u_nav_m": '
selvä, mitäs sulla on?
Enter = Tiedostoja (yksi tai useampi)\nESC = Yksi kansio (mukaan lukien alikansiot)', + "u_nav_b": 'TiedostojaYksi kansio', - "cl_opts": "opciones", - "cl_themes": "tema", - "cl_langs": "idioma", - "cl_ziptype": "descarga de carpeta", - "cl_uopts": "opciones up2k", + "cl_opts": "kytkimet", + "cl_themes": "teema", + "cl_langs": "kieli", + "cl_ziptype": "kansion lataus", + "cl_uopts": "up2k-kytkimet", "cl_favico": "favicon", - "cl_bigdir": "directorios grandes", - "cl_hsort": "#ordenar", - "cl_keytype": "notación musical", - "cl_hiddenc": "columnas ocultas", - "cl_hidec": "ocultar", - "cl_reset": "restablecer", - "cl_hpick": "toca en las cabeceras de columna para ocultarlas en la tabla de abajo", - "cl_hcancel": "ocultación de columna cancelada", + "cl_bigdir": "suuret hakemistot", + "cl_hsort": "#lajittelu", + "cl_keytype": "sävellajin notaatio", + "cl_hiddenc": "piilotetut sarakkeet", + "cl_hidec": "piilota", + "cl_reset": "palauta", + "cl_hpick": "napauta sarakeotsikoita piilottaaksesi alla olevassa taulukossa", + "cl_hcancel": "sarakkeiden piilotus peruttu", - "ct_grid": '田 cuadrícula', - "ct_ttips": '◔ ◡ ◔">ℹ️ tooltips', - "ct_thumb": 'en vista de cuadrícula, alternar iconos o miniaturas$NAtajo: T">🖼️ miniaturas', - "ct_csel": 'usa CTRL y SHIFT para seleccionar archivos en la vista de cuadrícula">sel', - "ct_ihop": 'al cerrar el visor de imágenes, desplazarse hasta el último archivo visto">g⮯', - "ct_dots": 'mostrar archivos ocultos (si el servidor lo permite)">archivos ocultos', - "ct_qdel": 'al eliminar archivos, pedir confirmación solo una vez">elim. rápida', - "ct_dir1st": 'ordenar carpetas antes que archivos">📁 primero', - "ct_nsort": 'orden natural (para nombres de archivo con dígitos iniciales)">ord. natural', - "ct_readme": 'mostrar README.md en los listados de carpetas">📜 léeme', - "ct_idxh": 'mostrar index.html en lugar del listado de carpetas">htm', - "ct_sbars": 'mostrar barra lateral">⟊', + "ct_grid": '田 RUUDUKKO', + "ct_ttips": '◔ ◡ ◔">ℹ️ ammattilaisvinkit', + "ct_thumb": 'RUUDUKKOnäkymässä, vaihda kuvakkeihin tai pienoiskuviin$NPikanäppäin: T">🖼️ pienoiskuvat', + "ct_csel": 'käytä CTRL ja SHIFT tiedostojen valintaan RUUDUKKOnäkymässä">val', + "ct_ihop": 'kun kuvakatselin suljetaan, vieritä alas viimeksi katsottuun tiedostoon">g⮯', + "ct_dots": 'näytä piilotetut tiedostot (jos palvelin sallii)">piilotiedostot', + "ct_qdel": 'tiedostoja poistaessa, kysy vahvistusta vain kerran">qdel', + "ct_dir1st": 'lajittele hakemistot ennen tiedostoja">📁 ensin', + "ct_nsort": 'luonnollinen lajittelu (tiedostonimille jotka ovat numeroalkuisia)">nsort', + "ct_utc": 'näytä kaikki aikaleimat UTC-ajassa">UTC', + "ct_readme": 'näytä README.md hakemistolistauksissa">📜 readme', + "ct_idxh": 'näytä index.html hakemistolistan sijasta">htm', + "ct_sbars": 'näytä vierityspalkit">⟊', - "cut_umod": 'si un archivo ya existe en el servidor, actualiza la fecha de última modificación del servidor para que coincida con tu archivo local (requiere permisos de escritura+eliminación)">re📅', + "cut_umod": "jos tiedosto on jo olemassa palvelimella, päivitä palvelimen viimeksi muokattu aikaleima vastaamaan paikallista tiedostoasi (vaatii write- ja delete-oikeudet)\">re📅", - "cut_turbo": 'el botón yolo, probablemente NO quieras activarlo:$N$Núsalo si estabas subiendo una gran cantidad de archivos y tuviste que reiniciar por alguna razón, y quieres continuar la subida lo antes posible$N$Nesto reemplaza la comprobación de hash por un simple "¿tiene este el mismo tamaño de archivo en el servidor?" así que si el contenido del archivo es diferente, NO se subirá$N$Ndeberías desactivar esto cuando la subida termine, y luego "subir" los mismos archivos de nuevo para que el cliente los verifique">turbo', + "cut_turbo": "yolo-painike, ET todennäköisesti halua ottaa tätä käyttöön:$N$Nkäytä tätä jos latasit valtavan määrän tiedostoja ja jouduit käynnistämään uudelleen jostain syystä, ja haluat jatkaa latausta ASAP$N$Ntämä korvaa tiivistetarkistuksen yksinkertaisella "onko tällä sama tiedostokoko palvelimella?" joten jos tiedoston sisältö on erilainen sitä EI ladata$N$Nsinun pitäisi poistaa tämä käytöstä kun lataus on valmis, ja sitten "ladata" samat tiedostot uudelleen antaaksesi asiakkaan varmistaa ne\">turbo", - "cut_datechk": 'no tiene efecto a menos que el botón turbo esté activado$N$Nreduce el factor yolo en una pequeña cantidad; comprueba si las fechas de los archivos en el servidor coinciden con las tuyas$N$Nteóricamente debería detectar la mayoría de las subidas inacabadas / corruptas, pero no es un sustituto de hacer una pasada de verificación con el turbo desactivado después">verif. fecha', + "cut_datechk": "ei vaikutusta ellei turbo-painike ole käytössä$N$Nvähentää yolo-tekijää hieman; tarkistaa vastaavatko tiedostojen aikaleimat palvelimella omia$N$Npitäisi teoriassa napata useimmat keskeneräiset / vioittuneet lataukset, mutta ei ole korvike varmistuskierrokselle turbo poistettuna käytöstä jälkeenpäin\">päiväys-tark", - "cut_u2sz": "tamaño (en MiB) de cada trozo de subida; los valores grandes vuelan mejor a través del atlántico. Prueba valores bajos en conexiones muy poco fiables", + "cut_u2sz": "kunkin lähetyspalan koko (MiB:ssä); suuret arvot lentävät paremmin atlantin yli. kokeile pieniä arvoja erittäin heikoilla yhteyksillä", - "cut_flag": "asegura que solo una pestaña esté subiendo a la vez $N -- otras pestañas también deben tener esto activado $N -- solo afecta a pestañas en el mismo dominio", + "cut_flag": "varmista että vain yksi välilehti lataa kerrallaan $N -- muissa välilehdissä täytyy olla tämä käytössä myös $N -- vaikuttaa vain saman verkkotunnuksen välilehtiin", - "cut_az": "subir archivos en orden alfabético, en lugar de los más pequeños primero$N$Nel orden alfabético puede facilitar la detección visual de si algo salió mal en el servidor, pero hace la subida ligeramente más lenta en fibra / LAN", + "cut_az": "lähetä tiedostot aakkosjärjestyksessä, eikä pienin-tiedosto-ensiksi$N$Naakkosjärjestys voi tehdä helpommaksi silmäillä jos jokin meni vikaan palvelimella, mutta se tekee latauksesta hieman hitaamman kuitu- ja lähiverkossa", - "cut_nag": "notificación del SO cuando la subida se complete$N(solo si el navegador o la pestaña no están activos)", - "cut_sfx": "alerta sonora cuando la subida se complete$N(solo si el navegador o la pestaña no están activos)", + "cut_nag": "käyttöjärjestelmäilmoitus kun lataus valmistuu$N(vain jos selain tai välilehti ei ole aktiivinen)", + "cut_sfx": "äänivaroitus kun lataus valmistuu$N(vain jos selain tai välilehti ei ole aktiivinen)", - "cut_mt": 'usar multithreading para acelerar el hashing de archivos$N$Nesto usa web-workers y requiere$Nmás RAM (hasta 512 MiB extra)$N$Nhace https un 30% más rápido, http 4.5x más rápido">mt', + "cut_mt": "monisäikeistä tiedostojen tiivistysarvojen laskeminen$N$Ntämä käyttää web-workereitä ja vaatii$Nenemmän RAM-muistia (jopa 512 MiB ekstraa)$N$Ntekee https:n 30% nopeammaksi, http:n 4.5x nopeammaksi\">mt", - "cut_wasm": 'usar wasm en lugar del hasher incorporado del navegador; mejora la velocidad en navegadores basados en chrome pero aumenta la carga de la CPU, y muchas versiones antiguas de chrome tienen errores que hacen que el navegador consuma toda la RAM y se bloquee si esto está activado">wasm', + "cut_wasm": "käytä wasm:ia selaimen sisäänrakennetun tiivistäjän sijaan; parantaa nopeutta chrome-pohjaisissa selaimissa mutta lisää prosessorikuormaa, ja monissa vanhemmissa chrome-versioissa on bugeja jotka saavat selaimen kuluttamaan kaiken RAM-muistin ja kaatumaan jos tämä on käytössä\">wasm", - "cft_text": "texto del favicon (dejar en blanco y refrescar para desactivar)", - "cft_fg": "color de primer plano", - "cft_bg": "color de fondo", + "cft_text": "favicon-teksti (tyhjennä ja päivitä poistaaksesi käytöstä)", + "cft_fg": "edustaväri", + "cft_bg": "taustaväri", - "cdt_lim": "número máximo de archivos a mostrar en una carpeta", - "cdt_ask": "al llegar al final,$Nen lugar de cargar más archivos,$Npreguntar qué hacer", - "cdt_hsort": "cuántas reglas de ordenación (<code>,sorthref</code>) incluir en las URLs de medios. Ponerlo a 0 también ignorará las reglas de ordenación incluidas en los enlaces de medios al hacer clic en ellos", + "cdt_lim": "tiedostojen enimmäismäärä näytettäväksi kansiossa", + "cdt_ask": "sivun lopussa,$Nsen sijaan että lataa automaattisesti lisää tiedostoja,$Nkysy mitä tehdä", + "cdt_hsort": "kuinka monta lajittelusääntöä (<code>,sorthref</code>) sisällyttää media-URL:eihin. Tämän asettaminen nollaan jättää myös huomioimatta media-linkeissä sisällytetyt lajittelusäännöt kun napsautat niitä", - "tt_entree": "mostrar panel de navegación (barra lateral con árbol de directorios)$NAtajo: B", - "tt_detree": "mostrar breadcrumbs$NAtajo: B", - "tt_visdir": "desplazarse a la carpeta seleccionada", - "tt_ftree": "alternar árbol de carpetas / archivos de texto$NAtajo: V", - "tt_pdock": "mostrar carpetas de niveles superiores en un panel acoplado en la parte superior", - "tt_dynt": "crecimiento automático a medida que el árbol se expande", - "tt_wrap": "ajuste de línea", - "tt_hover": "revelar líneas que se desbordan al pasar el ratón$N( rompe el desplazamiento a menos que el $N  cursor esté en el margen izquierdo )", + "tt_entree": "näytä navigointipaneeli$NPikanäppäin: B", + "tt_detree": "näytä linkkipolku$NPikanäppäin: B", + "tt_visdir": "näytä valittu kansio", + "tt_ftree": "vaihda linkkipolku- / tekstitiedostonäkymään$NPikanäppäin: V", + "tt_pdock": "näytä yläkansiot telakoitussa paneelissa ylhäällä", + "tt_dynt": "kasvata automaattisesti hakemistosyvyyden kasvaessa", + "tt_wrap": "rivitys", + "tt_hover": "paljasta ylivuotavat rivit leijutettaessa$N( rikkoo vierityksen ellei hiiri $N  ole vasemmassa marginaalissa )", - "ml_pmode": "al final de la carpeta...", - "ml_btns": "acciones", - "ml_tcode": "transcodificar", - "ml_tcode2": "transcodificar a", - "ml_tint": "tinte", - "ml_eq": "ecualizador de audio", - "ml_drc": "compresor de rango dinámico", + "ml_pmode": "kansion lopussa...", + "ml_btns": "komennot", + "ml_tcode": "transkoodaa", + "ml_tcode2": "transkoodaa muotoon", + "ml_tint": "sävy", + "ml_eq": "taajuuskorjain", + "ml_drc": "dynaaminen alueen kompressori", - "mt_loop": 'poner en bucle/repetir una canción">🔁', - "mt_one": 'parar después de una canción">1️⃣', - "mt_shuf": 'reproducir aleatoriamente las canciones en cada carpeta">🔀', - "mt_aplay": 'reproducir automaticamente si hay un ID de canción en el enlace en el que hiciste clic para acceder al servidor$N$Ndesactivar esto también evitará que la URL de la página se actualice con IDs de canción al reproducir música, para prevenir la reproducción automática si se pierden estos ajustes pero la URL permanece">a▶', - "mt_preload": 'empezar a cargar la siguiente canción cerca del final para una reproducción sin pausas">precarga', - "mt_prescan": 'ir a la siguiente carpeta antes de que la última canción$Ntermine, manteniendo contento al navegador$Npara que no detenga la reproducción">nav', - "mt_fullpre": 'intentar precargar la canción entera;$N✅ activar en conexiones inestables,$N❌ desactivar probablemente en conexiones lentas">completa', - "mt_fau": 'en teléfonos, evitar que la música se detenga si la siguiente canción no se precarga lo suficientemente rápido (puede causar fallos en la visualización de etiquetas)">☕️', - "mt_waves": 'barra de búsqueda con forma de onda:$Nmostrar la amplitud del audio en la barra de progreso">~s', - "mt_npclip": 'mostrar botones para copiar al portapapeles la canción actual">/np', - "mt_m3u_c": 'mostrar botones para copiar al portapapeles las$Ncanciones seleccionadas como entradas de lista m3u8">📻', - "mt_octl": 'integración con SO (teclas multimedia / OSD)">ctl-so', - "mt_oseek": 'permitir buscar a través de la integración con el SO$N$Nnota: en algunos dispositivos (iPhones),$Nesto reemplaza el botón de siguiente canción">búsqueda', - "mt_oscv": 'mostrar carátula del álbum en OSD">arte', - "mt_follow": 'mantener la pista en reproducción visible en pantalla">🎯', - "mt_compact": 'controles compactos">⟎', - "mt_uncache": 'limpiar caché  (prueba esto si tu navegador guardó en caché$Nuna copia rota de una canción que se niega a reproducir)">limpiar caché', - "mt_mloop": 'repetir la carpeta actual">🔁 bucle', - "mt_mnext": 'cargar la siguiente carpeta y continuar">📂 sig', - "mt_mstop": 'detener reproducción">⏸ parar', - "mt_cflac": 'convertir flac / wav a opus">flac', - "mt_caac": 'convertir aac / m4a a opus">aac', - "mt_coth": 'convertir todos los demás (no mp3) a opus">oth', - "mt_c2opus": 'la mejor opción para ordenadores, portátiles, android">opus', - "mt_c2owa": 'opus-weba, para iOS 17.5 y superior">owa', - "mt_c2caf": 'opus-caf, para iOS 11 a 17">caf', - "mt_c2mp3": 'usar en dispositivos muy antiguos">mp3', - "mt_c2ok": "bien, buena elección", - "mt_c2nd": "ese no es el formato de salida recomendado para tu dispositivo, pero está bien", - "mt_c2ng": "tu dispositivo no parece soportar este formato de salida, pero intentémoslo de todas formas", - "mt_xowa": "hay errores en iOS que impiden la reproducción en segundo plano con este formato; por favor, usa caf o mp3 en su lugar", - "mt_tint": "nivel de fondo (0-100) en la barra de búsqueda$Npara hacer el buffering menos molesto", - "mt_eq": "activa el ecualizador y el control de ganancia;$N$Nganancia <code>0</code> = volumen estándar 100% (sin modificar)$N$Nancho <code>1  </code> = estéreo estándar (sin modificar)$Nancho <code>0.5</code> = 50% de crossfeed izq-der$Nancho <code>0  </code> = mono$N$Nganancia <code>-0.8</code> y ancho <code>10</code> = eliminación de voz :^)$N$Nactivar el ecualizador hace que los álbumes sin pausas sean completamente sin pausas, así que déjalo activado con todos los valores a cero (excepto ancho = 1) si eso te importa", - "mt_drc": "activa el compresor de rango dinámico (aplanador de volumen / brickwaller); también activará el EQ para equilibrar el espagueti, así que pon todos los campos de EQ excepto 'ancho' a 0 si no lo quieres$N$Nbaja el volumen del audio por encima de THRESHOLD dB; por cada RATIO dB pasado THRESHOLD hay 1 dB de salida, así que los valores por defecto de umbral -24 y ratio 12 significan que nunca debería sonar más fuerte de -22 dB y es seguro aumentar la ganancia del ecualizador a 0.8, o incluso 1.8 con ATK 0 y un RLS enorme como 90 (solo funciona en firefox; RLS es máx. 1 en otros navegadores)$N$N(ver wikipedia, lo explican mucho mejor)", + "mt_loop": "toista samaa kappaletta\">🔁", + "mt_one": "lopeta yhden kappaleen jälkeen\">1️⃣", + "mt_shuf": "satunnaistoisto\">🔀", + "mt_aplay": "automaattitoisto jos linkissä jolla pääsit palvelimelle oli kappale-ID$N$Ntämän poistaminen käytöstä pysäyttää myös sivun URL:n päivittämisen kappale-ID:lla musiikkia toistettaessa, estääksesi automaattitoiston jos nämä asetukset menetetään mutta URL säilyy\">a▶", + "mt_preload": "aloita seuraavan kappaleen lataaminen lähellä loppua, mahdollistaen saumattoman toiston\">esilataus", + "mt_prescan": "siirry seuraavaan kansioon ennen viimeisen kappaleen$Nloppumista, pitäen verkkoselaimen tyytyväisenä$Njotta se ei pysäytä toistoa\">nav", + "mt_fullpre": "yritä esiladata koko kappale;$N✅ ota käyttöön heikoilla yhteyksillä,$N❌ poista käytöstä hitailla yhteyksillä\">täysi", + "mt_fau": "puhelimissa: estä musiikin pysähtyminen jos seuraava kappale ei esilataudu tarpeeksi nopeasti (voi aiheuttaa ongelmia kappaletietojen näyttämisessä)\">☕️", + "mt_waves": "aaltomuoto-hakupalkki:$Nnäytä äänenvahvuus selaimessa\">~s", + "mt_npclip": "näytä painikkeet parhaillaan soivan kappaleen leikepöydälle kopioimiseen\">/np", + "mt_m3u_c": "näytä painikkeet valittujen$Nkappaleiden kopioimiseen m3u8-soittolistana leikepöydälle\">📻", + "mt_octl": "käyttöjärjestelmäintegraatio (medianäppäimet / osd)\">os-ctl", + "mt_oseek": "salli haku käyttöjärjestelmäintegraation kautta$N$Nhuom: joissakin laitteissa (iPhonet),$Ntämä korvaa seuraava-kappale-painikkeen\">kelaus", + "mt_oscv": "näytä albumin kansi osd:ssä\">taide", + "mt_follow": "pidä soiva kappale näkyvissä\">🎯", + "mt_compact": "kompaktit säätimet\">⟎", + "mt_uncache": "tyhjennä välimuisti  (kokeile tätä jos selaimesi välimuistissa on$Nrikkinäinen kopio kappaleesta)\">uncache", + "mt_mloop": "toista avoinna olevaa hakemistoa loputtomasti\">🔁 silmukka", + "mt_mnext": "lataa seuraava kansio ja jatka\">📂 seuraava", + "mt_mstop": "pysäytä toisto\">⏸ pysäytä", + "mt_cflac": "muunna flac / wav opus-muotoon\">flac", + "mt_caac": "muunna aac / m4a opus-muotoon\">aac", + "mt_coth": "muunna kaikki muut paitsi mp3 opus-muotoon\">muut", + "mt_c2opus": "paras valinta pöytäkoneille, kannettaville, androidille\">opus", + "mt_c2owa": "opus-weba, iOS 17.5:lle ja uudemmille\">owa", + "mt_c2caf": "opus-caf, iOS 11:lle - 17:lle\">caf", + "mt_c2mp3": "käytä tätä erittäin vanhoissa laitteissa\">mp3", + "mt_c2ok": "hienoa, hyvä valinta", + "mt_c2nd": "tuo ei ole suositeltu formaatti laitteellesi, mutta tee miten lystäät", + "mt_c2ng": "laitteesi ei näytä tukevan tätä formaattia, mutta yritetään nyt silti", + "mt_xowa": "iOS:ssä on bugeja jotka estävät taustatoiston tällä formaatilla; käytä caf:ia tai mp3:a sen sijaan", + "mt_tint": "taustan taso (0-100) liukupalkissa$Ntehden puskuroinnista vähemmän häiritsevän", + "mt_eq": "aktivoi taajuuskorjaimen ja vahvistussäätimen;$N$Nvahvistus <code>0</code> = normaali 100% äänenvoimakkuus (muokkaamaton)$N$Nleveys <code>1  </code> = normaali stereo (muokkaamaton)$Nleveys <code>0.5</code> = 50% vasen-oikea ristisyöttö$Nleveys <code>0  </code> = mono$N$Nvahvistus <code>-0.8</code> & leveys <code>10</code> = laulun poisto :^)$N$Nequalizerin käyttöönotto tekee saumattomista albumeista täysin saumattomia, joten jätä se päälle kaikilla arvoilla nollassa (paitsi leveys = 1) jos välität siitä", + "mt_drc": "aktivoi dynaamisen alueen kompressorin (äänenvoimakkuuden tasoittaja / tiiliseinättäjä); ottaa myös käyttöön EQ:n tasapainottamaan spagettia, joten aseta kaikki EQ-kentät paitsi 'leveys' 0:aan jos et halua sitä$N$Nalentaa äänenvoimakkuutta KYNNYS dB:n yläpuolella; jokaisesta SUHDE dB:stä KYNNYKSEN yli tulee 1 dB ulos, joten oletusarvot kynnys -24 ja suhde 12 tarkoittaa ettei sen pitäisi koskaan tulla kovempaa kuin -22 dB ja on turvallista nostaa equalizerin vahvistus 0.8:aan, tai jopa 1.8:aan ATK 0:lla ja valtavalla RLS:llä kuten 90 (toimii vain firefoxissa; RLS on max 1 muissa selaimissa)$N$N(katso wikipedia, he selittävät sen paljon paremmin)", - "mb_play": "reproducir", - "mm_hashplay": "¿reproducir este archivo de audio?", - "mm_m3u": "pulsa Intro/Aceptar para Reproducir\npulsa ESC/Cancelar para Editar", - "mp_breq": "se necesita firefox 82+ o chrome 73+ o iOS 15+", - "mm_bload": "cargando...", - "mm_bconv": "convirtiendo a {0}, por favor espera...", - "mm_opusen": "tu navegador no puede reproducir archivos aac / m4a;\nse ha activado la transcodificación a opus", - "mm_playerr": "fallo de reproducción: ", - "mm_eabrt": "El intento de reproducción fue cancelado", - "mm_enet": "Tu conexión a internet es inestable", - "mm_edec": "¿Este archivo está supuestamente corrupto?", - "mm_esupp": "Tu navegador no entiende este formato de audio", - "mm_eunk": "Error desconocido", - "mm_e404": "No se pudo reproducir el audio; error 404: Archivo no encontrado.", - "mm_e403": "No se pudo reproducir el audio; error 403: Acceso denegado.\n\nIntenta pulsar F5 para recargar, quizás se cerró tu sesión", - "mm_e500": "No se pudo reproducir el audio; error 500: Revisa los registros del servidor.", - "mm_e5xx": "No se pudo reproducir el audio; error del servidor ", - "mm_nof": "no se encuentran más archivos de audio cerca", - "mm_prescan": "Buscando música para reproducir a continuación...", - "mm_scank": "Encontrada la siguiente canción:", - "mm_uncache": "caché limpiada; todas las canciones se volverán a descargar en la próxima reproducción", - "mm_hnf": "esa canción ya no existe", + "mb_play": "toista", + "mm_hashplay": "soita tämä äänitiedosto?", + "mm_m3u": "paina Enter/OK Toistaaksesi\npaina ESC/Peruuta Muokataksesi", + "mp_breq": "tarvitset firefox 82+ tai chrome 73+ tai iOS 15+", + "mm_bload": "ladataan...", + "mm_bconv": "muunnetaan muotoon {0}, odota...", + "mm_opusen": "selaimesi ei voi toistaa aac / m4a -tiedostoja;\ntranskoodaus opukseen on nyt käytössä", + "mm_playerr": "toisto epäonnistui: ", + "mm_eabrt": "Toistoyritys peruttiin", + "mm_enet": "Internet-yhteytesi on epävakaa", + "mm_edec": "Tämä tiedosto on väitetysti vioittunut??", + "mm_esupp": "Selaimesi ei ymmärrä tätä äänimuotoa", + "mm_eunk": "Tuntematon virhe", + "mm_e404": "Kappaletta ei voitu toistaa; virhe 404: Tiedostoa ei löydy.", + "mm_e403": "Kappaletta ei voitu toistaa; virhe 403: Pääsy kielletty.\n\nKokeile painaa F5 päivittääksesi, ehkä kirjauduit ulos", + "mm_e500": "Kappaletta ei voitu toistaa; virhe 500: Tarkista palvelinlokit.", + "mm_e5xx": "Kappaletta ei voitu toistaa; palvelinvirhe ", + "mm_nof": "ei löydy enempää äänitiedostoja lähistöltä", + "mm_prescan": "Etsitään musiikkia toistettavaksi seuraavaksi...", + "mm_scank": "Löytyi seuraava kappale:", + "mm_uncache": "välimuisti tyhjennetty; kaikki kappaleet ladataan uudelleen seuraavalla toistolla", + "mm_hnf": "tuota kappaletta ei enää ole olemassa", - "im_hnf": "esa imagen ya no existe", +" im_hnf": "tuota kuvaa ei enää ole olemassa", - "f_empty": "esta carpeta está vacía", - "f_chide": "esto ocultará la columna «{0}»\n\npuedes volver a mostrar las columnas en la pestaña de configuración", - "f_bigtxt": "este archivo pesa {0} MiB -- ¿realmente verlo como texto?", - "f_bigtxt2": "¿ver solo el final del archivo en su lugar? esto también activará el seguimiento, mostrando las líneas de texto recién añadidas en tiempo real", - "fbd_more": '
mostrando {0} de {1} archivos; mostrar {2} o mostrar todos
', - "fbd_all": '
mostrando {0} de {1} archivos; mostrar todos
', - "f_anota": "solo {0} de los {1} elementos fueron seleccionados;\npara seleccionar la carpeta completa, primero desplázate hasta el final", + "f_empty": 'tämä kansio on tyhjä', + "f_chide": 'tämä piilottaa sarakkeen «{0}»\n\nvoit palauttaa sarakkeet asetuksista', + "f_bigtxt": "tämä tiedosto on {0} Mt kokoinen -- näytetäänkö silti tekstinä?", + "f_bigtxt2": "näytetäänkö vain tiedoston loppu? tämä myös mahdollistaa seuraamisen/tailing, näyttäen uudet tekstirivit reaaliaikaisesti", + "fbd_more": '
näytetään {0} / {1} tiedostoa; näytä {2} tai näytä kaikki
', + "fbd_all": '
näytetään {0} / {1} tiedostoa; näytä kaikki
', + "f_anota": "vain {0} / {1} kohdetta valittiin;\nvalitaksesi koko kansion, vieritä ensin loppuun", - "f_dls": "los enlaces a archivos en la carpeta actual se han\nconvertido en enlaces de descarga", + "f_dls": 'nykyisen kansion tiedostolinkit on\nvaihdettu latauslinkeiksi', - "f_partial": "Para descargar de forma segura un archivo que se está subiendo actualmente, por favor haz clic en el archivo con el mismo nombre, pero sin la extensión .PARTIAL. Por favor, pulsa CANCELAR o Escape para hacer esto.\n\nPulsar ACEPTAR o Intro ignorará esta advertencia y continuará descargando el archivo temporal .PARTIAL, lo que casi con toda seguridad te dará datos corruptos.", + "f_partial": "Ladataksesi turvallisesti tiedoston joka on parhaillaan latautumassa, klikkaa tiedostoa jolla on sama nimi mutta ilman .PARTIAL päätettä. Paina PERUUTA tai Escape tehdäksesi tämän.\n\nOK / Enter painaminen sivuuttaa tämän varoituksen ja jatkaa .PARTIAL väliaikaistiedoston lataamista, mikä todennäköisesti antaa sinulle vioittunutta dataa.", - "ft_paste": "pegar {0} elementos$NAtajo: ctrl-V", - "fr_eperm": "no se puede renombrar:\nno tienes permiso de “mover” en esta carpeta", - "fd_eperm": "no se puede eliminar:\nno tienes permiso de “eliminar” en esta carpeta", - "fc_eperm": "no se puede cortar:\nno tienes permiso de “mover” en esta carpeta", - "fp_eperm": "no se puede pegar:\nno tienes permiso de “escribir” en esta carpeta", - "fr_emore": "selecciona al menos un elemento para renombrar", - "fd_emore": "selecciona al menos un elemento para eliminar", - "fc_emore": "selecciona al menos un elemento para cortar", - "fcp_emore": "selecciona al menos un elemento para copiar al portapapeles", + "ft_paste": "liitä {0} kohdetta$NPikanäppäin: ctrl-V", + "fr_eperm": 'ei voida nimetä uudelleen:\nsinulla ei ole "move"-oikeutta tässä kansiossa', + "fd_eperm": 'ei voida poistaa:\nsinulla ei ole "delete" oikeutta tässä kansiossa', + "fc_eperm": 'ei voida leikata:\nsinulla ei ole "move" oikeutta tässä kansiossa', + "fp_eperm": 'ei voida liittää:\nsinulla ei ole "write" oikeutta tässä kansiossa', + "fr_emore": "valitse vähintään yksi kohde uudelleennimettäväksi", + "fd_emore": "valitse vähintään yksi kohde poistettavaksi", + "fc_emore": "valitse vähintään yksi kohde leikattavaksi", + "fcp_emore": "valitse vähintään yksi kohde kopioitavaksi leikepöydälle", - "fs_sc": "compartir la carpeta en la que estás", - "fs_ss": "compartir los archivos seleccionados", - "fs_just1d": "no puedes seleccionar más de una carpeta,\no mezclar archivos y carpetas en una selección", - "fs_abrt": "❌ abortar", - "fs_rand": "🎲 nombre aleatorio", - "fs_go": "✅ crear enlace", - "fs_name": "nombre", - "fs_src": "origen", - "fs_pwd": "contraseña", - "fs_exp": "caducidad", - "fs_tmin": "minutos", - "fs_thrs": "horas", - "fs_tdays": "días", - "fs_never": "eterno", - "fs_pname": "nombre opcional del enlace; será aleatorio si se deja en blanco", - "fs_tsrc": "el archivo o carpeta a compartir", - "fs_ppwd": "contraseña opcional", - "fs_w8": "creando enlace...", - "fs_ok": "pulsa Intro/Aceptar para Copiar al Portapapeles\npulsa ESC/Cancelar para Cerrar", + "fs_sc": "jaa kansio jossa olet", + "fs_ss": "jaa valitut tiedostot", + "fs_just1d": "et voi valita useampaa kuin yhtä kansiota,\ntai sekoittaa tiedostoja ja kansioita yhdessä valinnassa", + "fs_abrt": "❌ keskeytä", + "fs_rand": "🎲 joku.nimi", + "fs_go": "✅ luo share", + "fs_name": "nimi", + "fs_src": "lähde", + "fs_pwd": "salasana", + "fs_exp": "vanheneminen", + "fs_tmin": "min", + "fs_thrs": "tuntia", + "fs_tdays": "päivää", + "fs_never": "ikuinen", + "fs_pname": "valinnainen linkin nimi; on satunnainen jos tyhjä", + "fs_tsrc": "jaettava tiedosto tai kansio", + "fs_ppwd": "valinnainen salasana", + "fs_w8": "luodaan sharea...", + "fs_ok": "paina Enter/OK lisätäksesi leikepöydälle\npaina ESC/Cancel sulkeaksesi", - "frt_dec": "puede arreglar algunos casos de nombres de archivo rotos\">url-decode", - "frt_rst": "restaurar los nombres de archivo modificados a los originales\">↺ restablecer", - "frt_abrt": "abortar y cerrar esta ventana\">❌ cancelar", - "frb_apply": "APLICAR RENOMBRADO", - "fr_adv": "renombrado por lotes / metadatos / patrones\">avanzado", - "fr_case": "regex sensible a mayúsculas\">mayús", - "fr_win": "nombres seguros para windows; reemplaza <>:"\\|?* con caracteres japoneses de ancho completo\">win", - "fr_slash": "reemplaza / con un carácter que no cree nuevas carpetas\">sin /", - "fr_re": "patrón de búsqueda regex para aplicar a los nombres de archivo originales; los grupos de captura se pueden referenciar en el campo de formato de abajo como <code>(1)</code> y <code>(2)</code> y así sucesivamente", - "fr_fmt": "inspirado en foobar2000:$N<code>(title)</code> se reemplaza por el título de la canción,$N<code>[(artist) - ](title)</code> omite la parte [entre corchetes] si el artista está en blanco$N<code>$lpad((tn),2,0)</code> rellena el número de pista a 2 dígitos", - "fr_pdel": "eliminar", - "fr_pnew": "guardar como", - "fr_pname": "proporciona un nombre para tu nuevo preajuste", - "fr_aborted": "abortado", - "fr_lold": "nombre antiguo", - "fr_lnew": "nombre nuevo", - "fr_tags": "etiquetas para los archivos seleccionados (solo lectura, como referencia):", - "fr_busy": "renombrando {0} elementos...\n\n{1}", - "fr_efail": "fallo al renombrar:\n", - "fr_nchg": "{0} de los nuevos nombres fueron alterados debido a win y/o sin /\n\n¿Aceptar para continuar con estos nuevos nombres alterados?", + "frt_dec": "saattaa korjata joitakin rikkinäisiä tiedostonimiä\">url-decode", + "frt_rst": "palauta muokatut tiedostonimet takaisin alkuperäisiksi\">↺ palauta", + "frt_abrt": "keskeytä ja sulje tämä ikkuna\">❌ peruuta", + "frb_apply": "UUDELLEENNIMEÄ", + "fr_adv": "erä / liitännäistiedot / kaava uudelleennimeäminen\">lisäasetukset", + "fr_case": "isot ja pienet kirjaimet erottava regex\">kirjainkoko", + "fr_win": "windows-yhteensopivat nimet; korvaa <>:"\\|?* japanilaisilla leveillä merkeillä\">win", + "fr_slash": "korvaa / merkillä joka ei aiheuta uusien kansioiden luomista\">ei /", + "fr_re": "regex hakukuvio jota käytetään alkuperäisiin tiedostonimiin; kaappausryhmiin voi viitata alla olevassa muotoilukentässä kuten <code>(1)</code> ja <code>(2)</code> ja niin edelleen", + "fr_fmt": "foobar2000 innoittama:$N<code>(title)</code> korvataan kappaleen nimellä,$N<code>[(artist) - ](title)</code> sivuuttaa [tämän] osan jos artisti on tyhjä$N<code>$lpad((tn),2,0)</code> ", + "fr_pdel": "poista", + "fr_pnew": "tallenna nimellä", + "fr_pname": "anna nimi uudelle esiasetuksellesi", + "fr_aborted": "keskeytetty", + "fr_lold": "vanha nimi", + "fr_lnew": "uusi nimi", + "fr_tags": "valittujen tiedostojen tagit (vain luku, viitetarkoituksiin):", + "fr_busy": "nimetään uudelleen {0} kohdetta...\n\n{1}", + "fr_efail": "uudelleennimeäminen epäonnistui:\n", + "fr_nchg": "{0} uusista nimistä muutettiin win ja/tai ei / vuoksi\n\nJatketaanko näillä muutetuilla uusilla nimillä?", - "fd_ok": "eliminación correcta", - "fd_err": "fallo al eliminar:\n", - "fd_none": "no se eliminó nada; quizás bloqueado por la configuración del servidor (xbd)?", - "fd_busy": "eliminando {0} elementos...\n\n{1}", - "fd_warn1": "¿ELIMINAR estos {0} elementos?", - "fd_warn2": "¡Última oportunidad! No se puede deshacer. ¿Eliminar?", + "fd_ok": "poisto OK", + "fd_err": "poisto epäonnistui:\n", + "fd_none": "mitään ei poistettu; ehkä palvelimen asetukset estivät (xbd)?", + "fd_busy": "poistetaan {0} kohdetta...\n\n{1}", + "fd_warn1": "POISTA nämä {0} kohdetta?", + "fd_warn2": "Viimeinen varoitus! Haluatko varmasti poistaa?", - "fc_ok": "cortados {0} elementos", - "fc_warn": "cortados {0} elementos\n\npero: solo esta pestaña del navegador puede pegarlos\n(dado que la selección es absolutamente masiva)", + "fc_ok": "siirettiin {0} kohdetta leikepöydälle", + "fc_warn": 'siirettiin {0} kohdetta leikepöydälle\n\nmutta: vain tämä selain-välilehti voi liittää ne\n(koska valinta on niin valtavan suuri)', - "fcc_ok": "copiados {0} elementos al portapapeles", - "fcc_warn": "copiados {0} elementos al portapapeles\n\npero: solo esta pestaña del navegador puede pegarlos\n(dado que la selección es absolutamente masiva)", + "fcc_ok": "kopioitiin {0} kohdetta leikepöydälle", + "fcc_warn": 'kopioitiin {0} kohdetta leikepöydälle\n\nmutta: vain tämä selain-välilehti voi liittää ne\n(koska valinta on niin valtavan suuri)', - "fp_apply": "usar estos nombres", - "fp_ecut": "primero corta o copia algunos archivos / carpetas para pegar / mover\n\nnota: puedes cortar / pegar entre diferentes pestañas del navegador", - "fp_ename": "{0} elementos no se pueden mover aquí porque los nombres ya existen. Dales nuevos nombres abajo para continuar, o deja el nombre en blanco para omitirlos:", - "fcp_ename": "{0} elementos no se pueden copiar aquí porque los nombres ya existen. Dales nuevos nombres abajo para continuar, o deja el nombre en blanco para omitirlos:", - "fp_emore": "todavía quedan algunas colisiones de nombres por resolver", - "fp_ok": "movimiento correcto", - "fcp_ok": "copia correcta", - "fp_busy": "moviendo {0} elementos...\n\n{1}", - "fcp_busy": "copiando {0} elementos...\n\n{1}", - "fp_err": "fallo al mover:\n", - "fcp_err": "fallo al copiar:\n", - "fp_confirm": "¿mover estos {0} elementos aquí?", - "fcp_confirm": "¿copiar estos {0} elementos aquí?", - "fp_etab": "fallo al leer el portapapeles de otra pestaña del navegador", - "fp_name": "subiendo un archivo desde tu dispositivo. Dale un nombre:", - "fp_both_m": "
elige qué pegar
Intro = Mover {0} archivos desde «{1}»\nESC = Subir {2} archivos desde tu dispositivo", - "fcp_both_m": "
elige qué pegar
Intro = Copiar {0} archivos desde «{1}»\nESC = Subir {2} archivos desde tu dispositivo", - "fp_both_b": "MoverSubir", - "fcp_both_b": "CopiarSubir", + "fp_apply": "käytä näitä nimiä", + "fp_ecut": "leikkaa tai kopioi ensin joitakin tiedostoja / kansioita liitettäväksi / siirrettäväksi\n\nhuom: voit leikata / liittää eri selain-välilehtien välillä", + "fp_ename": "{0} kohdetta ei voida siirtää tänne koska nimet ovat jo käytössä. Anna niille uudet nimet alla jatkaaksesi, tai tyhjennä nimi ohittaaksesi ne:", + "fcp_ename": "{0} kohdetta ei voida kopioida tänne koska nimet ovat jo käytössä. Anna niille uudet nimet alla jatkaaksesi, tai tyhjennä nimi ohittaaksesi ne:", + "fp_emore": "tiedostonimien törmäyksiä on vielä korjaamatta", + "fp_ok": "siirto OK", + "fcp_ok": "kopiointi OK", + "fp_busy": "siirretään {0} kohdetta...\n\n{1}", + "fcp_busy": "kopioidaan {0} kohdetta...\n\n{1}", + "fp_err": "siirto epäonnistui:\n", + "fcp_err": "kopiointi epäonnistui:\n", + "fp_confirm": "siirrä nämä {0} kohdetta tänne?", + "fcp_confirm": "kopioi nämä {0} kohdetta tänne?", + "fp_etab": 'leikepöydän lukeminen toisesta selain-välilehdestä epäonnistui', + "fp_name": "ladataan tiedostoa laitteeltasi. Anna sille nimi:", + "fp_both_m": '
valitse mitä liittää
Enter = Siirrä {0} tiedostoa kohteesta «{1}»\nESC = Lataa {2} tiedostoa laitteeltasi', + "fcp_both_m": '
valitse mitä liittää
Enter = Kopioi {0} tiedostoa kohteesta «{1}»\nESC = Lataa {2} tiedostoa laitteeltasi', + "fp_both_b": 'SiirräLähetä', + "fcp_both_b": 'KopioiLähetä', - "mk_noname": "escribe un nombre en el campo de texto de la izquierda antes de hacer eso :p", + "mk_noname": "kirjoita nimi vasemmalla olevaan tekstikenttään ennen kuin teet tuon :p", - "tv_load": "Cargando documento de texto:\n\n{0}\n\n{1}% ({2} de {3} MiB cargados)", - "tv_xe1": "no se pudo cargar el archivo de texto:\n\nerror ", - "tv_xe2": "404, archivo no encontrado", - "tv_lst": "lista de archivos de texto en", - "tvt_close": "volver a la vista de carpetas$NAtajo: M (o Esc)\">❌ cerrar", - "tvt_dl": "descargar este archivo$NAtajo: Y\">💾 descargar", - "tvt_prev": "mostrar documento anterior$NAtajo: i\">⬆ ant", - "tvt_next": "mostrar siguiente documento$NAtajo: K\">⬇ sig", - "tvt_sel": "seleccionar archivo   ( para cortar / copiar / eliminar / ... )$NAtajo: S\">sel", - "tvt_edit": "abrir archivo en editor de texto$NAtajo: E\">✏️ editar", - "tvt_tail": "monitorizar cambios en el archivo; mostrar nuevas líneas en tiempo real\">📡 seguir", - "tvt_wrap": "ajuste de línea\">↵", - "tvt_atail": "bloquear el desplazamiento al final de la página\">⚓", - "tvt_ctail": "decodificar colores de terminal (códigos de escape ansi)\">🌈", - "tvt_ntail": "límite de historial (cuántos bytes de texto mantener cargados)", + "tv_load": "Ladataan tekstidokumenttia:\n\n{0}\n\n{1}% ({2} / {3} Mt ladattu)", + "tv_xe1": "tekstitiedoston lataaminen epäonnistui:\n\nvirhe ", + "tv_xe2": "404, tiedostoa ei löydy", + "tv_lst": "tekstitiedostojen lista kansiossa", + "tvt_close": "palaa kansionäkymään$NPikanäppäin: M (tai Esc)\">❌ sulje", + "tvt_dl": "lataa tämä tiedosto$NPikanäppäin: Y\">💾 lataa", + "tvt_prev": "näytä edellinen dokumentti$NPikanäppäin: i\">⬆ edell", + "tvt_next": "näytä seuraava dokumentti$NPikanäppäin: K\">⬇ seur", + "tvt_sel": "valitse tiedosto   ( leikkausta / kopiointia / poistoa / ... varten )$NPikanäppäin: S\">val", + "tvt_edit": "avaa tiedosto tekstieditorissa$NPikanäppäin: E\">✏️ muokkaa", + "tvt_tail": "seuraa tiedoston muutoksia; näytä uudet rivit reaaliaikaisesti\">📡 seuraa", + "tvt_wrap": "rivitys\">↵", + "tvt_atail": "lukitse vieritys sivun alaosaan\">⚓", + "tvt_ctail": "dekoodaa terminaalin värit (ansi escape koodit)\">🌈", + "tvt_ntail": "vieritysbufferin raja (kuinka monta tavua tekstiä pidetään ladattuna)", - "m3u_add1": "canción añadida a la lista m3u", - "m3u_addn": "{0} canciones añadidas a la lista m3u", - "m3u_clip": "lista m3u copiada al portapapeles\n\ndebes crear un nuevo archivo de texto llamado algo.m3u y pegar la lista en ese documento; esto lo hará reproducible", + "m3u_add1": "kappale lisätty m3u soittolistaan", + "m3u_addn": "{0} kappaletta lisätty m3u soittolistaan", + "m3u_clip": "m3u soittolista nyt kopioitu leikepöydälle\n\nsinun tulisi luoda uusi tekstitiedosto nimeltä jotain.m3u ja liittää soittolista siihen dokumenttiin; tämä tekee siitä soitettavan", - "gt_vau": "no mostrar vídeos, solo reproducir el audio\">🎧", - "gt_msel": "activar selección de archivos; ctrl-clic en un archivo para anular$N$N<em>cuando está activo: doble clic en un archivo / carpeta para abrirlo</em>$N$NAtajo: S\">multiselección", - "gt_crop": "recortar miniaturas\">recortar", - "gt_3x": "miniaturas de alta resolución\">3x", - "gt_zoom": "zoom", - "gt_chop": "recortar", - "gt_sort": "ordenar por", - "gt_name": "nombre", - "gt_sz": "tamaño", - "gt_ts": "fecha", - "gt_ext": "tipo", - "gt_c1": "truncar más los nombres de archivo (mostrar menos)", - "gt_c2": "truncar menos los nombres de archivo (mostrar más)", + "gt_vau": "älä näytä videoita, toista vain ääni\">🎧", + "gt_msel": "ota käyttöön tiedostojen valinta; ctrl-klikkaa tiedostoa ohittaaksesi$N$N<em>kun aktiivinen: tuplaklikkaa tiedostoa / kansiota avataksesi sen</em>$N$NPikanäppäin: S\">monivalinta", + "gt_crop": "keskitä-rajaa pienoiskuvat\">rajaa", + "gt_3x": "korkearesoluutioiset pienoiskuvat\">3x", + "gt_zoom": "zoomaa", + "gt_chop": "pilko", + "gt_sort": "järjestä", + "gt_name": "nimi", + "gt_sz": "koko", + "gt_ts": "päiväys", + "gt_ext": "tyyppi", + "gt_c1": "rajaa tiedostonimiä enemmän (näytä vähemmän)", + "gt_c2": "rajaa tiedostonimiä vähemmän (näytä enemmän)", - "sm_w8": "buscando...", - "sm_prev": "los resultados de búsqueda a continuación son de una consulta anterior:\n ", - "sl_close": "cerrar resultados de búsqueda", - "sl_hits": "mostrando {0} resultados", - "sl_moar": "cargar más", + "sm_w8": "haetaan...", + "sm_prev": "alla olevat hakutulokset ovat edellisestä hausta:\n ", + "sl_close": "sulje hakutulokset", + "sl_hits": "näytetään {0} osumaa", + "sl_moar": "lataa lisää", - "s_sz": "tamaño", - "s_dt": "fecha", - "s_rd": "ruta", - "s_fn": "nombre", - "s_ta": "etiquetas", - "s_ua": "subido@", - "s_ad": "avanzado", - "s_s1": "MiB mínimo", - "s_s2": "MiB máximo", - "s_d1": "mín. iso8601", - "s_d2": "máx. iso8601", - "s_u1": "subido después de", - "s_u2": "y/o antes de", - "s_r1": "la ruta contiene   (separado por espacios)", - "s_f1": "el nombre contiene   (negar con -no)", - "s_t1": "las etiquetas contienen   (^=inicio, fin=$)", - "s_a1": "propiedades de metadatos específicas", + "s_sz": "koko", + "s_dt": "päiväys", + "s_rd": "polku", + "s_fn": "nimi", + "s_ta": "tagit", + "s_ua": "ylös@", + "s_ad": "edist.", + "s_s1": "minimi Mt", + "s_s2": "maksimi Mt", + "s_d1": "min. iso8601", + "s_d2": "maks. iso8601", + "s_u1": "ladattu jälkeen", + "s_u2": "ja/tai ennen", + "s_r1": "polku sisältää   (välilyönnillä erotetuttuina)", + "s_f1": "nimi sisältää   (negatoi käyttämällä -nope)", + "s_t1": "tagit sisältää   (^=alku, loppu=$)", + "s_a1": "tietyt metadatan ominaisuudet", - "md_eshow": "no se puede renderizar ", - "md_off": "[📜léeme] desactivado en [⚙️] -- documento oculto", + "md_eshow": "ei voida renderoida ", + "md_off": "[📜readme] poistettu käytöstä [⚙️] -- dokumentti piilotettu", - "badreply": "Fallo al procesar la respuesta del servidor", + "badreply": "Palvelimen vastauksen jäsentäminen epäonnistui.", - "xhr403": "403: Acceso denegado\n\nintenta pulsar F5, quizás se cerró tu sesión", - "xhr0": "desconocido (probablemente se perdió la conexión con el servidor, o el servidor está desconectado)", - "cf_ok": "perdón por eso -- la protección DD" + wah + "oS se activó\n\nlas cosas deberían reanudarse en unos 30 segundos\n\nsi no pasa nada, pulsa F5 para recargar la página", - "tl_xe1": "no se pudieron listar las subcarpetas:\n\nerror ", - "tl_xe2": "404: Carpeta no encontrada", - "fl_xe1": "no se pudieron listar los archivos en la carpeta:\n\nerror ", - "fl_xe2": "404: Carpeta no encontrada", - "fd_xe1": "no se pudo crear la subcarpeta:\n\nerror ", - "fd_xe2": "404: Carpeta de nivel superior no encontrada", - "fsm_xe1": "no se pudo enviar el mensaje:\n\nerror ", - "fsm_xe2": "404: Carpeta de nivel superior no encontrada", - "fu_xe1": "fallo al cargar la lista de deshacer del servidor:\n\nerror ", - "fu_xe2": "404: ¿Archivo no encontrado?", + "xhr403": "403: Pääsy kielletty\n\nkokeile painaa F5, ehkä sinut kirjattiin ulos", + "xhr0": "tuntematon (todennäköisesti yhteys palvelimeen katosi, tai palvelin on pois päältä)", + "cf_ok": "sori siitä -- DD" + wah + "oS suojaus aktivoitui\n\nasioiden pitäisi jatkua noin 30 sekunnissa\n\njos mitään ei tapahdu, paina F5 ladataksesi sivun uudelleen", + "tl_xe1": "alikansioiden listaaminen epäonnistui:\n\nvirhe ", + "tl_xe2": "404: Kansiota ei löydy", + "fl_xe1": "kansion tiedostojen listaaminen epäonnistui:\n\nvirhe ", + "fl_xe2": "404: Kansiota ei löydy", + "fd_xe1": "alikansion luominen epäonnistui:\n\nvirhe ", + "fd_xe2": "404: Yläkansiota ei löydy", + "fsm_xe1": "viestin lähettäminen epäonnistui:\n\nvirhe ", + "fsm_xe2": "404: Yläkansiota ei löydy", + "fu_xe1": "unpost-listan lataaminen palvelimelta epäonnistui:\n\nvirhe ", + "fu_xe2": "404: Tiedostoa ei löydy??", - "fz_tar": "archivo gnu-tar sin comprimir (linux / mac)", - "fz_pax": "tar formato pax sin comprimir (más lento)", - "fz_targz": "gnu-tar con compresión gzip nivel 3$N$Nesto suele ser muy lento, así que$Nusa tar sin comprimir en su lugar", - "fz_tarxz": "gnu-tar con compresión xz nivel 1$N$Nesto suele ser muy lento, así que$Nusa tar sin comprimir en su lugar", - "fz_zip8": "zip con nombres de archivo utf8 (puede dar problemas en windows 7 y anteriores)", - "fz_zipd": "zip con nombres de archivo cp437 tradicionales, para software muy antiguo", - "fz_zipc": "cp437 con crc32 calculado tempranamente,$Npara MS-DOS PKZIP v2.04g (octubre 1993)$N(tarda más en procesar antes de que la descarga pueda empezar)", + "fz_tar": "pakkaamaton gnu-tar tiedosto (linux / mac)", + "fz_pax": "pakkaamaton pax-formaatin tar (hitaampi)", + "fz_targz": "gnu-tar gzip tason 3 pakkauksella$N$NTämä on yleensä hyvin hidasta, $Nkäytä pakkamatonta tar:ia sen sijaan", + "fz_tarxz": "gnu-tar xz tason 1 pakkauksella$N$NTämä on yleensä hyvin hidasta, $Nkäytä pakkamatonta tar:ia sen sijaan", + "fz_zip8": "zip utf8-tiedostonimillä (suattaapi olla epävakaa windows 7:ssa ja vanhemmissa)", + "fz_zipd": "zip perinteisillä cp437 tiedostonimillä esihistoriallisille ohjelmistoille", + "fz_zipc": "cp437, jossa crc32 laskettu aikaisin,$NMS-DOS PKZIP v2.04g:lle (lokakuu 1993)$N(kestää kauemmin käsitellä ennen latauksen alkua)", - "un_m1": "puedes eliminar tus subidas recientes (o abortar las inacabadas) a continuación", - "un_upd": "actualizar", - "un_m4": "o compartir los archivos visibles a continuación:", - "un_ulist": "mostrar", - "un_ucopy": "copiar", - "un_flt": "filtro opcional:  la URL debe contener", - "un_fclr": "limpiar filtro", - "un_derr": "fallo al deshacer-eliminar:\n", - "un_f5": "algo se rompió, por favor intenta actualizar o pulsa F5", - "un_uf5": "lo siento pero tienes que refrescar la página (por ejemplo pulsando F5 o CTRL-R) antes de que esta subida pueda ser abortada", - "un_nou": "aviso: servidor demasiado ocupado para mostrar subidas inacabadas; haz clic en el enlace \"actualizar\" en un momento", - "un_noc": "aviso: la opción de deshacer subidas completadas no está activada/permitida en la configuración del servidor", - "un_max": "mostrando los primeros 2000 archivos (usa el filtro)", - "un_avail": "{0} subidas recientes se pueden eliminar
{1} inacabadas se pueden abortar", - "un_m2": "ordenado por tiempo de subida; más recientes primero:", - "un_no1": "¡pues no! ninguna subida es suficientemente reciente", - "un_no2": "¡pues no! ninguna subida que coincida con ese filtro es suficientemente reciente", - "un_next": "eliminar los siguientes {0} archivos a continuación", - "un_abrt": "abortar", - "un_del": "eliminar", - "un_m3": "cargando tus subidas recientes...", - "un_busy": "eliminando {0} archivos...", - "un_clip": "{0} enlaces copiados al portapapeles", + "un_m1": "voit poistaa viimeaikaiset latauksesi (tai keskeyttää keskeneräiset) alla", + "un_upd": "päivitä", + "un_m4": "tai jaa alla näkyvät tiedostot:", + "un_ulist": "näytä", + "un_ucopy": "kopioi", + "un_flt": "valinnainen suodatin:  URL:n täytyy sisältää", + "un_fclr": "tyhjennä suodatin", + "un_derr": 'unpost-poisto epäonnistui:\n', + "un_f5": 'jotain hajosi, kokeile päivitystä tai paina F5', + "un_uf5": "pahoittelen mutta sinun täytyy päivittää sivu (esimerkiksi painamalla F5 tai CTRL-R) ennen kuin tämä lataus voidaan keskeyttää", + "un_nou": 'varoitus: palvelin liian kiireinen näyttääkseen keskeneräiset lataukset; klikkaa "päivitä" linkkiä hetken kuluttua', + "un_noc": 'varoitus: täysin ladattujen tiedostojen unpost ei ole käytössä/sallittu palvelimen asetuksissa', + "un_max": "näytetään ensimmäiset 2000 tiedostoa (käytä suodatinta)", + "un_avail": "{0} viimeaikaista latausta voidaan poistaa
{1} keskeneräistä voidaan keskeyttää", + "un_m2": "järjestetty latausajan mukaan; viimeisimmät ensin:", + "un_no1": "hupsis! yksikään lataus ei ole riittävän tuore", + "un_no2": "hupsis! yksikään tuota suodatinta vastaava lataus ei ole riittävän tuore", + "un_next": "poista seuraavat {0} tiedostoa alla", + "un_abrt": "keskeytä", + "un_del": "poista", + "un_m3": "ladataan viimeaikana lähettämiäsi tiedostoja...", + "un_busy": "poistetaan {0} tiedostoa...", + "un_clip": "{0} linkkiä kopioitu leikepöydälle", - "u_https1": "deberías", - "u_https2": "cambiar a https", - "u_https3": "para un mejor rendimiento", - "u_ancient": "tu navegador es impresionantemente antiguo -- quizás deberías usar bup en su lugar", - "u_nowork": "se necesita firefox 53+ o chrome 57+ o iOS 11+", - "tail_2old": "se necesita firefox 105+ o chrome 71+ o iOS 14.5+", - "u_nodrop": "tu navegador es demasiado antiguo para subir arrastrando y soltando", - "u_notdir": "¡eso no es una carpeta!\n\ntu navegador es demasiado antiguo,\npor favor intenta arrastrar y soltar en su lugar", - "u_uri": "para arrastrar y soltar imágenes desde otras ventanas del navegador,\npor favor suéltalas sobre el gran botón de subida", - "u_enpot": "cambiar a UI ligera (puede mejorar la velocidad de subida)", - "u_depot": "cambiar a UI elegante (puede reducir la velocidad de subida)", - "u_gotpot": "cambiando a la UI ligera para mejorar la velocidad de subida,\n\n¡siéntete libre de no estar de acuerdo y volver a cambiar!", - "u_pott": "

archivos:   {0} finalizados,   {1} fallidos,   {2} ocupados,   {3} en cola

", - "u_ever": "este es el uploader básico; up2k necesita al menos
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", - "u_su2k": "este es el uploader básico; up2k es mejor", - "u_uput": "optimizar para velocidad (omitir checksum)", - "u_ewrite": "no tienes acceso de escritura a esta carpeta", - "u_eread": "no tienes acceso de lectura a esta carpeta", - "u_enoi": "la búsqueda de archivos no está activada en la configuración del servidor", - "u_enoow": "sobrescribir no funcionará aquí; se necesita permiso de eliminación", - "u_badf": "Estos {0} archivos (de un total de {1}) se omitieron, posiblemente debido a permisos del sistema de archivos:\n\n", - "u_blankf": "Estos {0} archivos (de un total de {1}) están en blanco / vacíos; ¿subirlos de todos modos?\n\n", - "u_applef": "Estos {0} archivos (de un total de {1}) probablemente no son deseables;\nPulsa Aceptar/Intro para OMITIR los siguientes archivos,\nPulsa Cancelar/ESC para NO excluir, y SUBIR esos también:\n\n", - "u_just1": "\nQuizás funcione mejor si seleccionas solo un archivo", - "u_ff_many": "si usas Linux / MacOS / Android, esta cantidad de archivos podría bloquear Firefox!\nsi eso ocurre, por favor inténtalo de nuevo (o usa Chrome).", - "u_up_life": "Esta subida será eliminada del servidor\n{0} después de que se complete", - "u_asku": "subir estos {0} archivos a {1}", - "u_unpt": "puedes deshacer / eliminar esta subida usando el 🧯 de arriba a la izquierda", - "u_bigtab": "a punto de mostrar {0} archivos\n\nesto podría bloquear tu navegador, ¿estás seguro?", - "u_scan": "Escaneando archivos...", - "u_dirstuck": "el iterador de directorios se atascó intentando acceder a los siguientes {0} elementos; se omitirán:", - "u_etadone": "Hecho ({0}, {1} archivos)", - "u_etaprep": "(preparando para subir)", - "u_hashdone": "hashing completado", - "u_hashing": "hash", - "u_hs": "negociando...", - "u_started": "los archivos se están subiendo ahora; mira en [🚀]", - "u_dupdefer": "duplicado; se procesará después de todos los demás archivos", - "u_actx": "haz clic en este texto para evitar la pérdida de
rendimiento al cambiar a otras ventanas/pestañas", - "u_fixed": "¡OK!  Arreglado 👍", - "u_cuerr": "fallo al subir el trozo {0} de {1};\nprobablemente inofensivo, continuando\n\narchivo: {2}", - "u_cuerr2": "el servidor rechazó la subida (trozo {0} de {1});\nse reintentará más tarde\n\narchivo: {2}\n\nerror ", - "u_ehstmp": "se reintentará; mira abajo a la derecha", - "u_ehsfin": "el servidor rechazó la solicitud para finalizar la subida; reintentando...", - "u_ehssrch": "el servidor rechazó la solicitud para realizar la búsqueda; reintentando...", - "u_ehsinit": "el servidor rechazó la solicitud para iniciar la subida; reintentando...", - "u_eneths": "error de red al realizar la negociación de subida; reintentando...", - "u_enethd": "error de red al comprobar la existencia del destino; reintentando...", - "u_cbusy": "esperando a que el servidor vuelva a confiar en nosotros después de un fallo de red...", - "u_ehsdf": "¡el servidor se quedó sin espacio en disco!\n\nse seguirá reintentando, por si alguien\nlibera suficiente espacio para continuar", - "u_emtleak1": "parece que tu navegador podría tener una fuga de memoria;\npor favor", - "u_emtleak2": " cambia a https (recomendado) o ", - "u_emtleak3": " ", - "u_emtleakc": "prueba lo siguiente:\n
  • pulsa F5 para refrescar la página
  • luego desactiva el botón  mt  en los  ⚙️ ajustes
  • e intenta esa subida de nuevo
Las subidas serán un poco más lentas, pero bueno.\n¡Perdón por las molestias!\n\nPD: chrome v107 tiene una solución para esto", - "u_emtleakf": "prueba lo siguiente:\n
  • pulsa F5 para refrescar la página
  • luego activa 🥔 (ligera) en la interfaz de subida
  • e intenta esa subida de nuevo
\nPD: firefox con suerte tendrá una solución en algún momento", - "u_s404": "no encontrado en el servidor", - "u_expl": "explicar", - "u_maxconn": "la mayoría de los navegadores limitan esto a 6, pero firefox te permite aumentarlo con connections-per-server en about:config", - "u_tu": '

AVISO: turbo activado,  el cliente puede no detectar y reanudar subidas incompletas; ver tooltip del botón turbo

', - "u_ts": '

AVISO: turbo activado,  los resultados de búsqueda pueden ser incorrectos; ver tooltip del botón turbo

', - "u_turbo_c": "turbo está desactivado en la configuración del servidor", - "u_turbo_g": "desactivando turbo porque no tienes\nprivilegios para listar directorios en este volumen", - "u_life_cfg": 'autoeliminar después de min (o horas)', - "u_life_est": 'la subida se eliminará ---', - "u_life_max": "esta carpeta impone una\nvida máxima de {0}", - "u_unp_ok": "se permite deshacer la subida durante {0}", - "u_unp_ng": "NO se permitirá deshacer la subida", - "ue_ro": "tu acceso a esta carpeta es de solo lectura\n\n", - "ue_nl": "actualmente no has iniciado sesión", - "ue_la": "actualmente has iniciado sesión como \"{0}\"", - "ue_sr": "actualmente estás en modo de búsqueda de archivos\n\ncambia a modo de subida haciendo clic en la lupa 🔎 (junto al gran botón BUSCAR), e intenta subir de nuevo\n\nlo siento", - "ue_ta": "intenta subir de nuevo, ahora debería funcionar", - "ue_ab": "este archivo ya se está subiendo a otra carpeta, y esa subida debe completarse antes de que el archivo pueda ser subido a otro lugar.\n\nPuedes abortar y olvidar la subida inicial usando el 🧯 de arriba a la izquierda", - "ur_1uo": "OK: Archivo subido con éxito", - "ur_auo": "OK: Todos los {0} archivos subidos con éxito", - "ur_1so": "OK: Archivo encontrado en el servidor", - "ur_aso": "OK: Todos los {0} archivos encontrados en el servidor", - "ur_1un": "Subida fallida, lo siento", - "ur_aun": "Todas las {0} subidas fallaron, lo siento", - "ur_1sn": "El archivo NO se encontró en el servidor", - "ur_asn": "Los {0} archivos NO se encontraron en el servidor", - "ur_um": "Finalizado;\n{0} subidas OK,\n{1} subidas fallidas, lo siento", - "ur_sm": "Finalizado;\n{0} archivos encontrados en el servidor,\n{1} archivos NO encontrados en el servidor", + "u_https1": "sinun kannattaisi", + "u_https2": "vaihtaa https:ään", + "u_https3": "paremman suorituskyvyn vuoksi", + "u_ancient": 'selaimesi on ns. vaikuttavan ikivanha -- kannattais varmaan käyttää bup:ia tän sijaan', + "u_nowork": "tarvitaan firefox 53+ tai chrome 57+ tai iOS 11+", + "tail_2old": "tarvitaan firefox 105+ tai chrome 71+ tai iOS 14.5+", + "u_nodrop": 'selaimesi on liian vanha vedä-ja-pudota lataamiseen', + "u_notdir": "tuo ei ole hakemisto!\n\nselaimesi on liian vanha,\nkokeile sen sijaan 'vedä-pudota'-tekniikkaa.", + "u_uri": "'vedä-pudottaaksesi' kuvia muista selainikkunoista,\npudota se isoon latausnapppiin", + "u_enpot": 'vaihda peruna UI:hin (voi parantaa latausnopeutta)', + "u_depot": 'vaihda ylelliseen UI:hin (voi vähentää latausnopeutta)', + "u_gotpot": 'vaihdetaan peruna UI:hin paremman latausnopeuden vuoksi,\n\ntee miten lystäät, jos ei kelpaa!', + "u_pott": "

tiedostot:   {0} valmis,   {1} epäonnistui,   {2} kiireinen,   {3} jonossa

", + "u_ever": "tämä on peruslatain; up2k tarvitsee vähintään
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'tämä on peruslatain; up2k on parempi', + "u_uput": 'optimoi nopeuteen (ohita tarkistussumma)', + "u_ewrite": 'sinulla ei ole move-oikeutta tähän kansioon', + "u_eread": 'sinulla ei ole read-oikeutta tähän kansioon', + "u_enoi": 'tiedostohaku ei ole käytössä palvelimen asetuksissa', + "u_enoow": "ylikirjoitus ei toimi täällä; tarvitaan "Delete"-oikeus", + "u_badf": 'Nämä {0} tiedostoa ({1} yhteensä) ohitettiin, mahdollisesti tiedostojärjestelmän oikeuksien vuoksi:\n\n', + "u_blankf": 'Nämä {0} tiedostoa ({1} yhteensä) ovat tyhjiä; ladataanko ne silti?\n\n', + "u_applef": 'Nämä {0} tiedostoa ({1} yhteensä) ovat todennäköisesti ei-toivottuja;\nPaina OK/Enter OHITTAAKSESI seuraavat tiedostot,\nPaina Cancel/ESC jos ET halua sulkea pois, ja LATAA nekin:\n\n', + "u_just1": '\nEhkä toimii paremmin jos valitset vain yhden tiedoston', + "u_ff_many": "jos käytät Linux / MacOS / Android, niin tämä määrä tiedostoja saattaa kaataa Firefoxin!\njos niin käy, kokeile uudelleen (tai käytä Chromea).", + "u_up_life": "Tämä lataus poistetaan palvelimelta\n{0} sen valmistumisen jälkeen", + "u_asku": 'lataa nämä {0} tiedostoa kohteeseen {1}', + "u_unpt": "voit perua / poistaa tämän latauksen käyttämällä vasemmalla ylhäällä olevaa 🧯", + "u_bigtab": 'näytetään {0} tiedostoa\n\ntämä voi kaataa selaimesi, oletko varma?', + "u_scan": 'Skannataan tiedostoja...', + "u_dirstuck": 'hakemistoiteraattori jumittui yrittäessään käyttää seuraavia {0} kohdetta; ohitetaan:', + "u_etadone": 'Valmis ({0}, {1} tiedostoa)', + "u_etaprep": '(valmistellaan latausta)', + "u_hashdone": 'hajautus valmis', + "u_hashing": 'hajautus', + "u_hs": 'kätellään...', + "u_started": "tiedostoja ladataan nyt; tsekkaa [🚀]", + "u_dupdefer": "duplikaatti; käsitellään kaikkien muiden tiedostojen jälkeen", + "u_actx": "klikkaa tätä tekstiä estääksesi suorituskyvyn
heikkenemisen vaihtaessasi muihin ikkunoihin/välilehtiin", + "u_fixed": "OK!  Hommat hoidossa 👍", + "u_cuerr": "chunk {0} / {1} lataus epäonnistui;\ntuskin haittaa, jatketaan\n\ntiedosto: {2}", + "u_cuerr2": "palvelin hylkäsi latauksen (chunk {0} / {1});\nyritetään myöhemmin uudelleen\n\ntiedosto: {2}\n\nvirhe ", + "u_ehstmp": "yritetään uudelleen; katso oikealta alhaalta", + "u_ehsfin": "palvelin hylkäsi pyynnön viimeistellä lataus; yritetään uudelleen...", + "u_ehssrch": "palvelin hylkäsi pyynnön suorittaa haku; yritetään uudelleen...", + "u_ehsinit": "palvelin hylkäsi pyynnön aloittaa lataus; yritetään uudelleen...", + "u_eneths": "verkkovirhe latauksen kättelyssä; yritetään uudelleen...", + "u_enethd": "verkkovirhe kohteen olemassaolon testauksessa; yritetään uudelleen...", + "u_cbusy": "odotetaan palvelimen luottavan meihin taas verkko-ongelman jälkeen...", + "u_ehsdf": "palvelimen levytila loppui!\n\nyritetään jatkuvasti, siinä tapauksessa että joku\nvapauttaa tarpeeksi tilaa jatkamiseen", + "u_emtleak1": "näyttää siltä että selaimessasi saattaa olla muistivuoto;\nole hyvä ja", + "u_emtleak2": ' vaihda https:ään (suositeltu) tai ', + "u_emtleak3": ' ', + "u_emtleakc": 'kokeile seuraavaa:\n
  • paina F5 päivittääksesi sivun
  • sitten poista käytöstä  mt  nappi  ⚙️ asetuksissa
  • ja kokeile latausta uudelleen
Lataukset ovat hieman hitaampia, minkäs teet.\nSori siitä!\n\nPS: chrome v107 sisältää bugfixin tätä varten', + "u_emtleakf": 'kokeile seuraavaa:\n
  • paina F5 päivittääksesi sivun
  • sitten ota käyttöön 🥔 (peruna) lataus UI:ssa
  • ja kokeile latausta uudelleen
\nPS: firefox toivottavasti saa kerättyä itsensä kasaan jossain vaiheessa', + "u_s404": "ei löydy palvelimelta", + "u_expl": "selitä", + "u_maxconn": "useimmat selaimet rajoittavat tämän 6:een, mutta firefox antaa nostaa sitä connections-per-server asetuksella about:config:issa", + "u_tu": '

VAROITUS: turbo päällä,  asiakasohjelma ei välttämättä huomaa jatkaa keskeneräisiä latauksia; katso turbo-napin vihje

', + "u_ts": '

VAROITUS: turbo päällä,  hakutulokset voivat olla vääriä; katso turbo-napin vihje

', + "u_turbo_c": "turbo on poistettu käytöstä palvelimen asetuksissa", + "u_turbo_g": "poistetaan turbo käytöstä koska sinulla ei ole\nhakemistolistausoikeuksia tässä asemassa", + "u_life_cfg": 'automaattinen poisto min kuluttua (tai tuntia)', + "u_life_est": 'lataus poistetaan ---', + "u_life_max": 'tämä kansio pakottaa\nmaksimi elinajan {0}', + "u_unp_ok": 'unpost on sallittu {0}', + "u_unp_ng": 'unpost EI ole sallittu', + "ue_ro": 'sinulla on vain read-oikeus tähän kansioon\n\n', + "ue_nl": 'et ole tällä hetkellä kirjautunut sisään', + "ue_la": 'olet tällä hetkellä kirjautunut sisään nimellä "{0}"', + "ue_sr": 'olet tällä hetkellä tiedostohaku-tilassa\n\nvaihda lataus-tilaan klikkaamalla suurennuslasia 🔎 (suuren HAKU napin vieressä), ja yritä latausta uudelleen\n\npahoittelen', + "ue_ta": 'yritä latausta uudelleen, sen pitäisi toimia nyt', + "ue_ab": "tätä tiedostoa ladataan jo toiseen kansioon, ja se lataus täytyy suorittaa loppuun ennen kuin tiedostoa voidaan ladata muualle.\n\nVoit keskeyttää ja unohtaa alkuperäisen latauksen käyttämällä vasemmalla ylhäällä olevaa 🧯", + "ur_1uo": "OK: Tiedosto ladattu onnistuneesti", + "ur_auo": "OK: Kaikki {0} tiedostoa ladattu onnistuneesti", + "ur_1so": "OK: Tiedosto löytyi palvelimelta", + "ur_aso": "OK: Kaikki {0} tiedostoa löytyi palvelimelta", + "ur_1un": "Lataus epäonnistui, pahoittelen", + "ur_aun": "Kaikki {0} latausta epäonnistui, pahoittelen", + "ur_1sn": "Tiedostoa EI löytynyt palvelimelta", + "ur_asn": "{0} tiedostoa EI löytynyt palvelimelta", + "ur_um": "Valmis;\n{0} latausta OK,\n{1} latausta epäonnistui, pahoittelen", + "ur_sm": "Valmis;\n{0} tiedostoa löytyi palvelimelta,\n{1} tiedostoa EI löytynyt palvelimelta", - "lang_set": "¿refrescar para que el cambio surta efecto?" + "lang_set": "päivitetäänkö sivu muutoksen voimaansaattamiseksi?", }, "rus": { "tt": "Русский", @@ -3762,6 +3763,631 @@ var Ls = { "lang_set": "перезагрузить страницу, чтобы применить изменения?", }, + "spa": { + "tt": "Español", + + "cols": { + "c": "acciones", + "dur": "duración", + "q": "calidad / bitrate", + "Ac": "códec de audio", + "Vc": "códec de vídeo", + "Fmt": "formato / contenedor", + "Ahash": "checksum de audio", + "Vhash": "checksum de vídeo", + "Res": "resolución", + "T": "tipo de archivo", + "aq": "calidad de audio / bitrate", + "vq": "calidad de vídeo / bitrate", + "pixfmt": "submuestreo / estructura de píxel", + "resw": "resolución horizontal", + "resh": "resolución vertical", + "chs": "canales de audio", + "hz": "frecuencia de muestreo" + }, + + "hks": [ + [ + "varios", + ["ESC", "cerrar varias cosas"], + + "gestor de archivos", + ["G", "alternar vista de lista / cuadrícula"], + ["T", "alternar miniaturas / iconos"], + ["⇧ A/D", "tamaño de miniatura"], + ["ctrl-K", "eliminar seleccionados"], + ["ctrl-X", "cortar selección al portapapeles"], + ["ctrl-C", "copiar selección al portapapeles"], + ["ctrl-V", "pegar (mover/copiar) aquí"], + ["Y", "descargar seleccionados"], + ["F2", "renombrar seleccionados"], + + "selección en lista de archivos", + ["space", "alternar selección de archivo"], + ["↑/↓", "mover cursor de selección"], + ["ctrl ↑/↓", "mover cursor y vista"], + ["⇧ ↑/↓", "seleccionar anterior/siguiente archivo"], + ["ctrl-A", "seleccionar todos los archivos / carpetas"] + ], [ + "navegación", + ["B", "alternar breadcrumbs / panel de navegación"], + ["I/K", "anterior/siguiente carpeta"], + ["M", "carpeta de nivel superior (o contraer actual)"], + ["V", "alternar carpetas / archivos en panel de navegación"], + ["A/D", "tamaño del panel de navegación"] + ], [ + "reproductor de audio", + ["J/L", "anterior/siguiente canción"], + ["U/O", "saltar 10s atrás/adelante"], + ["0..9", "saltar a 0%..90%"], + ["P", "reproducir/pausar (también inicia)"], + ["S", "seleccionar canción en reproducción"], + ["Y", "descargar canción"] + ], [ + "visor de imágenes", + ["J/L, ←/→", "anterior/siguiente imagen"], + ["Home/End", "primera/última imagen"], + ["F", "pantalla completa"], + ["R", "rotar en sentido horario"], + ["⇧ R", "rotar en sentido antihorario"], + ["S", "seleccionar imagen"], + ["Y", "descargar imagen"] + ], [ + "reproductor de vídeo", + ["U/O", "saltar 10s atrás/adelante"], + ["P/K/Space", "reproducir/pausar"], + ["C", "continuar con el siguiente"], + ["V", "bucle"], + ["M", "silenciar"], + ["[ y ]", "establecer intervalo de bucle"] + ], [ + "visor de texto", + ["I/K", "anterior/siguiente archivo"], + ["M", "cerrar archivo"], + ["E", "editar archivo"], + ["S", "seleccionar archivo (para cortar/copiar/renombrar)"] + ] + ], + + "m_ok": "Aceptar", + "m_ng": "Cancelar", + + "enable": "Activar", + "danger": "PELIGRO", + "clipped": "copiado al portapapeles", + + "ht_s1": "segundo", + "ht_s2": "segundos", + "ht_m1": "minuto", + "ht_m2": "minutos", + "ht_h1": "hora", + "ht_h2": "horas", + "ht_d1": "día", + "ht_d2": "días", + "ht_and": " y ", + + "goh": "panel de control", + "gop": 'hermano anterior">anterior', + "gou": 'carpeta de nivel superior">subir', + "gon": 'siguiente carpeta">siguiente', + "logout": "Cerrar sesión ", + "access": " acceso", + "ot_close": "cerrar submenú", + "ot_search": "buscar archivos por atributos, ruta / nombre, etiquetas de música, o cualquier combinación$N$N<code>foo bar</code> = debe contener «foo» y «bar»,$N<code>foo -bar</code> = debe contener «foo» pero no «bar»,$N<code>^yana .opus$</code> = empieza con «yana» y es un archivo «opus»$N<code>"try unite"</code> = contiene exactamente «try unite»$N$Nel formato de fecha es iso-8601, como$N<code>2009-12-31</code> o <code>2020-09-12 23:30:00</code>", + "ot_unpost": "dessubir: elimina tus subidas recientes, o aborta las inacabadas", + "ot_bup": "bup: uploader básico, soporta hasta netscape 4.0", + "ot_mkdir": "mkdir: crear un nuevo directorio", + "ot_md": "new-md: crear un nuevo documento markdown", + "ot_msg": "msg: enviar un mensaje al registro del servidor", + "ot_mp": "opciones del reproductor multimedia", + "ot_cfg": "opciones de configuración", + "ot_u2i": "up2k: subir archivos (si tienes acceso de escritura) o cambiar a modo de búsqueda para ver si existen en el servidor$N$Nlas subidas se pueden reanudar, usan múltiples hilos y conservan la fecha de los archivos, pero consume más CPU que [🎈]  (el uploader básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", + "ot_u2w": "up2k: subir archivos con soporte para reanudación (cierra tu navegador y arrastra los mismos archivos más tarde)$N$NMultihilo y conserva las fechas de los archivos, pero usa más CPU que [🎈]  (el uploader básico)

¡Durante las subidas, este icono se convierte en un indicador de progreso!", + "ot_noie": "Por favor, usa Chrome / Firefox / Edge", + + "ab_mkdir": "crear directorio", + "ab_mkdoc": "nuevo documento markdown", + "ab_msg": "enviar msg al registro del servidor", + + "ay_path": "saltar a carpetas", + "ay_files": "saltar a archivos", + + "wt_ren": "renombrar elementos seleccionados$NAtajo: F2", + "wt_del": "eliminar elementos seleccionados$NAtajo: ctrl-K", + "wt_cut": "cortar elementos seleccionados <small>(luego pegar en otro lugar)</small>$NAtajo: ctrl-X", + "wt_cpy": "copiar elementos seleccionados al portapapeles$N(para pegarlos en otro lugar)$NAtajo: ctrl-C", + "wt_pst": "pegar una selección previamente cortada / copiada$NAtajo: ctrl-V", + "wt_selall": "seleccionar todos los archivos$NAtajo: ctrl-A (con un archivo con foco)", + "wt_selinv": "invertir selección", + "wt_zip1": "descargar esta carpeta como un archivo comprimido", + "wt_selzip": "descargar selección como archivo comprimido", + "wt_seldl": "descargar selección como archivos separados$NAtajo: Y", + "wt_npirc": "copiar información de pista en formato IRC", + "wt_nptxt": "copiar información de pista en texto plano", + "wt_m3ua": "añadir a lista m3u (haz clic en 📻copiar después)", + "wt_m3uc": "copiar lista m3u al portapapeles", + "wt_grid": "alternar vista de cuadrícula / lista$NAtajo: G", + "wt_prev": "pista anterior$NAtajo: J", + "wt_play": "reproducir / pausar$NAtajo: P", + "wt_next": "siguiente pista$NAtajo: L", + + "ul_par": "subidas paralelas:", + "ut_rand": "aleatorizar nombres de archivo", + "ut_u2ts": 'copiar la fecha de última modificación$Nde tu sistema de archivos al servidor">📅', + "ut_ow": "sobrescribir archivos existentes en el servidor?$N🛡️: nunca (generará un nuevo nombre de archivo en su lugar)$N🕒: sobrescribir si el archivo del servidor es más antiguo que el tuyo$N♻️: siempre sobrescribir si los archivos son diferentes", + "ut_mt": "continuar generando hashes de otros archivos mientras se sube$N$Nquizás desactivar si tu CPU o HDD es un cuello de botella", + "ut_ask": 'pedir confirmación antes de iniciar la subida">💭', + "ut_pot": "mejorar la velocidad de subida en dispositivos lentos$Nsimplificando la interfaz de usuario", + "ut_srch": "no subir, en su lugar comprobar si los archivos ya $N existen en el servidor (escaneará todas las carpetas que puedas leer)", + "ut_par": "pausar subidas poniéndolo a 0$N$Naumentar si tu conexión es lenta / de alta latencia$N$Nmantener en 1 en LAN o si el HDD del servidor es un cuello de botella", + "ul_btn": "arrastra archivos / carpetas
aquí (o haz clic)", + "ul_btnu": "S U B I R", + "ul_btns": "B U S C A R", + + "ul_hash": "hash", + "ul_send": "envio", + "ul_done": "hecho", + "ul_idle1": "aún no hay subidas en cola", + "ut_etah": "velocidad media de <em>hashing</em>, y tiempo estimado para finalizar", + "ut_etau": "velocidad media de <em>subida</em> y tiempo estimado para finalizar", + "ut_etat": "velocidad media <em>total</em> y tiempo estimado para finalizar", + + "uct_ok": "completado con éxito", + "uct_ng": "fallido: error / rechazado / no encontrado", + "uct_done": "éxitos y fallos combinados", + "uct_bz": "generando hash o subiendo", + "uct_q": "inactivo, pendiente", + + "utl_name": "nombre de archivo", + "utl_ulist": "lista", + "utl_ucopy": "copiar", + "utl_links": "enlaces", + "utl_stat": "estado", + "utl_prog": "progreso", + + "utl_404": "404", + "utl_err": "ERROR", + "utl_oserr": "Error-SO", + "utl_found": "encontrado", + "utl_defer": "posponer", + "utl_yolo": "YOLO", + "utl_done": "hecho", + + "ul_flagblk": "los archivos se añadieron a la cola
sin embargo, hay un up2k ocupado en otra pestaña del navegador,
esperando a que termine primero", + "ul_btnlk": "la configuración del servidor ha bloqueado esta opción en este estado", + + "udt_up": "Subir", + "udt_srch": "Buscar", + "udt_drop": "suéltalo aquí", + + "u_nav_m": "
vale, ¿qué tienes?
Intro = Archivos (uno o más)\nESC = Una carpeta (incluyendo subcarpetas)", + "u_nav_b": "ArchivosUna carpeta", + + "cl_opts": "opciones", + "cl_themes": "tema", + "cl_langs": "idioma", + "cl_ziptype": "descarga de carpeta", + "cl_uopts": "opciones up2k", + "cl_favico": "favicon", + "cl_bigdir": "directorios grandes", + "cl_hsort": "#ordenar", + "cl_keytype": "notación musical", + "cl_hiddenc": "columnas ocultas", + "cl_hidec": "ocultar", + "cl_reset": "restablecer", + "cl_hpick": "toca en las cabeceras de columna para ocultarlas en la tabla de abajo", + "cl_hcancel": "ocultación de columna cancelada", + + "ct_grid": '田 cuadrícula', + "ct_ttips": '◔ ◡ ◔">ℹ️ tooltips', + "ct_thumb": 'en vista de cuadrícula, alternar iconos o miniaturas$NAtajo: T">🖼️ miniaturas', + "ct_csel": 'usa CTRL y SHIFT para seleccionar archivos en la vista de cuadrícula">sel', + "ct_ihop": 'al cerrar el visor de imágenes, desplazarse hasta el último archivo visto">g⮯', + "ct_dots": 'mostrar archivos ocultos (si el servidor lo permite)">archivos ocultos', + "ct_qdel": 'al eliminar archivos, pedir confirmación solo una vez">elim. rápida', + "ct_dir1st": 'ordenar carpetas antes que archivos">📁 primero', + "ct_nsort": 'orden natural (para nombres de archivo con dígitos iniciales)">ord. natural', + "ct_readme": 'mostrar README.md en los listados de carpetas">📜 léeme', + "ct_idxh": 'mostrar index.html en lugar del listado de carpetas">htm', + "ct_sbars": 'mostrar barra lateral">⟊', + + "cut_umod": 'si un archivo ya existe en el servidor, actualiza la fecha de última modificación del servidor para que coincida con tu archivo local (requiere permisos de escritura+eliminación)">re📅', + + "cut_turbo": 'el botón yolo, probablemente NO quieras activarlo:$N$Núsalo si estabas subiendo una gran cantidad de archivos y tuviste que reiniciar por alguna razón, y quieres continuar la subida lo antes posible$N$Nesto reemplaza la comprobación de hash por un simple "¿tiene este el mismo tamaño de archivo en el servidor?" así que si el contenido del archivo es diferente, NO se subirá$N$Ndeberías desactivar esto cuando la subida termine, y luego "subir" los mismos archivos de nuevo para que el cliente los verifique">turbo', + + "cut_datechk": 'no tiene efecto a menos que el botón turbo esté activado$N$Nreduce el factor yolo en una pequeña cantidad; comprueba si las fechas de los archivos en el servidor coinciden con las tuyas$N$Nteóricamente debería detectar la mayoría de las subidas inacabadas / corruptas, pero no es un sustituto de hacer una pasada de verificación con el turbo desactivado después">verif. fecha', + + "cut_u2sz": "tamaño (en MiB) de cada trozo de subida; los valores grandes vuelan mejor a través del atlántico. Prueba valores bajos en conexiones muy poco fiables", + + "cut_flag": "asegura que solo una pestaña esté subiendo a la vez $N -- otras pestañas también deben tener esto activado $N -- solo afecta a pestañas en el mismo dominio", + + "cut_az": "subir archivos en orden alfabético, en lugar de los más pequeños primero$N$Nel orden alfabético puede facilitar la detección visual de si algo salió mal en el servidor, pero hace la subida ligeramente más lenta en fibra / LAN", + + "cut_nag": "notificación del SO cuando la subida se complete$N(solo si el navegador o la pestaña no están activos)", + "cut_sfx": "alerta sonora cuando la subida se complete$N(solo si el navegador o la pestaña no están activos)", + + "cut_mt": 'usar multithreading para acelerar el hashing de archivos$N$Nesto usa web-workers y requiere$Nmás RAM (hasta 512 MiB extra)$N$Nhace https un 30% más rápido, http 4.5x más rápido">mt', + + "cut_wasm": 'usar wasm en lugar del hasher incorporado del navegador; mejora la velocidad en navegadores basados en chrome pero aumenta la carga de la CPU, y muchas versiones antiguas de chrome tienen errores que hacen que el navegador consuma toda la RAM y se bloquee si esto está activado">wasm', + + "cft_text": "texto del favicon (dejar en blanco y refrescar para desactivar)", + "cft_fg": "color de primer plano", + "cft_bg": "color de fondo", + + "cdt_lim": "número máximo de archivos a mostrar en una carpeta", + "cdt_ask": "al llegar al final,$Nen lugar de cargar más archivos,$Npreguntar qué hacer", + "cdt_hsort": "cuántas reglas de ordenación (<code>,sorthref</code>) incluir en las URLs de medios. Ponerlo a 0 también ignorará las reglas de ordenación incluidas en los enlaces de medios al hacer clic en ellos", + + "tt_entree": "mostrar panel de navegación (barra lateral con árbol de directorios)$NAtajo: B", + "tt_detree": "mostrar breadcrumbs$NAtajo: B", + "tt_visdir": "desplazarse a la carpeta seleccionada", + "tt_ftree": "alternar árbol de carpetas / archivos de texto$NAtajo: V", + "tt_pdock": "mostrar carpetas de niveles superiores en un panel acoplado en la parte superior", + "tt_dynt": "crecimiento automático a medida que el árbol se expande", + "tt_wrap": "ajuste de línea", + "tt_hover": "revelar líneas que se desbordan al pasar el ratón$N( rompe el desplazamiento a menos que el $N  cursor esté en el margen izquierdo )", + + "ml_pmode": "al final de la carpeta...", + "ml_btns": "acciones", + "ml_tcode": "transcodificar", + "ml_tcode2": "transcodificar a", + "ml_tint": "tinte", + "ml_eq": "ecualizador de audio", + "ml_drc": "compresor de rango dinámico", + + "mt_loop": 'poner en bucle/repetir una canción">🔁', + "mt_one": 'parar después de una canción">1️⃣', + "mt_shuf": 'reproducir aleatoriamente las canciones en cada carpeta">🔀', + "mt_aplay": 'reproducir automaticamente si hay un ID de canción en el enlace en el que hiciste clic para acceder al servidor$N$Ndesactivar esto también evitará que la URL de la página se actualice con IDs de canción al reproducir música, para prevenir la reproducción automática si se pierden estos ajustes pero la URL permanece">a▶', + "mt_preload": 'empezar a cargar la siguiente canción cerca del final para una reproducción sin pausas">precarga', + "mt_prescan": 'ir a la siguiente carpeta antes de que la última canción$Ntermine, manteniendo contento al navegador$Npara que no detenga la reproducción">nav', + "mt_fullpre": 'intentar precargar la canción entera;$N✅ activar en conexiones inestables,$N❌ desactivar probablemente en conexiones lentas">completa', + "mt_fau": 'en teléfonos, evitar que la música se detenga si la siguiente canción no se precarga lo suficientemente rápido (puede causar fallos en la visualización de etiquetas)">☕️', + "mt_waves": 'barra de búsqueda con forma de onda:$Nmostrar la amplitud del audio en la barra de progreso">~s', + "mt_npclip": 'mostrar botones para copiar al portapapeles la canción actual">/np', + "mt_m3u_c": 'mostrar botones para copiar al portapapeles las$Ncanciones seleccionadas como entradas de lista m3u8">📻', + "mt_octl": 'integración con SO (teclas multimedia / OSD)">ctl-so', + "mt_oseek": 'permitir buscar a través de la integración con el SO$N$Nnota: en algunos dispositivos (iPhones),$Nesto reemplaza el botón de siguiente canción">búsqueda', + "mt_oscv": 'mostrar carátula del álbum en OSD">arte', + "mt_follow": 'mantener la pista en reproducción visible en pantalla">🎯', + "mt_compact": 'controles compactos">⟎', + "mt_uncache": 'limpiar caché  (prueba esto si tu navegador guardó en caché$Nuna copia rota de una canción que se niega a reproducir)">limpiar caché', + "mt_mloop": 'repetir la carpeta actual">🔁 bucle', + "mt_mnext": 'cargar la siguiente carpeta y continuar">📂 sig', + "mt_mstop": 'detener reproducción">⏸ parar', + "mt_cflac": 'convertir flac / wav a opus">flac', + "mt_caac": 'convertir aac / m4a a opus">aac', + "mt_coth": 'convertir todos los demás (no mp3) a opus">oth', + "mt_c2opus": 'la mejor opción para ordenadores, portátiles, android">opus', + "mt_c2owa": 'opus-weba, para iOS 17.5 y superior">owa', + "mt_c2caf": 'opus-caf, para iOS 11 a 17">caf', + "mt_c2mp3": 'usar en dispositivos muy antiguos">mp3', + "mt_c2ok": "bien, buena elección", + "mt_c2nd": "ese no es el formato de salida recomendado para tu dispositivo, pero está bien", + "mt_c2ng": "tu dispositivo no parece soportar este formato de salida, pero intentémoslo de todas formas", + "mt_xowa": "hay errores en iOS que impiden la reproducción en segundo plano con este formato; por favor, usa caf o mp3 en su lugar", + "mt_tint": "nivel de fondo (0-100) en la barra de búsqueda$Npara hacer el buffering menos molesto", + "mt_eq": "activa el ecualizador y el control de ganancia;$N$Nganancia <code>0</code> = volumen estándar 100% (sin modificar)$N$Nancho <code>1  </code> = estéreo estándar (sin modificar)$Nancho <code>0.5</code> = 50% de crossfeed izq-der$Nancho <code>0  </code> = mono$N$Nganancia <code>-0.8</code> y ancho <code>10</code> = eliminación de voz :^)$N$Nactivar el ecualizador hace que los álbumes sin pausas sean completamente sin pausas, así que déjalo activado con todos los valores a cero (excepto ancho = 1) si eso te importa", + "mt_drc": "activa el compresor de rango dinámico (aplanador de volumen / brickwaller); también activará el EQ para equilibrar el espagueti, así que pon todos los campos de EQ excepto 'ancho' a 0 si no lo quieres$N$Nbaja el volumen del audio por encima de THRESHOLD dB; por cada RATIO dB pasado THRESHOLD hay 1 dB de salida, así que los valores por defecto de umbral -24 y ratio 12 significan que nunca debería sonar más fuerte de -22 dB y es seguro aumentar la ganancia del ecualizador a 0.8, o incluso 1.8 con ATK 0 y un RLS enorme como 90 (solo funciona en firefox; RLS es máx. 1 en otros navegadores)$N$N(ver wikipedia, lo explican mucho mejor)", + + "mb_play": "reproducir", + "mm_hashplay": "¿reproducir este archivo de audio?", + "mm_m3u": "pulsa Intro/Aceptar para Reproducir\npulsa ESC/Cancelar para Editar", + "mp_breq": "se necesita firefox 82+ o chrome 73+ o iOS 15+", + "mm_bload": "cargando...", + "mm_bconv": "convirtiendo a {0}, por favor espera...", + "mm_opusen": "tu navegador no puede reproducir archivos aac / m4a;\nse ha activado la transcodificación a opus", + "mm_playerr": "fallo de reproducción: ", + "mm_eabrt": "El intento de reproducción fue cancelado", + "mm_enet": "Tu conexión a internet es inestable", + "mm_edec": "¿Este archivo está supuestamente corrupto?", + "mm_esupp": "Tu navegador no entiende este formato de audio", + "mm_eunk": "Error desconocido", + "mm_e404": "No se pudo reproducir el audio; error 404: Archivo no encontrado.", + "mm_e403": "No se pudo reproducir el audio; error 403: Acceso denegado.\n\nIntenta pulsar F5 para recargar, quizás se cerró tu sesión", + "mm_e500": "No se pudo reproducir el audio; error 500: Revisa los registros del servidor.", + "mm_e5xx": "No se pudo reproducir el audio; error del servidor ", + "mm_nof": "no se encuentran más archivos de audio cerca", + "mm_prescan": "Buscando música para reproducir a continuación...", + "mm_scank": "Encontrada la siguiente canción:", + "mm_uncache": "caché limpiada; todas las canciones se volverán a descargar en la próxima reproducción", + "mm_hnf": "esa canción ya no existe", + + "im_hnf": "esa imagen ya no existe", + + "f_empty": "esta carpeta está vacía", + "f_chide": "esto ocultará la columna «{0}»\n\npuedes volver a mostrar las columnas en la pestaña de configuración", + "f_bigtxt": "este archivo pesa {0} MiB -- ¿realmente verlo como texto?", + "f_bigtxt2": "¿ver solo el final del archivo en su lugar? esto también activará el seguimiento, mostrando las líneas de texto recién añadidas en tiempo real", + "fbd_more": '
mostrando {0} de {1} archivos; mostrar {2} o mostrar todos
', + "fbd_all": '
mostrando {0} de {1} archivos; mostrar todos
', + "f_anota": "solo {0} de los {1} elementos fueron seleccionados;\npara seleccionar la carpeta completa, primero desplázate hasta el final", + + "f_dls": "los enlaces a archivos en la carpeta actual se han\nconvertido en enlaces de descarga", + + "f_partial": "Para descargar de forma segura un archivo que se está subiendo actualmente, por favor haz clic en el archivo con el mismo nombre, pero sin la extensión .PARTIAL. Por favor, pulsa CANCELAR o Escape para hacer esto.\n\nPulsar ACEPTAR o Intro ignorará esta advertencia y continuará descargando el archivo temporal .PARTIAL, lo que casi con toda seguridad te dará datos corruptos.", + + "ft_paste": "pegar {0} elementos$NAtajo: ctrl-V", + "fr_eperm": "no se puede renombrar:\nno tienes permiso de “mover” en esta carpeta", + "fd_eperm": "no se puede eliminar:\nno tienes permiso de “eliminar” en esta carpeta", + "fc_eperm": "no se puede cortar:\nno tienes permiso de “mover” en esta carpeta", + "fp_eperm": "no se puede pegar:\nno tienes permiso de “escribir” en esta carpeta", + "fr_emore": "selecciona al menos un elemento para renombrar", + "fd_emore": "selecciona al menos un elemento para eliminar", + "fc_emore": "selecciona al menos un elemento para cortar", + "fcp_emore": "selecciona al menos un elemento para copiar al portapapeles", + + "fs_sc": "compartir la carpeta en la que estás", + "fs_ss": "compartir los archivos seleccionados", + "fs_just1d": "no puedes seleccionar más de una carpeta,\no mezclar archivos y carpetas en una selección", + "fs_abrt": "❌ abortar", + "fs_rand": "🎲 nombre aleatorio", + "fs_go": "✅ crear enlace", + "fs_name": "nombre", + "fs_src": "origen", + "fs_pwd": "contraseña", + "fs_exp": "caducidad", + "fs_tmin": "minutos", + "fs_thrs": "horas", + "fs_tdays": "días", + "fs_never": "eterno", + "fs_pname": "nombre opcional del enlace; será aleatorio si se deja en blanco", + "fs_tsrc": "el archivo o carpeta a compartir", + "fs_ppwd": "contraseña opcional", + "fs_w8": "creando enlace...", + "fs_ok": "pulsa Intro/Aceptar para Copiar al Portapapeles\npulsa ESC/Cancelar para Cerrar", + + "frt_dec": "puede arreglar algunos casos de nombres de archivo rotos\">url-decode", + "frt_rst": "restaurar los nombres de archivo modificados a los originales\">↺ restablecer", + "frt_abrt": "abortar y cerrar esta ventana\">❌ cancelar", + "frb_apply": "APLICAR RENOMBRADO", + "fr_adv": "renombrado por lotes / metadatos / patrones\">avanzado", + "fr_case": "regex sensible a mayúsculas\">mayús", + "fr_win": "nombres seguros para windows; reemplaza <>:"\\|?* con caracteres japoneses de ancho completo\">win", + "fr_slash": "reemplaza / con un carácter que no cree nuevas carpetas\">sin /", + "fr_re": "patrón de búsqueda regex para aplicar a los nombres de archivo originales; los grupos de captura se pueden referenciar en el campo de formato de abajo como <code>(1)</code> y <code>(2)</code> y así sucesivamente", + "fr_fmt": "inspirado en foobar2000:$N<code>(title)</code> se reemplaza por el título de la canción,$N<code>[(artist) - ](title)</code> omite la parte [entre corchetes] si el artista está en blanco$N<code>$lpad((tn),2,0)</code> rellena el número de pista a 2 dígitos", + "fr_pdel": "eliminar", + "fr_pnew": "guardar como", + "fr_pname": "proporciona un nombre para tu nuevo preajuste", + "fr_aborted": "abortado", + "fr_lold": "nombre antiguo", + "fr_lnew": "nombre nuevo", + "fr_tags": "etiquetas para los archivos seleccionados (solo lectura, como referencia):", + "fr_busy": "renombrando {0} elementos...\n\n{1}", + "fr_efail": "fallo al renombrar:\n", + "fr_nchg": "{0} de los nuevos nombres fueron alterados debido a win y/o sin /\n\n¿Aceptar para continuar con estos nuevos nombres alterados?", + + "fd_ok": "eliminación correcta", + "fd_err": "fallo al eliminar:\n", + "fd_none": "no se eliminó nada; quizás bloqueado por la configuración del servidor (xbd)?", + "fd_busy": "eliminando {0} elementos...\n\n{1}", + "fd_warn1": "¿ELIMINAR estos {0} elementos?", + "fd_warn2": "¡Última oportunidad! No se puede deshacer. ¿Eliminar?", + + "fc_ok": "cortados {0} elementos", + "fc_warn": "cortados {0} elementos\n\npero: solo esta pestaña del navegador puede pegarlos\n(dado que la selección es absolutamente masiva)", + + "fcc_ok": "copiados {0} elementos al portapapeles", + "fcc_warn": "copiados {0} elementos al portapapeles\n\npero: solo esta pestaña del navegador puede pegarlos\n(dado que la selección es absolutamente masiva)", + + "fp_apply": "usar estos nombres", + "fp_ecut": "primero corta o copia algunos archivos / carpetas para pegar / mover\n\nnota: puedes cortar / pegar entre diferentes pestañas del navegador", + "fp_ename": "{0} elementos no se pueden mover aquí porque los nombres ya existen. Dales nuevos nombres abajo para continuar, o deja el nombre en blanco para omitirlos:", + "fcp_ename": "{0} elementos no se pueden copiar aquí porque los nombres ya existen. Dales nuevos nombres abajo para continuar, o deja el nombre en blanco para omitirlos:", + "fp_emore": "todavía quedan algunas colisiones de nombres por resolver", + "fp_ok": "movimiento correcto", + "fcp_ok": "copia correcta", + "fp_busy": "moviendo {0} elementos...\n\n{1}", + "fcp_busy": "copiando {0} elementos...\n\n{1}", + "fp_err": "fallo al mover:\n", + "fcp_err": "fallo al copiar:\n", + "fp_confirm": "¿mover estos {0} elementos aquí?", + "fcp_confirm": "¿copiar estos {0} elementos aquí?", + "fp_etab": "fallo al leer el portapapeles de otra pestaña del navegador", + "fp_name": "subiendo un archivo desde tu dispositivo. Dale un nombre:", + "fp_both_m": "
elige qué pegar
Intro = Mover {0} archivos desde «{1}»\nESC = Subir {2} archivos desde tu dispositivo", + "fcp_both_m": "
elige qué pegar
Intro = Copiar {0} archivos desde «{1}»\nESC = Subir {2} archivos desde tu dispositivo", + "fp_both_b": "MoverSubir", + "fcp_both_b": "CopiarSubir", + + "mk_noname": "escribe un nombre en el campo de texto de la izquierda antes de hacer eso :p", + + "tv_load": "Cargando documento de texto:\n\n{0}\n\n{1}% ({2} de {3} MiB cargados)", + "tv_xe1": "no se pudo cargar el archivo de texto:\n\nerror ", + "tv_xe2": "404, archivo no encontrado", + "tv_lst": "lista de archivos de texto en", + "tvt_close": "volver a la vista de carpetas$NAtajo: M (o Esc)\">❌ cerrar", + "tvt_dl": "descargar este archivo$NAtajo: Y\">💾 descargar", + "tvt_prev": "mostrar documento anterior$NAtajo: i\">⬆ ant", + "tvt_next": "mostrar siguiente documento$NAtajo: K\">⬇ sig", + "tvt_sel": "seleccionar archivo   ( para cortar / copiar / eliminar / ... )$NAtajo: S\">sel", + "tvt_edit": "abrir archivo en editor de texto$NAtajo: E\">✏️ editar", + "tvt_tail": "monitorizar cambios en el archivo; mostrar nuevas líneas en tiempo real\">📡 seguir", + "tvt_wrap": "ajuste de línea\">↵", + "tvt_atail": "bloquear el desplazamiento al final de la página\">⚓", + "tvt_ctail": "decodificar colores de terminal (códigos de escape ansi)\">🌈", + "tvt_ntail": "límite de historial (cuántos bytes de texto mantener cargados)", + + "m3u_add1": "canción añadida a la lista m3u", + "m3u_addn": "{0} canciones añadidas a la lista m3u", + "m3u_clip": "lista m3u copiada al portapapeles\n\ndebes crear un nuevo archivo de texto llamado algo.m3u y pegar la lista en ese documento; esto lo hará reproducible", + + "gt_vau": "no mostrar vídeos, solo reproducir el audio\">🎧", + "gt_msel": "activar selección de archivos; ctrl-clic en un archivo para anular$N$N<em>cuando está activo: doble clic en un archivo / carpeta para abrirlo</em>$N$NAtajo: S\">multiselección", + "gt_crop": "recortar miniaturas\">recortar", + "gt_3x": "miniaturas de alta resolución\">3x", + "gt_zoom": "zoom", + "gt_chop": "recortar", + "gt_sort": "ordenar por", + "gt_name": "nombre", + "gt_sz": "tamaño", + "gt_ts": "fecha", + "gt_ext": "tipo", + "gt_c1": "truncar más los nombres de archivo (mostrar menos)", + "gt_c2": "truncar menos los nombres de archivo (mostrar más)", + + "sm_w8": "buscando...", + "sm_prev": "los resultados de búsqueda a continuación son de una consulta anterior:\n ", + "sl_close": "cerrar resultados de búsqueda", + "sl_hits": "mostrando {0} resultados", + "sl_moar": "cargar más", + + "s_sz": "tamaño", + "s_dt": "fecha", + "s_rd": "ruta", + "s_fn": "nombre", + "s_ta": "etiquetas", + "s_ua": "subido@", + "s_ad": "avanzado", + "s_s1": "MiB mínimo", + "s_s2": "MiB máximo", + "s_d1": "mín. iso8601", + "s_d2": "máx. iso8601", + "s_u1": "subido después de", + "s_u2": "y/o antes de", + "s_r1": "la ruta contiene   (separado por espacios)", + "s_f1": "el nombre contiene   (negar con -no)", + "s_t1": "las etiquetas contienen   (^=inicio, fin=$)", + "s_a1": "propiedades de metadatos específicas", + + "md_eshow": "no se puede renderizar ", + "md_off": "[📜léeme] desactivado en [⚙️] -- documento oculto", + + "badreply": "Fallo al procesar la respuesta del servidor", + + "xhr403": "403: Acceso denegado\n\nintenta pulsar F5, quizás se cerró tu sesión", + "xhr0": "desconocido (probablemente se perdió la conexión con el servidor, o el servidor está desconectado)", + "cf_ok": "perdón por eso -- la protección DD" + wah + "oS se activó\n\nlas cosas deberían reanudarse en unos 30 segundos\n\nsi no pasa nada, pulsa F5 para recargar la página", + "tl_xe1": "no se pudieron listar las subcarpetas:\n\nerror ", + "tl_xe2": "404: Carpeta no encontrada", + "fl_xe1": "no se pudieron listar los archivos en la carpeta:\n\nerror ", + "fl_xe2": "404: Carpeta no encontrada", + "fd_xe1": "no se pudo crear la subcarpeta:\n\nerror ", + "fd_xe2": "404: Carpeta de nivel superior no encontrada", + "fsm_xe1": "no se pudo enviar el mensaje:\n\nerror ", + "fsm_xe2": "404: Carpeta de nivel superior no encontrada", + "fu_xe1": "fallo al cargar la lista de deshacer del servidor:\n\nerror ", + "fu_xe2": "404: ¿Archivo no encontrado?", + + "fz_tar": "archivo gnu-tar sin comprimir (linux / mac)", + "fz_pax": "tar formato pax sin comprimir (más lento)", + "fz_targz": "gnu-tar con compresión gzip nivel 3$N$Nesto suele ser muy lento, así que$Nusa tar sin comprimir en su lugar", + "fz_tarxz": "gnu-tar con compresión xz nivel 1$N$Nesto suele ser muy lento, así que$Nusa tar sin comprimir en su lugar", + "fz_zip8": "zip con nombres de archivo utf8 (puede dar problemas en windows 7 y anteriores)", + "fz_zipd": "zip con nombres de archivo cp437 tradicionales, para software muy antiguo", + "fz_zipc": "cp437 con crc32 calculado tempranamente,$Npara MS-DOS PKZIP v2.04g (octubre 1993)$N(tarda más en procesar antes de que la descarga pueda empezar)", + + "un_m1": "puedes eliminar tus subidas recientes (o abortar las inacabadas) a continuación", + "un_upd": "actualizar", + "un_m4": "o compartir los archivos visibles a continuación:", + "un_ulist": "mostrar", + "un_ucopy": "copiar", + "un_flt": "filtro opcional:  la URL debe contener", + "un_fclr": "limpiar filtro", + "un_derr": "fallo al deshacer-eliminar:\n", + "un_f5": "algo se rompió, por favor intenta actualizar o pulsa F5", + "un_uf5": "lo siento pero tienes que refrescar la página (por ejemplo pulsando F5 o CTRL-R) antes de que esta subida pueda ser abortada", + "un_nou": "aviso: servidor demasiado ocupado para mostrar subidas inacabadas; haz clic en el enlace \"actualizar\" en un momento", + "un_noc": "aviso: la opción de deshacer subidas completadas no está activada/permitida en la configuración del servidor", + "un_max": "mostrando los primeros 2000 archivos (usa el filtro)", + "un_avail": "{0} subidas recientes se pueden eliminar
{1} inacabadas se pueden abortar", + "un_m2": "ordenado por tiempo de subida; más recientes primero:", + "un_no1": "¡pues no! ninguna subida es suficientemente reciente", + "un_no2": "¡pues no! ninguna subida que coincida con ese filtro es suficientemente reciente", + "un_next": "eliminar los siguientes {0} archivos a continuación", + "un_abrt": "abortar", + "un_del": "eliminar", + "un_m3": "cargando tus subidas recientes...", + "un_busy": "eliminando {0} archivos...", + "un_clip": "{0} enlaces copiados al portapapeles", + + "u_https1": "deberías", + "u_https2": "cambiar a https", + "u_https3": "para un mejor rendimiento", + "u_ancient": "tu navegador es impresionantemente antiguo -- quizás deberías usar bup en su lugar", + "u_nowork": "se necesita firefox 53+ o chrome 57+ o iOS 11+", + "tail_2old": "se necesita firefox 105+ o chrome 71+ o iOS 14.5+", + "u_nodrop": "tu navegador es demasiado antiguo para subir arrastrando y soltando", + "u_notdir": "¡eso no es una carpeta!\n\ntu navegador es demasiado antiguo,\npor favor intenta arrastrar y soltar en su lugar", + "u_uri": "para arrastrar y soltar imágenes desde otras ventanas del navegador,\npor favor suéltalas sobre el gran botón de subida", + "u_enpot": "cambiar a UI ligera (puede mejorar la velocidad de subida)", + "u_depot": "cambiar a UI elegante (puede reducir la velocidad de subida)", + "u_gotpot": "cambiando a la UI ligera para mejorar la velocidad de subida,\n\n¡siéntete libre de no estar de acuerdo y volver a cambiar!", + "u_pott": "

archivos:   {0} finalizados,   {1} fallidos,   {2} ocupados,   {3} en cola

", + "u_ever": "este es el uploader básico; up2k necesita al menos
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": "este es el uploader básico; up2k es mejor", + "u_uput": "optimizar para velocidad (omitir checksum)", + "u_ewrite": "no tienes acceso de escritura a esta carpeta", + "u_eread": "no tienes acceso de lectura a esta carpeta", + "u_enoi": "la búsqueda de archivos no está activada en la configuración del servidor", + "u_enoow": "sobrescribir no funcionará aquí; se necesita permiso de eliminación", + "u_badf": "Estos {0} archivos (de un total de {1}) se omitieron, posiblemente debido a permisos del sistema de archivos:\n\n", + "u_blankf": "Estos {0} archivos (de un total de {1}) están en blanco / vacíos; ¿subirlos de todos modos?\n\n", + "u_applef": "Estos {0} archivos (de un total de {1}) probablemente no son deseables;\nPulsa Aceptar/Intro para OMITIR los siguientes archivos,\nPulsa Cancelar/ESC para NO excluir, y SUBIR esos también:\n\n", + "u_just1": "\nQuizás funcione mejor si seleccionas solo un archivo", + "u_ff_many": "si usas Linux / MacOS / Android, esta cantidad de archivos podría bloquear Firefox!\nsi eso ocurre, por favor inténtalo de nuevo (o usa Chrome).", + "u_up_life": "Esta subida será eliminada del servidor\n{0} después de que se complete", + "u_asku": "subir estos {0} archivos a {1}", + "u_unpt": "puedes deshacer / eliminar esta subida usando el 🧯 de arriba a la izquierda", + "u_bigtab": "a punto de mostrar {0} archivos\n\nesto podría bloquear tu navegador, ¿estás seguro?", + "u_scan": "Escaneando archivos...", + "u_dirstuck": "el iterador de directorios se atascó intentando acceder a los siguientes {0} elementos; se omitirán:", + "u_etadone": "Hecho ({0}, {1} archivos)", + "u_etaprep": "(preparando para subir)", + "u_hashdone": "hashing completado", + "u_hashing": "hash", + "u_hs": "negociando...", + "u_started": "los archivos se están subiendo ahora; mira en [🚀]", + "u_dupdefer": "duplicado; se procesará después de todos los demás archivos", + "u_actx": "haz clic en este texto para evitar la pérdida de
rendimiento al cambiar a otras ventanas/pestañas", + "u_fixed": "¡OK!  Arreglado 👍", + "u_cuerr": "fallo al subir el trozo {0} de {1};\nprobablemente inofensivo, continuando\n\narchivo: {2}", + "u_cuerr2": "el servidor rechazó la subida (trozo {0} de {1});\nse reintentará más tarde\n\narchivo: {2}\n\nerror ", + "u_ehstmp": "se reintentará; mira abajo a la derecha", + "u_ehsfin": "el servidor rechazó la solicitud para finalizar la subida; reintentando...", + "u_ehssrch": "el servidor rechazó la solicitud para realizar la búsqueda; reintentando...", + "u_ehsinit": "el servidor rechazó la solicitud para iniciar la subida; reintentando...", + "u_eneths": "error de red al realizar la negociación de subida; reintentando...", + "u_enethd": "error de red al comprobar la existencia del destino; reintentando...", + "u_cbusy": "esperando a que el servidor vuelva a confiar en nosotros después de un fallo de red...", + "u_ehsdf": "¡el servidor se quedó sin espacio en disco!\n\nse seguirá reintentando, por si alguien\nlibera suficiente espacio para continuar", + "u_emtleak1": "parece que tu navegador podría tener una fuga de memoria;\npor favor", + "u_emtleak2": " cambia a https (recomendado) o ", + "u_emtleak3": " ", + "u_emtleakc": "prueba lo siguiente:\n
  • pulsa F5 para refrescar la página
  • luego desactiva el botón  mt  en los  ⚙️ ajustes
  • e intenta esa subida de nuevo
Las subidas serán un poco más lentas, pero bueno.\n¡Perdón por las molestias!\n\nPD: chrome v107 tiene una solución para esto", + "u_emtleakf": "prueba lo siguiente:\n
  • pulsa F5 para refrescar la página
  • luego activa 🥔 (ligera) en la interfaz de subida
  • e intenta esa subida de nuevo
\nPD: firefox con suerte tendrá una solución en algún momento", + "u_s404": "no encontrado en el servidor", + "u_expl": "explicar", + "u_maxconn": "la mayoría de los navegadores limitan esto a 6, pero firefox te permite aumentarlo con connections-per-server en about:config", + "u_tu": '

AVISO: turbo activado,  el cliente puede no detectar y reanudar subidas incompletas; ver tooltip del botón turbo

', + "u_ts": '

AVISO: turbo activado,  los resultados de búsqueda pueden ser incorrectos; ver tooltip del botón turbo

', + "u_turbo_c": "turbo está desactivado en la configuración del servidor", + "u_turbo_g": "desactivando turbo porque no tienes\nprivilegios para listar directorios en este volumen", + "u_life_cfg": 'autoeliminar después de min (o horas)', + "u_life_est": 'la subida se eliminará ---', + "u_life_max": "esta carpeta impone una\nvida máxima de {0}", + "u_unp_ok": "se permite deshacer la subida durante {0}", + "u_unp_ng": "NO se permitirá deshacer la subida", + "ue_ro": "tu acceso a esta carpeta es de solo lectura\n\n", + "ue_nl": "actualmente no has iniciado sesión", + "ue_la": "actualmente has iniciado sesión como \"{0}\"", + "ue_sr": "actualmente estás en modo de búsqueda de archivos\n\ncambia a modo de subida haciendo clic en la lupa 🔎 (junto al gran botón BUSCAR), e intenta subir de nuevo\n\nlo siento", + "ue_ta": "intenta subir de nuevo, ahora debería funcionar", + "ue_ab": "este archivo ya se está subiendo a otra carpeta, y esa subida debe completarse antes de que el archivo pueda ser subido a otro lugar.\n\nPuedes abortar y olvidar la subida inicial usando el 🧯 de arriba a la izquierda", + "ur_1uo": "OK: Archivo subido con éxito", + "ur_auo": "OK: Todos los {0} archivos subidos con éxito", + "ur_1so": "OK: Archivo encontrado en el servidor", + "ur_aso": "OK: Todos los {0} archivos encontrados en el servidor", + "ur_1un": "Subida fallida, lo siento", + "ur_aun": "Todas las {0} subidas fallaron, lo siento", + "ur_1sn": "El archivo NO se encontró en el servidor", + "ur_asn": "Los {0} archivos NO se encontraron en el servidor", + "ur_um": "Finalizado;\n{0} subidas OK,\n{1} subidas fallidas, lo siento", + "ur_sm": "Finalizado;\n{0} archivos encontrados en el servidor,\n{1} archivos NO encontrados en el servidor", + + "lang_set": "¿refrescar para que el cambio surta efecto?" + }, "ukr": { "tt": "Українська", @@ -4390,7 +5016,7 @@ var Ls = { }, }; -var LANGS = ["eng", "nor", "chi", "deu", "spa", "rus", "ukr"]; +var LANGS = ["eng", "nor", "chi", "deu", "fin", "rus", "spa", "ukr"]; if (window.langmod) langmod(); diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 22e32381..e55f7ff0 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -51,7 +51,6 @@ var Ls = { "ta2": "repeat to confirm new password:", "ta3": "found a typo; please try again", }, - "chi": { "a1": "更新", "b1": "你好   (你尚未登录)", @@ -94,7 +93,6 @@ var Ls = { "af1": "显示最近上传的文件", //m "ag1": "查看已知 IdP 用户", //m }, - "deu": { "a1": "Neu laden", "b1": "Tach, wie geht's?   (Du bist nicht angemeldet)", @@ -136,7 +134,48 @@ var Ls = { "ae1": "Aktive Downloads:", "af1": "Zeige neue Uploads", }, - + "fin": { + "a1": "päivitä", + "b1": "hei sie muukalainen   (et ole kirjautunut sisään)", + "c1": "kirjaudu ulos", + "d1": "tulosta pinojälki", + "d2": "näytä kaikkien aktiivisten säikeiden tila", + "e1": "päivitä konffit", + "e2": "lataa konfiguraatiotiedostot uudelleen (käyttäjätilit/asemat/asemaflagit),$Nja skannaa kaikki e2ds asemat uudelleen$N$Nhuom: kaikki global-asetuksiin$Ntehdyt muutokset vaativat täyden$Nuudelleenkäynnistyksen", + "f1": "voit selata:", + "g1": "voit ladata:", + "cc1": "muuta:", + "h1": "poista k304 käytöstä", + "i1": "ota k304 käyttöön", + "j1": "k304 katkaisee yhteytesi jokaisella HTTP 304:llä, mikä voi estää joitain bugisia välityspalvelimia jumittumasta/lopettamasta sivujen lataamista, mutta se myös vähentää suorituskykyä", + "k1": "nollaa asetukset", + "l1": "kirjaudu sisään:", + "m1": "tervetuloa takaisin,", + "n1": "404: ei löytynyt mitään  ┐( ´ -`)┌", + "o1": 'tai ehkä sinulla ei vain ole käyttöoikeuksia? kokeile salasanaa tai mene kotiin', + "p1": "403: pääsy kielletty  ~┻━┻", + "q1": 'kokeile salasanaa tai mene kotiin', + "r1": "mene kotiin", + ".s1": "uudelleenkartoita", + "t1": "toiminto", + "u2": "aika viimeisestä palvelimen kirjoituksesta$N( lataus / uudelleennimeäminen / tms. )$N$N17d = 17 päivää$N1h23 = 1 tunti 23 minuuttia$N4m56 = 4 minuuttia 56 sekuntia", + "v1": "yhdistä", + "v2": "käytä tätä palvelinta paikallisena kiintolevynä", + "w1": "vaihda https:ään", + "x1": "vaihda salasana", + "y1": "muokkaa jakoja", + "z1": "avaa tämä jako:", + "ta1": "täytä ensin uusi salasana", + "ta2": "toista vahvistaaksesi uuden salasanan:", + "ta3": "löytyi kirjoitusvirhe; yritä uudelleen", + "aa1": "saapuvat:", + "ab1": "poista no304 käytöstä", + "ac1": "ota no304 käyttöön", + "ad1": "no304:n lopettaa välimuistin käytön kokonaan; kokeile tätä jos k304 ei riittänyt. Tuhlaa valtavan määrän verkkoliikennettä!", + "ae1": "lähtevät:", + "af1": "näytä viimeaikaiset lataukset", + "ag1": "näytä tunnetut IdP-käyttäjät", + }, "spa": { "a1": "actualizar", "b1": "hola   (no has iniciado sesión)", From d0d2f206a9c0917eabd3f468fe7f0f7ec40121cd Mon Sep 17 00:00:00 2001 From: ed Date: Fri, 1 Aug 2025 17:49:23 +0000 Subject: [PATCH 049/174] log creator of new/blank markdown files too --- copyparty/httpcli.py | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 16f94a11..5ce45a38 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -3109,6 +3109,20 @@ class HttpCli(object): if "fperms" in vfs.flags: set_fperms(f, vfs.flags) + dbv, vrem = vfs.get_dbv(rem) + self.conn.hsrv.broker.say( + "up2k.hash_file", + dbv.realpath, + dbv.vpath, + dbv.flags, + vrem, + sanitized, + self.ip, + bos.stat(fn).st_mtime, + self.uname, + True, + ) + vpath = "{}/{}".format(self.vpath, sanitized).lstrip("/") self.redirect(vpath, "?edit") return True From ad23b253dc033ef19eace3816bc6976b1b38090b Mon Sep 17 00:00:00 2001 From: ed Date: Fri, 1 Aug 2025 17:55:34 +0000 Subject: [PATCH 050/174] add `--localtime` for ui; closes #312 --- copyparty/__main__.py | 1 + copyparty/authsrv.py | 1 + copyparty/svchub.py | 2 ++ copyparty/web/browser.js | 20 ++++++++++++++++---- copyparty/web/rups.html | 1 + copyparty/web/rups.js | 2 +- copyparty/web/shares.html | 1 + copyparty/web/shares.js | 2 +- copyparty/web/up2k.js | 6 +++--- copyparty/web/util.js | 25 +++++++++++++++++++++++++ tests/util.py | 2 +- 11 files changed, 53 insertions(+), 10 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 97944c13..609a534e 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1548,6 +1548,7 @@ def add_ui(ap, retry): ap2 = ap.add_argument_group('ui options') ap2.add_argument("--grid", action="store_true", help="show grid/thumbnails by default (volflag=grid)") ap2.add_argument("--gsel", action="store_true", help="select files in grid by ctrl-click (volflag=gsel)") + ap2.add_argument("--localtime", action="store_true", help="default to local timezone instead of UTC") ap2.add_argument("--lang", metavar="LANG", type=u, default="eng", help="language; one of the following: \033[32meng nor chi\033[0m") ap2.add_argument("--theme", metavar="NUM", type=int, default=0, help="default theme to use (0..7)") ap2.add_argument("--themes", metavar="NUM", type=int, default=8, help="number of themes installed") diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 28441114..030209fe 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -2775,6 +2775,7 @@ class AuthSrv(object): "dth3x": vf["th3x"], "dvol": self.args.au_vol, "idxh": int(self.args.ih), + "dutc": not self.args.localtime, "themes": self.args.themes, "turbolvl": self.args.turbo, "nosubtle": self.args.nosubtle, diff --git a/copyparty/svchub.py b/copyparty/svchub.py index 73855bf1..bd7562e4 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -1026,6 +1026,8 @@ class SvcHub(object): except: raise Exception("invalid --mv-retry [%s]" % (self.args.mv_retry,)) + al.js_utc = "false" if al.localtime else "true" + al.tcolor = al.tcolor.lstrip("#") if len(al.tcolor) == 3: # fc5 => ffcc55 al.tcolor = "".join([x * 2 for x in al.tcolor]) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 17fe489a..0f9d7698 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -230,6 +230,7 @@ var Ls = { "ct_qdel": 'when deleting files, only ask for confirmation once">qdel', "ct_dir1st": 'sort folders before files">📁 first', "ct_nsort": 'natural sort (for filenames with leading digits)">nsort', + "ct_utc": 'show all datetimes in UTC">UTC', "ct_readme": 'show README.md in folder listings">📜 readme', "ct_idxh": 'show index.html instead of folder listing">htm', "ct_sbars": 'show scrollbars">⟊', @@ -857,6 +858,7 @@ var Ls = { "ct_qdel": 'sletteknappen spør bare én gang om bekreftelse">hurtig🗑️', "ct_dir1st": 'sorter slik at mapper kommer foran filer">📁 først', "ct_nsort": 'naturlig sortering (forstår tall i filnavn)">nsort', + "ct_utc": 'bruk UTC for alle klokkeslett">UTC', "ct_readme": 'vis README.md nedenfor filene">📜 readme', "ct_idxh": 'vis index.html istedenfor fil-liste">htm', "ct_sbars": 'vis rullgardiner / skrollefelt">⟊', @@ -1483,6 +1485,7 @@ var Ls = { "ct_qdel": '删除文件时,只需确认一次">快删', //m "ct_dir1st": '在文件之前排序文件夹">📁 排序', "ct_nsort": '正确排序以数字开头的文件名">数字排序', //m + "ct_utc": '所有时间请使用UTC">UTC', //m "ct_readme": '在文件夹列表中显示 README.md">📜 readme', "ct_idxh": '显示 index.html 代替文件夹列表">htm', "ct_sbars": '显示滚动条">⟊', @@ -2109,6 +2112,7 @@ var Ls = { "ct_qdel": 'Nur einmal fragen, wenn mehrere Dateien gelöscht werden">qdel', "ct_dir1st": 'Ordner vor Dateien sortieren">📁 zuerst', "ct_nsort": 'Natürliche Sortierung (für Dateinamen mit führenden Ziffern)">nsort', + "ct_utc": 'Verwenden Sie UTC für alle Zeitangaben">UTC', //m "ct_readme": 'README.md in Dateiliste anzeigen">📜 readme', "ct_idxh": 'index.html anstelle von Dateiliste anzeigen">htm', "ct_sbars": 'Scrollbars zeigen">⟊', @@ -3362,6 +3366,7 @@ var Ls = { "ct_qdel": 'спрашивать подтверждение только один раз перед удалением файлов">быстр. удал.', "ct_dir1st": 'разместить папки над файлами">📁 сверху', "ct_nsort": 'сортировка по числам$N(например, файл с >code<2>/code< в начале названия идёт перед >code<11>/code<)">нат. сорт.', + "ct_utc": 'используйте UTC для всех временных меток">UTC', //m "ct_readme": 'показывать содержимое README.md в описании папки">📜 ридми', "ct_idxh": 'показывать страницу index.html в текущей папке вместо интерфейса">htm', "ct_sbars": 'показывать полосы прокрутки">⟊', @@ -3987,6 +3992,7 @@ var Ls = { "ct_qdel": 'al eliminar archivos, pedir confirmación solo una vez">elim. rápida', "ct_dir1st": 'ordenar carpetas antes que archivos">📁 primero', "ct_nsort": 'orden natural (para nombres de archivo con dígitos iniciales)">ord. natural', + "ct_utc": 'use UTC para todas las horas">UTC', //m "ct_readme": 'mostrar README.md en los listados de carpetas">📜 léeme', "ct_idxh": 'mostrar index.html en lugar del listado de carpetas">htm', "ct_sbars": 'mostrar barra lateral">⟊', @@ -4613,6 +4619,7 @@ var Ls = { "ct_qdel": 'при видаленні файлів, запитати підтвердження лише один раз">швидке видалення', "ct_dir1st": 'сортувати папки перед файлами">спочатку 📁', "ct_nsort": 'природне сортування (для імен файлів з початковими цифрами)">природне сортування', + "ct_utc": 'використовуйте UTC для всіх часових позначень">UTC', //m "ct_readme": 'показати README.md у списках папок">📜 readme', "ct_idxh": 'показати index.html замість списку папки">htm', "ct_sbars": 'показати смуги прокрутки">⟊', @@ -5232,6 +5239,7 @@ ebi('op_cfg').innerHTML = ( ' ' + sv; } - ln = ln.concat([tn.ext, unix2iso(tn.ts)]).join('
'); } html.push(''); @@ -13184,7 +13196,7 @@ var unpost = (function () { var done = res[a].pd === undefined; html.push( '' + - '' + + '' + '' + (done ? '' : '') + ''); diff --git a/copyparty/web/rups.html b/copyparty/web/rups.html index 5d286cae..af90eefe 100644 --- a/copyparty/web/rups.html +++ b/copyparty/web/rups.html @@ -33,6 +33,7 @@ var SR="{{ r }}", lang="{{ lang }}", + dutc={{ this.args.js_utc }}, dfavico="{{ favico }}"; var STG = window.localStorage; diff --git a/copyparty/web/rups.js b/copyparty/web/rups.js index e476f0fe..9559be8c 100644 --- a/copyparty/web/rups.js +++ b/copyparty/web/rups.js @@ -10,7 +10,7 @@ function render() { fn = esc(uricom_dec(vsp[1])), at = f.at, td = now - f.at, - ts = !at ? '(?)' : unix2iso(at), + ts = !at ? '(?)' : unix2ui(at), sa = !at ? '(?)' : td > 60 ? shumantime(td) : (td + 's'), sz = ('' + f.sz).replace(/\B(?=(\d{3})+(?!\d))/g, " "); diff --git a/copyparty/web/shares.html b/copyparty/web/shares.html index 9c7dcd27..381fe853 100644 --- a/copyparty/web/shares.html +++ b/copyparty/web/shares.html @@ -66,6 +66,7 @@ var SR="{{ r }}", shr="{{ shr }}", lang="{{ lang }}", + dutc={{ this.args.js_utc }}, dfavico="{{ favico }}"; var STG = window.localStorage; diff --git a/copyparty/web/shares.js b/copyparty/web/shares.js index e3faf029..d14dfeb7 100644 --- a/copyparty/web/shares.js +++ b/copyparty/web/shares.js @@ -66,7 +66,7 @@ function showqr(href) { for (var b = 7; b < 9; b++) { var v = buf[ibuf++]; tr[a].cells[b].innerHTML = - v ? unix2iso(v).replace(' ', ', ') : 'never'; + v ? unix2ui(v).replace(' ', ', ') : 'never'; } for (var a = 0; a < tr.length; a++) diff --git a/copyparty/web/up2k.js b/copyparty/web/up2k.js index edf0eee8..2f588446 100644 --- a/copyparty/web/up2k.js +++ b/copyparty/web/up2k.js @@ -2511,8 +2511,8 @@ function up2k_init(subtle) { var msg = []; for (var a = 0, aa = Math.min(20, response.hits.length); a < aa; a++) { var hit = response.hits[a], - tr = unix2iso(hit.ts), - tu = unix2iso(t.lmod), + tr = unix2ui(hit.ts), + tu = unix2ui(t.lmod), diff = parseInt(t.lmod) - parseInt(hit.ts), cdiff = (Math.abs(diff) <= 2) ? '3c0' : 'f0b', sdiff = 'diff ' + diff; @@ -3187,7 +3187,7 @@ function up2k_init(subtle) { return; try { - ebi('lifew').innerHTML = unix2iso((st.lifetime || lifetime) + + ebi('lifew').innerHTML = unix2ui((st.lifetime || lifetime) + Date.now() / 1000 - new Date().getTimezoneOffset() * 60 ).replace(' ', ', ').slice(0, -3); } diff --git a/copyparty/web/util.js b/copyparty/web/util.js index b246e145..94f73f24 100644 --- a/copyparty/web/util.js +++ b/copyparty/web/util.js @@ -907,11 +907,29 @@ function noq_href(el) { } +function pad2(v) { + return ('0' + v).slice(-2); +} + + function unix2iso(ts) { return new Date(ts * 1000).toISOString().replace("T", " ").slice(0, -5); } +function unix2iso_localtime(ts) { + var o = new Date(ts * 1000), + p = pad2; + return "{0}-{1}-{2} {3}:{4}:{5}".format( + o.getFullYear(), + p(o.getMonth() + 1), + p(o.getDate()), + p(o.getHours()), + p(o.getMinutes()), + p(o.getSeconds())); +} + + function s2ms(s) { s = Math.floor(s); var m = Math.floor(s / 60); @@ -1203,6 +1221,13 @@ function scfg_bind(obj, oname, cname, defval, cb) { } +window.unix2ui = (function () { + var v = sread('utctid'); + v = v ? (v === '0') : (window.dutc === false); + return v ? unix2iso_localtime : unix2iso; +})(); + + function hist_push(url) { console.log("h-push " + url); try { diff --git a/tests/util.py b/tests/util.py index 8027c372..42a47907 100644 --- a/tests/util.py +++ b/tests/util.py @@ -143,7 +143,7 @@ class Cfg(Namespace): def __init__(self, a=None, v=None, c=None, **ka0): ka = {} - ex = "chpw cookie_lax daw dav_auth dav_mac dav_rt e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp early_ban ed emp exp force_js getmod grid gsel hardlink hardlink_only ih ihead magic nid nih no_acode no_athumb no_bauth no_clone no_cp no_dav no_db_ip no_del no_dirsz no_dupe no_lifetime no_logues no_mv no_pipe no_poll no_readme no_robots no_sb_md no_sb_lg no_scandir no_tail no_tarcmp no_thumb no_vthumb no_zip nrand nsort nw og og_no_head og_s_title ohead q rand re_dirsz reflink rmagic rss smb srch_dbg srch_excl stats uqe vague_403 vc ver wo_up_readme write_uplog xdev xlink xvol zipmaxu zs" + ex = "chpw cookie_lax daw dav_auth dav_mac dav_rt e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp early_ban ed emp exp force_js getmod grid gsel hardlink hardlink_only ih ihead localtime magic nid nih no_acode no_athumb no_bauth no_clone no_cp no_dav no_db_ip no_del no_dirsz no_dupe no_lifetime no_logues no_mv no_pipe no_poll no_readme no_robots no_sb_md no_sb_lg no_scandir no_tail no_tarcmp no_thumb no_vthumb no_zip nrand nsort nw og og_no_head og_s_title ohead q rand re_dirsz reflink rmagic rss smb srch_dbg srch_excl stats uqe vague_403 vc ver wo_up_readme write_uplog xdev xlink xvol zipmaxu zs" ka.update(**{k: False for k in ex.split()}) ex = "dav_inf dedup dotpart dotsrch hook_v no_dhash no_fastboot no_fpool no_htp no_rescan no_sendfile no_ses no_snap no_up_list no_voldump re_dhash see_dots plain_ip" From c160428810c1e128bc45119fea20f33d6186dc16 Mon Sep 17 00:00:00 2001 From: ed Date: Fri, 1 Aug 2025 18:00:48 +0000 Subject: [PATCH 051/174] support tabs in configfiles; closes #400 --- copyparty/authsrv.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 030209fe..01fad895 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -3458,7 +3458,7 @@ def expand_config_file( ipath += " -> " + fp ret.append("#\033[36m opening cfg file{}\033[0m".format(ipath)) - cfg_lines = read_utf8(log, fp, True).split("\n") + cfg_lines = read_utf8(log, fp, True).replace("\t", " ").split("\n") if True: # diff-golf for oln in [x.rstrip() for x in cfg_lines]: ln = oln.split(" #")[0].strip() From 941761e6e7eb93c80afb660351895994e4de8391 Mon Sep 17 00:00:00 2001 From: ed Date: Fri, 1 Aug 2025 19:50:38 +0000 Subject: [PATCH 052/174] tl cleanup --- copyparty/web/browser.js | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 0f9d7698..02549c9f 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -2861,10 +2861,10 @@ var Ls = { "f_partial": "Ladataksesi turvallisesti tiedoston joka on parhaillaan latautumassa, klikkaa tiedostoa jolla on sama nimi mutta ilman .PARTIAL päätettä. Paina PERUUTA tai Escape tehdäksesi tämän.\n\nOK / Enter painaminen sivuuttaa tämän varoituksen ja jatkaa .PARTIAL väliaikaistiedoston lataamista, mikä todennäköisesti antaa sinulle vioittunutta dataa.", "ft_paste": "liitä {0} kohdetta$NPikanäppäin: ctrl-V", - "fr_eperm": 'ei voida nimetä uudelleen:\nsinulla ei ole "move"-oikeutta tässä kansiossa', - "fd_eperm": 'ei voida poistaa:\nsinulla ei ole "delete" oikeutta tässä kansiossa', - "fc_eperm": 'ei voida leikata:\nsinulla ei ole "move" oikeutta tässä kansiossa', - "fp_eperm": 'ei voida liittää:\nsinulla ei ole "write" oikeutta tässä kansiossa', + "fr_eperm": 'ei voida nimetä uudelleen:\nsinulla ei ole “move”-oikeutta tässä kansiossa', + "fd_eperm": 'ei voida poistaa:\nsinulla ei ole “delete” oikeutta tässä kansiossa', + "fc_eperm": 'ei voida leikata:\nsinulla ei ole “move” oikeutta tässä kansiossa', + "fp_eperm": 'ei voida liittää:\nsinulla ei ole “write” oikeutta tässä kansiossa', "fr_emore": "valitse vähintään yksi kohde uudelleennimettäväksi", "fd_emore": "valitse vähintään yksi kohde poistettavaksi", "fc_emore": "valitse vähintään yksi kohde leikattavaksi", @@ -3074,7 +3074,7 @@ var Ls = { "u_ewrite": 'sinulla ei ole move-oikeutta tähän kansioon', "u_eread": 'sinulla ei ole read-oikeutta tähän kansioon', "u_enoi": 'tiedostohaku ei ole käytössä palvelimen asetuksissa', - "u_enoow": "ylikirjoitus ei toimi täällä; tarvitaan "Delete"-oikeus", + "u_enoow": "ylikirjoitus ei toimi täällä; tarvitaan “Delete”-oikeus", "u_badf": 'Nämä {0} tiedostoa ({1} yhteensä) ohitettiin, mahdollisesti tiedostojärjestelmän oikeuksien vuoksi:\n\n', "u_blankf": 'Nämä {0} tiedostoa ({1} yhteensä) ovat tyhjiä; ladataanko ne silti?\n\n', "u_applef": 'Nämä {0} tiedostoa ({1} yhteensä) ovat todennäköisesti ei-toivottuja;\nPaina OK/Enter OHITTAAKSESI seuraavat tiedostot,\nPaina Cancel/ESC jos ET halua sulkea pois, ja LATAA nekin:\n\n', From 4fa7be2a48013cd156685afe7957dcac44ade0d3 Mon Sep 17 00:00:00 2001 From: ed Date: Fri, 1 Aug 2025 19:56:16 +0000 Subject: [PATCH 053/174] change "ack" to "continue"; longer text so easier to tap on mobile, and less confusing for people who aren't network engineers and/or kernel hackers thx @JanluOfficial for the idea --- copyparty/httpcli.py | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 5ce45a38..8f28fe5e 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -2940,7 +2940,7 @@ class HttpCli(object): msg = "new password OK" redir = (self.args.SRS + "?h") if ok else "" - h2 = 'ack' + h2 = 'continue' html = self.j2s("msg", h1=msg, h2=h2, redir=redir) self.reply(html.encode("utf-8")) return True @@ -2969,7 +2969,8 @@ class HttpCli(object): dst += "_=1#" + html_escape(uhash, True, True) _, msg = self.get_pwd_cookie(pwd) - html = self.j2s("msg", h1=msg, h2='ack', redir=dst) + h2 = 'continue' + html = self.j2s("msg", h1=msg, h2=h2, redir=dst) self.reply(html.encode("utf-8")) return True @@ -2983,7 +2984,7 @@ class HttpCli(object): self.get_pwd_cookie("x") dst = self.args.SRS + "?h" - h2 = 'ack' + h2 = 'continue' html = self.j2s("msg", h1="ok bye", h2=h2, redir=dst) self.reply(html.encode("utf-8")) return True From 3c6f0b17d00bb52676d1a0b1d972a1e3e6cb6857 Mon Sep 17 00:00:00 2001 From: Beethoven <44652883+Beethoven-n@users.noreply.github.com> Date: Fri, 1 Aug 2025 15:10:03 -0500 Subject: [PATCH 054/174] add Debian packaging via MPR (#385) * (scripts/prep.sh) prep mpr package at the same time as the arch package * (contrib/package/makedeb-mpr/) add MPR package i wanted this on my raspberry pi and i could've done it with docker but this gives me a systemd service. i haven't actually uploaded this at all --------- Signed-off-by: Beethoven <44652883+Beethoven-n@users.noreply.github.com> --- contrib/package/makedeb-mpr/PKGBUILD | 44 +++++++++++++++++++ contrib/package/makedeb-mpr/copyparty.conf | 7 +++ contrib/package/makedeb-mpr/copyparty.service | 32 ++++++++++++++ contrib/package/makedeb-mpr/index.md | 3 ++ .../package/makedeb-mpr/prisonparty.service | 33 ++++++++++++++ scripts/prep.sh | 21 +++++++++ 6 files changed, 140 insertions(+) create mode 100644 contrib/package/makedeb-mpr/PKGBUILD create mode 100644 contrib/package/makedeb-mpr/copyparty.conf create mode 100644 contrib/package/makedeb-mpr/copyparty.service create mode 100644 contrib/package/makedeb-mpr/index.md create mode 100644 contrib/package/makedeb-mpr/prisonparty.service diff --git a/contrib/package/makedeb-mpr/PKGBUILD b/contrib/package/makedeb-mpr/PKGBUILD new file mode 100644 index 00000000..7f1b4b1e --- /dev/null +++ b/contrib/package/makedeb-mpr/PKGBUILD @@ -0,0 +1,44 @@ +# Contributor: Beethoven + + +pkgname=copyparty +pkgver=1.18.8 +pkgrel=1 +pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" +arch=("any") +url="https://github.com/9001/${pkgname}" +license=('MIT') +depends=("bash" "python3" "lsof" "python3-jinja2") +makedepends=("python3-wheel" "python3-setuptools" "python3-build" "python3-installer" "make" "pigz") +optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tags" + "golang-cfssl: generate TLS certificates on startup" + "python3-mutagen: music tags (alternative)" + "python3-pil: thumbnails for images" + "python3-openssl: ftps functionality" + "python3-zmq: send zeromq messages from event-hooks" + "python3-argon2: hashed passwords in config" +) +source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") +backup=("/etc/${pkgname}.d/init" ) +sha256sums=("5bbda1e67f20a4a7fc10887235dba441544642d33c6d918f8a46867cda684572") + +build() { + cd "${srcdir}/${pkgname}-${pkgver}/copyparty/web" + make + + cd "${srcdir}/${pkgname}-${pkgver}" + python -m build --wheel --no-isolation +} + +package() { + cd "${srcdir}/${pkgname}-${pkgver}" + python -m installer --destdir="$pkgdir" dist/*.whl + + install -dm755 "${pkgdir}/etc/${pkgname}.d" + install -Dm755 "bin/prisonparty.sh" "${pkgdir}/usr/bin/prisonparty" + install -Dm644 "contrib/package/makedeb-mpr/${pkgname}.conf" "${pkgdir}/etc/${pkgname}.d/init" + install -Dm644 "contrib/package/makedeb-mpr/${pkgname}.service" "${pkgdir}/usr/lib/systemd/system/${pkgname}.service" + install -Dm644 "contrib/package/makedeb-mpr/prisonparty.service" "${pkgdir}/usr/lib/systemd/system/prisonparty.service" + install -Dm644 "contrib/package/makedeb-mpr/index.md" "${pkgdir}/var/lib/${pkgname}-jail/README.md" + install -Dm644 "LICENSE" "${pkgdir}/usr/share/licenses/${pkgname}/LICENSE" +} diff --git a/contrib/package/makedeb-mpr/copyparty.conf b/contrib/package/makedeb-mpr/copyparty.conf new file mode 100644 index 00000000..1d90d772 --- /dev/null +++ b/contrib/package/makedeb-mpr/copyparty.conf @@ -0,0 +1,7 @@ +## import all *.conf files from the current folder (/etc/copyparty.d) +% ./ + +# add additional .conf files to this folder; +# see example config files for reference: +# https://github.com/9001/copyparty/blob/hovudstraum/docs/example.conf +# https://github.com/9001/copyparty/tree/hovudstraum/docs/copyparty.d diff --git a/contrib/package/makedeb-mpr/copyparty.service b/contrib/package/makedeb-mpr/copyparty.service new file mode 100644 index 00000000..22dac3d6 --- /dev/null +++ b/contrib/package/makedeb-mpr/copyparty.service @@ -0,0 +1,32 @@ +# this will start `/usr/bin/copyparty-sfx.py` +# and read config from `/etc/copyparty.d/*.conf` +# +# you probably want to: +# change "User=cpp" and "/home/cpp/" to another user +# +# unless you add -q to disable logging, you may want to remove the +# following line to allow buffering (slightly better performance): +# Environment=PYTHONUNBUFFERED=x + +[Unit] +Description=copyparty file server + +[Service] +Type=notify +SyslogIdentifier=copyparty +Environment=PYTHONUNBUFFERED=x +WorkingDirectory=/var/lib/copyparty-jail +ExecReload=/bin/kill -s USR1 $MAINPID + +# user to run as + where the TLS certificate is (if any) +User=cpp +Environment=XDG_CONFIG_HOME=/home/cpp/.config + +# stop systemd-tmpfiles-clean.timer from deleting copyparty while it's running +ExecStartPre=+/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf' + +# run copyparty +ExecStart=/usr/bin/python3 /usr/bin/copyparty -c /etc/copyparty.d/init + +[Install] +WantedBy=multi-user.target diff --git a/contrib/package/makedeb-mpr/index.md b/contrib/package/makedeb-mpr/index.md new file mode 100644 index 00000000..016c0b0a --- /dev/null +++ b/contrib/package/makedeb-mpr/index.md @@ -0,0 +1,3 @@ +this is `/var/lib/copyparty-jail`, the fallback webroot when copyparty has not yet been configured + +please add some `*.conf` files to `/etc/copyparty.d/` diff --git a/contrib/package/makedeb-mpr/prisonparty.service b/contrib/package/makedeb-mpr/prisonparty.service new file mode 100644 index 00000000..cd35ba99 --- /dev/null +++ b/contrib/package/makedeb-mpr/prisonparty.service @@ -0,0 +1,33 @@ +# this will start `/usr/bin/copyparty-sfx.py` +# in a chroot, preventing accidental access elsewhere, +# and read copyparty config from `/etc/copyparty.d/*.conf` +# +# expose additional filesystem locations to copyparty +# by listing them between the last `cpp` and `--` +# +# `cpp cpp` = user/group to run copyparty as; can be IDs (1000 1000) +# +# unless you add -q to disable logging, you may want to remove the +# following line to allow buffering (slightly better performance): +# Environment=PYTHONUNBUFFERED=x + +[Unit] +Description=copyparty file server + +[Service] +SyslogIdentifier=prisonparty +Environment=PYTHONUNBUFFERED=x +WorkingDirectory=/var/lib/copyparty-jail +ExecReload=/bin/kill -s USR1 $MAINPID + +# stop systemd-tmpfiles-clean.timer from deleting copyparty while it's running +ExecStartPre=+/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf' + +# run copyparty +ExecStart=/bin/bash /usr/bin/prisonparty /var/lib/copyparty-jail cpp cpp \ + /etc/copyparty.d \ + -- \ + /usr/bin/python3 /usr/bin/copyparty -c /etc/copyparty.d/init + +[Install] +WantedBy=multi-user.target diff --git a/scripts/prep.sh b/scripts/prep.sh index 791d8eec..c08d09ed 100755 --- a/scripts/prep.sh +++ b/scripts/prep.sh @@ -22,10 +22,31 @@ update_arch_pkgbuild() { rm -rf x } +update_mpr_pkgbuild() { + cd "$self/../contrib/package/makedeb-mpr" + rm -rf x + mkdir x + + sha=$(sha256sum "$self/../dist/copyparty-$ver.tar.gz" | awk '{print$1}') + + # awk -v ver=$ver -v sha=$sha ' + # /^pkgver=/{sub(/[0-9\.]+/,ver)}; + # /^sha256sums=/{sub(/[0-9a-f]{64}/,sha)}; + # 1' PKGBUILD >a + # mv a PKGBUILD + + echo thing 1 + sed -s -i "s/pkgver=\"\"/pkgver=\"$ver\"/" PKGBUILD + sed -s -i "s/sha256sums=(\".*\")/sha256sums=(\"$sha\")/" PKGBUILD + + rm -rf x +} + update_nixos_pin() { ( cd $self/../contrib/package/nix/copyparty; ./update.py $self/../dist/copyparty-sfx.py ) } update_arch_pkgbuild +update_mpr_pkgbuild update_nixos_pin From 09910ba80784c3980947d92f45db696398c0fd83 Mon Sep 17 00:00:00 2001 From: ed Date: Fri, 1 Aug 2025 20:42:49 +0000 Subject: [PATCH 055/174] fix GHSA-5662-2rj7-f2v6 ; an unauthenticated user could make the server inaccessible by accessing the recent-uploads page and using an expensive filter fixed by making the filter not regex-based, only supporting bare-minimum anchoring (^foo bar$) --- copyparty/httpcli.py | 31 ++++++++++++++++++++++--------- copyparty/util.py | 15 +++++++++++++++ 2 files changed, 37 insertions(+), 9 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 8f28fe5e..bea4ce43 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -107,6 +107,7 @@ from .util import ( sendfile_py, set_fperms, stat_resource, + str_anchor, ub64dec, ub64enc, ujoin, @@ -5369,9 +5370,9 @@ class HttpCli(object): raise Pebkac(500, "sqlite3 not found on server; unpost is disabled") raise Pebkac(500, "server busy, cannot unpost; please retry in a bit") - zs = self.uparam.get("filter") or "" - filt = re.compile(zs, re.I) if zs else None - lm = "ups %r" % (zs,) + sfilt = self.uparam.get("filter") or "" + nfi, vfi = str_anchor(sfilt) + lm = "ups %d%r" % (nfi, sfilt) if self.args.shr and self.vpath.startswith(self.args.shr1): shr_dbv, shr_vrem = self.vn.get_dbv(self.rem) @@ -5431,8 +5432,14 @@ class HttpCli(object): q = "select sz, rd, fn, at from up where ip=? and at>? order by at desc" for sz, rd, fn, at in cur.execute(q, (self.ip, lim)): vp = "/" + "/".join(x for x in [vol.vpath, rd, fn] if x) - if filt and not filt.search(vp): - continue + if nfi == 0 or (nfi == 1 and vfi in vp): + pass + elif nfi == 2: + if not vp.startswith(vfi): + continue + elif nfi == 3: + if not vp.endswith(vfi): + continue n -= 1 if not n: @@ -5513,8 +5520,8 @@ class HttpCli(object): raise Pebkac(500, "server busy, cannot list recent uploads; please retry") sfilt = self.uparam.get("filter") or "" - filt = re.compile(sfilt, re.I) if sfilt else None - lm = "ru %r" % (sfilt,) + nfi, vfi = str_anchor(sfilt) + lm = "ru %d%r" % (nfi, sfilt) self.log(lm) ret: list[dict[str, Any]] = [] @@ -5549,8 +5556,14 @@ class HttpCli(object): q = "select sz, rd, fn, ip, at from up where at>0 order by at desc" for sz, rd, fn, ip, at in cur.execute(q): vp = "/" + "/".join(x for x in [vol.vpath, rd, fn] if x) - if filt and not filt.search(vp): - continue + if nfi == 0 or (nfi == 1 and vfi in vp): + pass + elif nfi == 2: + if not vp.startswith(vfi): + continue + elif nfi == 3: + if not vp.endswith(vfi): + continue if not dots and "/." in vp: continue diff --git a/copyparty/util.py b/copyparty/util.py index 73284bbb..15b1924b 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -2396,6 +2396,21 @@ def ujoin(rd: str, fn: str) -> str: return rd or fn +def str_anchor(txt) -> tuple[int, str]: + if not txt: + return 0, "" + txt = txt.lower() + a = txt.startswith("^") + b = txt.endswith("$") + if not b: + if not a: + return 1, txt # ~ + return 2, txt[1:] # ^ + if not a: + return 3, txt[:-1] # $ + return 4, txt[1:-1] # ^$ + + def log_reloc( log: "NamedLogger", re: dict[str, str], From ca22cd88534f48f13b18fc13eab9fdafca83a471 Mon Sep 17 00:00:00 2001 From: ed Date: Fri, 1 Aug 2025 20:56:27 +0000 Subject: [PATCH 056/174] v1.18.9 --- copyparty/__version__.py | 4 +-- copyparty/up2k.py | 2 +- docs/changelog.md | 78 ++++++++++++++++++++++++++++++++++++++++ 3 files changed, 81 insertions(+), 3 deletions(-) diff --git a/copyparty/__version__.py b/copyparty/__version__.py index cd73581d..a82053fb 100644 --- a/copyparty/__version__.py +++ b/copyparty/__version__.py @@ -1,8 +1,8 @@ # coding: utf-8 -VERSION = (1, 18, 8) +VERSION = (1, 18, 9) CODENAME = "logtail" -BUILD_DT = (2025, 7, 31) +BUILD_DT = (2025, 8, 1) S_VERSION = ".".join(map(str, VERSION)) S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) diff --git a/copyparty/up2k.py b/copyparty/up2k.py index ea2bbb57..ae5ed07f 100644 --- a/copyparty/up2k.py +++ b/copyparty/up2k.py @@ -402,7 +402,7 @@ class Up2k(object): def get_unfinished_by_user(self, uname, ip) -> dict[str, Any]: # returns dict due to ExceptionalQueue if PY2 or not self.reg_mutex.acquire(timeout=2): - return {"timeout":1} + return {"timeout": 1} ret: list[tuple[int, str, int, int, int]] = [] userset = set([(uname or "\n"), "*"]) diff --git a/docs/changelog.md b/docs/changelog.md index 7a9571f1..5e1d7717 100644 --- a/docs/changelog.md +++ b/docs/changelog.md @@ -1,3 +1,81 @@ +▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀ +# 2025-0731-0833 `v1.18.8` sfx hotfix + +## 🩹 bugfixes + +* #354 fix `copyparty-sfx.py` failing to start on certain versions of python c17ce4892ecdb4e11437bc2785d132bd8100eaec + + + +▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀ +# 2025-0730-2131 `v1.18.7` SECURITY: fix another XSS + +## ⚠️ ATTN: this release fixes an XSS vulnerability + +[GHSA-8mx2-rjh8-q3jq](https://github.com/9001/copyparty/security/advisories/GHSA-8mx2-rjh8-q3jq), could let an attacker execute arbitrary JS by tricking you into clicking a malicious URL + +Soon there won't be many of these left, surely. Huge thanks to @Ju0x for finding and reporting this. + +## 🧪 new features + +* #265 uid/gid for new files can be configured per-volume f1959988 + * has preconditions; [see readme](https://github.com/9001/copyparty#chmod-and-chown) +* #212 add German translation (thx @rGunti, @Scotsguy, @chocolateimage) 9d32564c + +## 🩹 bugfixes + +* GHSA-8mx2-rjh8-q3jq a8705e61 +* #276 windows: fix segfault (thx @kernel1994 for debugging!) a9d07c63 +* #272 webdav: send disk-size and disk-free to clients 4988a55e +* #285 use disk-free sans root-reserve on linux (thx @Arklaum!) c3cc2dde +* cors-check was funky on IPv6 e9684d40 +* #325 upgrade sharex example for newer versions 6016ec93 +* #300 restore support for old versions of python 2.7 b7ca6f4a + +## 🔧 other changes + +* shares: the config POST-target is now always the webroot (for ease of IdP configuration) fb7cbc42 +* unlist: now applies to the navpane too fbf17be2 +* windows: show disk-usage as well, not just disk-free 5c6341e9 +* #228 nix-pkg improvements (thx @dtomvan!) 4915b14b +* docker-compose: ensure logs appear in realtime 3cde1f3b +* mention that IdP-volumes and users [can now be persisted](https://github.com/9001/copyparty/blob/hovudstraum/docs/idp.md#but-you-can-enable-idp-volume-persistence) 6069bc9b +* #316 explain a scary-looking thing in the code 053de619 + + + +▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀ +# 2025-0728-2320 `v1.18.6` reflink-dedup + +## 🧪 new features + +* #201 add support for reflink-based dedup on cow filesystems df9feabc + * combine `--dedup` with `--reflink` to enable, or volflags with same name + * a better and safer alternative to the other dedup approaches (symlink/hardlink), but only possible to use in some cases: + * needs linux 5.3 or newer, python 3.14 or newer, btrfs/xfs/zfs + * not available in the docker images yet; needs a new version of python, so maybe next alpine release (november/december 2025) +* ratelimit password changes to impede bruteforcing a2601fd6 + * limit is set by `--ban-pwc` (default is 5 changes in 60min) + +## 🩹 bugfixes + +* #240 nixos: fix unixgroups issue (thx @chinponya!) 7c9c962b +* #246 cbz: use correct page for thumbnail (thx @Scotsguy!) 542a1de1 + +## 🔧 other changes + +* volflag `nosub` now also prevents mkdir 0f2c6235 +* improve documentation: + * #229 use the same example UDS path everywhere cb019afe + * [example nginx config](https://github.com/9001/copyparty/blob/hovudstraum/contrib/nginx/copyparty.conf) had misleading cloudflare comment (thx @jmi2k!) 674fc1fe + * more readable `--help-chmod` 03d23dae + * #244 fix typo in `--help` 4f013f64 +* #242 hide "use real pw" on connectpage if no accounts (thx @toast003!) 025942a7 +* #211 docker: remove deprecated attribute (thx @ptweezy!) 5b98e104 +* #190 add the feature-showcase video to the readme (thx @RustoMCSpit!) 43e6da34 + + + ▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀ # 2025-0727-2305 `v1.18.5` SECURITY: fix XSS in media tags From b7000721073097d3d7b8be03ed9a1cf6805af072 Mon Sep 17 00:00:00 2001 From: ed Date: Fri, 1 Aug 2025 21:27:28 +0000 Subject: [PATCH 057/174] update pkgs to 1.18.9 --- contrib/package/arch/PKGBUILD | 4 ++-- contrib/package/makedeb-mpr/PKGBUILD | 4 ++-- contrib/package/nix/copyparty/pin.json | 6 +++--- 3 files changed, 7 insertions(+), 7 deletions(-) diff --git a/contrib/package/arch/PKGBUILD b/contrib/package/arch/PKGBUILD index 92423af8..40c0b62b 100644 --- a/contrib/package/arch/PKGBUILD +++ b/contrib/package/arch/PKGBUILD @@ -1,6 +1,6 @@ # Maintainer: icxes pkgname=copyparty -pkgver="1.18.8" +pkgver="1.18.9" pkgrel=1 pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" arch=("any") @@ -22,7 +22,7 @@ optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tag ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") backup=("etc/${pkgname}.d/init" ) -sha256sums=("d8af012f552a6f1dfc501d0ce94990b932d422df74f6afb66ca2a31985f5809b") +sha256sums=("d5d33b50d6717e52427956beb1687061a6f28b467997506505151e3ae18c58e5") build() { cd "${srcdir}/${pkgname}-${pkgver}" diff --git a/contrib/package/makedeb-mpr/PKGBUILD b/contrib/package/makedeb-mpr/PKGBUILD index 7f1b4b1e..227fc4cf 100644 --- a/contrib/package/makedeb-mpr/PKGBUILD +++ b/contrib/package/makedeb-mpr/PKGBUILD @@ -2,7 +2,7 @@ pkgname=copyparty -pkgver=1.18.8 +pkgver=1.18.9 pkgrel=1 pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" arch=("any") @@ -20,7 +20,7 @@ optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tag ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") backup=("/etc/${pkgname}.d/init" ) -sha256sums=("5bbda1e67f20a4a7fc10887235dba441544642d33c6d918f8a46867cda684572") +sha256sums=("d5d33b50d6717e52427956beb1687061a6f28b467997506505151e3ae18c58e5") build() { cd "${srcdir}/${pkgname}-${pkgver}/copyparty/web" diff --git a/contrib/package/nix/copyparty/pin.json b/contrib/package/nix/copyparty/pin.json index 1afe1c79..734dec31 100644 --- a/contrib/package/nix/copyparty/pin.json +++ b/contrib/package/nix/copyparty/pin.json @@ -1,5 +1,5 @@ { - "url": "https://github.com/9001/copyparty/releases/download/v1.18.8/copyparty-sfx.py", - "version": "1.18.8", - "hash": "sha256-ZQtpGVcC6fmJCC65hXN4v3ERiB3jPz+oqvFHtbxR8+o=" + "url": "https://github.com/9001/copyparty/releases/download/v1.18.9/copyparty-sfx.py", + "version": "1.18.9", + "hash": "sha256-R1OVx4f8GERAG80ZcHAIP6HK2TlBbKJZpvnJmJbGPRY=" } \ No newline at end of file From 7e3825f8f5e6e31b2318347611b0194fe0862670 Mon Sep 17 00:00:00 2001 From: Kazi Date: Sat, 2 Aug 2025 16:06:57 +0200 Subject: [PATCH 058/174] More verbose help text for TLS certificate flag (#429) * Clarify TLS key in help text * More verbose help text --------- Signed-off-by: Kazi --- copyparty/__main__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 609a534e..71a06033 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1109,7 +1109,7 @@ def add_tls(ap, cert_path): ap2 = ap.add_argument_group('SSL/TLS options') ap2.add_argument("--http-only", action="store_true", help="disable ssl/tls -- force plaintext") ap2.add_argument("--https-only", action="store_true", help="disable plaintext -- force tls") - ap2.add_argument("--cert", metavar="PATH", type=u, default=cert_path, help="path to TLS certificate") + ap2.add_argument("--cert", metavar="PATH", type=u, default=cert_path, help="path to file containing a concatenation of TLS key and certificate chain") ap2.add_argument("--ssl-ver", metavar="LIST", type=u, default="", help="set allowed ssl/tls versions; [\033[32mhelp\033[0m] shows available versions; default is what your python version considers safe") ap2.add_argument("--ciphers", metavar="LIST", type=u, default="", help="set allowed ssl/tls ciphers; [\033[32mhelp\033[0m] shows available ciphers") ap2.add_argument("--ssl-dbg", action="store_true", help="dump some tls info") From 971360e91434e34095bb9d0a7a9115f7c14fec85 Mon Sep 17 00:00:00 2001 From: ed Date: Sat, 2 Aug 2025 23:24:32 +0000 Subject: [PATCH 059/174] set config from PRTY_CONFIG; closes #439 --- copyparty/__main__.py | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 71a06033..3a3d950e 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -93,6 +93,10 @@ u = unicode printed: list[str] = [] zsid = uuid.uuid4().urn[4:] +CFG_DEF = [os.environ.get("PRTY_CONFIG", "")] +if not CFG_DEF[0]: + CFG_DEF.pop() + class RiceFormatter(argparse.HelpFormatter): def __init__(self, *args: Any, **kwargs: Any) -> None: @@ -990,7 +994,7 @@ def build_flags_desc(): def add_general(ap, nc, srvname): ap2 = ap.add_argument_group('general options') - ap2.add_argument("-c", metavar="PATH", type=u, action="append", help="add config file") + ap2.add_argument("-c", metavar="PATH", type=u, default=CFG_DEF, action="append", help="add config file") ap2.add_argument("-nc", metavar="NUM", type=int, default=nc, help="max num clients") ap2.add_argument("-j", metavar="CORES", type=int, default=1, help="max num cpu cores, 0=all") ap2.add_argument("-a", metavar="ACCT", type=u, action="append", help="add account, \033[33mUSER\033[0m:\033[33mPASS\033[0m; example [\033[32med:wark\033[0m]") @@ -1786,7 +1790,7 @@ def main(argv: Optional[list[str]] = None) -> None: argv[idx] = nk + ov time.sleep(2) - da = len(argv) == 1 + da = len(argv) == 1 and not CFG_DEF try: if da: argv.extend(["--qr"]) From a68d5b03f1f18a8243448119f16633ea8d5e9361 Mon Sep 17 00:00:00 2001 From: icxes Date: Sun, 3 Aug 2025 02:52:42 +0300 Subject: [PATCH 060/174] fixup finnish translation here and there --- copyparty/web/browser.js | 160 +++++++++++++++++++-------------------- copyparty/web/splash.js | 4 +- 2 files changed, 82 insertions(+), 82 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 02549c9f..5340d5bb 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -2543,7 +2543,7 @@ var Ls = { ["ESC", "sulje asioita"], "file-manager", - ["G", "vaihda lista/ruudukkonäkymään"], + ["G", "vaihda lista/kuvanäkymään"], ["T", "vaihda pienoiskuviin/kuvakkeisiin"], ["⇧ A/D", "pienoiskuvien koko"], ["ctrl-K", "poista valitut"], @@ -2558,13 +2558,13 @@ var Ls = { ["↑/↓", "siirrä valintaosoitinta"], ["ctrl ↑/↓", "siirrä osoitinta ja näkymää"], ["⇧ ↑/↓", "valitse edellinen/seuraava tiedosto"], - ["ctrl-A", "valitse kaikki tiedostot / kansiot"], + ["ctrl-A", "valitse kaikki tiedostot / hakemistot"], ], [ "navigation", ["B", "näytä linkkipolku"], ["I/K", "siirry edelliseen/seuraavaan hakemistoon"], ["M", "siirry ylähakemistoon/supista nykyinen hakemisto"], - ["V", "näytä kansiot/tekstitiedostot navigointipaneelissa"], + ["V", "näytä hakemistot/tekstitiedostot navigointipaneelissa"], ["A/D", "navigointipaneelin koko"], ], [ "audio-player", @@ -2617,7 +2617,7 @@ var Ls = { "ht_d2": "päivää", "ht_and": " ja ", - "goh": "ohjauspaneeli", + "goh": "hallintapaneeli", "gop": 'viereinen hakemisto">edell', "gou": 'ylempi hakemisto">ylös', "gon": 'seuraava hakemisto">seur', @@ -2626,7 +2626,7 @@ var Ls = { "ot_close": "sulje alavalikko", "ot_search": "etsi tiedostoja ominaisuuksien, tiedostopolun tai -nimen, musiikkitägien tai näiden yhdistelmän perusteella$N$N<code>foo bar</code> = täytyy sisältää sekä «foo» että «bar»,$N<code>foo -bar</code> = täytyy sisältää «foo» mutta ei «bar»,$N<code>^yana .opus$</code> = alkaa «yana» ja on «opus»-tiedosto$N<code>"try unite"</code> = sisältää täsmälleen «try unite»$N$Npäivämäärän muoto on iso-8601, kuten$N<code>2009-12-31</code> tai <code>2020-09-12 23:30:00</code>", "ot_unpost": "unpost: poista viimeaikaiset tai keskeytä keskeneräiset lataukset", - "ot_bup": "bup: tiedostojen 'perus'lähetysohjelma, tukee jopa netscape 4.0:aa", + "ot_bup": "bup: tiedostojen 'perus'lähetysohjelma, tukee jopa netscape 4.0", "ot_mkdir": "mkdir: luo uusi hakemisto", "ot_md": "new-md: luo uusi markdown-dokumentti", "ot_msg": "msg: lähetä viesti palvelinlokiin", @@ -2640,7 +2640,7 @@ var Ls = { "ab_mkdoc": "luo markdown-tiedosto", "ab_msg": "lähetä viesti palvelinlokiin", - "ay_path": "siirry kansioihin", + "ay_path": "siirry hakemistoihin", "ay_files": "siirry tiedostoihin", "wt_ren": "uudelleennimeä valitut kohteet$NPikanäppäin: F2", @@ -2650,29 +2650,29 @@ var Ls = { "wt_pst": "liitä aiemmin leikatut / kopioidut valinnat$NPikanäppäin: ctrl-V", "wt_selall": "valitse kaikki tiedostot$NPikanäppäin: ctrl-A (kun tiedosto on kohdistettu)", "wt_selinv": "valitse vastakkaiset tiedostot", - "wt_zip1": "lataa tämä kansio pakattuna", + "wt_zip1": "lataa tämä hakemisto pakattuna", "wt_selzip": "lataa valitut kohteet pakattuna", "wt_seldl": "lataa valitut kohteet paketoimatta$NPikanäppäin: Y", "wt_npirc": "kopioi kappaletiedot IRC-muotoilulla", "wt_nptxt": "kopioi kappaletiedot ilman muotoilua", "wt_m3ua": "lisää m3u-soittolistaan (klikkaa 📻kopioi myöhemmin)", "wt_m3uc": "kopioi m3u-soittolista leikepöydälle", - "wt_grid": "vaihda ruudukko- ja listanäkymän välillä$NPikanäppäin: G", + "wt_grid": "vaihda kuva- ja listanäkymän välillä$NPikanäppäin: G", "wt_prev": "edellinen kappale$NPikanäppäin: J", "wt_play": "toista / pysäytä$NPikanäppäin: P", "wt_next": "seuraava kappale$NPikanäppäin: L", - "ul_par": "rinnakkaisten latausten lkm:", + "ul_par": "rinnakkaislatausten lkm:", "ut_rand": "satunnaisgeneroidut tiedostonimet", "ut_u2ts": "kopioi viimeksi muokattu aikaleima$Ntiedostojärjestelmästäsi palvelimelle\">📅", "ut_ow": "korvaa olemassa olevat tiedostot palvelimella?$N🛡️: ei koskaan (luo sen sijaan uuden tiedostonimen)$N🕒: korvaa jos palvelintiedosto on vanhempi kuin omasi$N♻️: korvaa aina jos tiedostot ovat erilaisia", - "ut_mt": "jatka muiden tiedostojen tiivisteiden laskemista latauksen aikana$N$kannattanee poistaa käytöstä, mikäli prosessori tai kovalevy on vanhempaa mallia", + "ut_mt": "jatka muiden tiedostojen tiivisteiden laskemista latauksen aikana$N$Nkannattanee poistaa käytöstä, mikäli prosessori tai kovalevy on vanhempaa mallia", "ut_ask": 'kysy vahvistusta ennen latauksen aloittamista">💭', - "ut_pot": "paranna latausnopeutta hitailla laitteilla$Ntekemällä käyttöliittymäästä vähemmän monimutkaisen", + "ut_pot": "paranna latausnopeutta hitailla laitteilla$Nvähentämällä käyttöliittymän monimutkaisuutta", "ut_srch": "lataamisen sijaan tarkista, ovatko tiedostot jo $N olemassa palvelimella (käy läpi kaikki hakemistot, joihin sinulla on read-oikeudet)", - "ut_par": "keskeytä lataukset asettamalla se 0:aan$N$Nnosta, jos yhteytesi on hidas tai viive on suuri$N$Npidä se 1:ssä lähiverkossa tai jos palvelimen kovalevy on pullonkaula", - "ul_btn": "vedä tiedostoja / kansioita
tähän (tai napsauta tätä)", - "ul_btnu": "L A T A A", + "ut_par": "keskeytä lataukset asettamalla se nollaan$N$Nnosta, jos yhteytesi on hidas tai viive on suuri$N$Npidä se 1:ssä lähiverkossa tai jos palvelimen kovalevy on pullonkaula", + "ul_btn": "vedä tiedostoja / hakemistoja tähän
(tai klikkaa minua)", + "ul_btnu": "L Ä H E T Ä", "ul_btns": "E T S I", "ul_hash": "tiiviste", @@ -2712,31 +2712,31 @@ var Ls = { "udt_srch": "Etsi", "udt_drop": "pudota se tähän", - "u_nav_m": '
selvä, mitäs sulla on?
Enter = Tiedostoja (yksi tai useampi)\nESC = Yksi kansio (mukaan lukien alikansiot)', - "u_nav_b": 'TiedostojaYksi kansio', + "u_nav_m": '
selvä, mitäs sulla on?
Enter = Tiedostoja (yksi tai useampi)\nESC = Yksi hakemisto (mukaan lukien alihakemistot)', + "u_nav_b": 'TiedostojaYksi hakemisto', - "cl_opts": "kytkimet", + "cl_opts": "asetukset", "cl_themes": "teema", "cl_langs": "kieli", - "cl_ziptype": "kansion lataus", + "cl_ziptype": "hakemiston pakkaustyyppi", "cl_uopts": "up2k-kytkimet", "cl_favico": "favicon", "cl_bigdir": "suuret hakemistot", - "cl_hsort": "#lajittelu", - "cl_keytype": "sävellajin notaatio", + "cl_hsort": "#sort", + "cl_keytype": "sävellajin notaatiotyyppi", "cl_hiddenc": "piilotetut sarakkeet", "cl_hidec": "piilota", "cl_reset": "palauta", "cl_hpick": "napauta sarakeotsikoita piilottaaksesi alla olevassa taulukossa", "cl_hcancel": "sarakkeiden piilotus peruttu", - "ct_grid": '田 RUUDUKKO', - "ct_ttips": '◔ ◡ ◔">ℹ️ ammattilaisvinkit', - "ct_thumb": 'RUUDUKKOnäkymässä, vaihda kuvakkeihin tai pienoiskuviin$NPikanäppäin: T">🖼️ pienoiskuvat', - "ct_csel": 'käytä CTRL ja SHIFT tiedostojen valintaan RUUDUKKOnäkymässä">val', + "ct_grid": '田 kuvanäkymä', + "ct_ttips": '◔ ◡ ◔">ℹ️ vihjelaatikot', + "ct_thumb": 'valitse kuvakkeiden / pienoiskuvien välillä kuvanäkymässä $NPikanäppäin: T">🖼️ pienoiskuvat', + "ct_csel": 'käytä CTRL ja SHIFT tiedostojen valintaan kuvanäkymässä">valitse', "ct_ihop": 'kun kuvakatselin suljetaan, vieritä alas viimeksi katsottuun tiedostoon">g⮯', "ct_dots": 'näytä piilotetut tiedostot (jos palvelin sallii)">piilotiedostot', - "ct_qdel": 'tiedostoja poistaessa, kysy vahvistusta vain kerran">qdel', + "ct_qdel": 'kysy vahvistusta vain kerran tiedostoja poistaessa">qdel', "ct_dir1st": 'lajittele hakemistot ennen tiedostoja">📁 ensin', "ct_nsort": 'luonnollinen lajittelu (tiedostonimille jotka ovat numeroalkuisia)">nsort', "ct_utc": 'näytä kaikki aikaleimat UTC-ajassa">UTC', @@ -2746,9 +2746,9 @@ var Ls = { "cut_umod": "jos tiedosto on jo olemassa palvelimella, päivitä palvelimen viimeksi muokattu aikaleima vastaamaan paikallista tiedostoasi (vaatii write- ja delete-oikeudet)\">re📅", - "cut_turbo": "yolo-painike, ET todennäköisesti halua ottaa tätä käyttöön:$N$Nkäytä tätä jos latasit valtavan määrän tiedostoja ja jouduit käynnistämään uudelleen jostain syystä, ja haluat jatkaa latausta ASAP$N$Ntämä korvaa tiivistetarkistuksen yksinkertaisella "onko tällä sama tiedostokoko palvelimella?" joten jos tiedoston sisältö on erilainen sitä EI ladata$N$Nsinun pitäisi poistaa tämä käytöstä kun lataus on valmis, ja sitten "ladata" samat tiedostot uudelleen antaaksesi asiakkaan varmistaa ne\">turbo", + "cut_turbo": "yolo-painike -- et todennäköisesti halua ottaa tätä käyttöön:$N$Nkäytä tätä jos latasit valtavan määrän tiedostoja ja jouduit käynnistämään uudelleen jostain syystä, ja haluat jatkaa latausta välittömästi$N$Ntämä korvaa tiivistetarkistuksen yksinkertaisella "onko tällä sama tiedostokoko palvelimella?" joten jos tiedoston sisältö on erilainen sitä EI ladata$N$Nsinun pitäisi poistaa tämä käytöstä kun lataus on valmis, ja sitten "ladata" samat tiedostot uudelleen antaaksesi selaimesi varmistaa ne\">turbo", - "cut_datechk": "ei vaikutusta ellei turbo-painike ole käytössä$N$Nvähentää yolo-tekijää hieman; tarkistaa vastaavatko tiedostojen aikaleimat palvelimella omia$N$Npitäisi teoriassa napata useimmat keskeneräiset / vioittuneet lataukset, mutta ei ole korvike varmistuskierrokselle turbo poistettuna käytöstä jälkeenpäin\">päiväys-tark", + "cut_datechk": "ei vaikutusta ellei turbo-painike ole käytössä$N$Nvähentää yolo-tekijää hieman; tarkistaa vastaavatko tiedostojen aikaleimat palvelimella omia$N$Npitäisi teoriassa napata useimmat keskeneräiset / vioittuneet lataukset, mutta ei ole korvike varmistuskierrokselle turbo poistettuna käytöstä jälkeenpäin\">päiväysvarmistin", "cut_u2sz": "kunkin lähetyspalan koko (MiB:ssä); suuret arvot lentävät paremmin atlantin yli. kokeile pieniä arvoja erittäin heikoilla yhteyksillä", @@ -2767,46 +2767,46 @@ var Ls = { "cft_fg": "edustaväri", "cft_bg": "taustaväri", - "cdt_lim": "tiedostojen enimmäismäärä näytettäväksi kansiossa", - "cdt_ask": "sivun lopussa,$Nsen sijaan että lataa automaattisesti lisää tiedostoja,$Nkysy mitä tehdä", + "cdt_lim": "tiedostojen enimmäismäärä näytettäväksi hakemistossa", + "cdt_ask": "sivun lopussa, sen sijaan että lataa $Nautomaattisesti lisää tiedostoja, kysy mitä tehdä", "cdt_hsort": "kuinka monta lajittelusääntöä (<code>,sorthref</code>) sisällyttää media-URL:eihin. Tämän asettaminen nollaan jättää myös huomioimatta media-linkeissä sisällytetyt lajittelusäännöt kun napsautat niitä", "tt_entree": "näytä navigointipaneeli$NPikanäppäin: B", "tt_detree": "näytä linkkipolku$NPikanäppäin: B", - "tt_visdir": "näytä valittu kansio", + "tt_visdir": "näytä valittu hakemisto", "tt_ftree": "vaihda linkkipolku- / tekstitiedostonäkymään$NPikanäppäin: V", - "tt_pdock": "näytä yläkansiot telakoitussa paneelissa ylhäällä", + "tt_pdock": "näytä ylähakemistot telakoitussa paneelissa ylhäällä", "tt_dynt": "kasvata automaattisesti hakemistosyvyyden kasvaessa", "tt_wrap": "rivitys", "tt_hover": "paljasta ylivuotavat rivit leijutettaessa$N( rikkoo vierityksen ellei hiiri $N  ole vasemmassa marginaalissa )", - "ml_pmode": "kansion lopussa...", + "ml_pmode": "hakemiston lopussa...", "ml_btns": "komennot", - "ml_tcode": "transkoodaa", - "ml_tcode2": "transkoodaa muotoon", + "ml_tcode": "muunna nämä", + "ml_tcode2": "tähän muotoon", "ml_tint": "sävy", "ml_eq": "taajuuskorjain", "ml_drc": "dynaaminen alueen kompressori", "mt_loop": "toista samaa kappaletta\">🔁", - "mt_one": "lopeta yhden kappaleen jälkeen\">1️⃣", - "mt_shuf": "satunnaistoisto\">🔀", + "mt_one": "lopeta yhden toiston jälkeen\">1️⃣", + "mt_shuf": "aktivoi satunnaistoisto\">🔀", "mt_aplay": "automaattitoisto jos linkissä jolla pääsit palvelimelle oli kappale-ID$N$Ntämän poistaminen käytöstä pysäyttää myös sivun URL:n päivittämisen kappale-ID:lla musiikkia toistettaessa, estääksesi automaattitoiston jos nämä asetukset menetetään mutta URL säilyy\">a▶", "mt_preload": "aloita seuraavan kappaleen lataaminen lähellä loppua, mahdollistaen saumattoman toiston\">esilataus", - "mt_prescan": "siirry seuraavaan kansioon ennen viimeisen kappaleen$Nloppumista, pitäen verkkoselaimen tyytyväisenä$Njotta se ei pysäytä toistoa\">nav", - "mt_fullpre": "yritä esiladata koko kappale;$N✅ ota käyttöön heikoilla yhteyksillä,$N❌ poista käytöstä hitailla yhteyksillä\">täysi", + "mt_prescan": "siirry seuraavaan hakemistoon ennen viimeisen kappaleen$Nloppumista, pitäen verkkoselaimen tyytyväisenä$Njotta se ei pysäytä toistoa\">nav", + "mt_fullpre": "yritä esiladata koko kappale;$N✅ ota käyttöön heikoilla yhteyksillä,$N❌ poista käytöstä hitailla yhteyksillä\">esi+", "mt_fau": "puhelimissa: estä musiikin pysähtyminen jos seuraava kappale ei esilataudu tarpeeksi nopeasti (voi aiheuttaa ongelmia kappaletietojen näyttämisessä)\">☕️", "mt_waves": "aaltomuoto-hakupalkki:$Nnäytä äänenvahvuus selaimessa\">~s", "mt_npclip": "näytä painikkeet parhaillaan soivan kappaleen leikepöydälle kopioimiseen\">/np", "mt_m3u_c": "näytä painikkeet valittujen$Nkappaleiden kopioimiseen m3u8-soittolistana leikepöydälle\">📻", "mt_octl": "käyttöjärjestelmäintegraatio (medianäppäimet / osd)\">os-ctl", - "mt_oseek": "salli haku käyttöjärjestelmäintegraation kautta$N$Nhuom: joissakin laitteissa (iPhonet),$Ntämä korvaa seuraava-kappale-painikkeen\">kelaus", - "mt_oscv": "näytä albumin kansi osd:ssä\">taide", + "mt_oseek": "salli haku käyttöjärjestelmäintegraation kautta$N$Nhuom: joissakin laitteissa (iPhonet),$Ntämä korvaa 'seuraava kappale' -painikkeen\">kelaus", + "mt_oscv": "näytä albumin kansi osd:ssä\">kansikuvat", "mt_follow": "pidä soiva kappale näkyvissä\">🎯", "mt_compact": "kompaktit säätimet\">⟎", "mt_uncache": "tyhjennä välimuisti  (kokeile tätä jos selaimesi välimuistissa on$Nrikkinäinen kopio kappaleesta)\">uncache", - "mt_mloop": "toista avoinna olevaa hakemistoa loputtomasti\">🔁 silmukka", - "mt_mnext": "lataa seuraava kansio ja jatka\">📂 seuraava", + "mt_mloop": "toista avoinna olevaa hakemistoa loputtomasti\">🔁 alkuun", + "mt_mnext": "lataa seuraava hakemisto ja jatka\">📂 seuraava", "mt_mstop": "pysäytä toisto\">⏸ pysäytä", "mt_cflac": "muunna flac / wav opus-muotoon\">flac", "mt_caac": "muunna aac / m4a opus-muotoon\">aac", @@ -2821,7 +2821,7 @@ var Ls = { "mt_xowa": "iOS:ssä on bugeja jotka estävät taustatoiston tällä formaatilla; käytä caf:ia tai mp3:a sen sijaan", "mt_tint": "taustan taso (0-100) liukupalkissa$Ntehden puskuroinnista vähemmän häiritsevän", "mt_eq": "aktivoi taajuuskorjaimen ja vahvistussäätimen;$N$Nvahvistus <code>0</code> = normaali 100% äänenvoimakkuus (muokkaamaton)$N$Nleveys <code>1  </code> = normaali stereo (muokkaamaton)$Nleveys <code>0.5</code> = 50% vasen-oikea ristisyöttö$Nleveys <code>0  </code> = mono$N$Nvahvistus <code>-0.8</code> & leveys <code>10</code> = laulun poisto :^)$N$Nequalizerin käyttöönotto tekee saumattomista albumeista täysin saumattomia, joten jätä se päälle kaikilla arvoilla nollassa (paitsi leveys = 1) jos välität siitä", - "mt_drc": "aktivoi dynaamisen alueen kompressorin (äänenvoimakkuuden tasoittaja / tiiliseinättäjä); ottaa myös käyttöön EQ:n tasapainottamaan spagettia, joten aseta kaikki EQ-kentät paitsi 'leveys' 0:aan jos et halua sitä$N$Nalentaa äänenvoimakkuutta KYNNYS dB:n yläpuolella; jokaisesta SUHDE dB:stä KYNNYKSEN yli tulee 1 dB ulos, joten oletusarvot kynnys -24 ja suhde 12 tarkoittaa ettei sen pitäisi koskaan tulla kovempaa kuin -22 dB ja on turvallista nostaa equalizerin vahvistus 0.8:aan, tai jopa 1.8:aan ATK 0:lla ja valtavalla RLS:llä kuten 90 (toimii vain firefoxissa; RLS on max 1 muissa selaimissa)$N$N(katso wikipedia, he selittävät sen paljon paremmin)", + "mt_drc": "aktivoi dynaamisen alueen kompressorin; ottaa myös käyttöön taajuuskorjaimen tasapainottamaan spagettia, joten aseta kaikki EQ-kentät paitsi 'leveys' nollaan jos et halua sitä$N$Nalentaa äänenvoimakkuutta KYNNYS dB:n yläpuolella; jokaisesta SUHDE dB:stä KYNNYKSEN yli tulee 1 dB ulos, joten oletusarvot kynnys -24 ja suhde 12 tarkoittaa ettei sen pitäisi koskaan tulla kovempaa kuin -22 dB ja on turvallista nostaa equalizerin vahvistus 0.8:aan, tai jopa 1.8:aan ATK 0:lla ja valtavalla RLS:llä kuten 90 (toimii vain firefoxissa; RLS on max 1 muissa selaimissa)$N$N(katso wikipedia, he selittävät sen paljon paremmin)", "mb_play": "toista", "mm_hashplay": "soita tämä äänitiedosto?", @@ -2848,31 +2848,31 @@ var Ls = { " im_hnf": "tuota kuvaa ei enää ole olemassa", - "f_empty": 'tämä kansio on tyhjä', + "f_empty": 'tämä hakemisto on tyhjä', "f_chide": 'tämä piilottaa sarakkeen «{0}»\n\nvoit palauttaa sarakkeet asetuksista', "f_bigtxt": "tämä tiedosto on {0} Mt kokoinen -- näytetäänkö silti tekstinä?", "f_bigtxt2": "näytetäänkö vain tiedoston loppu? tämä myös mahdollistaa seuraamisen/tailing, näyttäen uudet tekstirivit reaaliaikaisesti", "fbd_more": '
näytetään {0} / {1} tiedostoa; näytä {2} tai näytä kaikki
', "fbd_all": '
näytetään {0} / {1} tiedostoa; näytä kaikki
', - "f_anota": "vain {0} / {1} kohdetta valittiin;\nvalitaksesi koko kansion, vieritä ensin loppuun", + "f_anota": "vain {0} / {1} kohdetta valittiin;\nvalitaksesi koko hakemiston, vieritä ensin loppuun", - "f_dls": 'nykyisen kansion tiedostolinkit on\nvaihdettu latauslinkeiksi', + "f_dls": 'nykyisen hakemiston tiedostolinkit on\nvaihdettu latauslinkeiksi', "f_partial": "Ladataksesi turvallisesti tiedoston joka on parhaillaan latautumassa, klikkaa tiedostoa jolla on sama nimi mutta ilman .PARTIAL päätettä. Paina PERUUTA tai Escape tehdäksesi tämän.\n\nOK / Enter painaminen sivuuttaa tämän varoituksen ja jatkaa .PARTIAL väliaikaistiedoston lataamista, mikä todennäköisesti antaa sinulle vioittunutta dataa.", "ft_paste": "liitä {0} kohdetta$NPikanäppäin: ctrl-V", - "fr_eperm": 'ei voida nimetä uudelleen:\nsinulla ei ole “move”-oikeutta tässä kansiossa', - "fd_eperm": 'ei voida poistaa:\nsinulla ei ole “delete” oikeutta tässä kansiossa', - "fc_eperm": 'ei voida leikata:\nsinulla ei ole “move” oikeutta tässä kansiossa', - "fp_eperm": 'ei voida liittää:\nsinulla ei ole “write” oikeutta tässä kansiossa', + "fr_eperm": 'ei voida nimetä uudelleen:\nsinulla ei ole “move”-oikeutta tässä hakemistossa', + "fd_eperm": 'ei voida poistaa:\nsinulla ei ole “delete” oikeutta tässä hakemistossa', + "fc_eperm": 'ei voida leikata:\nsinulla ei ole “move” oikeutta tässä hakemistossa', + "fp_eperm": 'ei voida liittää:\nsinulla ei ole “write” oikeutta tässä hakemistossa', "fr_emore": "valitse vähintään yksi kohde uudelleennimettäväksi", "fd_emore": "valitse vähintään yksi kohde poistettavaksi", "fc_emore": "valitse vähintään yksi kohde leikattavaksi", "fcp_emore": "valitse vähintään yksi kohde kopioitavaksi leikepöydälle", - "fs_sc": "jaa kansio jossa olet", + "fs_sc": "jaa hakemisto jossa olet", "fs_ss": "jaa valitut tiedostot", - "fs_just1d": "et voi valita useampaa kuin yhtä kansiota,\ntai sekoittaa tiedostoja ja kansioita yhdessä valinnassa", + "fs_just1d": "et voi valita useampaa kuin yhtä,\ntai sekoittaa tiedostoja ja hakemistoja yhdessä valinnassa", "fs_abrt": "❌ keskeytä", "fs_rand": "🎲 joku.nimi", "fs_go": "✅ luo share", @@ -2885,7 +2885,7 @@ var Ls = { "fs_tdays": "päivää", "fs_never": "ikuinen", "fs_pname": "valinnainen linkin nimi; on satunnainen jos tyhjä", - "fs_tsrc": "jaettava tiedosto tai kansio", + "fs_tsrc": "jaettava tiedosto tai hakemisto", "fs_ppwd": "valinnainen salasana", "fs_w8": "luodaan sharea...", "fs_ok": "paina Enter/OK lisätäksesi leikepöydälle\npaina ESC/Cancel sulkeaksesi", @@ -2897,7 +2897,7 @@ var Ls = { "fr_adv": "erä / liitännäistiedot / kaava uudelleennimeäminen\">lisäasetukset", "fr_case": "isot ja pienet kirjaimet erottava regex\">kirjainkoko", "fr_win": "windows-yhteensopivat nimet; korvaa <>:"\\|?* japanilaisilla leveillä merkeillä\">win", - "fr_slash": "korvaa / merkillä joka ei aiheuta uusien kansioiden luomista\">ei /", + "fr_slash": "korvaa / merkillä joka ei aiheuta uusien hakemistoiden luomista\">ei /", "fr_re": "regex hakukuvio jota käytetään alkuperäisiin tiedostonimiin; kaappausryhmiin voi viitata alla olevassa muotoilukentässä kuten <code>(1)</code> ja <code>(2)</code> ja niin edelleen", "fr_fmt": "foobar2000 innoittama:$N<code>(title)</code> korvataan kappaleen nimellä,$N<code>[(artist) - ](title)</code> sivuuttaa [tämän] osan jos artisti on tyhjä$N<code>$lpad((tn),2,0)</code> ", "fr_pdel": "poista", @@ -2925,7 +2925,7 @@ var Ls = { "fcc_warn": 'kopioitiin {0} kohdetta leikepöydälle\n\nmutta: vain tämä selain-välilehti voi liittää ne\n(koska valinta on niin valtavan suuri)', "fp_apply": "käytä näitä nimiä", - "fp_ecut": "leikkaa tai kopioi ensin joitakin tiedostoja / kansioita liitettäväksi / siirrettäväksi\n\nhuom: voit leikata / liittää eri selain-välilehtien välillä", + "fp_ecut": "leikkaa tai kopioi ensin joitakin tiedostoja / hakemistoja liitettäväksi / siirrettäväksi\n\nhuom: voit leikata / liittää eri selain-välilehtien välillä", "fp_ename": "{0} kohdetta ei voida siirtää tänne koska nimet ovat jo käytössä. Anna niille uudet nimet alla jatkaaksesi, tai tyhjennä nimi ohittaaksesi ne:", "fcp_ename": "{0} kohdetta ei voida kopioida tänne koska nimet ovat jo käytössä. Anna niille uudet nimet alla jatkaaksesi, tai tyhjennä nimi ohittaaksesi ne:", "fp_emore": "tiedostonimien törmäyksiä on vielä korjaamatta", @@ -2949,8 +2949,8 @@ var Ls = { "tv_load": "Ladataan tekstidokumenttia:\n\n{0}\n\n{1}% ({2} / {3} Mt ladattu)", "tv_xe1": "tekstitiedoston lataaminen epäonnistui:\n\nvirhe ", "tv_xe2": "404, tiedostoa ei löydy", - "tv_lst": "tekstitiedostojen lista kansiossa", - "tvt_close": "palaa kansionäkymään$NPikanäppäin: M (tai Esc)\">❌ sulje", + "tv_lst": "tekstitiedostojen lista hakemistossa", + "tvt_close": "palaa hakemistonäkymään$NPikanäppäin: M (tai Esc)\">❌ sulje", "tvt_dl": "lataa tämä tiedosto$NPikanäppäin: Y\">💾 lataa", "tvt_prev": "näytä edellinen dokumentti$NPikanäppäin: i\">⬆ edell", "tvt_next": "näytä seuraava dokumentti$NPikanäppäin: K\">⬇ seur", @@ -2967,8 +2967,8 @@ var Ls = { "m3u_clip": "m3u soittolista nyt kopioitu leikepöydälle\n\nsinun tulisi luoda uusi tekstitiedosto nimeltä jotain.m3u ja liittää soittolista siihen dokumenttiin; tämä tekee siitä soitettavan", "gt_vau": "älä näytä videoita, toista vain ääni\">🎧", - "gt_msel": "ota käyttöön tiedostojen valinta; ctrl-klikkaa tiedostoa ohittaaksesi$N$N<em>kun aktiivinen: tuplaklikkaa tiedostoa / kansiota avataksesi sen</em>$N$NPikanäppäin: S\">monivalinta", - "gt_crop": "keskitä-rajaa pienoiskuvat\">rajaa", + "gt_msel": "aktivoi tiedostonvalintatila; ctrl-klikkaa ohittaaksesi valitsemisen väliaikaisesti$N$N<em>tuplaklikkaa tiedostoa / hakemistoa avataksesi sen</em>$N$NPikanäppäin: S\">valitsin", + "gt_crop": "rajaa pienoiskuvat keskeltä\">rajaa", "gt_3x": "korkearesoluutioiset pienoiskuvat\">3x", "gt_zoom": "zoomaa", "gt_chop": "pilko", @@ -3012,28 +3012,28 @@ var Ls = { "xhr403": "403: Pääsy kielletty\n\nkokeile painaa F5, ehkä sinut kirjattiin ulos", "xhr0": "tuntematon (todennäköisesti yhteys palvelimeen katosi, tai palvelin on pois päältä)", "cf_ok": "sori siitä -- DD" + wah + "oS suojaus aktivoitui\n\nasioiden pitäisi jatkua noin 30 sekunnissa\n\njos mitään ei tapahdu, paina F5 ladataksesi sivun uudelleen", - "tl_xe1": "alikansioiden listaaminen epäonnistui:\n\nvirhe ", - "tl_xe2": "404: Kansiota ei löydy", - "fl_xe1": "kansion tiedostojen listaaminen epäonnistui:\n\nvirhe ", - "fl_xe2": "404: Kansiota ei löydy", - "fd_xe1": "alikansion luominen epäonnistui:\n\nvirhe ", - "fd_xe2": "404: Yläkansiota ei löydy", + "tl_xe1": "alihakemistojen listaaminen epäonnistui:\n\nvirhe ", + "tl_xe2": "404: hakemistoa ei löydy", + "fl_xe1": "hakemiston tiedostojen listaaminen epäonnistui:\n\nvirhe ", + "fl_xe2": "404: hakemistoa ei löydy", + "fd_xe1": "alihakemiston luominen epäonnistui:\n\nvirhe ", + "fd_xe2": "404: Ylähakemistoa ei löydy", "fsm_xe1": "viestin lähettäminen epäonnistui:\n\nvirhe ", - "fsm_xe2": "404: Yläkansiota ei löydy", + "fsm_xe2": "404: Ylähakemistoa ei löydy", "fu_xe1": "unpost-listan lataaminen palvelimelta epäonnistui:\n\nvirhe ", "fu_xe2": "404: Tiedostoa ei löydy??", "fz_tar": "pakkaamaton gnu-tar tiedosto (linux / mac)", "fz_pax": "pakkaamaton pax-formaatin tar (hitaampi)", - "fz_targz": "gnu-tar gzip tason 3 pakkauksella$N$NTämä on yleensä hyvin hidasta, $Nkäytä pakkamatonta tar:ia sen sijaan", - "fz_tarxz": "gnu-tar xz tason 1 pakkauksella$N$NTämä on yleensä hyvin hidasta, $Nkäytä pakkamatonta tar:ia sen sijaan", + "fz_targz": "gnu-tar gzip tason 3 pakkauksella$N$Nyleensä hyvin hidas, $Nkäytä pakkamatonta tar:ia tämän sijasta", + "fz_tarxz": "gnu-tar xz tason 1 pakkauksella$N$Nyleensä hyvin hidas, $Nkäytä pakkamatonta tar:ia tämän sijasta", "fz_zip8": "zip utf8-tiedostonimillä (suattaapi olla epävakaa windows 7:ssa ja vanhemmissa)", "fz_zipd": "zip perinteisillä cp437 tiedostonimillä esihistoriallisille ohjelmistoille", "fz_zipc": "cp437, jossa crc32 laskettu aikaisin,$NMS-DOS PKZIP v2.04g:lle (lokakuu 1993)$N(kestää kauemmin käsitellä ennen latauksen alkua)", - "un_m1": "voit poistaa viimeaikaiset latauksesi (tai keskeyttää keskeneräiset) alla", + "un_m1": "voit poistaa tuoreet tai keskeyttää keskeneräiset latauksesi alta", "un_upd": "päivitä", - "un_m4": "tai jaa alla näkyvät tiedostot:", + "un_m4": "tai jakaa alla näkyvät tiedostot:", "un_ulist": "näytä", "un_ucopy": "kopioi", "un_flt": "valinnainen suodatin:  URL:n täytyy sisältää", @@ -3041,8 +3041,8 @@ var Ls = { "un_derr": 'unpost-poisto epäonnistui:\n', "un_f5": 'jotain hajosi, kokeile päivitystä tai paina F5', "un_uf5": "pahoittelen mutta sinun täytyy päivittää sivu (esimerkiksi painamalla F5 tai CTRL-R) ennen kuin tämä lataus voidaan keskeyttää", - "un_nou": 'varoitus: palvelin liian kiireinen näyttääkseen keskeneräiset lataukset; klikkaa "päivitä" linkkiä hetken kuluttua', - "un_noc": 'varoitus: täysin ladattujen tiedostojen unpost ei ole käytössä/sallittu palvelimen asetuksissa', + "un_nou": 'huom! palvelin liian kiireinen näyttääkseen keskeneräiset lataukset; klikkaa "päivitä" linkkiä hetken kuluttua', + "un_noc": 'huom! täysin ladattujen tiedostojen unpost ei ole käytössä/sallittu palvelimen asetuksissa', "un_max": "näytetään ensimmäiset 2000 tiedostoa (käytä suodatinta)", "un_avail": "{0} viimeaikaista latausta voidaan poistaa
{1} keskeneräistä voidaan keskeyttää", "un_m2": "järjestetty latausajan mukaan; viimeisimmät ensin:", @@ -3069,10 +3069,10 @@ var Ls = { "u_gotpot": 'vaihdetaan peruna UI:hin paremman latausnopeuden vuoksi,\n\ntee miten lystäät, jos ei kelpaa!', "u_pott": "

tiedostot:   {0} valmis,   {1} epäonnistui,   {2} kiireinen,   {3} jonossa

", "u_ever": "tämä on peruslatain; up2k tarvitsee vähintään
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", - "u_su2k": 'tämä on peruslatain; up2k on parempi', - "u_uput": 'optimoi nopeuteen (ohita tarkistussumma)', - "u_ewrite": 'sinulla ei ole move-oikeutta tähän kansioon', - "u_eread": 'sinulla ei ole read-oikeutta tähän kansioon', + "u_su2k": 'peruslatain; up2k on parempi', + "u_uput": 'optimoi latausnopeus (älä laske tarkistussummia)', + "u_ewrite": 'sinulla ei ole move-oikeutta tähän hakemistoon', + "u_eread": 'sinulla ei ole read-oikeutta tähän hakemistoon', "u_enoi": 'tiedostohaku ei ole käytössä palvelimen asetuksissa', "u_enoow": "ylikirjoitus ei toimi täällä; tarvitaan “Delete”-oikeus", "u_badf": 'Nämä {0} tiedostoa ({1} yhteensä) ohitettiin, mahdollisesti tiedostojärjestelmän oikeuksien vuoksi:\n\n', @@ -3119,15 +3119,15 @@ var Ls = { "u_turbo_g": "poistetaan turbo käytöstä koska sinulla ei ole\nhakemistolistausoikeuksia tässä asemassa", "u_life_cfg": 'automaattinen poisto min kuluttua (tai tuntia)', "u_life_est": 'lataus poistetaan ---', - "u_life_max": 'tämä kansio pakottaa\nmaksimi elinajan {0}', + "u_life_max": 'tämä hakemisto pakottaa\nmaksimi elinajan {0}', "u_unp_ok": 'unpost on sallittu {0}', "u_unp_ng": 'unpost EI ole sallittu', - "ue_ro": 'sinulla on vain read-oikeus tähän kansioon\n\n', + "ue_ro": 'sinulla on vain read-oikeus tähän hakemistoon\n\n', "ue_nl": 'et ole tällä hetkellä kirjautunut sisään', "ue_la": 'olet tällä hetkellä kirjautunut sisään nimellä "{0}"', "ue_sr": 'olet tällä hetkellä tiedostohaku-tilassa\n\nvaihda lataus-tilaan klikkaamalla suurennuslasia 🔎 (suuren HAKU napin vieressä), ja yritä latausta uudelleen\n\npahoittelen', "ue_ta": 'yritä latausta uudelleen, sen pitäisi toimia nyt', - "ue_ab": "tätä tiedostoa ladataan jo toiseen kansioon, ja se lataus täytyy suorittaa loppuun ennen kuin tiedostoa voidaan ladata muualle.\n\nVoit keskeyttää ja unohtaa alkuperäisen latauksen käyttämällä vasemmalla ylhäällä olevaa 🧯", + "ue_ab": "tätä tiedostoa ladataan jo toiseen hakemistoon, ja se lataus täytyy suorittaa loppuun ennen kuin tiedostoa voidaan ladata muualle.\n\nVoit keskeyttää ja unohtaa alkuperäisen latauksen käyttämällä vasemmalla ylhäällä olevaa 🧯", "ur_1uo": "OK: Tiedosto ladattu onnistuneesti", "ur_auo": "OK: Kaikki {0} tiedostoa ladattu onnistuneesti", "ur_1so": "OK: Tiedosto löytyi palvelimelta", @@ -3139,7 +3139,7 @@ var Ls = { "ur_um": "Valmis;\n{0} latausta OK,\n{1} latausta epäonnistui, pahoittelen", "ur_sm": "Valmis;\n{0} tiedostoa löytyi palvelimelta,\n{1} tiedostoa EI löytynyt palvelimelta", - "lang_set": "päivitetäänkö sivu muutoksen voimaansaattamiseksi?", + "lang_set": "ladataanko sivu uudestaan kielen vaihtamiseksi?", }, "rus": { "tt": "Русский", diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index e55f7ff0..4ad53b9b 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -142,8 +142,8 @@ var Ls = { "d2": "näytä kaikkien aktiivisten säikeiden tila", "e1": "päivitä konffit", "e2": "lataa konfiguraatiotiedostot uudelleen (käyttäjätilit/asemat/asemaflagit),$Nja skannaa kaikki e2ds asemat uudelleen$N$Nhuom: kaikki global-asetuksiin$Ntehdyt muutokset vaativat täyden$Nuudelleenkäynnistyksen", - "f1": "voit selata:", - "g1": "voit ladata:", + "f1": "voit selata näitä:", + "g1": "voit ladata näihin:", "cc1": "muuta:", "h1": "poista k304 käytöstä", "i1": "ota k304 käyttöön", From 0bc1b8f71534cb99de9d7fa627b4b6dafff5e09f Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 10:22:52 +0000 Subject: [PATCH 061/174] readme: webdav and opengraph are incompatible by default --- README.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/README.md b/README.md index 889ea8b8..72b104c8 100644 --- a/README.md +++ b/README.md @@ -1425,6 +1425,8 @@ note that this disables hotlinking because the opengraph spec demands it; to sne you can also hotlink files regardless by appending `?raw` to the url +> WARNING: if you plan to use WebDAV, then `--og-ua` / `og_ua` must be configured + if you want to entirely replace the copyparty response with your own jinja2 template, give the template filepath to `--og-tpl` or volflag `og_tpl` (all members of `HttpCli` are available through the `this` object) From 50f162935533b601e1811d96f838f302c0507d1e Mon Sep 17 00:00:00 2001 From: Techflash <72118300+techflashYT@users.noreply.github.com> Date: Fri, 1 Aug 2025 16:16:00 -0700 Subject: [PATCH 062/174] Add test results for Wii Internet Channel Signed-off-by: Techflash <72118300+techflashYT@users.noreply.github.com> --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 72b104c8..d038207b 100644 --- a/README.md +++ b/README.md @@ -2431,6 +2431,7 @@ quick summary of more eccentric web-browsers trying to view a directory index: | **SerenityOS** (7e98457) | hits a page fault, works with `?b=u`, file upload not-impl | | **sony psp** 5.50 | can browse, upload/mkdir/msg (thx dwarf) [screenshot](https://github.com/user-attachments/assets/9d21f020-1110-4652-abeb-6fc09c533d4f) | | **nintendo 3ds** | can browse, upload, view thumbnails (thx bnjmn) | +| **Nintendo Wii (Opera 9.0 "Internet Channel")** | can browse, can't upload or download (no local storage), can view images - works best with `?b=u`, default view broken |

From c69c7c8ac06e900906a35fa909991fa563ad1fee Mon Sep 17 00:00:00 2001 From: Chinpo Nya Date: Fri, 1 Aug 2025 23:21:42 +0200 Subject: [PATCH 063/174] additional instructions for --ah-cli --- README.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/README.md b/README.md index d038207b..1a500c5b 100644 --- a/README.md +++ b/README.md @@ -2702,6 +2702,10 @@ optionally also specify `--ah-cli` to enter an interactive mode where it will ha the default configs take about 0.4 sec and 256 MiB RAM to process a new password on a decent laptop +when generating hashes using `--ah-cli` for docker or systemd services, make sure it is using the same `--ah-salt` by: +* inspecting the generated salt using `--show-ah-salt` in copyparty service configuration +* setting the same `--ah-salt` in both environments + ## https From c805c60f4093077cc2a51408597667be7a905705 Mon Sep 17 00:00:00 2001 From: Beethoven <44652883+Beethoven-n@users.noreply.github.com> Date: Fri, 1 Aug 2025 16:09:29 -0500 Subject: [PATCH 064/174] (scripts/prep.sh) update i forgot to follow the style woops Signed-off-by: Beethoven <44652883+Beethoven-n@users.noreply.github.com> --- scripts/prep.sh | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/scripts/prep.sh b/scripts/prep.sh index c08d09ed..a4c8a132 100755 --- a/scripts/prep.sh +++ b/scripts/prep.sh @@ -34,10 +34,10 @@ update_mpr_pkgbuild() { # /^sha256sums=/{sub(/[0-9a-f]{64}/,sha)}; # 1' PKGBUILD >a # mv a PKGBUILD - - echo thing 1 - sed -s -i "s/pkgver=\"\"/pkgver=\"$ver\"/" PKGBUILD - sed -s -i "s/sha256sums=(\".*\")/sha256sums=(\"$sha\")/" PKGBUILD + # TODO: check if this still works. if so, remove the following 2 lines: + + sed -s -i "s/pkgver=\"\"/pkgver=\"$ver\"/" PKGBUILD + sed -s -i "s/sha256sums=(\".*\")/sha256sums=(\"$sha\")/" PKGBUILD rm -rf x } From 3798e19a26d975f0b4ef1596706b1c62a4b1baca Mon Sep 17 00:00:00 2001 From: DeStilleGast <3677706+DeStilleGast@users.noreply.github.com> Date: Sun, 3 Aug 2025 18:20:10 +0200 Subject: [PATCH 065/174] Dutch translation (#426) * Update browser.js to include Dutch language * Update splash.js to include some Dutch translations Signed-off-by: DeStilleGast <3677706+DeStilleGast@users.noreply.github.com> --- copyparty/web/browser.js | 629 ++++++++++++++++++++++++++++++++++++++- copyparty/web/splash.js | 42 +++ 2 files changed, 670 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 5340d5bb..af0c0e39 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3141,6 +3141,633 @@ var Ls = { "lang_set": "ladataanko sivu uudestaan kielen vaihtamiseksi?", }, + "nld": { + "tt": "Nederlands", + + "cols": { + "c": "Action knoppen", + "dur": "Duratie", + "q": "Kwaliteit / bitrate", + "Ac": "Audio codec", + "Vc": "Video codec", + "Fmt": "Formaat / container", + "Ahash": "Audio checksum", + "Vhash": "Video checksum", + "Res": "Resolution", + "T": "Bestandstype", + "aq": "Audio kwaliteit / bitrate", + "vq": "Video kwaliteit / bitrate", + "pixfmt": "Subsampling / pixel structure", + "resw": "Horizontale resolutie", + "resh": "Verticale resolutie", + "chs": "Audiokanalen", + "hz": "Samplefrequentie" + }, + + "hks": [ + [ + "diversen", + ["ESC", "Sluit verschillende dingen"], + + "bestand beheer", + ["G", "Verwissel tussen list / grid weergave"], + ["T", "Verwissel tussen miniaturen / iconen"], + ["⇧ A/D", "Thumbnail formaat"], + ["ctrl-K", "Verwijder geselecteerde"], + ["ctrl-X", "Knip selectie naar klembord"], + ["ctrl-C", "Kopieer selectie naar klembord"], + ["ctrl-V", "Hier plakken (verplaatsen/kopieëren)"], + ["Y", "Download geselecteerde"], + ["F2", "Hernoem geselecteerde"], + + "bestand-lijst-selectie", + ["space", "wissel bestand selectie"], + ["↑/↓", "verplaats selectie cursor"], + ["ctrl ↑/↓", "verplaats cursor en scherm"], + ["⇧ ↑/↓", "select vorige/volgende bestand"], + ["ctrl-A", "selecteer alle bestanden / mappen"], + ], [ + "navigatie", + ["B", "verwissel breadcrumbs / navpane"], + ["I/K", "Vorige/volgende map"], + ["M", "Bovenliggende map (of huidige uitvouwen)"], + ["V", "Berwissel map / tekstbestand in navpane"], + ["A/D", "Navpane formaat"], + ], [ + "muziek-speler", + ["J/L", "Vorige/volgende song"], + ["U/O", "Skip 10sec terug/vooruit"], + ["0..9", "Spring naar 0%..90%"], + ["P", "Speel/pauzeer (start ook)"], + ["S", "Selecteer afspelende song"], + ["Y", "Download song"], + ], [ + "afbeelding viewer", + ["J/L, ←/→", "Vorige/volgende afbeelding"], + ["Home/End", "Eerste/laatste afbeelding"], + ["F", "Volledig scherm"], + ["R", "Draai rechtsom"], + ["⇧ R", "Draai linksom"], + ["S", "Selecteer afbeelding"], + ["Y", "Download afbeelding"], + ], [ + "video-speler", + ["U/O", "Skip 10sec terug/vooruit"], + ["P/K/Space", "Speel/pauze"], + ["C", "Verder met volgende"], + ["V", "herhaal"], + ["M", "stil"], + ["[ and ]", "zet herhaal interval"], + ], [ + "tekstbestand-viewer", + ["I/K", "vorige/volgende bestand"], + ["M", "sluit tekst bestand"], + ["E", "bewerk tekst bestand"], + ["S", "selecteer bestand (voor knip/kopie/hernoem)"], + ] + ], + + "m_ok": "OK", + "m_ng": "Annuleren", + + "enable": "Inschakelen", + "danger": "GEVAARLIJK", + "clipped": "Gekopieërd naar klembord", + + "ht_s1": "seconde", + "ht_s2": "secondes", + "ht_m1": "minuut", + "ht_m2": "minuten", + "ht_h1": "uur", + "ht_h2": "uur", + "ht_d1": "dag", + "ht_d2": "dagen", + "ht_and": " en ", + + "goh": "Beheer-paneel", + "gop": 'Vorige map">Vorige', + "gou": 'Bovenligende map">Omhoog', + "gon": 'Volgende map">Volgende', + "logout": "Uitloggen ", + "access": " Toegang", + "ot_close": "Sluit onder-menu", + "ot_search": "Zoek voor bestanden bij attributes, pad / naam, muziek tags, of elk andere combinatie tussen$N$N<code>foo bar</code> = moet beide «foo» en «bar» bevatten,$N<code>foo -bar</code> = moet «foo» bevatten maar geen «bar»,$N<code>^yana .opus$</code> = start met «yana» en moet een «opus» bestand zijn$N<code>"try unite"</code> = moet precies «try unite» bevatten$N$Nde datum formaat is iso-8601, zoals$N<code>2009-12-31</code> of <code>2020-09-12 23:30:00</code>", + "ot_unpost": "unpost: verwijder je recente uploads, of onvoltooide uploads afbreken", + "ot_bup": "bup: Basisuploader, supports zelfs netscape 4.0", + "ot_mkdir": "mkdir: Maak een nieuwe map", + "ot_md": "new-md: Maak een nieuwe markdown bestand", + "ot_msg": "msg: Verstuur een bericht naar de server logs", + "ot_mp": "Media speler opties", + "ot_cfg": "Configuratie opties", + "ot_u2i": 'up2k: upload bestanden (als je schrijf toegang hebt) of verwissel naar zoek-mode om te zien of ze ergens bestaan op de server$N$Nuploads zijn hervatbaar, multithreaded, en bestandstijdstempels blijven behouden, maar het gebruikt meer CPU dan [🎈]  (de basic uploader)

tijdens het uploaden, dit icoon word dan een progress indicatie!', + "ot_u2w": 'up2k: upload bestanden met hervattings ondersteuning (sluit je webbrowser en selecteer dezelfde bestand later opnieuw)$N$Nmultithreaded, en bestandstijdstempels blijven behouden, maar het gebruikt meer CPU dan [🎈]  (de basic uploader)

tijdens het uploaden, dit icoon word dan een progress indicatie!', + "ot_noie": 'Gebruik alstublieft Chrome / Firefox / Edge', + + "ab_mkdir": "maak map", + "ab_mkdoc": "nieuw markdown doc", + "ab_msg": "verstuur msg naar srv log", + + "ay_path": "skip naar mappen", + "ay_files": "skip naar bestanden", + + "wt_ren": "Hernoem geselecteerde items$NHotkey: F2", + "wt_del": "Berwijder geselecteerde items$NHotkey: ctrl-K", + "wt_cut": "Knip geselecteerde items <small>(en plak het ergens anders)</small>$NHotkey: ctrl-X", + "wt_cpy": "Kopieer geselecteerde items naar klembord$N(om te plakken ergens anders)$NHotkey: ctrl-C", + "wt_pst": "Plak eeen laatst geknipte / gekopieërde selectie$NHotkey: ctrl-V", + "wt_selall": "Selecteer alle bestanden$NHotkey: ctrl-A (wanneer bestand gefocused is)", + "wt_selinv": "Selectie omkeren", + "wt_zip1": "Download deze map als archief", + "wt_selzip": "Download selectie als archief", + "wt_seldl": "Download selectie als losse bestanden$NHotkey: Y", + "wt_npirc": "Kopieer irc-geformarteerde track info", + "wt_nptxt": "Kopieer platte tekst track info", + "wt_m3ua": "Aan m3u afspeellijst toevoegen (klik 📻kopieer later)", + "wt_m3uc": "Kopieer m3u playlist naar klembord", + "wt_grid": "Verwissel grid / lijst weergave$NHotkey: G", + "wt_prev": "Vorig nummer$NHotkey: J", + "wt_play": "Afspelen / pauzeer$NHotkey: P", + "wt_next": "Volgend nummer$NHotkey: L", + + "ul_par": "Parallel uploads:", + "ut_rand": "Willekeurige bestandsnaam", + "ut_u2ts": "Kopieer de laatste-gewijzigde tijdstamp$Nvan je bestandsysteem naar de server\">📅", + "ut_ow": "Overschrijf bestaande bestanden op de server?$N🛡️: nooit (zal in plaats daarvan een nieuwe bestandsnaam genereren)$N🕒: overschrijven als de server-bestand ouder is dan het geüploade bestand$N♻️: altijd overschrijven als de bestanden verschillend zijn", + "ut_mt": "Ga door met hashen van andere bestanden tijdens het uploaden$N$Moet je misschien uitschakelen als je CPU of HDD het niet aan kan", + "ut_ask": 'Vraag voor bevestiging voordat het uploaden start">💭', + "ut_pot": "Verbeter de uploadsnelheid voor langzame apparaten$Ndoor de interface minder complex te maken", + "ut_srch": "Niet uploaden, maar check of de bestanden als op de server bestaan$N (checkt alle mappen die waar jij toegang op hebt)", + "ut_par": "Pauzeer bij zetten het op 0$N$Nverhoog als je verbinding traag is$N$Nhou het op 1 als je netwerk of server HDD het niet aankan", + "ul_btn": "Drop bestanden / mappen
hier (of klik mij)", + "ul_btnu": "U P L O A D", + "ul_btns": "Z O E K E N", + + "ul_hash": "Gash", + "ul_send": "Verstuur", + "ul_done": "Klaar", + "ul_idle1": "Geen uploads in wachtrij", + "ut_etah": "Gemiddelde <em>hashing</em> snelheid en geschatte tijd tot de voltooiing", + "ut_etau": "Gemiddelde <em>upload</em> snelheid en geschatte tijd tot voltooiing", + "ut_etat": "Gemiddelde <em>totale</em> snelheid en geschatte tijd tot voltooiing", + + "uct_ok": "Succesvol afgerond", + "uct_ng": "Niet goed: gefaald / geweigerd / niet gevonden", + "uct_done": "ok en ng gecombineerd", + "uct_bz": "Hashing van uploads", + "uct_q": "Inactief, in afwachting", + + "utl_name": "Bestandsnaam", + "utl_ulist": "Lijst", + "utl_ucopy": "Kopieer", + "utl_links": "Links", + "utl_stat": "Status", + "utl_prog": "Vooruitgang", + + // keep short: + "utl_404": "404", + "utl_err": "FOUT", + "utl_oserr": "OS-FOUT", + "utl_found": "gevonden", + "utl_defer": "Uitgesteld", + "utl_yolo": "YOLO", + "utl_done": "klaar", + + "ul_flagblk": "De bestanden zijn toegevoegd aan de wachtrij
maar er is een drukke up2k bezig in een andere tabblad,
wachten totdat die eerst klaar is", + "ul_btnlk": "De server configuratie heeft deze schakelaar versleuteld in deze staat", + + "udt_up": "Upload", + "udt_srch": "Zoeken", + "udt_drop": "Laat hier los", + + "u_nav_m": '
Hey, wat heb jij daar?
Enter = Bestanden (een of meer)\nESC = Een map (inclusief submappen)', + "u_nav_b": 'BestandenEen map', + + "cl_opts": "Switches", + "cl_themes": "Thema", + "cl_langs": "Taal", + "cl_ziptype": "Download map als", + "cl_uopts": "up2k switches", + "cl_favico": "Favicon", + "cl_bigdir": "Item limiet in map", + "cl_hsort": "#sorteer", + "cl_keytype": "Key notaties", + "cl_hiddenc": "Verborgen kolomen", + "cl_hidec": "Verborgen", + "cl_reset": "Reset", + "cl_hpick": "Tik op de kolomkoppen om ze in de onderstaande tabel te verbergen", + "cl_hcancel": "Kolumn verbergen geannuleerd", + + "ct_grid": '田 grid', + "ct_ttips": '◔ ◡ ◔">ℹ️ tooltips', + "ct_thumb": 'In grid-overzicht, wissel tussen iconen of thumbnails$NHotkey: T">🖼️ thumbs', + "ct_csel": 'Gebruik CTRL en SHIFT voor de bestand selectie in grid-overzicht>sel', + "ct_ihop": 'Als je afbeeldingviewer afsluit, scroll omlaag naar de laatst bekeken bestand">g⮯', + "ct_dots": 'Laat verborgen bestanden zien (als de server dat toestaat)">dotfiles', + "ct_qdel": 'Waneeer je een bestand verwijderd, vraag eenmalig om bevestiging">qdel', + "ct_dir1st": 'Sorteer mappen eerst en dan de bestanden">📁 first', + "ct_nsort": 'Natural sort (voor bestandsnamen dat beginnen met getallen)">nsort', + "ct_readme": 'Laat README.md in mappen lijst zien">📜 readme', + "ct_utc": 'Toon alle datums en tijden in UTC">UTC', + "ct_idxh": 'Laat index.html zien in plaats van de map overzicht">htm', + "ct_sbars": 'Laat scrollbars zien">⟊', + + "cut_umod": "Als een bestand al bestaat op de server, update de 'gewijzigd' waarde op het bestand wat op de server staat met het bestand wat je geupload hebt (vereist schrijf+verwijder rechten)\">re📅", + + "cut_turbo": "De yolo knop, die wil jij waarschijnlijk NIET actief wilt hebben:$N$Ngebruik dit als je heel veel bestanden gaat uploaden EN je moest het herstarten voor een reden en je wilt doorgaan met uploaden ASAP$N$Ndit vervangt de hash-check met een simpele "heeft dit dezelfde bestands groote op de server?", zo als de bestands inhoud verschillend is, dan worden ze NIET geupload$N$NJe zou deze optie weer uit moeten zetten als de upload klaar is en dan "upload" de zelfde bestanden opnieuw uploaden zo de client het kan verifieren\">turbo", + + "cut_datechk": "Heeft geen effect tenzij de turbo knop actief is$N$Nverminder de yolo factor (een klein beetje); controlleert of de bestand tijdstamp op de server hetzelfde is met het geuploade bestand$N$Ndit zou in theorie de meest onvoltooide/onvoledige uploads, maar dit is geen vervaning voor de verificatie-check met de turbo knop uitgeschakeld daarna\">date-chk", + + "cut_u2sz": "Grote (in MiB) voor elk geuploade stuk; grote waardes vliegen beter over de Atlantische Oceaan. Probeer lage waardes op zeer onstabiele verbindingen", + + "cut_flag": "Alleen een tabblad kan bestanden uploaden $N -- andere tabbladen moeten deze optie ook actief hebben $N -- dit heeft alleen effect op de tabbladen die op hetzelfde domain zijn", + + "cut_az": "Bestanden uploaden in alfabetische volgorde, in plaats van kleinste bestanden eerst$N$Nalfabetische volgorde kan het makkelijker maken om te zien of er wat fout is gegaan op de server, dit maakt het uploaden ietsjes trager op fiber / LAN", + + "cut_nag": "Systeem notificatie weergeven als een upload voltooid is$N(alleen als de browser of tabblad niet actief is)", + "cut_sfx": "Geluid waarschuwing afspelen als een upload voltooid is$N(alleen als de browser of tabblad niet actief is)", + + "cut_mt": "Gebruik multithreading om bestands-hashing te versnellen$N$Ndit gebruikt web-workers en vereist$Nmeer geheugen (tot wel 512 MiB extra)$N$Nmaakt https 30% sneller en http 4.5x sneller\">mt", + + "cut_wasm": "Gebruik wasm in plaats van de webbrowser ingebouwde hasher; verbetert de snelheid op chrome-gebaseerde webbrowsers maar verhoogd CPU gebruik, veel oude versie van chrome hebben een bug dat een geheugen lek heeft, dat kan alle geheugen in gebruik nemen en crashen als dit actief is\">wasm", + + "cft_text": "Favicon tekst (laat leeg en vernieuw om uit te schakelen)", + "cft_fg": "Voorgrondkleur", + "cft_bg": "Achtergrondkleur", + + "cdt_lim": "Max aantal bestanden laten zien in een map", + "cdt_ask": "Als helemaal naar beneden gescrolld bent,$Nin plaats van meer inladen,$Nvraag wat het moet doen", + "cdt_hsort": "Hoeveel sorteerregels (<code>,sorthref</code>) moeten er in media-URL's worden opgenomen? Als je dit op 0 instelt, worden de sorteerregels in medialinks ook genegeerd wanneer erop geklikt word.", + + "tt_entree": "Laat navpane zien (directoryboom zijbalk)$NHotkey: B", + "tt_detree": "Laat breadcrumbs zien$NHotkey: B", + "tt_visdir": "Scroll naar geselecteerde map", + "tt_ftree": "Verwissel tussen directoryboom / tekst bestanden$NHotkey: V", + "tt_pdock": "Laat bovenliggende mappen zien in een vastgezet deelvenster bovenaan", + "tt_dynt": "Automatisch groeien naarmate de directoryboom zich uitbreidt", + "tt_wrap": "Automatische terugloop", + "tt_hover": "Laat overlopenden lijnen zien bij zweven$N(stopt het scrollen tenzij de muis in de linker gedeelte van het scherm is)", + + "ml_pmode": "Aan het einde van de map...", + "ml_btns": "Cmds", + "ml_tcode": "Transcode", + "ml_tcode2": "Transcode naar", + "ml_tint": "Tint", + "ml_eq": "Audio-equalizer", + "ml_drc": "Dynamisch bereikcompressor", + + "mt_loop": "Loop/herhaal een nummer\">🔁", + "mt_one": "Stop na een nummer\">1️⃣", + "mt_shuf": "Shuffle alle muziek in alle mappen\">🔀", + "mt_aplay": "Autoplay als er een song-ID staat in de link waarop je hebt geklikt om naar de server te gaan$N$NAls u dit uitschakelt, wordt de pagina-URL ook niet meer bijgewerkt met nummer-ID's tijdens het afspelen van muziek. Dit voorkomt automatisch afspelen als deze instellingen verloren gaan, maar de URL behouden blijft.\">a▶", + "mt_preload": "Begin het laden van de volgende nummer vlak voordat de huidige nummer het einde bereikt voor gapless playback\">preload", + "mt_prescan": "Ga naar de volgende map voordat de laatste nummer eindigd$NMaakt de webbrower blij$NZo het afspelen van muziek niet gestopt word\">nav", + "mt_fullpre": "Probeer het hele nummer vooraf te laden;$N✅ activeer dit op onstabiele verbindingen,$N❌ zet uit als je waarschijnlijk een trage verbinding hebt\">full", + "mt_fau": "Op telefoons, voorkom muziek van stoppen als de volgende nummer niet snel genoeg voorgeladen is (kan de weergave van tags glitchy maken)\">☕️", + "mt_waves": "Waveform zoekbar:$NToon audio-amplitude in de zoekbar\">~s", + "mt_npclip": "Knoppen tonen voor het clipboarden van het nummer dat op dat moment wordt afgespeeld\">/np", + "mt_m3u_c": "Knoppen tonen om de geselecteerde nummers als m3u8-afspeellijstitems te clipboarden\">📻", + "mt_octl": "OS-integratie (media hotkeys / osd)\">os-ctl", + "mt_oseek": "Zoeken via os-integratie mogelijk maken$N$NNotitie: op sommige toestellen (iPhones) dit vervcangt de volgende-nummer knop\">seek", + "mt_oscv": "Albumhoes weergeven in osd\">art", + "mt_follow": "Het afgespeelde nummer in beeld houden\">🎯", + "mt_compact": "Compacte bedieningselementen\">⟎", + "mt_uncache": "Cache wissen  (Probeer dit als uw browser een kapotte kopie van een nummer heeft gecached, waardoor het niet afgespeeld kan worden)\">uncache", + "mt_mloop": "De open map herhalen\">🔁 loop", + "mt_mnext": "Laad de volgende map en ga verder\">📂 next", + "mt_mstop": "Stoppen met afspelen\">⏸ stop", + "mt_cflac": "flac / wav omzetten naar opus\">flac", + "mt_caac": "aac / m4a omzetten naar opus\">aac", + "mt_coth": "Alle andere bestanden (geen mp3) converteren naar opus\">oth", + "mt_c2opus": "Beste keuze voor computers, laptops, android\">opus", + "mt_c2owa": "opus-weba, voor iOS 17.5 en nieuwer\">owa", + "mt_c2caf": "opus-caf, voor iOS 11 tot en met iOS 17\">caf", + "mt_c2mp3": "Gebruik dit hele oude toestellen\">mp3", + "mt_c2ok": "Mooi, goede keuze", + "mt_c2nd": "Dat is niet het aanbevolen uitvoerformaat voor uw apparaat, maar dat is prima", + "mt_c2ng": "Uw apparaat lijkt dit uitvoerformaat niet te ondersteunen, maar we gaan het toch proberen", + "mt_xowa": "iOS bevat bugs waardoor dit formaat niet op de achtergrond kan worden afgespeeld; gebruik in plaats daarvan caf of mp3.", + "mt_tint": "Achtergrond helderheid (0-100) op de zoekbalk om bufferen minder storend te maken", + "mt_eq": "Schakelt de equalizer en gain-control in;$N$Nboost <code>0</code> = standaard 100% volume (ongeweijzigd)$N$Nwidth <code>1  </code> = standaard stereo (ongeweijzigd)$Nwidth <code>0.5</code> = 50% links-rechts crossfeed$Nwidth <code>0  </code> = mono$N$Nboost <code>-0.8</code> & width <code>10</code> = stemverwijdering :^)$N$NDoor de equalizer in te schakelen, worden gapless albums volledig gapless. Laat hem dus aanstaan met alle waarden op nul (behalve width = 1) als je dat belangrijk vindt.", + "mt_drc": "Schakelt de dynamic range compressor in (volume flattener / brickwaller); schakelt ook EQ in om de spaghetti te balanceren, dus zet alle EQ velden behalve ‘width’ op 0 als je dat niet wilt.$N$Nverlaagt het volume van audio boven THRESHOLD dB; voor elke RATIO dB voorbij THRESHOLD is er 1 dB output, dus standaardwaarden van tresh -24 en ratio 12 betekenen dat het nooit luider dan -22 dB zou moeten worden en het is veilig om de equalizer boost te verhogen tot 0.8, of zelfs 1.8 met ATK 0 en een enorme RLS zoals 90 (werkt alleen in firefox; RLS is max 1 in andere browsers)$N$N(zie wikipedia, die legt het veel beter uit)", + + "mb_play": "Afspelen", + "mm_hashplay": "Deze audio bestand afspelen?", + "mm_m3u": "Druk op Enter/OK om af te spelen\nDruk op ESC/Cancel om te bewerken", + "mp_breq": "Heeft firefox 82+ of chrome 73+ of iOS 15+", + "mm_bload": "Aan het laden...", + "mm_bconv": "Opmzetten naar {0}, even geduld...", + "mm_opusen": "Uw browser kan geen aac / m4a-bestanden afspelen;\ntranscodering naar opus is nu ingeschakeld", + "mm_playerr": "Afspelen mislukt: ", + "mm_eabrt": "De afspeelpoging is geannuleerd", + "mm_enet": "Je internetverbinding is onstabiel", + "mm_edec": "Dit bestand is vermoedelijk beschadigd??", + "mm_esupp": "Uw browser begrijpt deze audio-formaat niet", + "mm_eunk": "Onbekende fout", + "mm_e404": "Kan audio niet afspelen; fout 404: Bestand niet gevonden..", + "mm_e403": "Kan audio niet afspelen; fout 403: Toegang geweigerd.\n\nProbeer op F5 te drukken om opnieuw te laden, misschien ben je uitgelogd", + "mm_e500": "Kan geen audio afspelen; fout 500: Controleer serverlogs.", + "mm_e5xx": "Kan geen audio afspelen; serverfout ", + "mm_nof": "Geen audiobestanden meer vinden in de buurt", + "mm_prescan": "Op zoek naar muziek om als volgende te spelen...", + "mm_scank": "Het volgende nummer gevonden:", + "mm_uncache": "Cache gewist; alle nummers worden opnieuw gedownload bij de volgende keer afspelen", + "mm_hnf": "Dat liedje bestaat niet meer", + + "im_hnf": "Deze afbeelding bestaat niet meer", + + "f_empty": 'Deze map is leeg', + "f_chide": 'Dit verbergt kolom «{0}»\n\nje kunt kolommen verbergen op de instellingen tabblad', + "f_bigtxt": "Dit bestand is {0} MiB groot -- echt bekijken als tekst?", + "f_bigtxt2": "Wilt u alleen het einde van het bestand bekijken? Dit maakt ook volgen/tailen mogelijk, waarbij nieuw toegevoegde tekstregels in realtime worden weergegeven.", + "fbd_more": '
{0} van de {1} bestanden weergegeven; Toon {2} of Laat alles zien
', + "fbd_all": '
{0} van de {1} bestanden weergegeven; Laat alles zien
', + "f_anota": "Alleen {0} van de {1} items zijn geselecteerd;\nom de volledige map te selecteren, scrol je eerst naar beneden", + + "f_dls": 'de bestandslinks in de huidige map zijn veranderd in downloadlinks', + + "f_partial": "Om een bestand dat momenteel wordt geüpload veilig te downloaden, klikt u op het bestand met dezelfde bestandsnaam, maar zonder de bestandsextensie .PARTIAL. Druk op Annuleren of Escape om dit te doen.\n\nAls u op OK / Enter drukt, wordt deze waarschuwing genegeerd en gaat u verder met het downloaden van het gedeeltelijke .PARTIAL scratchbestand, waardoor u vrijwel zeker beschadigde gegevens krijgt.", + + "ft_paste": "plakken {0} items$NHotkey: ctrl-V", + "fr_eperm": 'kan de naam niet wijzigen:\nje hebt geen “move” rechten in deze map', + "fd_eperm": 'kan niet verwijderen:\nje hebt geen “delete” rechten in deze map', + "fc_eperm": 'kan niet knippen:\nje hebt geen “move” rechten in deze map', + "fp_eperm": 'kan niet plakken:\nje hebt geen “schrijf” rechten in deze map', + "fr_emore": "selecteer ten minste één item om te hernoemen", + "fd_emore": "selecteer minstens één item om te verwijderen", + "fc_emore": "selecteer ten minste één item om te knippen", + "fcp_emore": "selecteer ten minste één item om naar het klembord te kopiëren", + + "fs_sc": "Deel de map waarin je je bevindt", + "fs_ss": "De geselecteerde bestand(en) delen", + "fs_just1d": "U kunt niet meer dan één map selecteren\nof mix bestanden en mappen in één selectie", + "fs_abrt": "❌ Afbreken", + "fs_rand": "🎲 rand.naam", + "fs_go": "✅ Maak share", + "fs_name": "Naam", + "fs_src": "Bron", + "fs_pwd": "Wachtwoord", + "fs_exp": "Verloopt", + "fs_tmin": "min", + "fs_thrs": "uur", + "fs_tdays": "dag(en)", + "fs_never": "eeuwig", + "fs_pname": "Optionele linknaam; is willekeurig als deze leeg is", + "fs_tsrc": "Het bestand of de map die u wilt delen", + "fs_ppwd": "Optioneel wachtwoord", + "fs_w8": "Delen...", + "fs_ok": "Druk op Enter/OK naar klembord te zetten\Druk op ESC/Cancel om te sluiten", + + "frt_dec": "Kan sommige gevallen van gebroken bestandsnamen oplossen\">url-decode", + "frt_rst": "Gewijzigde bestandsnamen terugzetten naar de oorspronkelijke namen\">↺ reset", + "frt_abrt": "Afbreken en dit venster sluiten\">❌ Annuleren", + "frb_apply": "HERNOEMEN TOEPASSEN", + "fr_adv": "Batch / metadata / patroon hernoemen\">Geavanceerd", + "fr_case": "Hoofdlettergevoelige regex\">case", + "fr_win": "Windows-veilige namen; vervangen <>:"\\|?* met japanse tekens over de volledige breedte\">win", + "fr_slash": "Vervang / met een teken waardoor er geen nieuwe mappen worden gemaakt\">geen /", + "fr_re": "Regex zoekpatroon om toe te passen op originele bestandsnamen; naar capturing groups kan worden verwezen in het onderstaande opmaakveld zoals <code>(1)</code> en <code>(2)</code> enzovoort", + "fr_fmt": "Geïnspireerd door foobar2000 :$N<code>(titel)</code> wordt vervangen door de titel van het nummer,$N<code>[(artiest) - ](titel)</code> sla [dit] gedeelte over als artiest leeg is$N<code>$lpad((tn),2,0)</code> vult tracknummer op tot 2 cijfers (0X)", + "fr_pdel": "Verwijderen", + "fr_pnew": "Opslaan als", + "fr_pname": "Geef een naam op voor je nieuwe preset", + "fr_aborted": "Afgebroken", + "fr_lold": "Oude naam", + "fr_lnew": "Nieuwe naam", + "fr_tags": "Tags voor de geselecteerde bestanden (alleen-lezen, alleen ter referentie):", + "fr_busy": "Hernoemen van {0} items...\n\n{1}", + "fr_efail": "Hernoemen mislukt:\n", + "fr_nchg": "{0} van de nieuwe namen zijn gewijzigd als gevolg van win en/of geen /\n\nOK om door te gaan met deze gewijzigde nieuwe namen?", + + "fd_ok": "Verwijderen OK", + "fd_err": "Verwijderen mislukt:\n", + "fd_none": "Er is niets verwijderd; misschien geblokkeerd door serverconfiguratie (xbd)?", + "fd_busy": "{0} items verwijderen...\n\n{1}", + "fd_warn1": "VERWIJDER deze {0} items?", + "fd_warn2": "LAATSTE KANS! Geen manier om ongedaan te maken. Verwijderen?", + + "fc_ok": "Knip {0} items", + "fc_warn": 'Knip {0} items\n\nmaar: alleen deze browser-tabblad kan weer plakken\n(omdat de selectie zo enorm is)', + + "fcc_ok": "{0} items naar klembord gekopieerd", + "fcc_warn": '{0} items naar klembord gekopieerd\n\maar: alleen deze browser-tabblad kan weer plakken\n(omdat de selectie zo enorm is)', + + "fp_apply": "Gebruik deze namen", + "fp_ecut": "Knip of kopieer eerst enkele bestanden/mappen om te verplaatsen/plakken\n\nnotitie: je kunt knippen/plakken in verschillende browsertabbladen", + "fp_ename": "{0} items kunnen hier niet worden verplaatst omdat de namen al in gebruik zijn. Geef ze hieronder een nieuwe naam om verder te gaan, of verwijder de naam om ze over te slaan:", + "fcp_ename": "{0} items kunnen hier niet worden gekopieerd omdat de namen al in gebruik zijn. Geef ze hieronder een nieuwe naam om verder te gaan, of verwijder de naam om ze over te slaan:", + "fp_emore": "Er zijn nog enkele bestandsnaambotsingen die moeten worden opgelost", + "fp_ok": "Verplaatsen OK", + "fcp_ok": "Kopiëren OK", + "fp_busy": "{0} items verplaatsen...\n\n{1}", + "fcp_busy": "{0} items kopiëren...\n\n{1}", + "fp_err": "Verplaatsen mislukt:\n", + "fcp_err": "Kopieëren mislukt:\n", + "fp_confirm": "Verplaats deze {0} items hierheen?", + "fcp_confirm": "Kopieer deze {0} items hier?", + "fp_etab": 'Kan klembord van ander browsertabblad niet lezen', + "fp_name": "Een bestand uploaden vanaf uw apparaat. Geef het een naam:", + "fp_both_m": '
Kies wat je wilt plakken
Enter = Verplaatsen {0} bestanden van «{1}»\nESC = Upload {2} bestanden van je apparaat', + "fcp_both_m": '
Kies wat je wilt plakken
Enter = Kopieer {0} bestanden van «{1}»\nESC = Upload {2} bestanden van je apparaat', + "fp_both_b": 'VerplaatsUpload', + "fcp_both_b": 'KopieerUpload', + + "mk_noname": "Voer een naam in het tekstveld aan de linkerkant voordat je verder gaat :p", + + "tv_load": "Tekstdocument laden:\n\n{0}\n\n{1}% ({2} van de {3} MiB geladen)", + "tv_xe1": "Kon tekstbestand niet laden:\n\nfout ", + "tv_xe2": "404, bestand niet gevonden", + "tv_lst": "Lijst met tekstbestanden in", + "tvt_close": "Terugkeren naar mapweergave$NHotkey: M (of Esc)\">❌ Sluiten", + "tvt_dl": "Download dit bestand$NHotkey: Y\">💾 download", + "tvt_prev": "Vorig document tonen$NHotkey: i\">⬆ prev", + "tvt_next": "Volgende document tonen$NHotkey: K\">⬇ next", + "tvt_sel": "Selecteer bestand   ( voor knip / verplaats / verwijder / ... )$NHotkey: S\">sel", + "tvt_edit": "Bestand openen in teksteditor$NHotkey: E\">✏️ bewerk", + "tvt_tail": "Bestand controleren op wijzigingen; nieuwe regels in realtime weergeven\">📡 volgen", + "tvt_wrap": "Automatische terugloop\">↵", + "tvt_atail": "Vergrendelen scroll naar onderkant van pagina\">⚓", + "tvt_ctail": "Kleuren van terminals decoderen (ansi escape codes)\">🌈", + "tvt_ntail": "Terugrollimiet (hoeveel tekst geladen moeten blijven)", + + "m3u_add1": "Nummer toegevoegd aan m3u afspeellijst", + "m3u_addn": "{0} nummers toegevoegd aan m3u-afspeellijst", + "m3u_clip": "m3u-afspeellijst nu gekopieerd naar klembord\n\nje moet een nieuw tekstbestand maken met de naam iets.m3u en de afspeellijst in dat document plakken; dit maakt het afspeelbaar", + + "gt_vau": "Laat geen video's zien, speel alleen de audio af\">🎧", + "gt_msel": "Schakel bestandsselectie in; ctrl-klik op een bestand om te openen$N$N<em>indien actief: dubbelklik op een bestand / map om het te openen</em>$N$NHotkey: S\">multiselect", + "gt_crop": "Gecentreerde miniaturen\">crop", + "gt_3x": "Hi-res miniaturen\">3x", + "gt_zoom": "Zoom", + "gt_chop": "Verkorten", + "gt_sort": "Sorteer bij", + "gt_name": "naam", + "gt_sz": "grootte", + "gt_ts": "datum", + "gt_ext": "type", + "gt_c1": "Bestandsnamen meer inkorten (minder tonen)", + "gt_c2": "Bestandsnamen minder inkorten (meer tonen)", + + "sm_w8": "Zoeken...", + "sm_prev": "Onderstaande zoekresultaten zijn afkomstig van een eerdere zoekopdracht:\n ", + "sl_close": "Zoekresultaten sluiten", + "sl_hits": "Toont {0} treffers", + "sl_moar": "Laad meer", + + "s_sz": "grootte", + "s_dt": "datum", + "s_rd": "pad", + "s_fn": "naam", + "s_ta": "tags", + "s_ua": "op@", + "s_ad": "adv.", + "s_s1": "Minimaal MiB", + "s_s2": "Maximaal MiB", + "s_d1": "Min. iso8601", + "s_d2": "Max. iso8601", + "s_u1": "Uploaded na", + "s_u2": "en/of voor", + "s_r1": "Pad bevad   (spatie-gescheiden)", + "s_f1": "Naam bevat   (ontkennen met -nope)", + "s_t1": "Tags bevat   (^=start, einde=$)", + "s_a1": "Specifieke metadata-eigenschappen", + + "md_eshow": "Kan niet weergeven ", + "md_off": "[📜readme] uitgeschakeld in [⚙️] -- document verborgen", + + "badreply": "Mislukt om antwoord van server te parsen", + + "xhr403": "403: Toegang geweigerd\n\nprobeer F5 in te drukken, misschien ben je uitgelogd", + "xhr0": "Onbekend (waarschijnlijk verbinding met server verloren of server is offline)", + "cf_ok": "Sorry daarvoor -- DD" + wah + "OS-bescherming ingeschakeld\n\nalles zou binnen ongeveer 30 seconden moeten hervatten\n\nals er niets gebeurt, druk dan op F5 om de pagina opnieuw te laden", + "tl_xe1": "Kon submappen niet weergeven:\n\nfout ", + "tl_xe2": "404: Map niet gevonden", + "fl_xe1": "Kon bestanden in map niet weergeven:\n\nfout ", + "fl_xe2": "404: Map niet gevonden", + "fd_xe1": "Kon submap niet aanmaken:\n\nfout ", + "fd_xe2": "404: Bovenliggende map niet gevonden", + "fsm_xe1": "Kon bericht niet verzenden:\n\nfout ", + "fsm_xe2": "404: Bovenliggende map niet gevonden", + "fu_xe1": "Mislukt om unpost lijst van server te laden:\n\nfout ", + "fu_xe2": "404: Bestand niet gevonden??", + + "fz_tar": "gnu-tar bestand uitpakken (linux / mac)", + "fz_pax": "pax-formaat tar uitpakken (trager)", + "fz_targz": "gnu-tar met gzip niveau 3 compressie$N$Ndit is meestal erg langzaam, dus gebruik in plaats daarvan ongecomprimeerde tar", + "fz_tarxz": "gnu-tar met xz-niveau 1 compressie$N$Ndit is meestal erg langzaam, dus gebruik in plaats daarvan ongecomprimeerde tar", + "fz_zip8": "Zip met utf8 bestandsnamen (misschien onhandig op windows 7 en ouder)", + "fz_zipd": "Zip met traditionele cp437-bestandsnamen, voor echt oude software", + "fz_zipc": "cp437 met crc32 vroeg berekend$Nvoor MS-DOS PKZIP v2.04g (oktober 1993)$N(het duurt langer voordat het downloaden kan beginnen)", + + "un_m1": "Hieronder kunt u uw recente uploads verwijderen (of onvoltooide uploads afbreken)", + "un_upd": "Vernieuwen", + "un_m4": "of deel de bestanden die hieronder zichtbaar zijn:", + "un_ulist": "Toon", + "un_ucopy": "Kopieer", + "un_flt": "Optionele filter:  URL moet het volgende bevatten", + "un_fclr": "Reset filter", + "un_derr": 'unpost-verwijderen mislukt:\n', + "un_f5": 'Er is iets kapot, probeer te verversen of druk op F5', + "un_uf5": "Sorry, maar u moet de pagina vernieuwen (bijvoorbeeld door op F5 of CTRL-R te drukken) voordat deze upload kan worden afgebroken.", + "un_nou": 'Waarschuwing: server te druk om onvoltooide uploads weer te geven; klik straks op de "refresh" link', + "un_noc": 'Waarschuwing: unpost van volledig geüploade bestanden is niet ingeschakeld/toegestaan in de serverconfiguratie', + "un_max": "Toont de eerste 2000 bestanden (gebruik de filter)", + "un_avail": "{0} recente uploads kunnen worden verwijderd
{1} onvoltooide kunnen worden afgebroken", + "un_m2": "Gesorteerd op uploadtijd; meest recente eerst:", + "un_no1": "sike! geen enkele upload is recent genoeg", + "un_no2": "sike! geen uploads die aan dat filter voldoen zijn voldoende recent", + "un_next": "Verwijder de volgende {0} bestanden", + "un_abrt": "Afbreken", + "un_del": "Verwijderen", + "un_m3": "Je recente uploads laden...", + "un_busy": "Verwijderen van {0} bestanden...", + "un_clip": "{0} links gekopieerd naar klembord", + + "u_https1": "Je moet", + "u_https2": "overschakelen naar https", + "u_https3": "voor betere prestaties", + "u_ancient": 'Je browser is indrukwekkend oud -- misschien moet je in plaats daarvan bup gebruiken', + "u_nowork": "Je moet firefox 53+ of chrome 57+ of iOS 11+ hebben", + "tail_2old": "Je moet firefox 105+ of chrome 71+ of iOS 14.5+ hebben", + "u_nodrop": 'Je browser is te oud voor uploaden via slepen en neerzetten', + "u_notdir": "Dat is geen map!\n\nuw browser is te oud,\nprobeer in plaats daarvan sleep en neerzetten", + "u_uri": "Om afbeeldingen te slepen vanuit andere browser tabblad,\nplaats deze dan op de grote uploadknop", + "u_enpot": 'Overschakelen naar potato UI (kan uploadsnelheid verbeteren)', + "u_depot": 'Overschakelen naar fancy UI (kan uploadsnelheid verminderen)', + "u_gotpot": 'Overschakelen naar de potato UI voor verbeterde uploadsnelheid,\n\nVoel je vrij om het er niet mee eens te zijn en schakel terug!', + "u_pott": "

Bestanden:   {0} klaar,   {1} mislukt,   {2} bezig,   {3} in de wachtrij

", + "u_ever": "Dit is de basis uploader; up2k heeft minstens het volgende nodig
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'Dit is de basis uploader; up2k is beter', + "u_uput": 'Optimaliseren voor snelheid (checksum overslaan)', + "u_ewrite": 'Je hebt geen schrijftoegang tot deze map', + "u_eread": 'Je hebt geen leestoegang tot deze map', + "u_enoi": 'Zoeken naar bestanden is niet ingeschakeld in de serverconfiguratie', + "u_enoow": "Overschrijven zal hier niet werken; je heb verwijder toestemming nodig", + "u_badf": 'Deze {0} bestanden (van {1} totaal) zijn overgeslagen, mogelijk door bestandssysteemmachtigingen:\n\n', + "u_blankf": 'Deze {0} bestanden (van {1} totaal) zijn leeg; alsnog uploaden?\n\n', + "u_applef": 'Deze {0} bestanden (van {1} totaal) zijn waarschijnlijk ongewenst;\nKlik op OK/Enter om de volgende bestanden over te slaan,\Klik op Cancel/ESC niet uit te sluiten en deze ook te uploaden:\n\n', + "u_just1": '\nMisschien werkt het beter als je slechts één bestand selecteert', + "u_ff_many": "Als je Linux / MacOS / Android, gebruikt dan kan deze hoeveelheid bestanden Firefox crashen!\nals dat gebeurt, probeer het dan opnieuw (of gebruik Chrome).", + "u_up_life": "Deze upload wordt verwijderd van de server\n{0} nadat het is voltooid", + "u_asku": 'Upload deze {0} bestanden naar {1}', + "u_unpt": "Je kunt deze upload ongedaan maken / verwijderen met de linkerbovenhoek 🧯", + "u_bigtab": 'We staan op het punt om {0} bestanden te tonen\n\nDit kan uw browser laten crashen, weet je het zeker??', + "u_scan": 'Bestanden scannen...', + "u_dirstuck": 'Directory iterator liep vast bij het benaderen van het volgende {0} items; zal het volgende overslaan:', + "u_etadone": 'Klaar ({0}, {1} bestanden)', + "u_etaprep": '(klaarmaken om te uploaden)', + "u_hashdone": 'hashing klaar', + "u_hashing": 'Hash', + "u_hs": 'Hallo zeggen...', + "u_started": "De bestanden worden nu geüpload; zie [🚀]", + "u_dupdefer": "Duplicaat; wordt verwerkt na alle andere bestanden", + "u_actx": "klik op deze tekst om prestatieverlies
bij het overschakelen naar andere vensters/tabbladen te voorkomen", + "u_fixed": "OK!  Fixed it 👍", + "u_cuerr": "Mislukt bij het uploaden van stuk {0} van {1};\nwaarschijnlijk ongevaarlijk, doorgaan\n\nbestand: {2}", + "u_cuerr2": "Upload door server geweigerd (stuk {0} van {1});\nzal later opnieuw proberen\n\nbestand: {2}\n\nfout ", + "u_ehstmp": "Zal opnieuw proberen; zie rechtsonder", + "u_ehsfin": "Server heeft het verzoek om de upload te finaliseren afgewezen; opnieuw proberen...", + "u_ehssrch": "Server heeft de zoekaanvraag afgewezen; opnieuw proberen...", + "u_ehsinit": "Server heeft het verzoek om het uploaden te starten afgewezen; opnieuw proberen...", + "u_eneths": "Netwerkfout tijdens het uitvoeren van de uploadhanddruk; opnieuw proberen...", + "u_enethd": "Netwerkfout tijdens het testen van het bestaan van het doel; opnieuw proberen...", + "u_cbusy": "Wachten tot de server ons weer vertrouwt na een netwerkstoring...", + "u_ehsdf": "Server heeft geen schijfruimte meer!\n\nzal blijven proberen, voor het geval iemand genoeg ruimte vrijmaakt om door te gaan", + "u_emtleak1": "Het lijkt erop dat uw webbrowser een geheugenlek heeft;\nprobeer", + "u_emtleak2": ' over te schakel over naar https (aanbevolen) of ', + "u_emtleak3": ' ', + "u_emtleakc": 'Probeer het volgende:\n
  • druk op F5 om de pagina te verversen
  • dan schakel de  mt  uit, deze knop staat in  ⚙️ instellingen
  • en probeer de upload opnieuw
Uploaden zal wat langzamer gaan, maar ja.\nSorry voor de problemen!\n\nPS: chrome v107 heeft een bugfix voor dit', + "u_emtleakf": '{robeer het volgende:\n
  • druk op F5 om de pagina te verversen
  • dan activeer 🥔 (aardappel) in de upload scherm
  • en probeer de upload opnieuw
\nPS: firefox heeft mogelijk een fix op een gegeven moment', + "u_s404": "Niet gevonden op server", + "u_expl": "Leg uit", + "u_maxconn": "De meeste browsers beperken dit tot 6, maar firefox laat je dit verhogen met network.http.max-persistent-connections-per-server in about:config", + "u_tu": '

WAARSCHUWING: turbo ingeschakeld,  webbrowser detecteert en hervat onvolledige uploads mogelijk niet; zie de tooltip van de turboknop

', + "u_ts": '

WAARSCHUWING: turbo ingeschakeld,  zoekresultaten kunnen onjuist zijn; zie turbo-knop tooltip

', + "u_turbo_c": "Turbo is uitgeschakeld in serverconfiguratie", + "u_turbo_g": "Turbo uitgeschakeld, je geen recht om mappen in deze volume te tonen", + "u_life_cfg": 'Automatisch verwijderen na minuten (of uur)', + "u_life_est": 'Upload wordt verwijderd ---', + "u_life_max": 'Deze map dwingt een\nmaximale levensduur van {0} af', + "u_unp_ok": 'unpost is toegestaan voor {0}', + "u_unp_ng": 'unpost zijn NIET toegestaan', + "ue_ro": 'Je toegang tot deze map is alleen-lezen\n\n', + "ue_nl": 'Je bent momenteel niet ingelogd', + "ue_la": 'Je bent momenteel aangemeld als "{0}"', + "ue_sr": 'U bevindt zich momenteel in de bestandszoekmodus\n\nschakel over naar uploadmodus door op het vergrootglas te klikken 🔎 (naast de grote ZOEK-knop), en probeer opnieuw te uploaden\n\nsorry', + "ue_ta": 'Probeer opnieuw te uploaden, het zou nu moeten werken', + "ue_ab": "Dit bestand wordt al geüpload naar een andere map en die upload moet worden voltooid voordat het bestand naar een andere map kan worden geüpload.\n\nU kunt de eerste upload afbreken en laten vergeten met de linkerbovenhoek 🧯", + "ur_1uo": "OK: Bestand succesvol geüpload", + "ur_auo": "OK: Alle {0} bestanden succesvol geüpload", + "ur_1so": "OK: Bestand gevonden op server", + "ur_aso": "OK: Alle {0} bestanden gevonden op server", + "ur_1un": "Uploaden mislukt, sorry", + "ur_aun": "Alle {0} uploads mislukt, sorry", + "ur_1sn": "Bestand NIET gevonden op server", + "ur_asn": "De {0} bestanden zijn NIET gevonden op de server", + "ur_um": "Voltooid;\n{0} upload(s) OK,\n{1} upload(s) mislukt, sorry", + "ur_sm": "Voltooid;\n{0} bestand(en) gevonden op de server,\n{1} bestand(en) NIET gevonden op de server", + + "lang_set": "Vernieuw de pagina om de wijziging door te voeren?", + }, "rus": { "tt": "Русский", @@ -5023,7 +5650,7 @@ var Ls = { }, }; -var LANGS = ["eng", "nor", "chi", "deu", "fin", "rus", "spa", "ukr"]; +var LANGS = ["eng", "nor", "chi", "deu", "fin", "nld", "rus", "spa", "ukr"]; if (window.langmod) langmod(); diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 4ad53b9b..06cb0bf7 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -176,6 +176,48 @@ var Ls = { "af1": "näytä viimeaikaiset lataukset", "ag1": "näytä tunnetut IdP-käyttäjät", }, + "nld": { + "a1": "Update", + "b1": "Hallo, hoe gaat het met jou?   (Je bent niet ingelogd)", + "c1": "Uitloggen", + "d1": "Voorwaarde", + "d2": "Toont de status van alle actieve threads", + "e1": "Configuratie opnieuw laden.", + "e2": "Leest configuratiebestanden opnieuw in$N(accounts, volumes, volumeschakelaars)$Nen brengt alle e2ds-volumes in kaart$N$Nopmerking: veranderingen in globale parameters$Nvereist een volledige herstart van de server", + "f1": "Je kan het volgende lezen:", + "g1": "Je kan naar het volgende uploaden:", + "cc1": "Schakelaars en dergelijke:", + "h1": "k304 uitschakelen", + "i1": "k304 inschakelen", + "j1": "k304 verbreekt de verbinding voor elke HTTP 304. Dit helpt tegen bepaalde proxy servers die kunnen vastlopen/plotseling stoppen met het laden van pagina's, maar het vermindert ook de prestaties aanzienlijk", + "k1": "Instellingen resetten", + "l1": "Inloggen:", + "m1": "Welkom terug,", + "n1": "404: bestand bestaat niet  ┐( ´ -`)┌", + "o1": 'of misschien heb je geen toegang? probeer een wachtwoord of ga naar startscherm', + "p1": "403: toegang geweigerd  ~┻━┻", + "q1": 'Probeer een wachtwoord of ga naar startscherm', + "r1": "Ga naar startscherm", + ".s1": "Kaart", + "t1": "Actie", + "u2": "Tijd sinds iemand voor het laatst naar de server schreef$N( upload / naamswijziging / ... )$N$N17d = 17 dagen$N1h23 = 1 uur 23 minuten$N4m56 = 4 minuten 56 secondes", + "v1": "Verbinden", + "v2": "Gebruik deze server als een lokale harde schijf", + "w1": "Overschakelen naar https", + "x1": "Wachtwoord wijzigen", + "y1": "Jou gedeelde items", + "z1": "Ontgrendel gebied:", + "ta1": "Je moet eerst een nieuw wachtwoord invoeren", + "ta2": "Herhaal om nieuw wachtwoord te bevestigen:", + "ta3": "Typefout gevonden; probeer het opnieuw", + "aa1": "Inkomend:", + "ab1": "Schakel nr. 304 uit", + "ac1": "Schakel nr. 304 in", + "ad1": "Nr. 304 stopt al het cachegebruik. Als k304 niet voldoende was, probeer dan deze. Vermenigvuldigt het dataverbruik.!", + "ae1": "Uitgaand:", + "af1": "Recent geüploade bestanden weergeven", + "ag1": "Bekende IdP-gebruikers weergeven", + }, "spa": { "a1": "actualizar", "b1": "hola   (no has iniciado sesión)", From a38e6e65d5e578fcfc6940dcb88c0269582e7556 Mon Sep 17 00:00:00 2001 From: AOTREVAI <46420278+AOTREVAI@users.noreply.github.com> Date: Sun, 3 Aug 2025 20:38:16 +0200 Subject: [PATCH 066/174] Translate to Italian (#458) * Translating to Italian * Update browser.js * sync splash.js, browser.js * Update browser.js * Update splash.js --------- Signed-off-by: AOTREVAI <46420278+AOTREVAI@users.noreply.github.com> --- copyparty/web/browser.js | 629 ++++++++++++++++++++++++++++++++++++++- copyparty/web/splash.js | 42 +++ 2 files changed, 670 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index af0c0e39..b26f02fc 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3141,6 +3141,633 @@ var Ls = { "lang_set": "ladataanko sivu uudestaan kielen vaihtamiseksi?", }, + "ita": { + "tt": "Italiano", + + "cols": { + "c": "pulsanti azione", + "dur": "durata", + "q": "qualità / bitrate", + "Ac": "codec audio", + "Vc": "codec video", + "Fmt": "formato / container", + "Ahash": "checksum audio", + "Vhash": "checksum video", + "Res": "risoluzione", + "T": "tipo file", + "aq": "qualità audio / bitrate", + "vq": "qualità video / bitrate", + "pixfmt": "subsampling / struttura pixel", + "resw": "risoluzione orizzontale", + "resh": "risoluzione verticale", + "chs": "canali audio", + "hz": "frequenza di campionamento" + }, + + "hks": [ + [ + "varie", + ["ESC", "chiudi vari elementi"], + + "file-manager", + ["G", "alterna vista lista / griglia"], + ["T", "alterna miniature / icone"], + ["⇧ A/D", "dimensione miniature"], + ["ctrl-K", "elimina selezionati"], + ["ctrl-X", "taglia selezione negli appunti"], + ["ctrl-C", "copia selezione negli appunti"], + ["ctrl-V", "incolla (sposta/copia) qui"], + ["Y", "scarica selezionati"], + ["F2", "rinomina selezionati"], + + "file-list-sel", + ["spazio", "alterna selezione file"], + ["↑/↓", "sposta cursore selezione"], + ["ctrl ↑/↓", "sposta cursore e viewport"], + ["⇧ ↑/↓", "seleziona file prec/succ"], + ["ctrl-A", "seleziona tutti i file / cartelle"], + ], [ + "navigation", + ["B", "alterna breadcrumb / pannello nav"], + ["I/K", "cartella prec/succ"], + ["M", "cartella genitore (o comprimi corrente)"], + ["V", "alterna cartelle / file di testo nel pannello nav"], + ["A/D", "dimensione pannello nav"], + ], [ + "audio-player", + ["J/L", "brano prec/succ"], + ["U/O", "salta 10sec indietro/avanti"], + ["0..9", "salta a 0%..90%"], + ["P", "play/pausa (avvia anche)"], + ["S", "seleziona brano in riproduzione"], + ["Y", "scarica brano"], + ], [ + "image-viewer", + ["J/L, ←/→", "immagine prec/succ"], + ["Home/End", "prima/ultima immagine"], + ["F", "schermo intero"], + ["R", "ruota in senso orario"], + ["⇧ R", "ruota in senso antiorario"], + ["S", "seleziona immagine"], + ["Y", "scarica immagine"], + ], [ + "video.player", + ["U/O", "salta 10sec indietro/avanti"], + ["P/K/Spazio", "play/pausa"], + ["C", "continua riproduzione successivo"], + ["V", "loop"], + ["M", "muto"], + ["[ e ]", "imposta intervallo loop"], + ], [ + "textfile-viewer", + ["I/K", "file prec/succ"], + ["M", "chiudi file di testo"], + ["E", "modifica file di testo"], + ["S", "seleziona file (per taglia/copia/rinomina)"], + ] + ], + + "m_ok": "OK", + "m_ng": "Annulla", + + "enable": "Abilita", + "danger": "PERICOLO", + "clipped": "copiato negli appunti", + + "ht_s1": "secondo", + "ht_s2": "secondi", + "ht_m1": "minuto", + "ht_m2": "minuti", + "ht_h1": "ora", + "ht_h2": "ore", + "ht_d1": "giorno", + "ht_d2": "giorni", + "ht_and": " e ", + + "goh": "control-panel", + "gop": 'cartella sorella precedente">prec', + "gou": 'cartella genitore">su', + "gon": 'prossima cartella">succ', + "logout": "Logout ", + "access": " accesso", + "ot_close": "chiudi sottomenu", + "ot_search": "cerca file per attributi, percorso / nome, tag musicali, o qualsiasi combinazione di questi$N$N<code>foo bar</code> = deve contenere sia «foo» che «bar»,$N<code>foo -bar</code> = deve contenere «foo» ma non «bar»,$N<code>^yana .opus$</code> = inizia con «yana» ed è un file «opus»$N<code>"try unite"</code> = contiene esattamente «try unite»$N$Nil formato data è iso-8601, come$N<code>2009-12-31</code> o <code>2020-09-12 23:30:00</code>", + "ot_unpost": "unpost: elimina i tuoi caricamenti recenti, o interrompi quelli non completati", + "ot_bup": "bup: uploader di base, supporta anche netscape 4.0", + "ot_mkdir": "mkdir: crea una nuova directory", + "ot_md": "new-md: crea un nuovo documento markdown", + "ot_msg": "msg: invia un messaggio al log del server", + "ot_mp": "opzioni lettore multimediale", + "ot_cfg": "opzioni di configurazione", + "ot_u2i": 'up2k: carica file (se hai accesso in scrittura) o attiva la modalità ricerca per vedere se esistono già da qualche parte sul server$N$NI caricamenti sono ripristinabili, multithreaded, e i timestamp dei file vengono preservati, ma usa più CPU di [🎈]  (l\'uploader di base)

durante i caricamenti, questa icona diventa un indicatore di progresso!', + "ot_u2w": 'up2k: carica file con supporto per il ripristino (chiudi il browser e trascina gli stessi file più tardi)$N$NMultithreaded, e i timestamp dei file vengono preservati, ma usa più CPU di [🎈]  (l\'uploader di base)

durante i caricamenti, questa icona diventa un indicatore di progresso!', + "ot_noie": 'Perfavore usa Chrome / Firefox / Edge', + + "ab_mkdir": "crea directory", + "ab_mkdoc": "nuovo doc markdown", + "ab_msg": "invia msg al log srv", + + "ay_path": "salta alle cartelle", + "ay_files": "salta ai file", + + "wt_ren": "rinomina elementi selezionati$NTasto rapido: F2", + "wt_del": "elimina elementi selezionati$NTasto rapido: ctrl-K", + "wt_cut": "taglia elementi selezionati <small>(poi incolla altrove)</small>$NTasto rapido: ctrl-X", + "wt_cpy": "copia elementi selezionati negli appunti$N(per incollarli altrove)$NTasto rapido: ctrl-C", + "wt_pst": "incolla una selezione precedentemente tagliata / copiata$NTasto rapido: ctrl-V", + "wt_selall": "seleziona tutti i file$NTasto rapido: ctrl-A (quando il file è focalizzato)", + "wt_selinv": "inverti selezione", + "wt_zip1": "scarica questa cartella come archivio", + "wt_selzip": "scarica selezione come archivio", + "wt_seldl": "scarica selezione come file separati$NTasto rapido: Y", + "wt_npirc": "copia info traccia formato irc", + "wt_nptxt": "copia info traccia testo semplice", + "wt_m3ua": "aggiungi alla playlist m3u (clicca 📻copia dopo)", + "wt_m3uc": "copia playlist m3u negli appunti", + "wt_grid": "alterna vista griglia / lista$NTasto rapido: G", + "wt_prev": "traccia precedente$NTasto rapido: J", + "wt_play": "play / pausa$NTasto rapido: P", + "wt_next": "traccia successiva$NTasto rapido: L", + + "ul_par": "caricamenti paralleli:", + "ut_rand": "randomizza nomi file", + "ut_u2ts": "copia il timestamp di ultima modifica$Ndal tuo filesystem al server\">📅", + "ut_ow": "sovrascrivere file esistenti sul server?$N🛡️: mai (genererà un nuovo nome file)$N🕒: sovrascrivi se il file del server è più vecchio del tuo$N♻️: sovrascrivi sempre se i file sono diversi", + "ut_mt": "continua l'hashing di altri file durante il caricamento$N$NProva a disabilitare se la tua CPU o HDD è un collo di bottiglia", + "ut_ask": 'chiedi conferma prima che inizi il caricamento">💭', + "ut_pot": "migliora la velocità di caricamento su dispositivi lenti$Nrendendo l'interfaccia meno complessa", + "ut_srch": "non caricare realmente, invece controlla se i file esistono già $N sul server (scansionerà tutte le cartelle che puoi leggere)", + "ut_par": "metti in pausa i caricamenti impostandolo a 0$N$NAumenta se la tua connessione è lenta / alta latenza$N$NMantienilo a 1 su LAN o se l'HDD del server è un collo di bottiglia", + "ul_btn": "trascina file / cartelle
qui (o cliccami)", + "ul_btnu": "C A R I C A", + "ul_btns": "C E R C A", + + "ul_hash": "hash", + "ul_send": "invia", + "ul_done": "fatto", + "ul_idle1": "nessun caricamento ancora in coda", + "ut_etah": "velocità media di <em>hashing</em>, e tempo stimato al completamento", + "ut_etau": "velocità media di <em>caricamento</em> e tempo stimato al completamento", + "ut_etat": "velocità <em>totale</em> media e tempo stimato al completamento", + + "uct_ok": "completato con successo", + "uct_ng": "non-valido: fallito / rifiutato / non-trovato", + "uct_done": "ok e ng combinati", + "uct_bz": "hashing o caricamento", + "uct_q": "inattivo, in attesa", + + "utl_name": "nome file", + "utl_ulist": "lista", + "utl_ucopy": "copia", + "utl_links": "link", + "utl_stat": "stato", + "utl_prog": "progresso", + + // keep short: + "utl_404": "404", + "utl_err": "ERRORE", + "utl_oserr": "Errore-SO", + "utl_found": "trovato", + "utl_defer": "rinvia", + "utl_yolo": "YOLO", + "utl_done": "finito", + + "ul_flagblk": "i file sono stati aggiunti alla coda
tuttavia c'è un up2k occupato in un'altra scheda del browser,
quindi aspetto che quello finisca prima", + "ul_btnlk": "la configurazione del server ha bloccato questo interruttore in questo stato", + + "udt_up": "Carica", + "udt_srch": "Cerca", + "udt_drop": "lascialo qui", + + "u_nav_m": '
ok, cosa hai?
Invio = File (uno o più)\nESC = Una cartella (incluse sottocartelle)', + "u_nav_b": 'FileUna cartella', + + "cl_opts": "opzioni", + "cl_themes": "tema", + "cl_langs": "lingua", + "cl_ziptype": "download cartella", + "cl_uopts": "opzioni up2k", + "cl_favico": "favicon", + "cl_bigdir": "cartelle grandi", + "cl_hsort": "#ordinamento", + "cl_keytype": "notazione tasti", + "cl_hiddenc": "colonne nascoste", + "cl_hidec": "nascondi", + "cl_reset": "reset", + "cl_hpick": "tocca le intestazioni delle colonne per nascondere nella tabella sottostante", + "cl_hcancel": "nascondere colonne annullato", + + "ct_grid": '田 griglia', + "ct_ttips": '◔ ◡ ◔">ℹ️ tooltip', + "ct_thumb": 'nella vista griglia, alterna icone o miniature$NTasto rapido: T">🖼️ miniature', + "ct_csel": 'usa CTRL e SHIFT per la selezione file nella vista griglia">sel', + "ct_ihop": 'quando il visualizzatore immagini è chiuso, scorri fino all\'ultimo file visualizzato">g⮯', + "ct_dots": 'mostra file nascosti (se il server lo permette)">dotfile', + "ct_qdel": 'quando elimini file, chiedi conferma solo una volta">qdel', + "ct_dir1st": 'ordina cartelle prima dei file">📁 prima', + "ct_nsort": 'ordinamento naturale (per nomi file con cifre iniziali)">nsort', + "ct_utc": 'mostra tutte le date/ore in UTC">UTC', + "ct_readme": 'mostra README.md negli elenchi cartelle">📜 readme', + "ct_idxh": 'mostra index.html invece dell\'elenco cartelle">htm', + "ct_sbars": 'mostra barre di scorrimento">⟊', + + "cut_umod": "se un file esiste già sul server, aggiorna il timestamp di ultima modifica del server per farlo coincidere con il tuo file locale (richiede permessi di scrittura+eliminazione)\">re📅", + + "cut_turbo": "il pulsante yolo, probabilmente NON lo vuoi abilitare:$N$NUsalo se stavi caricando una grande quantità di file e hai dovuto riavviare per qualche motivo, e vuoi continuare il caricamento il prima possibile$N$NQuesto sostituisce il controllo hash con un semplice "questo ha la stessa dimensione file sul server?" quindi se il contenuto del file è diverso NON verrà caricato$N$NDovresti spegnere questo quando il caricamento è finito, e poi "caricare" di nuovo gli stessi file per far verificare al client\">turbo", + + "cut_datechk": "non ha effetto a meno che il pulsante turbo sia abilitato$N$NRiduce il fattore yolo di una piccola quantità; controlla se i timestamp dei file sul server corrispondono ai tuoi$N$NDovrebbe teoricamente catturare la maggior parte dei caricamenti non finiti / corrotti, ma non è un sostituto per fare un passaggio di verifica con turbo disabilitato dopo\">date-chk", + + "cut_u2sz": "dimensione (in MiB) di ogni chunk di caricamento; valori grandi volano meglio attraverso l'atlantico. Prova valori bassi su connessioni molto inaffidabili", + + "cut_flag": "assicura che solo una scheda stia caricando alla volta $N -- anche le altre schede devono avere questo abilitato $N -- influisce solo sulle schede dello stesso dominio", + + "cut_az": "carica file in ordine alfabetico, invece che dal file più piccolo prima$N$NL'ordine alfabetico può rendere più facile controllare a occhio se qualcosa è andato storto sul server, ma rende il caricamento leggermente più lento su fibra / LAN", + + "cut_nag": "notifica SO quando il caricamento si completa$N(solo se il browser o la scheda non è attiva)", + "cut_sfx": "allarme sonoro quando il caricamento si completa$N(solo se il browser o la scheda non è attiva)", + + "cut_mt": "usa multithreading per accelerare l'hashing dei file$N$NQuesto usa web-worker e richiede$Npiù RAM (fino a 512 MiB extra)$N$NRende https 30% più veloce, http 4.5x più veloce\">mt", + + "cut_wasm": "usa wasm invece dell'hasher integrato del browser; migliora la velocità sui browser basati su chrome ma aumenta il carico CPU, e molte versioni vecchie di chrome hanno bug che fanno consumare tutta la RAM al browser e crashare se questo è abilitato\">wasm", + + "cft_text": "testo favicon (vuoto e aggiorna per disabilitare)", + "cft_fg": "colore primo piano", + "cft_bg": "colore sfondo", + + "cdt_lim": "numero massimo di file da mostrare in una cartella", + "cdt_ask": "quando scorri verso il fondo,$Ninvece di caricare più file,$Nchiedi cosa fare", + "cdt_hsort": "quante regole di ordinamento (<code>,sorthref</code>) includere negli URL multimediali. Impostandolo a 0 ignorerà anche le regole di ordinamento incluse nei link multimediali quando li clicchi", + + "tt_entree": "mostra pannello nav (barra laterale albero directory)$NTasto rapido: B", + "tt_detree": "mostra breadcrumb$NTasto rapido: B", + "tt_visdir": "scorri alla cartella selezionata", + "tt_ftree": "alterna albero cartelle / file di testo$NTasto rapido: V", + "tt_pdock": "mostra cartelle genitore in un pannello ancorato in alto", + "tt_dynt": "crescita automatica mentre l'albero si espande", + "tt_wrap": "a capo parola", + "tt_hover": "rivela righe che traboccano al passaggio del mouse$N( interrompe lo scorrimento a meno che il cursore $N  del mouse non sia nella grondaia sinistra )", + + "ml_pmode": "alla fine della cartella...", + "ml_btns": "comandi", + "ml_tcode": "transcodifica", + "ml_tcode2": "transcodifica in", + "ml_tint": "tinta", + "ml_eq": "equalizzatore audio", + "ml_drc": "compressore gamma dinamica", + + "mt_loop": "loop/ripeti una canzone\">🔁", + "mt_one": "fermati dopo una canzone\">1️⃣", + "mt_shuf": "mescola le canzoni in ogni cartella\">🔀", + "mt_aplay": "autoplay se c'è un song-ID nel link che hai cliccato per accedere al server$N$NDisabilitando questo fermerà anche l'aggiornamento dell'URL della pagina con song-ID quando riproduci musica, per prevenire autoplay se queste impostazioni vengono perse ma l'URL rimane\">a▶", + "mt_preload": "inizia a caricare la prossima canzone verso la fine per riproduzione senza interruzioni\">preload", + "mt_prescan": "vai alla prossima cartella prima che finisca l'ultima canzone$Nmantenendo felice il browser web$Ncosì non si ferma la riproduzione\">nav", + "mt_fullpre": "prova a precaricare l'intera canzone;$N✅ abilita su connessioni inaffidabili,$N❌ disabilita su connessioni lente probabilmente\">full", + "mt_fau": "sui telefoni, previeni che la musica si fermi se la prossima canzone non si precarica abbastanza velocemente (può rendere glitchy la visualizzazione dei tag)\">☕️", + "mt_waves": "barra di ricerca forma d'onda:$Nmostra ampiezza audio nello scrubber\">~s", + "mt_npclip": "mostra pulsanti per copiare negli appunti la canzone attualmente in riproduzione\">/np", + "mt_m3u_c": "mostra pulsanti per copiare negli appunti le$Ncanzoni selezionate come voci playlist m3u8\">📻", + "mt_octl": "integrazione so (tasti multimediali / osd)\">os-ctl", + "mt_oseek": "permetti ricerca attraverso integrazione so$N$Nnota: su alcuni dispositivi (iPhone),$Nquesto sostituisce il pulsante canzone successiva\">seek", + "mt_oscv": "mostra copertina album in osd\">art", + "mt_follow": "mantieni la traccia in riproduzione scorrevole nella vista\">🎯", + "mt_compact": "controlli compatti\">⟎", + "mt_uncache": "pulisci cache  (prova ad attivare se il tuo browser ha messo in cache$Nuna copia rotta di una canzone e si rifiuta di riprodurla)\">uncache", + "mt_mloop": "loop della cartella aperta\">🔁 loop", + "mt_mnext": "carica la prossima cartella e continua\">📂 succ", + "mt_mstop": "ferma riproduzione\">⏸ stop", + "mt_cflac": "converti flac / wav in opus\">flac", + "mt_caac": "converti aac / m4a in opus\">aac", + "mt_coth": "converti tutti gli altri (non mp3) in opus\">oth", + "mt_c2opus": "scelta migliore per desktop, laptop, android\">opus", + "mt_c2owa": "opus-weba, per iOS 17.5 e più recenti\">owa", + "mt_c2caf": "opus-caf, per iOS 11 fino a 17\">caf", + "mt_c2mp3": "usa questo su dispositivi molto vecchi\">mp3", + "mt_c2ok": "bene, buona scelta", + "mt_c2nd": "quello non è il formato di output raccomandato per il tuo dispositivo, ma va bene", + "mt_c2ng": "il tuo dispositivo non sembra supportare questo formato di output, ma proviamo comunque", + "mt_xowa": "ci sono bug in iOS che prevengono la riproduzione in background usando questo formato; usa caf o mp3 invece", + "mt_tint": "livello sfondo (0-100) sulla barra di ricerca$Nper rendere il buffering meno distraente", + "mt_eq": "abilita l'equalizzatore e controllo guadagno;$N$Nboost <code>0</code> = volume standard 100% (non modificato)$N$Nwidth <code>1  </code> = stereo standard (non modificato)$Nwidth <code>0.5</code> = 50% crossfeed sinistra-destra$Nwidth <code>0  </code> = mono$N$Nboost <code>-0.8</code> & width <code>10</code> = rimozione vocale :^)$N$Nabilitando l'equalizzatore rende gli album senza interruzioni completamente senza interruzioni, quindi lascialo acceso con tutti i valori a zero (eccetto width = 1) se ti importa di quello", + "mt_drc": "abilita il compressore gamma dinamica (appiattitore volume / brickwaller); abiliterà anche EQ per bilanciare gli spaghetti, quindi imposta tutti i campi EQ eccetto 'width' a 0 se non lo vuoi$N$NAbbassa il volume dell'audio sopra THRESHOLD dB; per ogni RATIO dB oltre THRESHOLD c'è 1 dB di output, quindi i valori di default di tresh -24 e ratio 12 significa che non dovrebbe mai diventare più forte di -22 dB ed è sicuro aumentare il boost equalizzatore a 0.8, o anche 1.8 con ATK 0 e un RLS enorme come 90 (funziona solo in firefox; RLS è max 1 in altri browser)$N$N(vedi wikipedia, lo spiegano molto meglio)", + + "mb_play": "riproduci", + "mm_hashplay": "riprodurre questo file audio?", + "mm_m3u": "premi Invio/OK per Riprodurre\npremi ESC/Annulla per Modificare", + "mp_breq": "serve firefox 82+ o chrome 73+ o iOS 15+", + "mm_bload": "ora caricando...", + "mm_bconv": "convertendo in {0}, attendi...", + "mm_opusen": "il tuo browser non può riprodurre file aac / m4a;\ntranscodifica in opus ora abilitata", + "mm_playerr": "riproduzione fallita: ", + "mm_eabrt": "Il tentativo di riproduzione è stato cancellato", + "mm_enet": "La tua connessione internet è instabile", + "mm_edec": "Questo file è presumibilmente corrotto??", + "mm_esupp": "Il tuo browser non capisce questo formato audio", + "mm_eunk": "Errore Sconosciuto", + "mm_e404": "Non è stato possibile riprodurre audio; errore 404: File non trovato.", + "mm_e403": "Non è stato possibile riprodurre audio; errore 403: Accesso negato.\n\nProva a premere F5 per ricaricare, forse sei stato disconnesso", + "mm_e500": "Non è stato possibile riprodurre audio; errore 500: Controlla i log del server.", + "mm_e5xx": "Non è stato possibile riprodurre audio; errore server ", + "mm_nof": "non trovo altri file audio nelle vicinanze", + "mm_prescan": "Cercando musica da riprodurre dopo...", + "mm_scank": "Trovata la prossima canzone:", + "mm_uncache": "cache pulita; tutte le canzoni si riscaricheranno alla prossima riproduzione", + "mm_hnf": "quella canzone non esiste più", + + "im_hnf": "quell'immagine non esiste più", + + "f_empty": 'questa cartella è vuota', + "f_chide": 'questo nasconderà la colonna «{0}»\n\npuoi mostrare le colonne nella scheda impostazioni', + "f_bigtxt": "questo file è {0} MiB grande -- visualizzare davvero come testo?", + "f_bigtxt2": "visualizzare solo la fine del file invece? questo abiliterà anche following/tailing, mostrando righe di testo appena aggiunte in tempo reale", + "fbd_more": '
mostrando {0} di {1} file; mostra {2} o mostra tutti
', + "fbd_all": '
mostrando {0} di {1} file; mostra tutti
', + "f_anota": "solo {0} dei {1} elementi sono stati selezionati;\nper selezionare l'intera cartella, prima scorri fino in fondo", + + "f_dls": 'i link dei file nella cartella corrente sono stati\ncambiati in link di download', + + "f_partial": "Per scaricare in sicurezza un file che è attualmente in fase di caricamento, clicca il file che ha lo stesso nome, ma senza l'estensione .PARTIAL. Premi ANNULLA o Escape per farlo.\n\nPremendo OK / Invio ignorerai questo avviso e continuerai a scaricare il file .PARTIAL scratch, che quasi sicuramente ti darà dati corrotti.", + + "ft_paste": "incolla {0} elementi$NTasto rapido: ctrl-V", + "fr_eperm": 'impossibile rinominare:\nnon hai il permesso "sposta" in questa cartella', + "fd_eperm": 'impossibile eliminare:\nnon hai il permesso "elimina" in questa cartella', + "fc_eperm": 'impossibile tagliare:\nnon hai il permesso "sposta" in questa cartella', + "fp_eperm": 'impossibile incollare:\nnon hai il permesso "scrivi" in questa cartella', + "fr_emore": "seleziona almeno un elemento da rinominare", + "fd_emore": "seleziona almeno un elemento da eliminare", + "fc_emore": "seleziona almeno un elemento da tagliare", + "fcp_emore": "seleziona almeno un elemento da copiare negli appunti", + + "fs_sc": "condividi la cartella in cui ti trovi", + "fs_ss": "condividi i file selezionati", + "fs_just1d": "non puoi selezionare più di una cartella,\no mescolare file e cartelle in una selezione", + "fs_abrt": "❌ interrompi", + "fs_rand": "🎲 nome.casuale", + "fs_go": "✅ crea condivisione", + "fs_name": "nome", + "fs_src": "sorgente", + "fs_pwd": "password", + "fs_exp": "scadenza", + "fs_tmin": "min", + "fs_thrs": "ore", + "fs_tdays": "giorni", + "fs_never": "eterno", + "fs_pname": "nome link opzionale; sarà casuale se vuoto", + "fs_tsrc": "il file o cartella da condividere", + "fs_ppwd": "password opzionale", + "fs_w8": "creando condivisione...", + "fs_ok": "premi Invio/OK per Appunti\npremi ESC/Annulla per Chiudere", + + "frt_dec": "può risolvere alcuni casi di nomi file corrotti\">url-decode", + "frt_rst": "ripristina nomi file modificati a quelli originali\">↺ reset", + "frt_abrt": "interrompi e chiudi questa finestra\">❌ annulla", + "frb_apply": "APPLICA RINOMINA", + "fr_adv": "rinomina batch / metadata / pattern\">avanzato", + "fr_case": "regex case-sensitive\">maiusc", + "fr_win": "nomi sicuri per windows; sostituisce <>:"\\|?* con caratteri giapponesi fullwidth\">win", + "fr_slash": "sostituisce / con un carattere che non causa la creazione di nuove cartelle\">no /", + "fr_re": "pattern di ricerca regex da applicare ai nomi file originali; i gruppi di cattura possono essere referenziati nel campo formato sottostante come <code>(1)</code> e <code>(2)</code> e così via", + "fr_fmt": "ispirato da foobar2000:$N<code>(title)</code> è sostituito dal titolo della canzone,$N<code>[(artist) - ](title)</code> salta [questa] parte se artista è vuoto$N<code>$lpad((tn),2,0)</code> aggiunge padding al numero traccia a 2 cifre", + "fr_pdel": "elimina", + "fr_pnew": "salva come", + "fr_pname": "fornisci un nome per il tuo nuovo preset", + "fr_aborted": "interrotto", + "fr_lold": "nome vecchio", + "fr_lnew": "nome nuovo", + "fr_tags": "tag per i file selezionati (sola lettura, solo per riferimento):", + "fr_busy": "rinominando {0} elementi...\n\n{1}", + "fr_efail": "rinomina fallita:\n", + "fr_nchg": "{0} dei nuovi nomi sono stati alterati a causa di win e/o no /\n\nOK per continuare con questi nuovi nomi alterati?", + + "fd_ok": "eliminazione OK", + "fd_err": "eliminazione fallita:\n", + "fd_none": "niente è stato eliminato; forse bloccato dalla configurazione server (xbd)?", + "fd_busy": "eliminando {0} elementi...\n\n{1}", + "fd_warn1": "ELIMINARE questi {0} elementi?", + "fd_warn2": "Ultima possibilità! Nessun modo per annullare. Eliminare?", + + "fc_ok": "tagliati {0} elementi", + "fc_warn": 'tagliati {0} elementi\n\nma: solo questa scheda-browser può incollarli\n(dato che la selezione è così assolutamente massiva)', + + "fcc_ok": "copiati {0} elementi negli appunti", + "fcc_warn": 'copiati {0} elementi negli appunti\n\nma: solo questa scheda-browser può incollarli\n(dato che la selezione è così assolutamente massiva)', + + "fp_apply": "usa questi nomi", + "fp_ecut": "prima taglia o copia alcuni file / cartelle da incollare / spostare\n\nnota: puoi tagliare / incollare attraverso diverse schede del browser", + "fp_ename": "{0} elementi non possono essere spostati qui perché i nomi sono già presi. Dai loro nuovi nomi qui sotto per continuare, o lascia vuoto il nome per saltarli:", + "fcp_ename": "{0} elementi non possono essere copiati qui perché i nomi sono già presi. Dai loro nuovi nomi qui sotto per continuare, o lascia vuoto il nome per saltarli:", + "fp_emore": "ci sono ancora alcune collisioni di nomi file rimaste da risolvere", + "fp_ok": "spostamento OK", + "fcp_ok": "copia OK", + "fp_busy": "spostando {0} elementi...\n\n{1}", + "fcp_busy": "copiando {0} elementi...\n\n{1}", + "fp_err": "spostamento fallito:\n", + "fcp_err": "copia fallita:\n", + "fp_confirm": "spostare questi {0} elementi qui?", + "fcp_confirm": "copiare questi {0} elementi qui?", + "fp_etab": 'fallito leggere appunti da altra scheda browser', + "fp_name": "caricando un file dal tuo dispositivo. Dagli un nome:", + "fp_both_m": '
scegli cosa incollare
Invio = Sposta {0} file da «{1}»\nESC = Carica {2} file dal tuo dispositivo', + "fcp_both_m": '
scegli cosa incollare
Invio = Copia {0} file da «{1}»\nESC = Carica {2} file dal tuo dispositivo', + "fp_both_b": 'SpostaCarica', + "fcp_both_b": 'CopiaCarica', + + "mk_noname": "scrivi un nome nel campo di testo a sinistra prima di farlo :p", + + "tv_load": "Caricando documento di testo:\n\n{0}\n\n{1}% ({2} di {3} MiB caricati)", + "tv_xe1": "impossibile caricare file di testo:\n\nerrore ", + "tv_xe2": "404, file non trovato", + "tv_lst": "lista di file di testo in", + "tvt_close": "torna alla vista cartella$NTasto rapido: M (o Esc)\">❌ chiudi", + "tvt_dl": "scarica questo file$NTasto rapido: Y\">💾 scarica", + "tvt_prev": "mostra documento precedente$NTasto rapido: i\">⬆ prec", + "tvt_next": "mostra documento successivo$NTasto rapido: K\">⬇ succ", + "tvt_sel": "seleziona file   ( per taglia / copia / elimina / ... )$NTasto rapido: S\">sel", + "tvt_edit": "apri file nell'editor di testo$NTasto rapido: E\">✏️ modifica", + "tvt_tail": "monitora file per cambiamenti; mostra nuove righe in tempo reale\">📡 segui", + "tvt_wrap": "a capo parola\">↵", + "tvt_atail": "blocca scorrimento in fondo alla pagina\">⚓", + "tvt_ctail": "decodifica colori terminale (codici escape ansi)\">🌈", + "tvt_ntail": "limite scrollback (quanti byte di testo mantenere caricati)", + + "m3u_add1": "canzone aggiunta alla playlist m3u", + "m3u_addn": "{0} canzoni aggiunte alla playlist m3u", + "m3u_clip": "playlist m3u ora copiata negli appunti\n\ndovresti creare un nuovo file di testo chiamato qualcosa.m3u e incollare la playlist in quel documento; questo la renderà riproducibile", + + "gt_vau": "non mostrare video, riproduci solo l'audio\">🎧", + "gt_msel": "abilita selezione file; ctrl-click un file per sovrascrivere$N$N<em>quando attivo: doppio-click un file / cartella per aprirlo</em>$N$NTasto rapido: S\">multiselezione", + "gt_crop": "ritaglia miniature al centro\">ritaglia", + "gt_3x": "miniature hi-res\">3x", + "gt_zoom": "zoom", + "gt_chop": "taglia", + "gt_sort": "ordina per", + "gt_name": "nome", + "gt_sz": "dimensione", + "gt_ts": "data", + "gt_ext": "tipo", + "gt_c1": "tronca nomi file di più (mostra meno)", + "gt_c2": "tronca nomi file di meno (mostra di più)", + + "sm_w8": "cercando...", + "sm_prev": "i risultati di ricerca qui sotto sono da una query precedente:\n ", + "sl_close": "chiudi risultati ricerca", + "sl_hits": "mostrando {0} risultati", + "sl_moar": "carica altro", + + "s_sz": "dimensione", + "s_dt": "data", + "s_rd": "percorso", + "s_fn": "nome", + "s_ta": "tag", + "s_ua": "car@", + "s_ad": "avanz.", + "s_s1": "MiB minimo", + "s_s2": "MiB massimo", + "s_d1": "iso8601 min.", + "s_d2": "iso8601 max.", + "s_u1": "caricato dopo", + "s_u2": "e/o prima", + "s_r1": "percorso contiene   (separato da spazi)", + "s_f1": "nome contiene   (nega con -nope)", + "s_t1": "tag contiene   (^=inizio, fine=$)", + "s_a1": "proprietà metadata specifiche", + + "md_eshow": "impossibile renderizzare ", + "md_off": "[📜readme] disabilitato in [⚙️] -- documento nascosto", + + "badreply": "Fallito nel parsare risposta dal server", + + "xhr403": "403: Accesso negato\n\nprova a premere F5, forse sei stato disconnesso", + "xhr0": "sconosciuto (probabilmente persa connessione al server, o server offline)", + "cf_ok": "scusa per quello -- la protezione DD" + wah + "oS è entrata in azione\n\nle cose dovrebbero riprendere in circa 30 sec\n\nse non succede niente, premi F5 per ricaricare la pagina", + "tl_xe1": "impossibile elencare sottocartelle:\n\nerrore ", + "tl_xe2": "404: Cartella non trovata", + "fl_xe1": "impossibile elencare file nella cartella:\n\nerrore ", + "fl_xe2": "404: Cartella non trovata", + "fd_xe1": "impossibile creare sottocartella:\n\nerrore ", + "fd_xe2": "404: Cartella genitore non trovata", + "fsm_xe1": "impossibile inviare messaggio:\n\nerrore ", + "fsm_xe2": "404: Cartella genitore non trovata", + "fu_xe1": "fcaricamento fallito per la lista unpost dal server:\n\nerrore ", + "fu_xe2": "404: File non trovato??", + + "fz_tar": "file gnu-tar non compresso (linux / mac)", + "fz_pax": "tar formato pax non compresso (più lento)", + "fz_targz": "gnu-tar con compressione gzip livello 3$N$NSolitamente è molto lento, quindi$Nusa tar non compresso", + "fz_tarxz": "gnu-tar con compressione xz livello 1$N$NQuesto è solitamente molto lento, quindi$Nusa tar non compresso", + "fz_zip8": "zip con nomi file utf8 (forse instabile su windows 7 e precedenti)", + "fz_zipd": "zip con nomi file cp437 tradizionali, per software molto vecchio", + "fz_zipc": "cp437 con crc32 calcolato presto,$Nper MS-DOS PKZIP v2.04g (ottobre 1993)$N(ci vuole più tempo per elaborare prima che possa iniziare il download)", + + "un_m1": "puoi eliminare i tuoi caricamenti recenti (o interrompere quelli non finiti) qui sotto", + "un_upd": "aggiorna", + "un_m4": "o condividi i file visibili qui sotto:", + "un_ulist": "mostra", + "un_ucopy": "copia", + "un_flt": "filtro opzionale:  URL deve contenere", + "un_fclr": "resetta filtro", + "un_derr": 'unpost-delete fallito:\n', + "un_f5": 'qualcosa si è rotto, prova un aggiornamento o premi F5', + "un_uf5": "scusa ma devi aggiornare la pagina (per esempio premendo F5 o CTRL-R) prima che questo caricamento possa essere interrotto", + "un_nou": 'avviso: server troppo occupato per mostrare caricamenti non finiti; clicca il link "aggiorna" tra un po\'', + "un_noc": 'avviso: unpost di file completamente caricati non è abilitato/permesso nella configurazione server', + "un_max": "mostrando primi 2000 file (usa il filtro)", + "un_avail": "{0} caricamenti recenti possono essere eliminati
{1} non finiti possono essere interrotti", + "un_m2": "ordinati per tempo di caricamento; più recenti prima:", + "un_no1": "scherzo! nessun caricamento è abbastanza recente", + "un_no2": "scherzo! nessun caricamento che corrisponde a quel filtro è abbastanza recente", + "un_next": "elimina i prossimi {0} file qui sotto", + "un_abrt": "interrompi", + "un_del": "elimina", + "un_m3": "caricando i tuoi caricamenti recenti...", + "un_busy": "eliminando {0} file...", + "un_clip": "{0} link copiati negli appunti", + + "u_https1": "dovresti", + "u_https2": "passare a https", + "u_https3": "per prestazioni migliori", + "u_ancient": 'il tuo browser è incredibilmente antico -- forse dovresti usare bup invece', + "u_nowork": "serve firefox 53+ o chrome 57+ o iOS 11+", + "tail_2old": "serve firefox 105+ o chrome 71+ o iOS 14.5+", + "u_nodrop": 'il tuo browser è troppo vecchio per il caricamento drag-and-drop', + "u_notdir": "quella non è una cartella!\n\nil tuo browser è troppo vecchio,\nprova dragdrop invece", + "u_uri": "per trascinare immagini da altre finestre del browser,\nrilasciale sul pulsante upload grande", + "u_enpot": 'passa alla UI patata (può migliorare velocità upload)', + "u_depot": 'passa alla UI elegante (può ridurre velocità upload)', + "u_gotpot": 'passando alla UI patata per migliorare velocità upload,\n\nsentiti libero di non essere d\'accordo e tornare indietro!', + "u_pott": "

file:   {0} finiti,   {1} falliti,   {2} occupati,   {3} in coda

", + "u_ever": "questo è l'uploader di base; up2k necessita almeno
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'questo è l\'uploader di base; up2k è migliore', + "u_uput": 'velocizza (salta checksum)', + "u_ewrite": 'non hai accesso in scrittura a questa cartella', + "u_eread": 'non hai accesso in lettura a questa cartella', + "u_enoi": 'file-search non è abilitato nella configurazione server', + "u_enoow": "non puoi sovrascrivere qui; serve permesso Elimina", + "u_badf": 'Questi {0} file (di {1} totali) sono stati saltati, probabilmente a causa di permessi filesystem:\n\n', + "u_blankf": 'Questi {0} file (di {1} totali) sono vuoti; caricarli comunque?\n\n', + "u_applef": 'Questi {0} file (di {1} totali) sono probabilmente indesiderabili;\nPremi OK/Invio per SALTARE i seguenti file,\nPremi Annulla/ESC per NON escludere, e CARICARE anche quelli:\n\n', + "u_just1": '\nForse funziona meglio se selezioni solo un file', + "u_ff_many": "se stai usando Linux / MacOS / Android, allora questa quantità di file potrebbe far crashare Firefox!\nse succede, riprova (o usa Chrome).", + "u_up_life": "Questo caricamento sarà eliminato dal server\n{0} dopo che si completa", + "u_asku": 'caricare questi {0} file in {1}', + "u_unpt": "puoi annullare / eliminare questo caricamento usando 🧯 in alto a sinistra", + "u_bigtab": 'sto per mostrare {0} file\n\nquesto potrebbe far crashare il tuo browser, sei sicuro?', + "u_scan": 'Scansionando file...', + "u_dirstuck": 'iteratore directory si è bloccato tentando di accedere ai seguenti {0} elementi; salterò:', + "u_etadone": 'Fatto ({0}, {1} file)', + "u_etaprep": '(preparando per caricare)', + "u_hashdone": 'hashing completato', + "u_hashing": 'hash', + "u_hs": 'handshaking...', + "u_started": "i file ora sono in caricamento; vedi [🚀]", + "u_dupdefer": "duplicato; sarà processato dopo tutti gli altri file", + "u_actx": "clicca questo testo per prevenire perdita di
prestazioni quando cambi ad altre finestre/schede", + "u_fixed": "OK!  Risolto 👍", + "u_cuerr": "caricamento fallito del chunk {0} di {1};\nprobabilmente innocuo, continuo\n\nfile: {2}", + "u_cuerr2": "il server ha rifiutato il caricamento (chunk {0} di {1});\nriproverò più tardi\n\nfile: {2}\n\nerrore ", + "u_ehstmp": "riproverò; vedi in basso a destra", + "u_ehsfin": "il server ha rifiutato la richiesta di finalizzare caricamento; riprovando...", + "u_ehssrch": "il server ha rifiutato la richiesta di eseguire ricerca; riprovando...", + "u_ehsinit": "il server ha rifiutato la richiesta di iniziare caricamento; riprovando...", + "u_eneths": "errore di rete durante handshake per upload; riprovando...", + "u_enethd": "errore di rete durante test esistenza target; riprovando...", + "u_cbusy": "aspettando che il server si fidi di noi di nuovo dopo un problema di rete...", + "u_ehsdf": "il server ha finito lo spazio su disco!\n\ncontinuerò a riprovare, nel caso qualcuno\nliberi abbastanza spazio per continuare", + "u_emtleak1": "sembra che il tuo browser possa avere un memory leak;\nper favore", + "u_emtleak2": ' passa a https (raccomandato) o ', + "u_emtleak3": ' ', + "u_emtleakc": 'prova quanto segue:\n
  • premi F5 per aggiornare la pagina
  • poi disabilita il pulsante  mt  nelle  ⚙️ impostazioni
  • e riprova quel caricamento
I caricamenti saranno un po\' più lenti, ma pazienza.\nScusa per il disturbo !\n\nPS: chrome v107 ha un bugfix per questo', + "u_emtleakf": 'prova quanto segue:\n
  • premi F5 per aggiornare la pagina
  • poi abilita 🥔 (patata) nell\'UI caricamento
  • e riprova quel caricamento
\nPS: firefox avrà sperabilmente un bugfix ad un certo punto', + "u_s404": "non trovato sul server", + "u_expl": "spiega", + "u_maxconn": "la maggior parte dei browser limita questo a 6, ma firefox ti permette di alzarlo con connections-per-server in about:config", + "u_tu": '

AVVISO: turbo abilitato,  client potrebbe non rilevare e riprendere caricamenti incompleti; vedi tooltip pulsante turbo

', + "u_ts": '

AVVISO: turbo abilitato,  risultati ricerca possono essere incorretti; vedi tooltip pulsante turbo

', + "u_turbo_c": "turbo è disabilitato nella configurazione server", + "u_turbo_g": "disabilitando turbo perché non hai\nprivilegi di elenco directory all'interno di questo volume", + "u_life_cfg": 'auto-elimina dopo min (o ore)', + "u_life_est": 'caricamento sarà eliminato ---', + "u_life_max": 'questa cartella impone una\nvita massima di {0}', + "u_unp_ok": 'unpost è permesso per {0}', + "u_unp_ng": 'unpost NON sarà permesso', + "ue_ro": 'il tuo accesso a questa cartella è solo-Lettura\n\n', + "ue_nl": 'attualmente non sei loggato', + "ue_la": 'attualmente sei loggato come "{0}"', + "ue_sr": 'attualmente sei in modalità file-search\n\npassa alla modalità upload cliccando la lente d\'ingrandimento 🔎 (accanto al grande pulsante CERCA), e prova a caricare di nuovo\n\nscusa', + "ue_ta": 'prova a caricare di nuovo, dovrebbe funzionare ora', + "ue_ab": "questo file è già in caricamento in un'altra cartella, e quel caricamento deve essere completato prima che il file possa essere caricato altrove.\n\nPuoi interrompere e dimenticare il caricamento iniziale usando l'🧯 in alto a sinistra", + "ur_1uo": "OK: File caricato con successo", + "ur_auo": "OK: Tutti i {0} file caricati con successo", + "ur_1so": "OK: File trovato sul server", + "ur_aso": "OK: Tutti i {0} file trovati sul server", + "ur_1un": "Caricamento fallito, scusa", + "ur_aun": "Tutti i {0} caricamenti falliti, scusa", + "ur_1sn": "File NON trovato sul server", + "ur_asn": "I {0} file NON sono stati trovati sul server", + "ur_um": "Finito;\n{0} caricamenti OK,\n{1} caricamenti falliti, scusa", + "ur_sm": "Finito;\n{0} file trovati sul server,\n{1} file NON trovati sul server", + + "lang_set": "aggiornare per rendere effettivo il cambiamento?", + }, "nld": { "tt": "Nederlands", @@ -5650,7 +6277,7 @@ var Ls = { }, }; -var LANGS = ["eng", "nor", "chi", "deu", "fin", "nld", "rus", "spa", "ukr"]; +var LANGS = ["eng", "nor", "chi", "deu", "fin", "ita", "nld", "rus", "spa", "ukr"]; if (window.langmod) langmod(); diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 06cb0bf7..44f315ff 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -176,6 +176,48 @@ var Ls = { "af1": "näytä viimeaikaiset lataukset", "ag1": "näytä tunnetut IdP-käyttäjät", }, + "ita": { + "a1": "aggiorna", + "b1": "ciao   (non sei connesso)", + "c1": "disconnetti", + "d1": "stato", + "d2": "mostra lo stato di tutti i thread attivi", + "e1": "ricarica configurazione", + "e2": "ricarica i file di configurazione (account/volumi/flag dei volumi),\n e riesegue la scansione di tutti i volumi e2ds.\n\nNota: qualsiasi modifica alle impostazioni globali richiede un riavvio completo per avere effetto", + "f1": "puoi visualizzare:", + "g1": "puoi caricare su:", + "cc1": "altro:", + "h1": "disattiva k304", + "i1": "attiva k304", + "j1": "k304 interrompe la connessione per ogni HTTP 304. Questo aiuta contro alcuni proxy difettosi che possono bloccarsi o smettere improvvisamente di caricare pagine, ma riduce notevolmente le prestazioni", + "k1": "resetta impostazioni", + "l1": "accedi:", + "m1": "bentornato,", + "n1": "404: file non trovato  ┐( ´ -`)┌", + "o1": "oppure forse non hai accesso? prova una password o torna alla home", + "p1": "403: accesso negato  ~┻━┻", + "q1": "prova una password o torna alla home", + "r1": "torna alla home", + ".s1": "mappa", + "t1": "azione", + "u2": "tempo dall'ultima scrittura sul server\n (caricamento / rinomina / ...)\n\n17d = 17 giorni\n1h23 = 1 ora 23 minuti\n4m56 = 4 minuti 56 secondi", + "v1": "connetti", + "v2": "usa questo server come un disco locale", + "w1": "passa a https", + "x1": "cambia password", + "y1": "le tue condivisioni", + "z1": "sblocca area:", + "ta1": "devi prima inserire una nuova password", + "ta2": "ripeti per confermare la nuova password:", + "ta3": "errore di digitazione; riprova", + "aa1": "in arrivo:", + "ab1": "disattiva no304", + "ac1": "attiva no304", + "ad1": "no304 disabilita completamente la cache. Se k304 non è sufficiente, prova questa opzione. Aumenterà notevolmente il consumo di dati!", + "ae1": "in uscita:", + "af1": "mostra i file caricati di recente", + "ag1": "mostra utenti IdP conosciuti" +}, "nld": { "a1": "Update", "b1": "Hallo, hoe gaat het met jou?   (Je bent niet ingelogd)", From b469db3c621a93a258924b1bb399c5672607bc8e Mon Sep 17 00:00:00 2001 From: Toby Kohlhagen Date: Sun, 3 Aug 2025 23:26:13 +1000 Subject: [PATCH 067/174] Add wav transcoding option --- copyparty/httpcli.py | 2 +- copyparty/mtag.py | 2 ++ copyparty/th_cli.py | 4 ++-- copyparty/th_srv.py | 45 +++++++++++++++++++++++++++++++++++++--- copyparty/web/browser.js | 15 ++++++++++---- 5 files changed, 58 insertions(+), 10 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index bea4ce43..51620912 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -4684,7 +4684,7 @@ class HttpCli(object): # for f in fgen: print(repr({k: f[k] for k in ["vp", "ap"]})) cfmt = "" if self.thumbcli and not self.args.no_bacode: - for zs in ("opus", "mp3", "w", "j", "p"): + for zs in ("opus", "mp3", "wav", "w", "j", "p"): if zs in self.ouparam or uarg == zs: cfmt = zs diff --git a/copyparty/mtag.py b/copyparty/mtag.py index cacfda39..240b5a83 100644 --- a/copyparty/mtag.py +++ b/copyparty/mtag.py @@ -270,6 +270,8 @@ def parse_ffprobe(txt: str) -> tuple[dict[str, tuple[int, Any]], dict[str, list[ ["channel_layout", "chs"], ["sample_rate", ".hz"], ["bit_rate", ".aq"], + ["bits_per_sample", ".bps"], + ["bits_per_raw_sample", ".bprs"], ["duration", ".dur"], ] diff --git a/copyparty/th_cli.py b/copyparty/th_cli.py index cfb148c2..6c2632b5 100644 --- a/copyparty/th_cli.py +++ b/copyparty/th_cli.py @@ -88,7 +88,7 @@ class ThumbCli(object): if rem.startswith(".hist/th/") and rem.split(".")[-1] in ["webp", "jpg", "png"]: return os.path.join(ptop, rem) - if fmt[:1] in "jw": + if fmt[:1] in "jw" and fmt != "wav": sfmt = fmt[:1] if sfmt == "j" and self.args.th_no_jpg: @@ -129,7 +129,7 @@ class ThumbCli(object): tpath = thumb_path(histpath, rem, mtime, fmt, self.fmt_ffa) tpaths = [tpath] - if fmt[:1] == "w": + if fmt[:1] == "w" and fmt != "wav": # also check for jpg (maybe webp is unavailable) tpaths.append(tpath.rsplit(".", 1)[0] + ".jpg") diff --git a/copyparty/th_srv.py b/copyparty/th_srv.py index 67413c10..8eaf9287 100644 --- a/copyparty/th_srv.py +++ b/copyparty/th_srv.py @@ -50,7 +50,7 @@ HAVE_AVIF = False HAVE_WEBP = False EXTS_TH = set(["jpg", "webp", "png"]) -EXTS_AC = set(["opus", "owa", "caf", "mp3"]) +EXTS_AC = set(["opus", "owa", "caf", "mp3", "wav"]) EXTS_SPEC_SAFE = set("aif aiff flac mp3 opus wav".split()) PTN_TS = re.compile("^-?[0-9a-f]{8,10}$") @@ -355,8 +355,9 @@ class ThumbSrv(object): tex = tpath.rsplit(".", 1)[-1] want_mp3 = tex == "mp3" want_opus = tex in ("opus", "owa", "caf") + want_wav = tex == "wav" want_png = tex == "png" - want_au = want_mp3 or want_opus + want_au = want_mp3 or want_opus or want_wav for lib in self.args.th_dec: can_au = lib == "ff" and ( ext in self.fmt_ffa or ext in self.fmt_ffv @@ -371,6 +372,8 @@ class ThumbSrv(object): funs.append(self.conv_opus) elif want_mp3: funs.append(self.conv_mp3) + elif want_wav: + funs.append(self.conv_wav) elif want_png: funs.append(self.conv_waves) png_ok = True @@ -580,7 +583,7 @@ class ThumbSrv(object): self._run_ff(cmd, vn) def _run_ff(self, cmd: list[bytes], vn: VFS, oom: int = 400) -> None: - # self.log((b" ".join(cmd)).decode("utf-8")) + self.log((b" ".join(cmd)).decode("utf-8")) ret, _, serr = runcmd(cmd, timeout=vn.flags["convt"], nice=True, oom=oom) if not ret: return @@ -807,6 +810,42 @@ class ThumbSrv(object): # fmt: on self._run_ff(cmd, vn, oom=300) + def conv_wav(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: + if self.args.no_acode or not self.args.q_wav: + raise Exception("disabled in server config") + + self.wait4ram(0.2, tpath) + tags, rawtags = ffprobe(abspath, int(vn.flags["convt"] / 2)) + if "ac" not in tags: + raise Exception("not audio") + + bits = tags[".bps"][1] + if bits == 0.0: + bits = tags[".bprs"][1] + + codec = b"pcm_s32le" + if bits == 16.0: + codec = b"pcm_s16le" + elif bits == 24.0: + codec = b"pcm_s24le" + + self.log("conv2 wav-tmp", 6) + + # fmt: off + cmd = [ + b"ffmpeg", + b"-nostdin", + b"-v", b"error", + b"-hide_banner", + b"-i", fsenc(abspath), + b"-map", b"0:a:0", + b"-c:a", codec, + fsenc(tpath) + ] + # fmt: on + print(cmd) + self._run_ff(cmd, vn, oom=300) + def conv_opus(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: if self.args.no_acode or not self.args.q_opus: raise Exception("disabled in server config") diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index b26f02fc..bd9fbbe7 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -306,6 +306,7 @@ var Ls = { "mt_c2owa": "opus-weba, for iOS 17.5 and newer\">owa", "mt_c2caf": "opus-caf, for iOS 11 through 17\">caf", "mt_c2mp3": "use this on very old devices\">mp3", + "mt_c2wav": "use this for uncompressed playback\">wav", "mt_c2ok": "nice, good choice", "mt_c2nd": "that's not the recommended output format for your device, but that's fine", "mt_c2ng": "your device does not seem to support this output format, but let's try anyways", @@ -6802,6 +6803,7 @@ var mpl = (function () { ' None: - self.log((b" ".join(cmd)).decode("utf-8")) + # self.log((b" ".join(cmd)).decode("utf-8")) ret, _, serr = runcmd(cmd, timeout=vn.flags["convt"], nice=True, oom=oom) if not ret: return @@ -810,8 +813,33 @@ class ThumbSrv(object): # fmt: on self._run_ff(cmd, vn, oom=300) + def conv_flac(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: + if self.args.no_acode or self.args.no_flac: + raise Exception("disabled in server config") + + self.wait4ram(0.2, tpath) + tags, rawtags = ffprobe(abspath, int(vn.flags["convt"] / 2)) + if "ac" not in tags: + raise Exception("not audio") + + self.log("conv2 flac", 6) + + # fmt: off + cmd = [ + b"ffmpeg", + b"-nostdin", + b"-v", b"error", + b"-hide_banner", + b"-i", fsenc(abspath), + b"-map", b"0:a:0", + b"-c:a", b"flac", + fsenc(tpath) + ] + # fmt: on + self._run_ff(cmd, vn, oom=300) + def conv_wav(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: - if self.args.no_acode or not self.args.q_wav: + if self.args.no_acode or self.args.no_wav: raise Exception("disabled in server config") self.wait4ram(0.2, tpath) @@ -824,12 +852,12 @@ class ThumbSrv(object): bits = tags[".bprs"][1] codec = b"pcm_s32le" - if bits == 16.0: + if bits <= 16.0: codec = b"pcm_s16le" - elif bits == 24.0: + elif bits <= 24.0: codec = b"pcm_s24le" - self.log("conv2 wav-tmp", 6) + self.log("conv2 wav", 6) # fmt: off cmd = [ @@ -843,7 +871,6 @@ class ThumbSrv(object): fsenc(tpath) ] # fmt: on - print(cmd) self._run_ff(cmd, vn, oom=300) def conv_opus(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index bd9fbbe7..201b3588 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -306,7 +306,8 @@ var Ls = { "mt_c2owa": "opus-weba, for iOS 17.5 and newer\">owa", "mt_c2caf": "opus-caf, for iOS 11 through 17\">caf", "mt_c2mp3": "use this on very old devices\">mp3", - "mt_c2wav": "use this for uncompressed playback\">wav", + "mt_c2flac": "best sound quality\">flac", + "mt_c2wav": "uncompressed playback\">wav", "mt_c2ok": "nice, good choice", "mt_c2nd": "that's not the recommended output format for your device, but that's fine", "mt_c2ng": "your device does not seem to support this output format, but let's try anyways", @@ -6803,6 +6804,7 @@ var mpl = (function () { 'owa", "mt_c2caf": "opus-caf, for iOS 11 through 17\">caf", "mt_c2mp3": "use this on very old devices\">mp3", + "mt_c2flac": "best sound quality\">flac", + "mt_c2wav": "uncompressed playback\">wav", "mt_c2ok": "nice, good choice", "mt_c2nd": "that's not the recommended output format for your device, but that's fine", "mt_c2ng": "your device does not seem to support this output format, but let's try anyways", From 0d09fb681869932b16015e5cba0b068ed4a27508 Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 21:23:41 +0000 Subject: [PATCH 069/174] audio transcoding tweaks --- README.md | 3 +++ copyparty/__main__.py | 4 ++-- copyparty/authsrv.py | 2 ++ copyparty/mtag.py | 4 +++- copyparty/th_srv.py | 8 ++++---- copyparty/web/browser.js | 37 ++++++++++++++++++++++++++----------- copyparty/web/splash.js | 1 + scripts/tl.js | 5 +++-- 8 files changed, 44 insertions(+), 20 deletions(-) diff --git a/README.md b/README.md index 1a500c5b..93b6a464 100644 --- a/README.md +++ b/README.md @@ -1084,6 +1084,9 @@ open the `[🎺]` media-player-settings tab to configure it, * `[awo]` is `opus` in a `weba` file, good for iPhones (iOS 17.5 and newer) but Apple is still fixing some state-confusion bugs as of iOS 18.2.1 * `[caf]` is `opus` in a `caf` file, good for iPhones (iOS 11 through 17), technically unsupported by Apple but works for the most part * `[mp3]` -- the myth, the legend, the undying master of mediocre sound quality that definitely works everywhere + * `[flac]` -- lossless but compressed, for LAN and/or fiber playback on electrostatic headphones + * `[wav]` -- lossless and uncompressed, for LAN and/or fiber playback on electrostatic headphones connected to very old equipment + * `flac` and `wav` must be enabled with `--allow-flac` / `--allow-wav` to allow spending the disk space * "tint" reduces the contrast of the playback bar diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 183fb916..21bda508 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1447,10 +1447,10 @@ def add_transcoding(ap): ap2 = ap.add_argument_group('transcoding options') ap2.add_argument("--q-opus", metavar="KBPS", type=int, default=128, help="target bitrate for transcoding to opus; set 0 to disable") ap2.add_argument("--q-mp3", metavar="QUALITY", type=u, default="q2", help="target quality for transcoding to mp3, for example [\033[32m192k\033[0m] (CBR) or [\033[32mq0\033[0m] (CQ/CRF, q0=maxquality, q9=smallest); set 0 to disable") + ap2.add_argument("--allow-wav", action="store_true", help="allow transcoding to wav (lossless, uncompressed)") + ap2.add_argument("--allow-flac", action="store_true", help="allow transcoding to flac (lossless, compressed)") ap2.add_argument("--no-caf", action="store_true", help="disable transcoding to caf-opus (affects iOS v12~v17), will use mp3 instead") ap2.add_argument("--no-owa", action="store_true", help="disable transcoding to webm-opus (iOS v18 and later), will use mp3 instead") - ap2.add_argument("--no-wav", action="store_true", help="disable transcoding to wav (lossless, uncompressed)") - ap2.add_argument("--no-flac", action="store_true", help="disable transcoding to flac (lossless, compressed)") ap2.add_argument("--no-acode", action="store_true", help="disable audio transcoding") ap2.add_argument("--no-bacode", action="store_true", help="disable batch audio transcoding by folder download (zip/tar)") ap2.add_argument("--ac-maxage", metavar="SEC", type=int, default=86400, help="delete cached transcode output after \033[33mSEC\033[0m seconds") diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 01fad895..76a09ecd 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -2751,6 +2751,8 @@ class AuthSrv(object): "s_name": self.args.bname, "have_up2k_idx": "e2d" in vf, "have_acode": not self.args.no_acode, + "have_c2flac": self.args.allow_flac, + "have_c2wav": self.args.allow_wav, "have_shr": self.args.shr, "have_zip": not self.args.no_zip, "have_mv": not self.args.no_mv, diff --git a/copyparty/mtag.py b/copyparty/mtag.py index 240b5a83..e9428b79 100644 --- a/copyparty/mtag.py +++ b/copyparty/mtag.py @@ -67,6 +67,8 @@ HAVE_FFPROBE = not os.environ.get("PRTY_NO_FFPROBE") and have_ff("ffprobe") CBZ_PICS = set("png jpg jpeg gif bmp tga tif tiff webp avif".split()) CBZ_01 = re.compile(r"(^|[^0-9v])0+[01]\b") +FMT_AU = set("mp3 ogg flac wav".split()) + class MParser(object): def __init__(self, cmdline: str) -> None: @@ -242,7 +244,7 @@ def parse_ffprobe(txt: str) -> tuple[dict[str, tuple[int, Any]], dict[str, list[ ret: dict[str, Any] = {} # processed md: dict[str, list[Any]] = {} # raw tags - is_audio = fmt.get("format_name") in ["mp3", "ogg", "flac", "wav"] + is_audio = fmt.get("format_name") in FMT_AU if fmt.get("filename", "").split(".")[-1].lower() in ["m4a", "aac"]: is_audio = True diff --git a/copyparty/th_srv.py b/copyparty/th_srv.py index e8094b5a..f40d0a43 100644 --- a/copyparty/th_srv.py +++ b/copyparty/th_srv.py @@ -814,8 +814,8 @@ class ThumbSrv(object): self._run_ff(cmd, vn, oom=300) def conv_flac(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: - if self.args.no_acode or self.args.no_flac: - raise Exception("disabled in server config") + if self.args.no_acode or not self.args.allow_flac: + raise Exception("flac not permitted in server config") self.wait4ram(0.2, tpath) tags, rawtags = ffprobe(abspath, int(vn.flags["convt"] / 2)) @@ -839,8 +839,8 @@ class ThumbSrv(object): self._run_ff(cmd, vn, oom=300) def conv_wav(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: - if self.args.no_acode or self.args.no_wav: - raise Exception("disabled in server config") + if self.args.no_acode or not self.args.allow_wav: + raise Exception("wav not permitted in server config") self.wait4ram(0.2, tpath) tags, rawtags = ffprobe(abspath, int(vn.flags["convt"] / 2)) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 201b3588..3dca9476 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -4,6 +4,7 @@ var XHR = XMLHttpRequest, img_re = /\.(a?png|avif|bmp|gif|heif|jpe?g|jfif|svg|webp|webm|mkv|mp4|m4v|mov)(\?|$)/i; // please add translations in alphabetic order, but keep "eng" and "nor" first +// (lines ending with //m are machine translations) var Ls = { "eng": { "tt": "English", @@ -306,8 +307,8 @@ var Ls = { "mt_c2owa": "opus-weba, for iOS 17.5 and newer\">owa", "mt_c2caf": "opus-caf, for iOS 11 through 17\">caf", "mt_c2mp3": "use this on very old devices\">mp3", - "mt_c2flac": "best sound quality\">flac", - "mt_c2wav": "uncompressed playback\">wav", + "mt_c2flac": "best sound quality, but huge downloads\">flac", + "mt_c2wav": "uncompressed playback (even bigger)\">wav", "mt_c2ok": "nice, good choice", "mt_c2nd": "that's not the recommended output format for your device, but that's fine", "mt_c2ng": "your device does not seem to support this output format, but let's try anyways", @@ -936,6 +937,8 @@ var Ls = { "mt_c2owa": "opus-weba, for iOS 17.5 og nyere\">owa", "mt_c2caf": "opus-caf, for iOS 11 tilogmed 17\">caf", "mt_c2mp3": "bra valg for steinalder-utstyr (slår aldri feil)\">mp3", + "mt_c2flac": "gir best lydkvalitet, men eter nettet ditt\">flac", + "mt_c2wav": "helt rå lydstrøm (bruker enda mere data enn flac)\">wav", "mt_c2ok": "bra valg!", "mt_c2nd": "ikke det foretrukne valget for din enhet, men funker sikkert greit", "mt_c2ng": "ser virkelig ikke ut som enheten din takler dette formatet... men ok, vi prøver", @@ -1563,6 +1566,8 @@ var Ls = { "mt_c2owa": "opus-weba(适用于 iOS 17.5 及更新版本)\">owa", //m "mt_c2caf": "opus-caf(适用于 iOS 11 到 iOS 17)\">caf", //m "mt_c2mp3": "适用于非常旧的设备\">mp3", //m + "mt_c2flac": "最佳音质,但下载量很大\">flac", //m + "mt_c2wav": "无压缩播放(更占空间)\">wav", //m "mt_c2ok": "不错的选择!", //m "mt_c2nd": "这不是您的设备推荐的输出格式,但应该没问题。", //m "mt_c2ng": "您的设备似乎不支持此输出格式,不过我们还是试试看吧。", //m @@ -2190,6 +2195,8 @@ var Ls = { "mt_c2owa": "opus-weba, für iOS 17.5 und neuer\">owa", "mt_c2caf": "opus-caf, für iOS 11 bis 17\">caf", "mt_c2mp3": "benutze dieses Format für ältere Geräte\">mp3", + "mt_c2flac": "beste Klangqualität, aber große Downloads\">flac", //m + "mt_c2wav": "unkomprimierte Wiedergabe (noch größer)\">wav", //m "mt_c2ok": "Gute Wahl, Chef!", "mt_c2nd": "Das ist nicht das empfohlene Ausgabeformat für dein Gerät, aber passt schon", "mt_c2ng": "Dein Gerät scheint dieses Ausgabeformat nicht zu unterstützen, aber lass trotzdem mal probieren", @@ -2817,6 +2824,8 @@ var Ls = { "mt_c2owa": "opus-weba, iOS 17.5:lle ja uudemmille\">owa", "mt_c2caf": "opus-caf, iOS 11:lle - 17:lle\">caf", "mt_c2mp3": "käytä tätä erittäin vanhoissa laitteissa\">mp3", + "mt_c2flac": "paras äänenlaatu, mutta isot lataukset\">flac", //m + "mt_c2wav": "pakkaamaton toisto (vielä suurempi tiedosto)\">wav", //m "mt_c2ok": "hienoa, hyvä valinta", "mt_c2nd": "tuo ei ole suositeltu formaatti laitteellesi, mutta tee miten lystäät", "mt_c2ng": "laitteesi ei näytä tukevan tätä formaattia, mutta yritetään nyt silti", @@ -3444,6 +3453,8 @@ var Ls = { "mt_c2owa": "opus-weba, per iOS 17.5 e più recenti\">owa", "mt_c2caf": "opus-caf, per iOS 11 fino a 17\">caf", "mt_c2mp3": "usa questo su dispositivi molto vecchi\">mp3", + "mt_c2flac": "qualità audio migliore, ma download pesanti\">flac", //m + "mt_c2wav": "riproduzione non compressa (ancora più grande)\">wav", //m "mt_c2ok": "bene, buona scelta", "mt_c2nd": "quello non è il formato di output raccomandato per il tuo dispositivo, ma va bene", "mt_c2ng": "il tuo dispositivo non sembra supportare questo formato di output, ma proviamo comunque", @@ -4071,6 +4082,8 @@ var Ls = { "mt_c2owa": "opus-weba, voor iOS 17.5 en nieuwer\">owa", "mt_c2caf": "opus-caf, voor iOS 11 tot en met iOS 17\">caf", "mt_c2mp3": "Gebruik dit hele oude toestellen\">mp3", + "mt_c2flac": "Beste geluidskwaliteit, maar grote downloads\">flac", //m + "mt_c2wav": "Ongemprimeerde weergave (nog groter)\">wav", //m "mt_c2ok": "Mooi, goede keuze", "mt_c2nd": "Dat is niet het aanbevolen uitvoerformaat voor uw apparaat, maar dat is prima", "mt_c2ng": "Uw apparaat lijkt dit uitvoerformaat niet te ondersteunen, maar we gaan het toch proberen", @@ -4698,6 +4711,8 @@ var Ls = { "mt_c2owa": "opus-weba, для iOS 17.5 и выше\">owa", "mt_c2caf": "opus-caf, для iOS 11-17\">caf", "mt_c2mp3": "для очень старых устройств\">mp3", + "mt_c2flac": "лучшее качество звука, но большие файлы\">flac", //m + "mt_c2wav": "не сжатое воспроизведение (ещё больше)\">wav", //m "mt_c2ok": "хороший выбор", "mt_c2nd": "это не рекомендованный вариант формата для вашего устройства, но сойдёт", "mt_c2ng": "не похоже, что ваше устройство поддерживает этот формат, но давайте попробуем и узнаем наверняка", @@ -5324,6 +5339,8 @@ var Ls = { "mt_c2owa": 'opus-weba, para iOS 17.5 y superior">owa', "mt_c2caf": 'opus-caf, para iOS 11 a 17">caf', "mt_c2mp3": 'usar en dispositivos muy antiguos">mp3', + "mt_c2flac": "la mejor calidad de sonido,$Npero descargas muy grandes\">flac", //m + "mt_c2wav": "reproducción sin comprimir (aún más grande)\">wav", //m "mt_c2ok": "bien, buena elección", "mt_c2nd": "ese no es el formato de salida recomendado para tu dispositivo, pero está bien", "mt_c2ng": "tu dispositivo no parece soportar este formato de salida, pero intentémoslo de todas formas", @@ -5951,6 +5968,8 @@ var Ls = { "mt_c2owa": "opus-weba, для iOS 17.5 і новіших\">owa", "mt_c2caf": "opus-caf, для iOS 11 до 17\">caf", "mt_c2mp3": "використовуйте це на дуже старих пристроях\">mp3", + "mt_c2flac": "найкраща якість звуку, але великі завантаження\">flac", //m + "mt_c2wav": "відтворення без стиснення (ще більше)\">wav", //m "mt_c2ok": "гарно, хороший вибір", "mt_c2nd": "це не рекомендований вихідний формат для вашого пристрою, але це нормально", "mt_c2ng": "ваш пристрій, здається, не підтримує цей вихідний формат, але давайте все одно спробуємо", @@ -6921,10 +6940,8 @@ var mpl = (function () { if (!have_acode) c = false; - else if (/\.flac$/i.exec(cs)) + else if (/\.(wav|flac)$/i.exec(cs)) c = r.ac_flac; - else if (/\.wav$/i.exec(cs)) - c = r.ac_wav; else if (/\.(aac|m4a)$/i.exec(cs)) c = r.ac_aac; else if (/\.(oga|ogg|opus)$/i.exec(cs) && (!can_ogg || mpl.ac2 == 'mp3')) @@ -6932,6 +6949,7 @@ var mpl = (function () { else if (re_au_native.exec(cs)) c = false; + // allow flac->flac (bitstream fixup) if (!c) return url; @@ -6949,9 +6967,7 @@ var mpl = (function () { } var dv = can_ogg ? 'opus' : - can_caf ? 'caf' : - can_mp3 ? 'mp3' : - can_flac ? 'flac' : 'wav', + can_caf ? 'caf' : 'mp3', fmts = ['opus', 'owa', 'caf', 'mp3', 'flac', 'wav'], btns = []; @@ -6963,7 +6979,6 @@ var mpl = (function () { if ((v == 'opus' && !can_ogg) || (v == 'caf' && !can_caf) || (v == 'owa' && !can_owa) || - (v == 'mp3' && !can_mp3) || (v == 'flac' && !can_flac)) toast.warn(15, L.mt_c2ng); @@ -6979,6 +6994,8 @@ var mpl = (function () { } if (!IPHONE) btns[1].style.display = btns[2].style.display = 'none'; + btns[4].style.display = have_c2flac ? '' : 'none'; + btns[5].style.display = have_c2wav ? '' : 'none'; if (v) swrite('acode2', v); @@ -7102,14 +7119,12 @@ var mpl = (function () { var za, can_ogg = true, can_owa = false, - can_mp3 = false, can_flac = false, can_caf = APPLE && !/ OS ([1-9]|1[01])_/.test(UA); try { za = new Audio(); can_ogg = za.canPlayType('audio/ogg; codecs=opus') === 'probably'; can_owa = za.canPlayType('audio/webm; codecs=opus') === 'probably'; - can_mp3 = za.canPlayType('audio/mpeg') === 'probably'; can_flac = za.canPlayType('audio/flac') === 'probably'; can_caf = za.canPlayType('audio/x-caf') && can_caf; //'maybe' } diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 44f315ff..bad3615f 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -1,4 +1,5 @@ // please add translations in alphabetic order, but keep "nor" and "eng" first +// (lines ending with //m are machine translations) var Ls = { "nor": { "a1": "oppdater", diff --git a/scripts/tl.js b/scripts/tl.js index e24d9451..97bf061d 100644 --- a/scripts/tl.js +++ b/scripts/tl.js @@ -315,6 +315,7 @@ var tl_browser = { "ct_qdel": 'when deleting files, only ask for confirmation once">qdel', "ct_dir1st": 'sort folders before files">📁 first', "ct_nsort": 'natural sort (for filenames with leading digits)">nsort', + "ct_utc": 'show all datetimes in UTC">UTC', "ct_readme": 'show README.md in folder listings">📜 readme', "ct_idxh": 'show index.html instead of folder listing">htm', "ct_sbars": 'show scrollbars">⟊', @@ -390,8 +391,8 @@ var tl_browser = { "mt_c2owa": "opus-weba, for iOS 17.5 and newer\">owa", "mt_c2caf": "opus-caf, for iOS 11 through 17\">caf", "mt_c2mp3": "use this on very old devices\">mp3", - "mt_c2flac": "best sound quality\">flac", - "mt_c2wav": "uncompressed playback\">wav", + "mt_c2flac": "best sound quality, but huge downloads\">flac", + "mt_c2wav": "uncompressed playback (even bigger)\">wav", "mt_c2ok": "nice, good choice", "mt_c2nd": "that's not the recommended output format for your device, but that's fine", "mt_c2ng": "your device does not seem to support this output format, but let's try anyways", From 66a5bf365bff16448b0843fd2fdda72939588fa0 Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 21:33:08 +0000 Subject: [PATCH 070/174] fix ipv6 qrcode port; closes #449 --- copyparty/tcpsrv.py | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/copyparty/tcpsrv.py b/copyparty/tcpsrv.py index 65f343d0..b0f9391e 100644 --- a/copyparty/tcpsrv.py +++ b/copyparty/tcpsrv.py @@ -583,8 +583,7 @@ class TcpSrv(object): if not ip: return "" - if ":" in ip: - ip = "[{}]".format(ip) + hip = "[%s]" % (ip,) if ":" in ip else ip if self.args.http_only: https = "" @@ -596,7 +595,7 @@ class TcpSrv(object): ports = t1.get(ip, t2.get(ip, [])) dport = 443 if https else 80 port = "" if dport in ports or not ports else ":{}".format(ports[0]) - txt = "http{}://{}{}/{}".format(https, ip, port, self.args.qrl) + txt = "http{}://{}{}/{}".format(https, hip, port, self.args.qrl) btxt = txt.encode("utf-8") if PY2: From 153d240d0de06f0925d6bd5c3ead7b8e457a495d Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 21:35:33 +0000 Subject: [PATCH 071/174] mention the -nc option in the max-conn errormsg --- copyparty/httpsrv.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/copyparty/httpsrv.py b/copyparty/httpsrv.py index 56638003..ddf5e7bc 100644 --- a/copyparty/httpsrv.py +++ b/copyparty/httpsrv.py @@ -324,7 +324,8 @@ class HttpSrv(object): spins = 0 while self.ncli >= self.nclimax: if not spins: - self.log(self.name, "at connection limit; waiting", 3) + t = "at connection limit (global-option 'nc'); waiting" + self.log(self.name, t, 3) spins += 1 time.sleep(0.1) From 7d3a5c1e977b260cefa3d080a752a5fd12119836 Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 21:35:52 +0000 Subject: [PATCH 072/174] black --- bin/u2c.py | 2 +- copyparty/tcpsrv.py | 2 +- copyparty/th_srv.py | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/bin/u2c.py b/bin/u2c.py index 85298730..e92c1e4e 100755 --- a/bin/u2c.py +++ b/bin/u2c.py @@ -52,7 +52,7 @@ if PY2: sys.dont_write_bytecode = True bytes = str - files_decoder = lambda s: unicode(s, 'utf8') + files_decoder = lambda s: unicode(s, "utf8") else: from urllib.parse import quote_from_bytes as quote from urllib.parse import unquote_to_bytes as unquote diff --git a/copyparty/tcpsrv.py b/copyparty/tcpsrv.py index b0f9391e..53388c5e 100644 --- a/copyparty/tcpsrv.py +++ b/copyparty/tcpsrv.py @@ -583,7 +583,7 @@ class TcpSrv(object): if not ip: return "" - hip = "[%s]" % (ip,) if ":" in ip else ip + hip = "[%s]" % (ip,) if ":" in ip else ip if self.args.http_only: https = "" diff --git a/copyparty/th_srv.py b/copyparty/th_srv.py index f40d0a43..7b07376a 100644 --- a/copyparty/th_srv.py +++ b/copyparty/th_srv.py @@ -846,7 +846,7 @@ class ThumbSrv(object): tags, rawtags = ffprobe(abspath, int(vn.flags["convt"] / 2)) if "ac" not in tags: raise Exception("not audio") - + bits = tags[".bps"][1] if bits == 0.0: bits = tags[".bprs"][1] From d0499257c8c0a634a7a1fcaa3f1309008efab237 Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 21:36:38 +0000 Subject: [PATCH 073/174] fix tests --- tests/util.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/util.py b/tests/util.py index 42a47907..f555a37e 100644 --- a/tests/util.py +++ b/tests/util.py @@ -143,7 +143,7 @@ class Cfg(Namespace): def __init__(self, a=None, v=None, c=None, **ka0): ka = {} - ex = "chpw cookie_lax daw dav_auth dav_mac dav_rt e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp early_ban ed emp exp force_js getmod grid gsel hardlink hardlink_only ih ihead localtime magic nid nih no_acode no_athumb no_bauth no_clone no_cp no_dav no_db_ip no_del no_dirsz no_dupe no_lifetime no_logues no_mv no_pipe no_poll no_readme no_robots no_sb_md no_sb_lg no_scandir no_tail no_tarcmp no_thumb no_vthumb no_zip nrand nsort nw og og_no_head og_s_title ohead q rand re_dirsz reflink rmagic rss smb srch_dbg srch_excl stats uqe vague_403 vc ver wo_up_readme write_uplog xdev xlink xvol zipmaxu zs" + ex = "allow_flac allow_wav chpw cookie_lax daw dav_auth dav_mac dav_rt e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp early_ban ed emp exp force_js getmod grid gsel hardlink hardlink_only ih ihead localtime magic nid nih no_acode no_athumb no_bauth no_clone no_cp no_dav no_db_ip no_del no_dirsz no_dupe no_lifetime no_logues no_mv no_pipe no_poll no_readme no_robots no_sb_md no_sb_lg no_scandir no_tail no_tarcmp no_thumb no_vthumb no_zip nrand nsort nw og og_no_head og_s_title ohead q rand re_dirsz reflink rmagic rss smb srch_dbg srch_excl stats uqe vague_403 vc ver wo_up_readme write_uplog xdev xlink xvol zipmaxu zs" ka.update(**{k: False for k in ex.split()}) ex = "dav_inf dedup dotpart dotsrch hook_v no_dhash no_fastboot no_fpool no_htp no_rescan no_sendfile no_ses no_snap no_up_list no_voldump re_dhash see_dots plain_ip" From e85a71070eb086a2cf39bb0202d8978b0bb5ef2d Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 21:42:26 +0000 Subject: [PATCH 074/174] docs: groups; closes #461 --- README.md | 6 ++++++ copyparty/__main__.py | 2 +- 2 files changed, 7 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 93b6a464..46d691a0 100644 --- a/README.md +++ b/README.md @@ -513,12 +513,17 @@ anyone trying to bruteforce a password gets banned according to `--ban-pw`; defa and if you want to use config files instead of commandline args (good!) then here's the same examples as a configfile; save it as `foobar.conf` and use it like this: `python copyparty-sfx.py -c foobar.conf` +* you can also `PRTY_CONFIG=foobar.conf python copyparty-sfx.py` (convenient in docker etc) + ```yaml [accounts] u1: p1 # create account "u1" with password "p1" u2: p2 # (note that comments must have u3: p3 # two spaces before the # sign) +[groups] + g1: u1, u2 # create a group + [/] # this URL will be mapped to... /srv # ...this folder on the server filesystem accs: @@ -528,6 +533,7 @@ and if you want to use config files instead of commandline args (good!) then her /mnt/music # which is mapped to this folder accs: r: u1, u2 # only these accounts can read, + r: @g1 # (exactly the same, just with a group instead) rw: u3 # and only u3 can read-write [/inc] diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 21bda508..9db3a4f5 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -578,7 +578,7 @@ def get_sects(): --grp takes groupname:username1,username2,... and groupnames can be used instead of usernames in -v - by prefixing the groupname with % + by prefixing the groupname with @ list of permissions: "r" (read): list folder contents, download files From 16bbcce51bb1720b7a0cabd2052584c953e2767e Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 22:10:33 +0000 Subject: [PATCH 075/174] videos can be folder thumbnails; closes #459 --- copyparty/up2k.py | 34 ++++++++++++++++++---------------- 1 file changed, 18 insertions(+), 16 deletions(-) diff --git a/copyparty/up2k.py b/copyparty/up2k.py index ae5ed07f..785d0a4f 100644 --- a/copyparty/up2k.py +++ b/copyparty/up2k.py @@ -86,7 +86,10 @@ if TYPE_CHECKING: from .svchub import SvcHub zsg = "avif,avifs,bmp,gif,heic,heics,heif,heifs,ico,j2p,j2k,jp2,jpeg,jpg,jpx,png,tga,tif,tiff,webp" -CV_EXTS = set(zsg.split(",")) +ICV_EXTS = set(zsg.split(",")) + +zsg = "3gp,asf,av1,avc,avi,flv,m4v,mjpeg,mjpg,mkv,mov,mp4,mpeg,mpeg2,mpegts,mpg,mpg2,mts,nut,ogm,ogv,rm,vob,webm,wmv" +VCV_EXTS = set(zsg.split(",")) zsg = "nohash noidx xdev xvol" VF_AFFECTS_INDEXING = set(zsg.split(" ")) @@ -1478,7 +1481,7 @@ class Up2k(object): unreg: list[str] = [] files: list[tuple[int, int, str]] = [] fat32 = True - cv = "" + cv = vcv = "" th_cvd = self.args.th_coversd th_cvds = self.args.th_coversd_set @@ -1573,25 +1576,24 @@ class Up2k(object): rsz += sz files.append((sz, lmod, iname)) - liname = iname.lower() - if ( - sz - and ( + if sz: + liname = iname.lower() + ext = liname.rsplit(".", 1)[-1] + if ( liname in th_cvds - or ( - not cv - and liname.rsplit(".", 1)[-1] in CV_EXTS - and not iname.startswith(".") - ) - ) - and ( + or (not cv and ext in ICV_EXTS and not iname.startswith(".")) + ) and ( not cv or liname not in th_cvds or cv.lower() not in th_cvds or th_cvd.index(liname) < th_cvd.index(cv.lower()) - ) - ): - cv = iname + ): + cv = iname + elif not vcv and ext in VCV_EXTS and not iname.startswith("."): + vcv = iname + + if not cv: + cv = vcv if not self.args.no_dirsz: tnf += len(files) From 39e55824961a8870aad56b77ff6b0c6d9cd8aa26 Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 22:14:31 +0000 Subject: [PATCH 076/174] ignore browser-extension errors; closes #435 --- copyparty/web/util.js | 3 +++ 1 file changed, 3 insertions(+) diff --git a/copyparty/web/util.js b/copyparty/web/util.js index 94f73f24..929316e5 100644 --- a/copyparty/web/util.js +++ b/copyparty/web/util.js @@ -180,6 +180,9 @@ function vis_exh(msg, url, lineNo, columnNo, error) { if (!/\.js($|\?)/.exec(url)) return; // chrome debugger + if (url.indexOf('extension://') + 1) + return; + if (url.indexOf(' > eval') + 1 && !evalex_fatal) return; // md timer From 9db8037e39ea9330337a0aa5c52511432dc6a61e Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 22:24:36 +0000 Subject: [PATCH 077/174] remove old joke end of an era --- copyparty/__init__.py | 4 ---- copyparty/__main__.py | 2 +- copyparty/httpcli.py | 9 --------- copyparty/web/browser.css | 13 +------------ copyparty/web/dd/2.png | Bin 258 -> 0 bytes copyparty/web/dd/3.png | Bin 252 -> 0 bytes copyparty/web/dd/4.png | Bin 248 -> 0 bytes copyparty/web/dd/5.png | Bin 250 -> 0 bytes copyparty/web/dd/__init__.py | 0 docs/devnotes.md | 1 - pyproject.toml | 1 - scripts/copyparty-repack.sh | 8 ++++---- scripts/make-sfx.sh | 14 ++------------ scripts/sfx.ls | 6 ------ setup.py | 1 - 15 files changed, 8 insertions(+), 51 deletions(-) delete mode 100644 copyparty/web/dd/2.png delete mode 100644 copyparty/web/dd/3.png delete mode 100644 copyparty/web/dd/4.png delete mode 100644 copyparty/web/dd/5.png delete mode 100644 copyparty/web/dd/__init__.py diff --git a/copyparty/__init__.py b/copyparty/__init__.py index 86b7fe1e..d1630ba0 100644 --- a/copyparty/__init__.py +++ b/copyparty/__init__.py @@ -63,10 +63,6 @@ web/browser.js web/browser2.html web/cf.html web/copyparty.gif -web/dd/2.png -web/dd/3.png -web/dd/4.png -web/dd/5.png web/deps/busy.mp3 web/deps/easymde.css web/deps/easymde.js diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 9db3a4f5..9e406bea 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1567,7 +1567,7 @@ def add_ui(ap, retry): ap2.add_argument("--unlist", metavar="REGEX", type=u, default="", help="don't show files/folders matching \033[33mREGEX\033[0m in file list. WARNING: Purely cosmetic! Does not affect API calls, just the browser. Example: [\033[32m\\.(js|css)$\033[0m] (volflag=unlist)") ap2.add_argument("--favico", metavar="TXT", type=u, default="c 000 none" if retry else "🎉 000 none", help="\033[33mfavicon-text\033[0m [ \033[33mforeground\033[0m [ \033[33mbackground\033[0m ] ], set blank to disable") ap2.add_argument("--ext-th", metavar="E=VP", type=u, action="append", help="use thumbnail-image \033[33mVP\033[0m for file-extension \033[33mE\033[0m, example: [\033[32mexe=/.res/exe.png\033[0m] (volflag=ext_th)") - ap2.add_argument("--mpmc", metavar="URL", type=u, default="", help="change the mediaplayer-toggle mouse cursor; URL to a folder with {2..5}.png inside (or disable with [\033[32m.\033[0m])") + ap2.add_argument("--mpmc", type=u, default="", help=argparse.SUPPRESS) ap2.add_argument("--spinner", metavar="TXT", type=u, default="🌲", help="\033[33memoji\033[0m or \033[33memoji,css\033[0m Example: [\033[32m🥖,padding:0\033[0m]") ap2.add_argument("--css-browser", metavar="L", type=u, default="", help="URL to additional CSS to include in the filebrowser html") ap2.add_argument("--js-browser", metavar="L", type=u, default="", help="URL to additional JS to include in the filebrowser html") diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index db1cbcb8..44087e1f 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -1196,15 +1196,6 @@ class HttpCli(object): self.reply(b"ssdp is disabled in server config", 404) return False - if self.vpath.startswith(".cpr/dd/") and self.args.mpmc: - if self.args.mpmc == ".": - raise Pebkac(404) - - loc = self.args.mpmc.rstrip("/") + self.vpath[self.vpath.rfind("/") :] - h = {"Location": loc, "Cache-Control": "max-age=39"} - self.reply(b"", 301, headers=h) - return True - if self.vpath == ".cpr/metrics": return self.conn.hsrv.metrics.tx(self) diff --git a/copyparty/web/browser.css b/copyparty/web/browser.css index 35dbeeac..1f5a7da6 100644 --- a/copyparty/web/browser.css +++ b/copyparty/web/browser.css @@ -1113,18 +1113,7 @@ html.y #widget.open { top: -.12em; } #wtico { - cursor: url(dd/4.png), pointer; - animation: cursor 500ms; -} -#wtico:hover { - animation: cursor 500ms infinite; -} -@keyframes cursor { - 0% {cursor: url(dd/2.png), pointer} - 30% {cursor: url(dd/3.png), pointer} - 50% {cursor: url(dd/4.png), pointer} - 75% {cursor: url(dd/5.png), pointer} - 85% {cursor: url(dd/4.png), pointer} + cursor: pointer; } @keyframes spin { 100% {transform: rotate(360deg)} diff --git a/copyparty/web/dd/2.png b/copyparty/web/dd/2.png deleted file mode 100644 index 5f7509580fd235cfa30e0f9e06f20031bbf004d3..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 258 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz&H|6fVjv9yj0`Q6;vmizPZ!6K ziaBqmDGD8O5MX&8o)O~{*YHp|pkoiiapwvq!6)2;5ep0&)NY)aGo$OI+nS86;hSqi zqpRB=m)Z6(I@s)+@A-hs>GgN>W)TkY3vI!yQCnK9=QEuazMvRlvcSpp0mC8AFYDDO zGJRxvem70xF;m_4z5qqpQ>QciGxm!_%yTeQ*)$`p)H!2shQilJAH5q37`Jznbi_Cx yeY)IXPXl)VqxcKG1o1$wGkOc;yKZ_mW;EKDWPaALSh?;o$S0nzelF{r5}E)Z{$9cW diff --git a/copyparty/web/dd/3.png b/copyparty/web/dd/3.png deleted file mode 100644 index 524facc389c48bccf623fed592505335b93740d6..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 252 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz&H|6fVjv9yj0`Q6;vmjCPZ!6K ziaBqmF63o$6meOu{bi?dNW)Gy1??3K?ut_mNJcp%G$t^dZQL`{cSgbwm70%N@}BMY zRjc3lYE~i-!=sY@d!JtZeO$X)ghTv7gUAniM%{&r7;UQ-EfDZasJ9f>6wu!=^L(43 z-h;v$g1S*ot=nF+F8mzy;5*BzNhcP~^qJl=FP!%AFGrA91#U{NU;8=d#Wzp$Py9YFyj`&9{y!~Ul5VmF5BMCmPnf_gAtmwQQ6M{yE&Bp>r;?tB z3?=T9UNl(e$1i>W6y3wT?V8(v_hT(Q4(z-U1B`2{($xhGutr8x{I-FoAVHYr(FoB&z%w;j(g5As< jrbd~4?y?3K_!t=SUww`EuyW2Vko!Gd{an^LB{Ts50OL}6 diff --git a/copyparty/web/dd/5.png b/copyparty/web/dd/5.png deleted file mode 100644 index 3d1d181e31e8a78396654a4c66f10ac9beaff096..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 250 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz&H|6fVjv9yj0`Q6;vmi%PZ!6K ziaBqm?G|h{5O7(qoh0tS>UXC@k~QSP8Qu$x-UZ?&2lNfkJ)IKEDm~*NhhD|PviWIw zew&WJowR+1q(R;FSV_B%)`r~-;{UW0PBsWw>}TLtJj{4fXaU#Ex|t0EAH*x}uCMIo zIuXmhqUrl)#vS@B=hOpuICYn;%4^(n&LPm@dG~RBmOHFhlpZNQa?nc&spI&;y@Ek3 rpz#$`kznhAhfIGKTFepnAohdN_Hssf-jU-bARl Date: Sun, 3 Aug 2025 22:34:37 +0000 Subject: [PATCH 078/174] fix nosub with PUT uploads; closes #412 --- copyparty/httpcli.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 44087e1f..9463e982 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -2074,16 +2074,16 @@ class HttpCli(object): rnd, lifetime, xbu, xau = self.upload_flags(vfs) lim = vfs.get_dbv(rem)[0].lim fdir = vfs.canonical(rem) - if lim: - fdir, rem = lim.all( - self.ip, rem, remains, vfs.realpath, fdir, self.conn.hsrv.broker - ) - fn = None if rem and not self.trailing_slash and not bos.path.isdir(fdir): fdir, fn = os.path.split(fdir) rem, _ = vsplit(rem) + if lim: + fdir, rem = lim.all( + self.ip, rem, remains, vfs.realpath, fdir, self.conn.hsrv.broker + ) + bos.makedirs(fdir, vf=vfs.flags) open_ka: dict[str, Any] = {"fun": open} From 848315c0091f66fe0f43dc5f2fe3babca16d3744 Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 22:43:57 +0000 Subject: [PATCH 079/174] do not force d2d with default vfs; #295 fixes v1.18.3 regression --- copyparty/authsrv.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 76a09ecd..afbc24ba 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -1713,7 +1713,8 @@ class AuthSrv(object): t = "Read-access has been disabled due to failsafe: No volumes were defined by the config-file. This failsafe is to prevent unintended access if this is due to accidental loss of config. You can override this safeguard and allow read/write to the working-directory by adding the following arguments: -v .::rw" self.log(t, 1) axs = AXS() - vfs = VFS(self.log_func, absreal("."), "", "", axs, self.vf0()) + zvf = {"tcolor": self.args.tcolor} + vfs = VFS(self.log_func, absreal("."), "", "", axs, zvf) if not axs.uread: self.badcfg1 = True elif "" not in mount: From 6eaf8af15aba4937ff86edb22d9d28e09f335094 Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 22:50:12 +0000 Subject: [PATCH 080/174] pypi: add extras-group "all"; closes #398 --- pyproject.toml | 8 ++++++++ setup.py | 1 + 2 files changed, 9 insertions(+) diff --git a/pyproject.toml b/pyproject.toml index d1f59351..a48e6579 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -45,6 +45,14 @@ classifiers = [ "Demo Server" = "https://a.ocv.me/pub/demo/" [project.optional-dependencies] +all = [ + "argon2-cffi", + "partftpy>=0.4.0", + "Pillow", + "pyftpdlib", + "pyopenssl", + "pyzmq", +] thumbnails = ["Pillow"] thumbnails2 = ["pyvips"] audiotags = ["mutagen"] diff --git a/setup.py b/setup.py index b7554e2f..ce5afb47 100755 --- a/setup.py +++ b/setup.py @@ -136,6 +136,7 @@ args = { ], "install_requires": ["jinja2"], "extras_require": { + "all": ["argon2-cffi", "partftpy>=0.4.0", "Pillow", "pyftpdlib", "pyopenssl", "pyzmq"], "thumbnails": ["Pillow"], "thumbnails2": ["pyvips"], "audiotags": ["mutagen"], From ae5eefc52845c1404c78f762d49e5ea073e71a1c Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 23:02:56 +0000 Subject: [PATCH 081/174] add sfx explanation; #345 --- README.md | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 46d691a0..22d3a0d7 100644 --- a/README.md +++ b/README.md @@ -147,6 +147,8 @@ made in Norway 🇳🇴 just run **[copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py)** -- that's it! 🎉 +> ℹ️ the sfx is a [self-extractor](https://github.com/9001/copyparty/issues/270) which unpacks an embedded `tar.gz` into `$TEMP` -- if this looks too scary, you can use the [zipapp](#zipapp) which has slightly worse performance + * or install through [pypi](https://pypi.org/project/copyparty/): `python3 -m pip install --user -U copyparty` * or if you cannot install python, you can use [copyparty.exe](#copypartyexe) instead * or install [on arch](#arch-package) ╱ [on NixOS](#nixos-module) ╱ [through nix](#nix-package) @@ -2859,7 +2861,7 @@ then again, if you are already into downloading shady binaries from the internet ## zipapp -another emergency alternative, [copyparty.pyz](https://github.com/9001/copyparty/releases/latest/download/copyparty.pyz) has less features, is slow, requires python 3.7 or newer, worse compression, and more importantly is unable to benefit from more recent versions of jinja2 and such (which makes it less secure)... lots of drawbacks with this one really -- but it does not unpack any temporary files to disk, so it *may* just work if the regular sfx fails to start because the computer is messed up in certain funky ways, so it's worth a shot if all else fails +another emergency alternative, [copyparty.pyz](https://github.com/9001/copyparty/releases/latest/download/copyparty.pyz) has less features, is slow, requires python 3.7 or newer, worse compression, and more importantly is unable to benefit from more recent versions of jinja2 and such (which makes it less secure)... lots of drawbacks with this one really -- but, unlike the sfx, it is a completely normal zipfile which does not unpack any temporary files to disk, so it *may* just work if the regular sfx fails to start because the computer is messed up in certain funky ways, so it's worth a shot if all else fails run it by doubleclicking it, or try typing `python copyparty.pyz` in your terminal/console/commandline/telex if that fails From f9502c3df35a1c8eded8eb192b09fc11e8a7880d Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 23:27:53 +0000 Subject: [PATCH 082/174] add idp-cookie; for high-traffic / glitchy auth servers --- README.md | 2 ++ copyparty/__main__.py | 1 + copyparty/authsrv.py | 5 ++++- copyparty/httpcli.py | 16 ++++++++++++++++ 4 files changed, 23 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 22d3a0d7..f7c27be8 100644 --- a/README.md +++ b/README.md @@ -1894,6 +1894,8 @@ you can disable the built-in password-based login system, and instead replace it * the regular config-defined users will be used as a fallback for requests which don't include a valid (trusted) IdP username header +* if your IdP-server is slow, consider `--idp-cookie` and let requests with the cookie `cppws` bypass the IdP; experimental sessions-based feature added for a party + some popular identity providers are [Authelia](https://www.authelia.com/) (config-file based) and [authentik](https://goauthentik.io/) (GUI-based, more complex) there is a [docker-compose example](./docs/examples/docker/idp-authelia-traefik) which is hopefully a good starting point (alternatively see [./docs/idp.md](./docs/idp.md) if you're the DIY type) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 9e406bea..d92089d0 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1150,6 +1150,7 @@ def add_auth(ap): ap2.add_argument("--idp-db", metavar="PATH", type=u, default=idp_db, help="where to store the known IdP users/groups (if you run multiple copyparty instances, make sure they use different DBs)") ap2.add_argument("--idp-store", metavar="N", type=int, default=1, help="how to use \033[33m--idp-db\033[0m; [\033[32m0\033[0m] = entirely disable, [\033[32m1\033[0m] = write-only (effectively disabled), [\033[32m2\033[0m] = remember users, [\033[32m3\033[0m] = remember users and groups.\nNOTE: Will remember and restore the IdP-volumes of all users for all eternity if set to 2 or 3, even when user is deleted from your IdP") ap2.add_argument("--idp-adm", metavar="U,U", type=u, default="", help="comma-separated list of users allowed to use /?idp (the cache management UI)") + ap2.add_argument("--idp-cookie", metavar="S", type=int, default=0, help="generate a session-token for IdP users which is written to cookie \033[33mcppws\033[0m (or \033[33mcppwd\033[0m if plaintext), to reduce the load on the IdP server, lifetime \033[33mS\033[0m seconds.\n └─note: The expiration time is a client hint only; the actual lifetime of the session-token is infinite (until next restart with \033[33m--ses-db\033[0m wiped)") ap2.add_argument("--no-bauth", action="store_true", help="disable basic-authentication support; do not accept passwords from the 'Authenticate' header at all. NOTE: This breaks support for the android app") ap2.add_argument("--bauth-last", action="store_true", help="keeps basic-authentication enabled, but only as a last-resort; if a cookie is also provided then the cookie wins") ap2.add_argument("--ses-db", metavar="PATH", type=u, default=ses_db, help="where to store the sessions database (if you run multiple copyparty instances, make sure they use different DBs)") diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index afbc24ba..44629e62 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -2869,7 +2869,10 @@ class AuthSrv(object): n = [] q = "insert into us values (?,?,?)" - for uname in self.acct: + accs = list(self.acct) + if self.args.idp_h_usr and self.args.idp_cookie: + accs.extend(self.idp_accs.keys()) + for uname in accs: if uname not in ases: sid = ub64enc(os.urandom(blen)).decode("ascii") cur.execute(q, (uname, sid, int(time.time()))) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 9463e982..d562485b 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -658,6 +658,9 @@ class HttpCli(object): self.pw = "" self.uname = idp_usr self.html_head += "\n" + zs = self.asrv.ases.get(idp_usr) + if zs: + self.set_idp_cookie(zs) else: self.log("unknown username: %r" % (idp_usr,), 1) @@ -3029,6 +3032,19 @@ class HttpCli(object): return dur > 0, msg + def set_idp_cookie(self, ases) -> None: + k = "cppws" if self.is_https else "cppwd" + ck = gencookie( + k, + ases, + self.args.R, + self.args.cookie_lax, + self.is_https, + self.args.idp_cookie, + "; HttpOnly", + ) + self.out_headers["Set-Cookie"] = ck + def handle_mkdir(self) -> bool: assert self.parser # !rm new_dir = self.parser.require("name", 512) From 40d56bb3f0b218389c230dfb808c622307a43cda Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 23:35:08 +0000 Subject: [PATCH 083/174] indicate play-as-audio for video files --- copyparty/web/browser.js | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 3dca9476..f1cf6562 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -6166,7 +6166,7 @@ var Ls = { "xhr403": "403: Доступ заборонено\n\nспробуйте натиснути F5, можливо ви вийшли з системи", "xhr0": "невідома (ймовірно втрачено з'єднання з сервером, або сервер офлайн)", - "cf_ok": "вибачте за це -- захист від DD" + "oS спрацював\n\nречі повинні відновитися приблизно через 30 сек\n\nякщо нічого не відбувається, натисніть F5 для перезавантаження сторінки", + "cf_ok": "вибачте за це -- захист від DD" + wah + "oS спрацював\n\nречі повинні відновитися приблизно через 30 сек\n\nякщо нічого не відбувається, натисніть F5 для перезавантаження сторінки", "tl_xe1": "не вдалося перелічити підпапки:\n\nпомилка ", "tl_xe2": "404: Папка не знайдена", "fl_xe1": "не вдалося перелічити файли в папці:\n\nпомилка ", @@ -7139,6 +7139,7 @@ mpl.init_ac2(); var re_m3u = /\.(m3u8?)$/i; var re_au_native = (can_ogg || have_acode) ? /\.(aac|flac|m4a|mp3|oga|ogg|opus|wav)$/i : /\.(aac|flac|m4a|mp3|wav)$/i, + re_au_vid = /\.(3gp|asf|avi|flv|m4v|mkv|mov|mp4|mpeg|mpeg2|mpegts|mpg|mpg2|nut|ogm|ogv|rm|ts|vob|webm|wmv)$/i, re_au_all = /\.(aac|ac3|aif|aiff|alac|alaw|amr|ape|au|dfpwm|dts|flac|gsm|it|itgz|itxz|itz|m4a|mdgz|mdxz|mdz|mo3|mod|mp2|mp3|mpc|mptm|mt2|mulaw|oga|ogg|okt|opus|ra|s3m|s3gz|s3xz|s3z|tak|tta|ulaw|wav|wma|wv|xm|xmgz|xmxz|xmz|xpk|3gp|asf|avi|flv|m4v|mkv|mov|mp4|mpeg|mpeg2|mpegts|mpg|mpg2|nut|ogm|ogv|rm|ts|vob|webm|wmv)$/i; @@ -7167,10 +7168,11 @@ function MPlayer() { fn = url.split('?')[0]; if (re_audio.exec(fn)) { - var tid = link.getAttribute('id'); + var tid = link.getAttribute('id'), + txt = re_au_vid.exec(fn) ? '(🎧)' : L.mb_play; r.order.push(tid); r.tracks[tid] = url; - tds[0].innerHTML = '' + L.mb_play + ''; + tds[0].innerHTML = '' + txt + ''; ebi('a' + tid).onclick = ev_play; clmod(trs[a], 'au', 1); } From 715f8424b303fe8163c2463727f682bf60322e3c Mon Sep 17 00:00:00 2001 From: ed Date: Sun, 3 Aug 2025 23:42:51 +0000 Subject: [PATCH 084/174] tl cleanup --- copyparty/web/browser.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index f1cf6562..d7982c78 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3501,10 +3501,10 @@ var Ls = { "f_partial": "Per scaricare in sicurezza un file che è attualmente in fase di caricamento, clicca il file che ha lo stesso nome, ma senza l'estensione .PARTIAL. Premi ANNULLA o Escape per farlo.\n\nPremendo OK / Invio ignorerai questo avviso e continuerai a scaricare il file .PARTIAL scratch, che quasi sicuramente ti darà dati corrotti.", "ft_paste": "incolla {0} elementi$NTasto rapido: ctrl-V", - "fr_eperm": 'impossibile rinominare:\nnon hai il permesso "sposta" in questa cartella', - "fd_eperm": 'impossibile eliminare:\nnon hai il permesso "elimina" in questa cartella', - "fc_eperm": 'impossibile tagliare:\nnon hai il permesso "sposta" in questa cartella', - "fp_eperm": 'impossibile incollare:\nnon hai il permesso "scrivi" in questa cartella', + "fr_eperm": 'impossibile rinominare:\nnon hai il permesso “sposta” in questa cartella', + "fd_eperm": 'impossibile eliminare:\nnon hai il permesso “elimina” in questa cartella', + "fc_eperm": 'impossibile tagliare:\nnon hai il permesso “sposta” in questa cartella', + "fp_eperm": 'impossibile incollare:\nnon hai il permesso “scrivi” in questa cartella', "fr_emore": "seleziona almeno un elemento da rinominare", "fd_emore": "seleziona almeno un elemento da eliminare", "fc_emore": "seleziona almeno un elemento da tagliare", From d7e7e77f930f6887af8215a6d6a87400383031ca Mon Sep 17 00:00:00 2001 From: ed Date: Mon, 4 Aug 2025 00:13:54 +0000 Subject: [PATCH 085/174] v1.18.10 --- copyparty/__version__.py | 4 +- docs/changelog.md | 43 ++++++++ docs/logo-sq.svg | 216 +++++++++++++++++++++++++++++++++++++++ docs/logo256.svg | 209 +++++++++++++++++++++++++++++++++++++ tests/util.py | 2 +- 5 files changed, 471 insertions(+), 3 deletions(-) create mode 100644 docs/logo-sq.svg create mode 100644 docs/logo256.svg diff --git a/copyparty/__version__.py b/copyparty/__version__.py index a82053fb..30bd3fd6 100644 --- a/copyparty/__version__.py +++ b/copyparty/__version__.py @@ -1,8 +1,8 @@ # coding: utf-8 -VERSION = (1, 18, 9) +VERSION = (1, 18, 10) CODENAME = "logtail" -BUILD_DT = (2025, 8, 1) +BUILD_DT = (2025, 8, 4) S_VERSION = ".".join(map(str, VERSION)) S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) diff --git a/docs/changelog.md b/docs/changelog.md index 5e1d7717..3cc8bdf1 100644 --- a/docs/changelog.md +++ b/docs/changelog.md @@ -1,3 +1,46 @@ +▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀ +# 2025-0801-2056 `v1.18.9` fix Denial-of-Service + +## ⚠️ ATTN: this release fixes a Denial-of-Service vuln + +[CVE-2025-54796](https://github.com/9001/copyparty/security/advisories/GHSA-5662-2rj7-f2v6): an unauthenticated user could make the server grind to a halt by accessing a particular URL + +## recent important news + +* [v1.18.9 (2025-08-01)](https://github.com/9001/copyparty/releases/tag/v1.18.9) fixed [CVE-2025-54796](https://github.com/9001/copyparty/security/advisories/GHSA-5662-2rj7-f2v6) (Denial-of-Service) +* [v1.15.0 (2024-09-08)](https://github.com/9001/copyparty/releases/tag/v1.15.0) changed upload deduplication to be default-disabled +* [v1.14.3 (2024-08-30)](https://github.com/9001/copyparty/releases/tag/v1.14.3) fixed a bug that was introduced in v1.13.8 (2024-08-13); this bug could lead to **data loss** -- see the v1.14.3 release-notes for details + +## 🧪 new features + +* #310 translated to Spanish (thx @herruzo99!) a1dfd0be +* #350 translated to Ukrainian (thx @MrMebelMan!) fea45e45 +* #321 translated to Russian (thx @A1Asriel!) 0b05c726 +* #381 translated to Finnish (thx @icxes and @Permik!) 7ecedb2c + * haha it says surf +* #312 add option to use localtime in the UI ad23b253 +* #386 initial packaging for debian (thx @Beethoven-n!) 3c6f0b17 + +## 🩹 bugfixes + +* CVE-2025-54796 / GHSA-5662-2rj7-f2v6 09910ba8 +* #347 fix upload-abort when uploading to a share 6d6d79fc +* fix xiu backlog dropping on restart 3222ba3a +* #375 fix crash on really old versions of python2.7 (thx @bb!) b69d5901 +* #388 another python2.7 fix: improve unicode support in u2c (thx @KevinXuxuxu!) 9c197535 +* log creator of new/blank markdown docs d0d2f206 +* #400 config didn't support indenting with tabs c1604288 + +## 🔧 other changes + +* `ack` was changed to `continue` 4fa7be2a + +## 🌠 fun facts + +* the translations have made the sfx size balloon from 766 to 845 KiB in under a week... nice! keep em coming :tada: + + + ▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀ # 2025-0731-0833 `v1.18.8` sfx hotfix diff --git a/docs/logo-sq.svg b/docs/logo-sq.svg new file mode 100644 index 00000000..cca66519 --- /dev/null +++ b/docs/logo-sq.svg @@ -0,0 +1,216 @@ + + + + copyparty_logo + + + + + + + + + + + + image/svg+xml + + copyparty_logo + github.com/9001/copyparty + + + + + + + + + + + + + + + + + + + diff --git a/docs/logo256.svg b/docs/logo256.svg new file mode 100644 index 00000000..8fdb2782 --- /dev/null +++ b/docs/logo256.svg @@ -0,0 +1,209 @@ + + + copyparty_logo + + + + + + + + + + + + image/svg+xml + + copyparty_logo + github.com/9001/copyparty + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/util.py b/tests/util.py index f555a37e..ea9bfeb9 100644 --- a/tests/util.py +++ b/tests/util.py @@ -161,7 +161,7 @@ class Cfg(Namespace): ex = "au_vol dl_list mtab_age reg_cap s_thead s_tbody tail_tmax tail_who th_convt ups_who zip_who" ka.update(**{k: 9 for k in ex.split()}) - ex = "db_act forget_ip idp_store k304 loris no304 nosubtle re_maxage rproxy rsp_jtr rsp_slp s_wr_slp snap_wri theme themes turbo u2ow zipmaxn zipmaxs" + ex = "db_act forget_ip idp_cookie idp_store k304 loris no304 nosubtle re_maxage rproxy rsp_jtr rsp_slp s_wr_slp snap_wri theme themes turbo u2ow zipmaxn zipmaxs" ka.update(**{k: 0 for k in ex.split()}) ex = "ah_alg bname chmod_f chpw_db doctitle df exit favico idp_h_usr ipa html_head lg_sba lg_sbf log_fk md_sba md_sbf name og_desc og_site og_th og_title og_title_a og_title_v og_title_i shr tcolor textfiles unlist vname xff_src zipmaxt R RS SR" From d4397e7217c88c9b201666c5062dd1f60e4d432c Mon Sep 17 00:00:00 2001 From: ed Date: Mon, 4 Aug 2025 00:39:26 +0000 Subject: [PATCH 086/174] update pkgs to 1.18.10 --- contrib/package/arch/PKGBUILD | 4 ++-- contrib/package/makedeb-mpr/PKGBUILD | 4 ++-- contrib/package/nix/copyparty/pin.json | 6 +++--- 3 files changed, 7 insertions(+), 7 deletions(-) diff --git a/contrib/package/arch/PKGBUILD b/contrib/package/arch/PKGBUILD index 40c0b62b..9efb5c41 100644 --- a/contrib/package/arch/PKGBUILD +++ b/contrib/package/arch/PKGBUILD @@ -1,6 +1,6 @@ # Maintainer: icxes pkgname=copyparty -pkgver="1.18.9" +pkgver="1.18.10" pkgrel=1 pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" arch=("any") @@ -22,7 +22,7 @@ optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tag ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") backup=("etc/${pkgname}.d/init" ) -sha256sums=("d5d33b50d6717e52427956beb1687061a6f28b467997506505151e3ae18c58e5") +sha256sums=("49c5fedf7619437bc0af125cb4e8360d9bda9d87ef45d6314d7acf163ab4cf99") build() { cd "${srcdir}/${pkgname}-${pkgver}" diff --git a/contrib/package/makedeb-mpr/PKGBUILD b/contrib/package/makedeb-mpr/PKGBUILD index 227fc4cf..f16e5459 100644 --- a/contrib/package/makedeb-mpr/PKGBUILD +++ b/contrib/package/makedeb-mpr/PKGBUILD @@ -2,7 +2,7 @@ pkgname=copyparty -pkgver=1.18.9 +pkgver=1.18.10 pkgrel=1 pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" arch=("any") @@ -20,7 +20,7 @@ optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tag ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") backup=("/etc/${pkgname}.d/init" ) -sha256sums=("d5d33b50d6717e52427956beb1687061a6f28b467997506505151e3ae18c58e5") +sha256sums=("49c5fedf7619437bc0af125cb4e8360d9bda9d87ef45d6314d7acf163ab4cf99") build() { cd "${srcdir}/${pkgname}-${pkgver}/copyparty/web" diff --git a/contrib/package/nix/copyparty/pin.json b/contrib/package/nix/copyparty/pin.json index 734dec31..70657f2c 100644 --- a/contrib/package/nix/copyparty/pin.json +++ b/contrib/package/nix/copyparty/pin.json @@ -1,5 +1,5 @@ { - "url": "https://github.com/9001/copyparty/releases/download/v1.18.9/copyparty-sfx.py", - "version": "1.18.9", - "hash": "sha256-R1OVx4f8GERAG80ZcHAIP6HK2TlBbKJZpvnJmJbGPRY=" + "url": "https://github.com/9001/copyparty/releases/download/v1.18.10/copyparty-sfx.py", + "version": "1.18.10", + "hash": "sha256-2FdQ5aCwNdZ5Jy9mn8rr8g41+QHT5tsEc+GeAKvhGeg=" } \ No newline at end of file From 8c000fd68353ff087ea2208eb90ba9fb61605eef Mon Sep 17 00:00:00 2001 From: ed Date: Mon, 4 Aug 2025 00:40:27 +0000 Subject: [PATCH 087/174] cleanup --- README.md | 2 +- scripts/prep.sh | 14 +++++--------- 2 files changed, 6 insertions(+), 10 deletions(-) diff --git a/README.md b/README.md index f7c27be8..9746581b 100644 --- a/README.md +++ b/README.md @@ -1229,7 +1229,7 @@ using arguments or config files, or a mix of both: **NB:** as humongous as this readme is, there is also a lot of undocumented features. Run copyparty with `--help` to see all available global options; all of those can be used in the `[global]` section of config files, and everything listed in `--help-flags` can be used in volumes as volflags. * if running in docker/podman, try this: `docker run --rm -it copyparty/ac --help` -* or see this (probably outdated): https://ocv.me/copyparty/helptext.html +* or see this: https://ocv.me/copyparty/helptext.html * or if you prefer plaintext, https://ocv.me/copyparty/helptext.txt diff --git a/scripts/prep.sh b/scripts/prep.sh index a4c8a132..d522371e 100755 --- a/scripts/prep.sh +++ b/scripts/prep.sh @@ -29,15 +29,11 @@ update_mpr_pkgbuild() { sha=$(sha256sum "$self/../dist/copyparty-$ver.tar.gz" | awk '{print$1}') - # awk -v ver=$ver -v sha=$sha ' - # /^pkgver=/{sub(/[0-9\.]+/,ver)}; - # /^sha256sums=/{sub(/[0-9a-f]{64}/,sha)}; - # 1' PKGBUILD >a - # mv a PKGBUILD - # TODO: check if this still works. if so, remove the following 2 lines: - - sed -s -i "s/pkgver=\"\"/pkgver=\"$ver\"/" PKGBUILD - sed -s -i "s/sha256sums=(\".*\")/sha256sums=(\"$sha\")/" PKGBUILD + awk -v ver=$ver -v sha=$sha ' + /^pkgver=/{sub(/[0-9\.]+/,ver)}; + /^sha256sums=/{sub(/[0-9a-f]{64}/,sha)}; + 1' PKGBUILD >a + mv a PKGBUILD rm -rf x } From 0df1901fc0376ff751a8c0db7d28251f8935ccf6 Mon Sep 17 00:00:00 2001 From: geekalaa Date: Mon, 4 Aug 2025 15:46:03 +0100 Subject: [PATCH 088/174] fix: add missing L. prefix for un_clip localization string Fixes ReferenceError when copying links from recently uploaded files. The un_clip localization string was missing the L. prefix in up2k.js. Fixes #467 --- copyparty/web/up2k.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/web/up2k.js b/copyparty/web/up2k.js index 2f588446..31ed1d6a 100644 --- a/copyparty/web/up2k.js +++ b/copyparty/web/up2k.js @@ -1595,7 +1595,7 @@ function up2k_init(subtle) { ev(e); var txt = linklist(); cliptxt(txt + '\n', function () { - toast.inf(5, un_clip.format(txt.split('\n').length)); + toast.inf(5, L.un_clip.format(txt.split('\n').length)); }); }; From c2ac57a2a8a48afcd916384102a2875f15c8b6c7 Mon Sep 17 00:00:00 2001 From: ed Date: Mon, 4 Aug 2025 22:56:30 +0000 Subject: [PATCH 089/174] improve helptext for multi-value options (#475) --- copyparty/__main__.py | 122 +++++++++++++++++++++--------------------- 1 file changed, 61 insertions(+), 61 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index d92089d0..3bb8c69e 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -993,18 +993,18 @@ def build_flags_desc(): def add_general(ap, nc, srvname): - ap2 = ap.add_argument_group('general options') - ap2.add_argument("-c", metavar="PATH", type=u, default=CFG_DEF, action="append", help="add config file") + ap2 = ap.add_argument_group("general options") + ap2.add_argument("-c", metavar="PATH", type=u, default=CFG_DEF, action="append", help="\033[34mREPEATABLE:\033[0m add config file") ap2.add_argument("-nc", metavar="NUM", type=int, default=nc, help="max num clients") ap2.add_argument("-j", metavar="CORES", type=int, default=1, help="max num cpu cores, 0=all") - ap2.add_argument("-a", metavar="ACCT", type=u, action="append", help="add account, \033[33mUSER\033[0m:\033[33mPASS\033[0m; example [\033[32med:wark\033[0m]") - ap2.add_argument("-v", metavar="VOL", type=u, action="append", help="add volume, \033[33mSRC\033[0m:\033[33mDST\033[0m:\033[33mFLAG\033[0m; examples [\033[32m.::r\033[0m], [\033[32m/mnt/nas/music:/music:r:aed\033[0m], see --help-accounts") - ap2.add_argument("--grp", metavar="G:N,N", type=u, action="append", help="add group, \033[33mNAME\033[0m:\033[33mUSER1\033[0m,\033[33mUSER2\033[0m,\033[33m...\033[0m; example [\033[32madmins:ed,foo,bar\033[0m]") + ap2.add_argument("-a", metavar="ACCT", type=u, action="append", help="\033[34mREPEATABLE:\033[0m add account, \033[33mUSER\033[0m:\033[33mPASS\033[0m; example [\033[32med:wark\033[0m]") + ap2.add_argument("-v", metavar="VOL", type=u, action="append", help="\033[34mREPEATABLE:\033[0m add volume, \033[33mSRC\033[0m:\033[33mDST\033[0m:\033[33mFLAG\033[0m; examples [\033[32m.::r\033[0m], [\033[32m/mnt/nas/music:/music:r:aed\033[0m], see --help-accounts") + ap2.add_argument("--grp", metavar="G:N,N", type=u, action="append", help="\033[34mREPEATABLE:\033[0m add group, \033[33mNAME\033[0m:\033[33mUSER1\033[0m,\033[33mUSER2\033[0m,\033[33m...\033[0m; example [\033[32madmins:ed,foo,bar\033[0m]") ap2.add_argument("-ed", action="store_true", help="enable the ?dots url parameter / client option which allows clients to see dotfiles / hidden files (volflag=dots)") ap2.add_argument("--urlform", metavar="MODE", type=u, default="print,xm", help="how to handle url-form POSTs; see \033[33m--help-urlform\033[0m") ap2.add_argument("--wintitle", metavar="TXT", type=u, default="cpp @ $pub", help="server terminal title, for example [\033[32m$ip-10.1.2.\033[0m] or [\033[32m$ip-]") ap2.add_argument("--name", metavar="TXT", type=u, default=srvname, help="server name (displayed topleft in browser and in mDNS)") - ap2.add_argument("--mime", metavar="EXT=MIME", type=u, action="append", help="map file \033[33mEXT\033[0mension to \033[33mMIME\033[0mtype, for example [\033[32mjpg=image/jpeg\033[0m]") + ap2.add_argument("--mime", metavar="EXT=MIME", type=u, action="append", help="\033[34mREPEATABLE:\033[0m map file \033[33mEXT\033[0mension to \033[33mMIME\033[0mtype, for example [\033[32mjpg=image/jpeg\033[0m]") ap2.add_argument("--mimes", action="store_true", help="list default mimetype mapping and exit") ap2.add_argument("--rmagic", action="store_true", help="do expensive analysis to improve accuracy of returned mimetypes; will make file-downloads, rss, and webdav slower (volflag=rmagic)") ap2.add_argument("--license", action="store_true", help="show licenses and exit") @@ -1012,7 +1012,7 @@ def add_general(ap, nc, srvname): def add_qr(ap, tty): - ap2 = ap.add_argument_group('qr options') + ap2 = ap.add_argument_group("qr options") ap2.add_argument("--qr", action="store_true", help="show http:// QR-code on startup") ap2.add_argument("--qrs", action="store_true", help="show https:// QR-code on startup") ap2.add_argument("--qrl", metavar="PATH", type=u, default="", help="location to include in the url, for example [\033[32mpriv/?pw=hunter2\033[0m]") @@ -1034,7 +1034,7 @@ def add_fs(ap): def add_share(ap): db_path = os.path.join(E.cfg, "shares.db") - ap2 = ap.add_argument_group('share-url options') + ap2 = ap.add_argument_group("share-url options") ap2.add_argument("--shr", metavar="DIR", type=u, default="", help="toplevel virtual folder for shared files/folders, for example [\033[32m/share\033[0m]") ap2.add_argument("--shr-db", metavar="FILE", type=u, default=db_path, help="database to store shares in") ap2.add_argument("--shr-adm", metavar="U,U", type=u, default="", help="comma-separated list of users allowed to view/delete any share") @@ -1043,7 +1043,7 @@ def add_share(ap): def add_upload(ap): - ap2 = ap.add_argument_group('upload options') + ap2 = ap.add_argument_group("upload options") ap2.add_argument("--dotpart", action="store_true", help="dotfile incomplete uploads, hiding them from clients unless \033[33m-ed\033[0m") ap2.add_argument("--plain-ip", action="store_true", help="when avoiding filename collisions by appending the uploader's ip to the filename: append the plaintext ip instead of salting and hashing the ip") ap2.add_argument("--put-name", metavar="TXT", type=u, default="put-{now.6f}-{cip}.bin", help="filename for nameless uploads (when uploader doesn't provide a name); default is [\033[32mput-UNIXTIME-IP.bin\033[0m] (the \033[32m.6f\033[0m means six decimal places) (volflag=put_name)") @@ -1085,14 +1085,14 @@ def add_upload(ap): def add_network(ap): - ap2 = ap.add_argument_group('network options') - ap2.add_argument("-i", metavar="IP", type=u, default="::", help="IPs and/or unix-sockets to listen on (see \033[33m--help-bind\033[0m). Default: all IPv4 and IPv6") + ap2 = ap.add_argument_group("network options") + ap2.add_argument("-i", metavar="IP", type=u, default="::", help="IPs and/or unix-sockets to listen on (comma-separated list; see \033[33m--help-bind\033[0m). Default: all IPv4 and IPv6") ap2.add_argument("-p", metavar="PORT", type=u, default="3923", help="ports to listen on (comma/range); ignored for unix-sockets") ap2.add_argument("--ll", action="store_true", help="include link-local IPv4/IPv6 in mDNS replies, even if the NIC has routable IPs (breaks some mDNS clients)") ap2.add_argument("--rproxy", metavar="DEPTH", type=int, default=1, help="which ip to associate clients with; [\033[32m0\033[0m]=tcp, [\033[32m1\033[0m]=origin (first x-fwd, unsafe), [\033[32m2\033[0m]=outermost-proxy, [\033[32m3\033[0m]=second-proxy, [\033[32m-1\033[0m]=closest-proxy") ap2.add_argument("--xff-hdr", metavar="NAME", type=u, default="x-forwarded-for", help="if reverse-proxied, which http header to read the client's real ip from") - ap2.add_argument("--xff-src", metavar="CIDR", type=u, default="127.0.0.0/8, ::1/128", help="comma-separated list of trusted reverse-proxy CIDRs; only accept the real-ip header (\033[33m--xff-hdr\033[0m) and IdP headers if the incoming connection is from an IP within either of these subnets. Specify [\033[32mlan\033[0m] to allow all LAN / private / non-internet IPs. Can be disabled with [\033[32many\033[0m] if you are behind cloudflare (or similar) and are using \033[32m--xff-hdr=cf-connecting-ip\033[0m (or similar)") - ap2.add_argument("--ipa", metavar="CIDR", type=u, default="", help="only accept connections from IP-addresses inside \033[33mCIDR\033[0m; examples: [\033[32mlan\033[0m] or [\033[32m10.89.0.0/16, 192.168.33.0/24\033[0m]") + ap2.add_argument("--xff-src", metavar="CIDR", type=u, default="127.0.0.0/8, ::1/128", help="list of trusted reverse-proxy CIDRs (comma-separated); only accept the real-ip header (\033[33m--xff-hdr\033[0m) and IdP headers if the incoming connection is from an IP within either of these subnets. Specify [\033[32mlan\033[0m] to allow all LAN / private / non-internet IPs. Can be disabled with [\033[32many\033[0m] if you are behind cloudflare (or similar) and are using \033[32m--xff-hdr=cf-connecting-ip\033[0m (or similar)") + ap2.add_argument("--ipa", metavar="CIDR", type=u, default="", help="only accept connections from IP-addresses inside \033[33mCIDR\033[0m (comma-separated); examples: [\033[32mlan\033[0m] or [\033[32m10.89.0.0/16, 192.168.33.0/24\033[0m]") ap2.add_argument("--rp-loc", metavar="PATH", type=u, default="", help="if reverse-proxying on a location instead of a dedicated domain/subdomain, provide the base location here; example: [\033[32m/foo/bar\033[0m]") if ANYWIN: ap2.add_argument("--reuseaddr", action="store_true", help="set reuseaddr on listening sockets on windows; allows rapid restart of copyparty at the expense of being able to accidentally start multiple instances") @@ -1110,7 +1110,7 @@ def add_network(ap): def add_tls(ap, cert_path): - 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 -- force plaintext") ap2.add_argument("--https-only", action="store_true", help="disable plaintext -- force tls") ap2.add_argument("--cert", metavar="PATH", type=u, default=cert_path, help="path to file containing a concatenation of TLS key and certificate chain") @@ -1122,7 +1122,7 @@ def add_tls(ap, cert_path): def add_cert(ap, cert_path): cert_dir = os.path.dirname(cert_path) - ap2 = ap.add_argument_group('TLS certificate generator options') + ap2 = ap.add_argument_group("TLS certificate generator options") ap2.add_argument("--no-crt", action="store_true", help="disable automatic certificate creation") ap2.add_argument("--crt-ns", metavar="N,N", type=u, default="", help="comma-separated list of FQDNs (domains) to add into the certificate") ap2.add_argument("--crt-exact", action="store_true", help="do not add wildcard entries for each \033[33m--crt-ns\033[0m") @@ -1142,7 +1142,7 @@ def add_cert(ap, cert_path): def add_auth(ap): idp_db = os.path.join(E.cfg, "idp.db") ses_db = os.path.join(E.cfg, "sessions.db") - ap2 = ap.add_argument_group('IdP / identity provider / user authentication options') + ap2 = ap.add_argument_group("IdP / identity provider / user authentication options") ap2.add_argument("--idp-h-usr", metavar="HN", type=u, default="", help="bypass the copyparty authentication checks if the request-header \033[33mHN\033[0m contains a username to associate the request with (for use with authentik/oauth/...)\n\033[1;31mWARNING:\033[0m if you enable this, make sure clients are unable to specify this header themselves; must be washed away and replaced by a reverse-proxy") ap2.add_argument("--idp-h-grp", metavar="HN", type=u, default="", help="assume the request-header \033[33mHN\033[0m contains the groupname of the requesting user; can be referenced in config files for group-based access control") ap2.add_argument("--idp-h-key", metavar="HN", type=u, default="", help="optional but recommended safeguard; your reverse-proxy will insert a secret header named \033[33mHN\033[0m into all requests, and the other IdP headers will be ignored if this header is not present") @@ -1156,14 +1156,14 @@ def add_auth(ap): ap2.add_argument("--ses-db", metavar="PATH", type=u, default=ses_db, help="where to store the sessions database (if you run multiple copyparty instances, make sure they use different DBs)") ap2.add_argument("--ses-len", metavar="CHARS", type=int, default=20, help="session key length; default is 120 bits ((20//4)*4*6)") ap2.add_argument("--no-ses", action="store_true", help="disable sessions; use plaintext passwords in cookies") - ap2.add_argument("--ipu", metavar="CIDR=USR", type=u, action="append", help="users with IP matching \033[33mCIDR\033[0m are auto-authenticated as username \033[33mUSR\033[0m; example: [\033[32m172.16.24.0/24=dave]") + ap2.add_argument("--ipu", metavar="CIDR=USR", type=u, action="append", help="\033[34mREPEATABLE:\033[0m users with IP matching \033[33mCIDR\033[0m are auto-authenticated as username \033[33mUSR\033[0m; example: [\033[32m172.16.24.0/24=dave]") def add_chpw(ap): db_path = os.path.join(E.cfg, "chpw.json") - ap2 = ap.add_argument_group('user-changeable passwords options') + ap2 = ap.add_argument_group("user-changeable passwords options") ap2.add_argument("--chpw", action="store_true", help="allow users to change their own passwords") - ap2.add_argument("--chpw-no", metavar="U,U,U", type=u, action="append", help="do not allow password-changes for this comma-separated list of usernames") + ap2.add_argument("--chpw-no", metavar="U,U,U", type=u, action="append", help="\033[34mREPEATABLE:\033[0m do not allow password-changes for this comma-separated list of usernames") ap2.add_argument("--chpw-db", metavar="PATH", type=u, default=db_path, help="where to store the passwords database (if you run multiple copyparty instances, make sure they use different DBs)") ap2.add_argument("--chpw-len", metavar="N", type=int, default=8, help="minimum password length") ap2.add_argument("--chpw-v", metavar="LVL", type=int, default=2, help="verbosity of summary on config load [\033[32m0\033[0m] = nothing at all, [\033[32m1\033[0m] = number of users, [\033[32m2\033[0m] = list users with default-pw, [\033[32m3\033[0m] = list all users") @@ -1212,12 +1212,12 @@ def add_zc_ssdp(ap): def add_ftp(ap): - ap2 = ap.add_argument_group('FTP options (TCP only)') + ap2 = ap.add_argument_group("FTP options (TCP only)") ap2.add_argument("--ftp", metavar="PORT", type=int, default=0, help="enable FTP server on \033[33mPORT\033[0m, for example \033[32m3921") ap2.add_argument("--ftps", metavar="PORT", type=int, default=0, help="enable FTPS server on \033[33mPORT\033[0m, for example \033[32m3990") ap2.add_argument("--ftpv", action="store_true", help="verbose") ap2.add_argument("--ftp4", action="store_true", help="only listen on IPv4") - ap2.add_argument("--ftp-ipa", metavar="CIDR", type=u, default="", help="only accept connections from IP-addresses inside \033[33mCIDR\033[0m; specify [\033[32many\033[0m] to disable inheriting \033[33m--ipa\033[0m. Examples: [\033[32mlan\033[0m] or [\033[32m10.89.0.0/16, 192.168.33.0/24\033[0m]") + ap2.add_argument("--ftp-ipa", metavar="CIDR", type=u, default="", help="only accept connections from IP-addresses inside \033[33mCIDR\033[0m (comma-separated); specify [\033[32many\033[0m] to disable inheriting \033[33m--ipa\033[0m. Examples: [\033[32mlan\033[0m] or [\033[32m10.89.0.0/16, 192.168.33.0/24\033[0m]") ap2.add_argument("--ftp-no-ow", action="store_true", help="if target file exists, reject upload instead of overwrite") ap2.add_argument("--ftp-wt", metavar="SEC", type=int, default=7, help="grace period for resuming interrupted uploads (any client can write to any file last-modified more recently than \033[33mSEC\033[0m seconds ago)") ap2.add_argument("--ftp-nat", metavar="ADDR", type=u, default="", help="the NAT address to use for passive connections") @@ -1225,7 +1225,7 @@ def add_ftp(ap): def add_webdav(ap): - ap2 = ap.add_argument_group('WebDAV options') + ap2 = ap.add_argument_group("WebDAV options") ap2.add_argument("--daw", action="store_true", help="enable full write support, even if client may not be webdav. \033[1;31mWARNING:\033[0m This has side-effects -- PUT-operations will now \033[1;31mOVERWRITE\033[0m existing files, rather than inventing new filenames to avoid loss of data. You might want to instead set this as a volflag where needed. By not setting this flag, uploaded files can get written to a filename which the client does not expect (which might be okay, depending on client)") ap2.add_argument("--dav-inf", action="store_true", help="allow depth:infinite requests (recursive file listing); extremely server-heavy but required for spec compliance -- luckily few clients rely on this") ap2.add_argument("--dav-mac", action="store_true", help="disable apple-garbage filter -- allow macos to create junk files (._* and .DS_Store, .Spotlight-*, .fseventsd, .Trashes, .AppleDouble, __MACOS)") @@ -1235,7 +1235,7 @@ def add_webdav(ap): def add_tftp(ap): - ap2 = ap.add_argument_group('TFTP options (UDP only)') + ap2 = ap.add_argument_group("TFTP options (UDP only)") ap2.add_argument("--tftp", metavar="PORT", type=int, default=0, help="enable TFTP server on \033[33mPORT\033[0m, for example \033[32m69 \033[0mor \033[32m3969") ap2.add_argument("--tftp4", action="store_true", help="only listen on IPv4") ap2.add_argument("--tftpv", action="store_true", help="verbose") @@ -1243,12 +1243,12 @@ def add_tftp(ap): ap2.add_argument("--tftp-no-fast", action="store_true", help="debug: disable optimizations") ap2.add_argument("--tftp-lsf", metavar="PTN", type=u, default="\\.?(dir|ls)(\\.txt)?", help="return a directory listing if a file with this name is requested and it does not exist; defaults matches .ls, dir, .dir.txt, ls.txt, ...") ap2.add_argument("--tftp-nols", action="store_true", help="if someone tries to download a directory, return an error instead of showing its directory listing") - ap2.add_argument("--tftp-ipa", metavar="CIDR", type=u, default="", help="only accept connections from IP-addresses inside \033[33mCIDR\033[0m; specify [\033[32many\033[0m] to disable inheriting \033[33m--ipa\033[0m. Examples: [\033[32mlan\033[0m] or [\033[32m10.89.0.0/16, 192.168.33.0/24\033[0m]") + ap2.add_argument("--tftp-ipa", metavar="CIDR", type=u, default="", help="only accept connections from IP-addresses inside \033[33mCIDR\033[0m (comma-separated); specify [\033[32many\033[0m] to disable inheriting \033[33m--ipa\033[0m. Examples: [\033[32mlan\033[0m] or [\033[32m10.89.0.0/16, 192.168.33.0/24\033[0m]") ap2.add_argument("--tftp-pr", metavar="P-P", type=u, default="", help="the range of UDP ports to use for data transfer, for example \033[32m12000-13000") def add_smb(ap): - ap2 = ap.add_argument_group('SMB/CIFS options') + ap2 = ap.add_argument_group("SMB/CIFS options") ap2.add_argument("--smb", action="store_true", help="enable smb (read-only) -- this requires running copyparty as root on linux and macos unless \033[33m--smb-port\033[0m is set above 1024 and your OS does port-forwarding from 445 to that.\n\033[1;31mWARNING:\033[0m this protocol is DANGEROUS and buggy! Never expose to the internet!") ap2.add_argument("--smbw", action="store_true", help="enable write support (please dont)") ap2.add_argument("--smb1", action="store_true", help="disable SMBv2, only enable SMBv1 (CIFS)") @@ -1262,30 +1262,30 @@ def add_smb(ap): def add_handlers(ap): - ap2 = ap.add_argument_group('handlers (see --help-handlers)') - ap2.add_argument("--on404", metavar="PY", type=u, action="append", help="handle 404s by executing \033[33mPY\033[0m file") - ap2.add_argument("--on403", metavar="PY", type=u, action="append", help="handle 403s by executing \033[33mPY\033[0m file") + ap2 = ap.add_argument_group("handlers (see --help-handlers)") + ap2.add_argument("--on404", metavar="PY", type=u, action="append", help="\033[34mREPEATABLE:\033[0m handle 404s by executing \033[33mPY\033[0m file") + ap2.add_argument("--on403", metavar="PY", type=u, action="append", help="\033[34mREPEATABLE:\033[0m handle 403s by executing \033[33mPY\033[0m file") ap2.add_argument("--hot-handlers", action="store_true", help="recompile handlers on each request -- expensive but convenient when hacking on stuff") def add_hooks(ap): - ap2 = ap.add_argument_group('event hooks (see --help-hooks)') - ap2.add_argument("--xbu", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m before a file upload starts") - ap2.add_argument("--xau", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m after a file upload finishes") - ap2.add_argument("--xiu", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m after all uploads finish and volume is idle") - ap2.add_argument("--xbc", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m before a file copy") - ap2.add_argument("--xac", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m after a file copy") - ap2.add_argument("--xbr", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m before a file move/rename") - ap2.add_argument("--xar", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m after a file move/rename") - ap2.add_argument("--xbd", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m before a file delete") - ap2.add_argument("--xad", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m after a file delete") - ap2.add_argument("--xm", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m on message") - ap2.add_argument("--xban", metavar="CMD", type=u, action="append", help="execute \033[33mCMD\033[0m if someone gets banned (pw/404/403/url)") + ap2 = ap.add_argument_group("event hooks (see --help-hooks)") + ap2.add_argument("--xbu", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m before a file upload starts") + ap2.add_argument("--xau", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m after a file upload finishes") + ap2.add_argument("--xiu", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m after all uploads finish and volume is idle") + ap2.add_argument("--xbc", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m before a file copy") + ap2.add_argument("--xac", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m after a file copy") + ap2.add_argument("--xbr", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m before a file move/rename") + ap2.add_argument("--xar", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m after a file move/rename") + ap2.add_argument("--xbd", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m before a file delete") + ap2.add_argument("--xad", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m after a file delete") + ap2.add_argument("--xm", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m on message") + ap2.add_argument("--xban", metavar="CMD", type=u, action="append", help="\033[34mREPEATABLE:\033[0m execute \033[33mCMD\033[0m if someone gets banned (pw/404/403/url)") ap2.add_argument("--hook-v", action="store_true", help="verbose hooks") def add_stats(ap): - ap2 = ap.add_argument_group('grafana/prometheus metrics endpoint') + ap2 = ap.add_argument_group("grafana/prometheus metrics endpoint") ap2.add_argument("--stats", action="store_true", help="enable openmetrics at /.cpr/metrics for admin accounts") ap2.add_argument("--nos-hdd", action="store_true", help="disable disk-space metrics (used/free space)") ap2.add_argument("--nos-vol", action="store_true", help="disable volume size metrics (num files, total bytes, vmaxb/vmaxn)") @@ -1295,7 +1295,7 @@ def add_stats(ap): def add_yolo(ap): - ap2 = ap.add_argument_group('yolo options') + ap2 = ap.add_argument_group("yolo options") ap2.add_argument("--allow-csrf", action="store_true", help="disable csrf protections; let other domains/sites impersonate you through cross-site requests") ap2.add_argument("--cookie-lax", action="store_true", help="allow cookies from other domains (if you follow a link from another website into your server, you will arrive logged-in); this reduces protection against CSRF") ap2.add_argument("--getmod", action="store_true", help="permit ?move=[...] and ?delete as GET") @@ -1303,7 +1303,7 @@ def add_yolo(ap): def add_optouts(ap): - ap2 = ap.add_argument_group('opt-outs') + ap2 = ap.add_argument_group("opt-outs") ap2.add_argument("-nw", action="store_true", help="never write anything to disk (debug/benchmark)") ap2.add_argument("--keep-qem", action="store_true", help="do not disable quick-edit-mode on windows (it is disabled to avoid accidental text selection in the terminal window, as this would pause execution)") ap2.add_argument("--no-dav", action="store_true", help="disable webdav support") @@ -1329,7 +1329,7 @@ def add_optouts(ap): def add_safety(ap): - ap2 = ap.add_argument_group('safety options') + ap2 = ap.add_argument_group("safety options") ap2.add_argument("-s", action="count", default=0, help="increase safety: Disable thumbnails / potentially dangerous software (ffmpeg/pillow/vips), hide partial uploads, avoid crawlers.\n └─Alias of\033[32m --dotpart --no-thumb --no-mtag-ff --no-robots --force-js") ap2.add_argument("-ss", action="store_true", help="further increase safety: Prevent js-injection, accidental move/delete, broken symlinks, webdav, 404 on 403, ban on excessive 404s.\n └─Alias of\033[32m -s --unpost=0 --no-del --no-mv --hardlink --vague-403 -nih") ap2.add_argument("-sss", action="store_true", help="further increase safety: Enable logging to disk, scan for dangerous symlinks.\n └─Alias of\033[32m -ss --no-dav --no-logues --no-readme -lo=cpp-%%Y-%%m%%d-%%H%%M%%S.txt.xz --ls=**,*,ln,p,r") @@ -1360,7 +1360,7 @@ def add_safety(ap): def add_salt(ap, fk_salt, dk_salt, ah_salt): - ap2 = ap.add_argument_group('salting options') + ap2 = ap.add_argument_group("salting options") ap2.add_argument("--ah-alg", metavar="ALG", type=u, default="none", help="account-pw hashing algorithm; one of these, best to worst: \033[32margon2 scrypt sha2 none\033[0m (each optionally followed by alg-specific comma-sep. config)") ap2.add_argument("--ah-salt", metavar="SALT", type=u, default=ah_salt, help="account-pw salt; ignored if \033[33m--ah-alg\033[0m is none (default)") ap2.add_argument("--ah-gen", metavar="PW", type=u, default="", help="generate hashed password for \033[33mPW\033[0m, or read passwords from STDIN if \033[33mPW\033[0m is [\033[32m-\033[0m]") @@ -1374,14 +1374,14 @@ def add_salt(ap, fk_salt, dk_salt, ah_salt): def add_shutdown(ap): - ap2 = ap.add_argument_group('shutdown options') + ap2 = ap.add_argument_group("shutdown options") ap2.add_argument("--ign-ebind", action="store_true", help="continue running even if it's impossible to listen on some of the requested endpoints") ap2.add_argument("--ign-ebind-all", action="store_true", help="continue running even if it's impossible to receive connections at all") ap2.add_argument("--exit", metavar="WHEN", type=u, default="", help="shutdown after \033[33mWHEN\033[0m has finished; [\033[32mcfg\033[0m] config parsing, [\033[32midx\033[0m] volscan + multimedia indexing") def add_logging(ap): - ap2 = ap.add_argument_group('logging options') + ap2 = ap.add_argument_group("logging options") ap2.add_argument("-q", action="store_true", help="quiet; disable most STDOUT messages") ap2.add_argument("-lo", metavar="PATH", type=u, default="", help="logfile, example: \033[32mcpp-%%Y-%%m%%d-%%H%%M%%S.txt.xz\033[0m (NB: some errors may appear on STDOUT only)") ap2.add_argument("--no-ansi", action="store_true", default=not VT100, help="disable colors; same as environment-variable NO_COLOR") @@ -1399,7 +1399,7 @@ def add_logging(ap): def add_admin(ap): - ap2 = ap.add_argument_group('admin panel options') + ap2 = ap.add_argument_group("admin panel options") ap2.add_argument("--no-reload", action="store_true", help="disable ?reload=cfg (reload users/volumes/volflags from config file)") ap2.add_argument("--no-rescan", action="store_true", help="disable ?scan (volume reindexing)") ap2.add_argument("--no-stack", action="store_true", help="disable ?stack (list all stacks)") @@ -1413,7 +1413,7 @@ def add_admin(ap): def add_thumbnail(ap): th_ram = (RAM_AVAIL or RAM_TOTAL or 9) * 0.6 th_ram = int(max(min(th_ram, 6), 0.3) * 10) / 10 - ap2 = ap.add_argument_group('thumbnail options') + ap2 = ap.add_argument_group("thumbnail options") ap2.add_argument("--no-thumb", action="store_true", help="disable all thumbnails (volflag=dthumb)") ap2.add_argument("--no-vthumb", action="store_true", help="disable video thumbnails (volflag=dvthumb)") ap2.add_argument("--no-athumb", action="store_true", help="disable audio thumbnails (spectrograms) (volflag=dathumb)") @@ -1445,7 +1445,7 @@ def add_thumbnail(ap): def add_transcoding(ap): - ap2 = ap.add_argument_group('transcoding options') + ap2 = ap.add_argument_group("transcoding options") ap2.add_argument("--q-opus", metavar="KBPS", type=int, default=128, help="target bitrate for transcoding to opus; set 0 to disable") ap2.add_argument("--q-mp3", metavar="QUALITY", type=u, default="q2", help="target quality for transcoding to mp3, for example [\033[32m192k\033[0m] (CBR) or [\033[32mq0\033[0m] (CQ/CRF, q0=maxquality, q9=smallest); set 0 to disable") ap2.add_argument("--allow-wav", action="store_true", help="allow transcoding to wav (lossless, uncompressed)") @@ -1458,7 +1458,7 @@ def add_transcoding(ap): def add_tail(ap): - ap2 = ap.add_argument_group('tailing options (realtime streaming of a growing file)') + ap2 = ap.add_argument_group("tailing options (realtime streaming of a growing file)") ap2.add_argument("--tail-who", metavar="LVL", type=int, default=2, help="who can tail? [\033[32m0\033[0m]=nobody, [\033[32m1\033[0m]=admins, [\033[32m2\033[0m]=authenticated-with-read-access, [\033[32m3\033[0m]=everyone-with-read-access (volflag=tail_who)") ap2.add_argument("--tail-cmax", metavar="N", type=int, default=64, help="do not allow starting a new tail if more than \033[33mN\033[0m active downloads") ap2.add_argument("--tail-tmax", metavar="SEC", type=float, default=0, help="terminate connection after \033[33mSEC\033[0m seconds; [\033[32m0\033[0m]=never (volflag=tail_tmax)") @@ -1468,7 +1468,7 @@ def add_tail(ap): def add_rss(ap): - ap2 = ap.add_argument_group('RSS options') + ap2 = ap.add_argument_group("RSS options") ap2.add_argument("--rss", action="store_true", help="enable RSS output (experimental) (volflag=rss)") ap2.add_argument("--rss-nf", metavar="HITS", type=int, default=250, help="default number of files to return (url-param 'nf')") ap2.add_argument("--rss-fext", metavar="E,E", type=u, default="", help="default list of file extensions to include (url-param 'fext'); blank=all") @@ -1477,7 +1477,7 @@ def add_rss(ap): def add_db_general(ap, hcores): noidx = APPLESAN_TXT if MACOS else "" - ap2 = ap.add_argument_group('general db options') + ap2 = ap.add_argument_group("general db options") ap2.add_argument("-e2d", action="store_true", help="enable up2k database; this enables file search, upload-undo, improves deduplication") ap2.add_argument("-e2ds", action="store_true", help="scan writable folders for new files on startup; sets \033[33m-e2d\033[0m") ap2.add_argument("-e2dsa", action="store_true", help="scans all folders on startup; sets \033[33m-e2ds\033[0m") @@ -1506,7 +1506,7 @@ def add_db_general(ap, hcores): def add_db_metadata(ap): - ap2 = ap.add_argument_group('metadata db options') + ap2 = ap.add_argument_group("metadata db options") ap2.add_argument("-e2t", action="store_true", help="enable metadata indexing; makes it possible to search for artist/title/codec/resolution/...") ap2.add_argument("-e2ts", action="store_true", help="scan newly discovered files for metadata on startup; sets \033[33m-e2t\033[0m") ap2.add_argument("-e2tsr", action="store_true", help="delete all metadata from DB and do a full rescan; sets \033[33m-e2ts\033[0m") @@ -1516,14 +1516,14 @@ def add_db_metadata(ap): ap2.add_argument("--mtag-mt", metavar="CORES", type=int, default=CORES, help="num cpu cores to use for tag scanning") ap2.add_argument("--mtag-v", action="store_true", help="verbose tag scanning; print errors from mtp subprocesses and such") ap2.add_argument("--mtag-vv", action="store_true", help="debug mtp settings and mutagen/FFprobe parsers") - ap2.add_argument("-mtm", metavar="M=t,t,t", type=u, action="append", help="add/replace metadata mapping") + ap2.add_argument("-mtm", metavar="M=t,t,t", type=u, action="append", help="\033[34mREPEATABLE:\033[0m add/replace metadata mapping") ap2.add_argument("-mte", metavar="M,M,M", type=u, help="tags to index/display (comma-sep.); either an entire replacement list, or add/remove stuff on the default-list with +foo or /bar", default=DEF_MTE) ap2.add_argument("-mth", metavar="M,M,M", type=u, help="tags to hide by default (comma-sep.); assign/add/remove same as \033[33m-mte\033[0m", default=DEF_MTH) - ap2.add_argument("-mtp", metavar="M=[f,]BIN", type=u, action="append", help="read tag \033[33mM\033[0m using program \033[33mBIN\033[0m to parse the file") + ap2.add_argument("-mtp", metavar="M=[f,]BIN", type=u, action="append", help="\033[34mREPEATABLE:\033[0m read tag \033[33mM\033[0m using program \033[33mBIN\033[0m to parse the file") def add_txt(ap): - ap2 = ap.add_argument_group('textfile options') + ap2 = ap.add_argument_group("textfile options") ap2.add_argument("--md-hist", metavar="TXT", type=u, default="s", help="where to store old version of markdown files; [\033[32ms\033[0m]=subfolder, [\033[32mv\033[0m]=volume-histpath, [\033[32mn\033[0m]=nope/disabled (volflag=md_hist)") ap2.add_argument("-mcr", metavar="SEC", type=int, default=60, help="the textfile editor will check for serverside changes every \033[33mSEC\033[0m seconds") ap2.add_argument("-emp", action="store_true", help="enable markdown plugins -- neat but dangerous, big XSS risk") @@ -1534,7 +1534,7 @@ def add_txt(ap): def add_og(ap): - ap2 = ap.add_argument_group('og / open graph / discord-embed options') + ap2 = ap.add_argument_group("og / open graph / discord-embed options") ap2.add_argument("--og", action="store_true", help="disable hotlinking and return an html document instead; this is required by open-graph, but can also be useful on its own (volflag=og)") ap2.add_argument("--og-ua", metavar="RE", type=u, default="", help="only disable hotlinking / engage OG behavior if the useragent matches regex \033[33mRE\033[0m (volflag=og_ua)") ap2.add_argument("--og-tpl", metavar="PATH", type=u, default="", help="do not return the regular copyparty html, but instead load the jinja2 template at \033[33mPATH\033[0m (if path contains 'EXT' then EXT will be replaced with the requested file's extension) (volflag=og_tpl)") @@ -1552,7 +1552,7 @@ def add_og(ap): def add_ui(ap, retry): - ap2 = ap.add_argument_group('ui options') + ap2 = ap.add_argument_group("ui options") ap2.add_argument("--grid", action="store_true", help="show grid/thumbnails by default (volflag=grid)") ap2.add_argument("--gsel", action="store_true", help="select files in grid by ctrl-click (volflag=gsel)") ap2.add_argument("--localtime", action="store_true", help="default to local timezone instead of UTC") @@ -1567,7 +1567,7 @@ def add_ui(ap, retry): ap2.add_argument("--qdel", metavar="LVL", type=int, default=2, help="number of confirmations to show when deleting files (2/1/0)") ap2.add_argument("--unlist", metavar="REGEX", type=u, default="", help="don't show files/folders matching \033[33mREGEX\033[0m in file list. WARNING: Purely cosmetic! Does not affect API calls, just the browser. Example: [\033[32m\\.(js|css)$\033[0m] (volflag=unlist)") ap2.add_argument("--favico", metavar="TXT", type=u, default="c 000 none" if retry else "🎉 000 none", help="\033[33mfavicon-text\033[0m [ \033[33mforeground\033[0m [ \033[33mbackground\033[0m ] ], set blank to disable") - ap2.add_argument("--ext-th", metavar="E=VP", type=u, action="append", help="use thumbnail-image \033[33mVP\033[0m for file-extension \033[33mE\033[0m, example: [\033[32mexe=/.res/exe.png\033[0m] (volflag=ext_th)") + ap2.add_argument("--ext-th", metavar="E=VP", type=u, action="append", help="\033[34mREPEATABLE:\033[0m use thumbnail-image \033[33mVP\033[0m for file-extension \033[33mE\033[0m, example: [\033[32mexe=/.res/exe.png\033[0m] (volflag=ext_th)") ap2.add_argument("--mpmc", type=u, default="", help=argparse.SUPPRESS) ap2.add_argument("--spinner", metavar="TXT", type=u, default="🌲", help="\033[33memoji\033[0m or \033[33memoji,css\033[0m Example: [\033[32m🥖,padding:0\033[0m]") ap2.add_argument("--css-browser", metavar="L", type=u, default="", help="URL to additional CSS to include in the filebrowser html") @@ -1593,7 +1593,7 @@ def add_ui(ap, retry): def add_debug(ap): - ap2 = ap.add_argument_group('debug options') + ap2 = ap.add_argument_group("debug options") ap2.add_argument("--vc", action="store_true", help="verbose config file parser (explain config)") ap2.add_argument("--cgen", action="store_true", help="generate config file from current config (best-effort; probably buggy)") ap2.add_argument("--deps", action="store_true", help="list information about detected optional dependencies") From b664ebb01f28094749e233ac70795ccd11fed253 Mon Sep 17 00:00:00 2001 From: ed Date: Mon, 4 Aug 2025 22:56:37 +0000 Subject: [PATCH 090/174] add chungus.conf (#475) --- docs/chungus.conf | 2138 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 2138 insertions(+) create mode 100644 docs/chungus.conf diff --git a/docs/chungus.conf b/docs/chungus.conf new file mode 100644 index 00000000..7ac7ba9a --- /dev/null +++ b/docs/chungus.conf @@ -0,0 +1,2138 @@ +# not actually YAML but lets pretend: +# -*- mode: yaml -*- +# vim: ft=yaml: + + +## this config-file does not make any sense at all, and will not work anywhere +## +## it is a nearly-complete example of every config option and how each can be used, +## but will NOT be maintained, it's just to get a feel for how the config works +## +## this file was initially generated by and based on: +## cat copyparty/__main__.py | awk -F\" -vp1=$(printf %090d 0) '/arse.SUPPRE/{next}/add_argument_group/{printf"\n ###%s\\\n ###// %s \\\\%s\n",p1,$2,p1};{m=""}/metavar=/{m=": "$4}/add_argument\(/{h=$0;sub(/.*, help="/,"",h);sub(/"\)$/,"",h);k=$2;sub(/^-+/,"",k);printf"\n # %s\n %s%s\n",h,k,m}' | sed -r 's/^( ###\/\/ .{88})0+/\1\\/;s/([^#]{40}) +/\1/;s/\\033\[[^m]*[0-9]m//g;s/ \(volflag=([^)]+)\)$/\n # 📂 also available as volflag "\1"/' | xsel -ib +## grep -A9001 ^flagcats copyparty/cfg.py | grep -B9001 ^flagdescs | awk -F\" -vp1=$(printf %090d 0) '/^ "/{printf"\n ###%s\\\n ###// %s \\\\%s\n",p1,$2,p1};/^ [^ ]/{k=$0;h=$0;sub(/^ +./,"",k);sub(/.: .*/,"",k);sub(/[^:]+: ./,"",h);sub(/.,$/,"",h);sub(/=/,": ",k);printf"\n # %s\n %s\n",h,k}' | sed -r 's/^( ###\/\/ .{88})0+/\1\\/;s/([^#]{40}) +/\1/' | xsel -ib + + +[global] + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// general options \\0000000000000000000000000000000000000000000000000000000000000000000000\ + + # REPEATABLE: add a config file; add multiple by repeating the option + c: /etc/some.conf + c: /etc/another.conf + + # max num clients; will not accept more http/https connections at this point + nc: 1024 + + # max num cpu cores, 0=all, 1=default=recommended + j: 4 + + # enable the ?dots url parameter / client option which allows clients to see dotfiles / hidden files + # 📂 also available as volflag "dots" + ed + + # how to handle url-form POSTs; see --help-urlform + urlform: save,get + + # server terminal title, for example [$ip-10.1.2.] or [$ip-] + wintitle: $ip-10.1.2. + + # server name (displayed topleft in browser and in mDNS) + name: mogra + + # REPEATABLE: map file EXTension to MIMEtype, for example [jpg=image/jpeg] + mime: qoi=image/x-qoi + mime: adf=application/x-amiga-disk-format + + # list default mimetype mapping and exit + mimes + + # do expensive analysis to improve accuracy of returned mimetypes; will make file-downloads, rss, and webdav slower + # 📂 also available as volflag "rmagic" + rmagic + + # show licenses and exit + license + + # show versions and exit + version + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// qr options \\000000000000000000000000000000000000000000000000000000000000000000000000000\ + + # show http:// QR-code on startup + qr + + # show https:// QR-code on startup + qrs + + # location to include in the url, for example [priv/?pw=hunter2] + qrl: uploads/?pw=okletsgo # hint + + # select IP which starts with PREFIX; [.] to force default IP when mDNS URL would have been used instead + qri: 192.168. # hint + + # foreground; try [0] if the qr-code is unreadable + qr-fg: 46 # hint; default=0=black + + # background (white=255) + qr-bg: 92 # hint; default=229=parchment + + # padding (spec says 4 or more, but 1 is usually fine) + qrp: 2 # hint + + # [1]=1x, [2]=2x, [0]=auto (try [2] on broken fonts) + qrz: 2 # hint + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// filesystem options \\0000000000000000000000000000000000000000000000000000000000000000000\ + + # if a file cannot be deleted because it is busy, continue trying for T seconds, retry every R seconds; disable with 0/0 + # 📂 also available as volflag "rm_retry" + rm-retry: 15/0.1 # default on windows; 0 on unix + + # if a file cannot be renamed because it is busy, continue trying for T seconds, retry every R seconds; disable with 0/0 + # 📂 also available as volflag "mv_retry" + mv-retry: 15/0.1 # default on windows; 0 on unix + + # file I/O buffer-size; if your volumes are on a network drive, try increasing to 524288 or even 4194304 (and let me know if that improves your performance) + iobuf: 262144 # default + + # rebuild mountpoint cache every SEC to keep track of sparse-files support; keep low on servers with removable media + mtab-age: 60 # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// share-url options \\00000000000000000000000000000000000000000000000000000000000000000000\ + + # toplevel virtual folder for shared files/folders, for example [/share] + shr: /shares # hint; default is unset + + # database to store shares in + shr-db: $XDG_CONFIG_PATH/.copyparty/shares.db # default (and yes, $ENV will expand in configs) + + # comma-separated list of users allowed to view/delete any share + shr-adm: ame,same # hint + + # shares can be revived by their owner if they expired less than MIN minutes ago; [60]=hour, [1440]=day, [10080]=week + shr-rt: 1440 # default + + # debug + shr-v + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// upload options \\00000000000000000000000000000000000000000000000000000000000000000000000\ + + # dotfile incomplete uploads, hiding them from clients unless -ed + dotpart + + # when avoiding filename collisions by appending the uploader's ip to the filename: append the plaintext ip instead of salting and hashing the ip + plain-ip + + # filename for nameless uploads (when uploader doesn't provide a name); default is [put-UNIXTIME-IP.bin] (the .6f means six decimal places) + # 📂 also available as volflag "put_name" + put-name: put-{now.6f}-{cip}.bin # default + + # default checksum-hasher for PUT/WebDAV uploads: no / md5 / sha1 / sha256 / sha512 / b2 / blake2 / b2s / blake2s + # 📂 also available as volflag "put_ck" + put-ck: sha512 # default + + # default checksum-hasher for bup/basic-uploader: no / md5 / sha1 / sha256 / sha512 / b2 / blake2 / b2s / blake2s + # 📂 also available as volflag "bup_ck" + bup-ck: sha512 # default + + # grace period where uploads can be deleted by the uploader, even without delete permissions; 0=disabled, default=12h + unpost: 43200 # default (12h) + + # clients can abort incomplete uploads by using the unpost tab (requires -e2d). [0] = never allowed (disable feature), [1] = allow if client has the same IP as the upload AND is using the same account, [2] = just check the IP, [3] = just check account-name + # 📂 also available as volflag "u2abort" + u2abort: 1 # default + + # file write grace period (any client can write to a blank file last-modified more recently than SEC seconds ago) + blank-wt: 300 # default + + # max number of uploads to keep in memory when running without -e2d; roughly 1 MiB RAM per 600 + reg-cap: 38400 # default + + # disable file-handle pooling -- instead, repeatedly close and reopen files during upload (bad idea to enable this on windows and/or cow filesystems) + no-fpool + + # force file-handle pooling, even when it might be dangerous (multiprocessing, filesystems lacking sparse-files support, ...) + use-fpool + + # unix file permissions to use when creating files; default is probably 644 (OS-decided), see --help-chmod. Examples: [644] = owner-RW + all-R, [755] = owner-RWX + all-RX, [777] = full-yolo + # 📂 also available as volflag "chmod_f" + chmod-f: 644 # hint; default is unset + + # unix file permissions to use when creating directories; see --help-chmod. Examples: [755] = owner-RW + all-R, [777] = full-yolo + # 📂 also available as volflag "chmod_d" + chmod-d: 755 # default + + # unix user-id to chown new files/folders to; default = -1 = do-not-change + # 📂 also available as volflag "uid" + uid: 1000 # hint + + # unix group-id to chown new files/folders to; default = -1 = do-not-change + # 📂 also available as volflag "gid" + gid: 573 # hint + + # enable symlink-based upload deduplication + # 📂 also available as volflag "dedup" + dedup + + # how careful to be when deduplicating files; [1] = just verify the filesize, [50] = verify file contents have not been altered + # 📂 also available as volflag "safededup" + safe-dedup: 50 # default + + # enable hardlink-based dedup; will fallback on symlinks when that is impossible (across filesystems) + # 📂 also available as volflag "hardlink" + hardlink + + # do not fallback to symlinks when a hardlink cannot be made + # 📂 also available as volflag "hardlinkonly" + hardlink-only + + # enable reflink-based dedup; will fallback on full copies when that is impossible (non-CoW filesystem) + # 📂 also available as volflag "reflink" + reflink + + # reject duplicate files during upload; only matches within the same volume + # 📂 also available as volflag "nodupe" + no-dupe + + # do not use existing data on disk to satisfy dupe uploads; reduces server HDD reads in exchange for much more network load + # 📂 also available as volflag "noclone" + no-clone + + # disable snapshots -- forget unfinished uploads on shutdown; don't create .hist/up2k.snap files -- abandoned/interrupted uploads must be cleaned up manually + no-snap + + # write upload state to ./hist/up2k.snap every SEC seconds; allows resuming incomplete uploads after a server crash + snap-wri: 300 # default + + # forget unfinished uploads after MIN minutes; impossible to resume them after that (360=6h, 1440=24h) + snap-drop: 1440 # default + + # how to timestamp uploaded files; [c]=client-last-modified, [u]=upload-time, [fc]=force-c, [fu]=force-u + # 📂 also available as volflag "u2ts" + u2ts: c # default + + # force randomized filenames, --nrand chars long + # 📂 also available as volflag "rand" + rand + + # randomized filenames length + # 📂 also available as volflag "nrand" + nrand: 9 # default + + # enable filetype detection on nameless uploads + # 📂 also available as volflag "magic" + magic + + # ensure GiB free disk space by rejecting upload requests; assumes gigabytes unless a unit suffix is given: [256m], [4], [2T] + # 📂 also available as volflag "df" + df: 4 # hint; default=0 + + # windows-only: minimum size of incoming uploads through up2k before they are made into sparse files + sparse: 4 # default + + # configure turbo-mode in up2k client; [-1] = forbidden/always-off, [0] = default-off and warn if enabled, [1] = default-off, [2] = on, [3] = on and disable datecheck + turbo: 0 # default + + # when to use a wasm-hasher instead of the browser's builtin; faster on chrome, but buggy in older chrome versions. [0] = only when necessary (non-https), [1] = always (all browsers), [2] = always on chrome/firefox, [3] = always on chrome, [N] = chrome-version N and newer (recommendation: 137) + nosubtle: 0 # default + + # web-client: number of file chunks to upload in parallel; 1 or 2 is good when latency is low (same-country), 2~4 for android-clients, 2~6 for cross-atlantic. Max is 6 in most browsers. Big values increase network-speed but may reduce HDD-speed + u2j: 2 # default + + # web-client: default upload chunksize (MiB); sets min,default,max in the settings gui. Each HTTP POST will aim for default, and never exceed max. Cloudflare max is 96. Big values are good for cross-atlantic but may increase HDD fragmentation on some FS. Disable this optimization with [1,1,1] + u2sz: 1,64,96 # default + + # web-client: default setting for when to replace/overwrite existing files; [0]=never, [1]=if-client-newer, [2]=always + # 📂 also available as volflag "u2ow" + u2ow: 0 # default + + # upload order; [s]=smallest-first, [n]=alphabetical, [fs]=force-s, [fn]=force-n -- alphabetical is a bit slower on fiber/LAN but makes it easier to eyeball if everything went fine + u2sort: s # default + + # write POST reports to textfiles in working-directory + write-uplog + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// network options \\0000000000000000000000000000000000000000000000000000000000000000000000\ + + # IPs and/or unix-sockets to listen on, COMMA-SEPARATED LIST (see --help-bind). Default: all IPv4 and IPv6 + i: 192.168.0.1,::1,unix:770:www:/dev/shm/party.sock # hint; default=:: + + # ports to listen on (comma/range); ignored for unix-sockets + p: 3923,4001-4005 # hint; default=3923 + + # include link-local IPv4/IPv6 in mDNS replies, even if the NIC has routable IPs (breaks some mDNS clients) + ll + + # which ip to associate clients with; [0]=tcp, [1]=origin (first x-fwd, unsafe), [2]=outermost-proxy, [3]=second-proxy, [-1]=closest-proxy + rproxy: 1 # default + + # if reverse-proxied, which http header to read the client's real ip from + xff-hdr: x-forwarded-for # default + + # list of trusted reverse-proxy CIDRs (comma-separated); only accept the real-ip header (--xff-hdr) and IdP headers if the incoming connection is from an IP within either of these subnets. Specify [lan] to allow all LAN / private / non-internet IPs. Can be disabled with [any] if you are behind cloudflare (or similar) and are using --xff-hdr=cf-connecting-ip (or similar) + xff-src: 127.0.0.0/8, ::1/128 # default + + # only accept connections from IP-addresses inside CIDR (comma-separated); examples: [lan] or [10.89.0.0/16, 192.168.33.0/24] + ipa: 10.89.0.0/16, 192.168.33.0/24 # hint; default is unset + + # if reverse-proxying on a location instead of a dedicated domain/subdomain, provide the base location here; example: [/foo/bar] + rp-loc: /files # hint; default is unset + + # set reuseaddr on listening sockets on windows; allows rapid restart of copyparty at the expense of being able to accidentally start multiple instances + reuseaddr + + # allow listening on IPs which do not yet exist, for example if the network interfaces haven't finished going up. Only makes sense for IPs other than '0.0.0.0', '127.0.0.1', '::', and '::1'. May require running as root (unless net.ipv6.ip_nonlocal_bind) + freebind + + # write list of listening-on ip:port to textfile at PATH when http-servers have started + wr-h-eps: /dev/shm/listening-on.txt # hint; default is unset + + # write list of accessible-on ip:port to textfile at PATH when http-servers have started + wr-h-aon: /dev/shm/accessible-on.txt # hint; default is unset + + # socket timeout (read request header) + s-thead: 120 # default + + # socket timeout (read/write request/response bodies). Use 60 on fast servers (default is extremely safe). Disable with 0 if reverse-proxied for a 2% speed boost + s-tbody: 128 # default + + # socket read size in bytes (indirectly affects filesystem writes; recommendation: keep equal-to or lower-than --iobuf) + s-rd-sz: 262144 # default + + # socket write size in bytes + s-wr-sz: 262144 # default + + # debug: socket write delay in seconds + s-wr-slp: 0 # default + + # debug: response delay in seconds + rsp-slp: 0 # default + + # debug: response delay, random duration 0..SEC + rsp-jtr: 0 # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// SSL/TLS options \\0000000000000000000000000000000000000000000000000000000000000000000000\ + + # disable ssl/tls -- force plaintext + http-only + + # disable plaintext -- force tls + https-only + + # path to file containing a concatenation of TLS key and certificate chain + cert: $XDG_CONFIG_PATH/.copyparty/cert.pem # default + + # set allowed ssl/tls versions; [help] shows available versions; default is what your python version considers safe + ssl-ver: ssl3,tls10 # hint; default is unset + + # set allowed ssl/tls ciphers; [help] shows available ciphers + ciphers: xtea,rot13 # hint; default is unset + + # dump some tls info + ssl-dbg + + # log master secrets for later decryption in wireshark + ssl-log: /mnt/stash/gotem.log # hint; default is unset + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// TLS certificate generator options \\0000000000000000000000000000000000000000000000000000\ + + # disable automatic certificate creation + no-crt + + # comma-separated list of FQDNs (domains) to add into the certificate + crt-ns: example.com,fileshare.nasa.gov + + # do not add wildcard entries for each --crt-ns + crt-exact + + # do not add autodetected IP addresses into cert + crt-noip + + # do not add 127.0.0.1 / localhost into cert + crt-nolo + + # do not add mDNS names / hostname into cert + crt-nohn + + # where to save the CA cert + crt-dir: $XDG_CONFIG_PATH/.copyparty/ # default + + # ca-certificate expiration time in days + crt-cdays: 3650 # default + + # server-cert expiration time in days + crt-sdays: 365 # default + + # CA/server-cert common-name + crt-cn: partyco # default + + # override CA name + crt-cnc: --crt-cn # default = copy the above + + # override server-cert name + crt-cns: --crt-cnc # default = copy the above + + # backdate in hours + crt-back: 72 # default + + # algorithm and keysize; one of these: ecdsa-256 rsa-4096 rsa-2048 + crt-alg: ecdsa-256 # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// IdP / identity provider / user authentication options \\00000000000000000000000000000000\ + + # bypass the copyparty authentication checks if the request-header HN contains a username to associate the request with (for use with authentik/oauth/...) + # └─WARNING: if you enable this, make sure clients are unable to specify this header themselves; must be washed away and replaced by a reverse-proxy + idp-h-usr: idp-username # hint; default is unset + + # assume the request-header HN contains the groupname of the requesting user; can be referenced in config files for group-based access control + idp-h-grp: idp-groups # hint; default is unset + + # optional but recommended safeguard; your reverse-proxy will insert a secret header named HN into all requests, and the other IdP headers will be ignored if this header is not present + idp-h-key: supersecretmagicword # hint; default is unset + + # if there are multiple groups in --idp-h-grp, they are separated by one of the characters in RE + idp-gsep: |:;+, # default + + # where to store the known IdP users/groups (if you run multiple copyparty instances, make sure they use different DBs) + idp-db: $XDG_CONFIG_PATH/.copyparty/idp.db # default + + # how to use --idp-db; [0] = entirely disable, [1] = write-only (effectively disabled), [2] = remember users, [3] = remember users and groups. + # └─NOTE: Will remember and restore the IdP-volumes of all users for all eternity if set to 2 or 3, even when user is deleted from your IdP + idp-store: 1 # default + + # comma-separated list of users allowed to use /?idp (the cache management UI) + idp-adm: ben,jerry # hint; default is unset + + # generate a session-token for IdP users which is written to cookie cppws (or cppwd if plaintext), to reduce the load on the IdP server, lifetime S seconds. + # └─note: The expiration time is a client hint only; the actual lifetime of the session-token is infinite (until next restart with --ses-db wiped) + idp-cookie: 600 # hint; default=0=disabled + + # disable basic-authentication support; do not accept passwords from the 'Authenticate' header at all. NOTE: This breaks support for the android app + no-bauth + + # keeps basic-authentication enabled, but only as a last-resort; if a cookie is also provided then the cookie wins + bauth-last + + # where to store the sessions database (if you run multiple copyparty instances, make sure they use different DBs) + ses-db: $XDG_CONFIG_PATH/.copyparty/sessions.db # default + + # session key length; default is 120 bits ((20//4)*4*6) + ses-len: 20 + + # disable sessions; use plaintext passwords in cookies + no-ses + + # REPEATABLE: users with IP matching CIDR are auto-authenticated as username USR; example: [172.16.24.0/24=dave] + ipu: CIDR=USR # placeholder + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// user-changeable passwords options \\0000000000000000000000000000000000000000000000000000\ + + # allow users to change their own passwords + chpw + + # REPEATABLE: do not allow password-changes for this comma-separated list of usernames + chpw-no: ole,dole,doffen # hint; default is unset + + # where to store the passwords database (if you run multiple copyparty instances, make sure they use different DBs) + chpw-db: $XDG_CONFIG_PATH/.copyparty/chpw.json + + # minimum password length + chpw-len: 8 # default + + # verbosity of summary on config load [0] = nothing at all, [1] = number of users, [2] = list users with default-pw, [3] = list all users + chpw-v: 2 # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// Zeroconf options \\000000000000000000000000000000000000000000000000000000000000000000000\ + + # enable all zeroconf backends (mdns, ssdp) + z + + # enable zeroconf ONLY on the comma-separated list of subnets and/or interface names/indexes + # └─example: eth0, wlo1, virhost0, 192.168.123.0/24, fd00:fda::/96 + z-on: NETS # placeholder; default is unset + + # disable zeroconf on the comma-separated list of subnets and/or interface names/indexes + z-off: NETS # placeholder; default is unset + + # check for network changes every SEC seconds (0=disable) + z-chk: 10 # default + + # verbose all zeroconf backends + zv + + # rejoin multicast groups every SEC seconds (workaround for some switches/routers which cause mDNS to suddenly stop working after some time); try [300] or [180] + # └─note: can be due to firewalls; make sure UDP port 5353 is open in both directions (on clients too) + mc-hop: 0 # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// Zeroconf-mDNS options; also see --help-zm \\00000000000000000000000000000000000000000000\ + + # announce the enabled protocols over mDNS (multicast DNS-SD) -- compatible with KDE, gnome, macOS, ... + zm + + # enable mDNS ONLY on the comma-separated list of subnets and/or interface names/indexes + zm-on: NETS # placeholder; default is unset + + # disable mDNS on the comma-separated list of subnets and/or interface names/indexes + zm-off: NETS # placeholder; default is unset + + # IPv4 only -- try this if some clients can't connect + zm4 + + # IPv6 only + zm6 + + # verbose mdns + zmv + + # verboser mdns + zmvv + + # mute parser errors (invalid incoming MDNS packets) + zm-no-pe + + # disable workaround for avahi-bug #379 (corruption in Avahi's mDNS reflection feature) + zm-nwa-1 + + # list of services to announce -- d=webdav h=http f=ftp s=smb -- lowercase=plaintext uppercase=TLS -- default: all enabled services except http/https (Ddfs if --ftp and --smb is set, Dd otherwise) + zms: dhf # probably default + + # link a specific folder for webdav shares + zm-ld: /public/stuff # hint; default is unset (webroot) + + # link a specific folder for http shares + zm-lh: /public/stuff # hint; default is unset (webroot) + + # link a specific folder for ftp shares + zm-lf: /public/stuff # hint; default is unset (webroot) + + # link a specific folder for smb shares + zm-ls: /public/stuff # hint; default is unset (webroot) + + # merge NICs which share subnets; assume that same subnet means same network + zm-mnic + + # merge subnets on each NIC -- always enabled for ipv6 -- reduces network load, but gnome-gvfs clients may stop working, and clients cannot be in subnets that the server is not + zm-msub + + # disable NSEC replies -- try this if some clients don't see copyparty + zm-noneg + + # send unsolicited announce every SEC; useful if clients have IPs in a subnet which doesn't overlap with the server, or to avoid some firewall issues + zm-spam: 0 # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// Zeroconf-SSDP options \\0000000000000000000000000000000000000000000000000000000000000000\ + + # announce the enabled protocols over SSDP -- compatible with Windows + zs + + # enable SSDP ONLY on the comma-separated list of subnets and/or interface names/indexes + zs-on: NETS # placeholder; default is unset + + # disable SSDP on the comma-separated list of subnets and/or interface names/indexes + zs-off: NETS # placeholder; default is unset + + # verbose SSDP + zsv + + # location to include in the url (or a complete external URL), for example [priv/?pw=hunter2] (goes directly to /priv/ with password hunter2) or [?hc=priv&pw=hunter2] (shows mounting options for /priv/ with password) + zsl: /?hc # default + + # USN (device identifier) to announce + zsid: UUID # placeholder; default is an autogenerated UUID + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// FTP options (TCP only) \\000000000000000000000000000000000000000000000000000000000000000\ + + # enable FTP server on PORT, for example 3921 + ftp: 3921 # hint; default is unset + + # enable FTPS server on PORT, for example 3990 + ftps: 3990 # hint; default is unset + + # verbose + ftpv + + # only listen on IPv4 + ftp4 + + # only accept connections from IP-addresses inside CIDR (comma-separated); specify [any] to disable inheriting --ipa. Examples: [lan] or [10.89.0.0/16, 192.168.33.0/24] + ftp-ipa: CIDR # placeholder + + # if target file exists, reject upload instead of overwrite + ftp-no-ow + + # grace period for resuming interrupted uploads (any client can write to any file last-modified more recently than SEC seconds ago) + ftp-wt: 7 # default + + # the NAT address to use for passive connections + ftp-nat: 192.168.1.13 # hint; default is unset + + # the range of TCP ports to use for passive connections, for example 12000-13000 + ftp-pr: 12000-12099 # hint; default is unset + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// WebDAV options \\00000000000000000000000000000000000000000000000000000000000000000000000\ + + # enable full write support, even if client may not be webdav. WARNING: This has side-effects -- PUT-operations will now OVERWRITE existing files, rather than inventing new filenames to avoid loss of data. You might want to instead set this as a volflag where needed. By not setting this flag, uploaded files can get written to a filename which the client does not expect (which might be okay, depending on client) + daw + + # allow depth:infinite requests (recursive file listing); extremely server-heavy but required for spec compliance -- luckily few clients rely on this + dav-inf + + # disable apple-garbage filter -- allow macos to create junk files (._* and .DS_Store, .Spotlight-*, .fseventsd, .Trashes, .AppleDouble, __MACOS) + dav-mac + + # show symlink-destination's lastmodified instead of the link itself; always enabled for recursive listings + # 📂 also available as volflag "davrt" + dav-rt + + # force auth for all folders (required by davfs2 when only some folders are world-readable) + # 📂 also available as volflag "davauth" + dav-auth + + # regex of tricky user-agents which expect 401 from GET requests; disable with [no] or blank + dav-ua1: kioworker/ # default (KDE) + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// TFTP options (UDP only) \\00000000000000000000000000000000000000000000000000000000000000\ + + # enable TFTP server on PORT, for example 69 or 3969 + tftp: 3969 # hint; default is unset + + # only listen on IPv4 + tftp4 + + # verbose + tftpv + + # verboser + tftpvv + + # debug: disable optimizations + tftp-no-fast + + # return a directory listing if a file with this name is requested and it does not exist; defaults matches .ls, dir, .dir.txt, ls.txt, ... + tftp-lsf: \.?(dir|ls)(\.txt)? # default + + # if someone tries to download a directory, return an error instead of showing its directory listing + tftp-nols + + # only accept connections from IP-addresses inside CIDR (comma-separated); specify [any] to disable inheriting --ipa. Examples: [lan] or [10.89.0.0/16, 192.168.33.0/24] + tftp-ipa: CIDR # placeholder + + # the range of UDP ports to use for data transfer, for example 12000-13000 + tftp-pr: 12100-12199 # hint; default is unset + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// SMB/CIFS options \\000000000000000000000000000000000000000000000000000000000000000000000\ + + # enable smb (read-only) -- this requires running copyparty as root on linux and macos unless --smb-port is set above 1024 and your OS does port-forwarding from 445 to that. + # └─WARNING: this protocol is DANGEROUS and buggy! Never expose to the internet! + smb + + # enable write support (please dont) + smbw + + # disable SMBv2, only enable SMBv1 (CIFS) + smb1 + + # port to listen on -- if you change this value, you must NAT from TCP:445 to this port using iptables or similar + smb-port: 445 # default + + # truncate directory listings to 64kB (~400 files); avoids impacket-0.11 bug, fixes impacket-0.12 performance + smb-nwa-1 + + # disable impacket workaround for filecopy globs + smb-nwa-2 + + # small performance boost: disable per-account permissions, enables account coalescing instead (if one user has write/delete-access, then everyone does) + smba + + # verbose + smbv + + # verboser + smbvv + + # verbosest + smbvvv + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// handlers (see --help-handlers) \\0000000000000000000000000000000000000000000000000000000\ + + # REPEATABLE: handle 404s by executing PY file + on404: ~/bin/on404.py # hint; default is unset + on404: ~/bin/on404season2.py + + # REPEATABLE: handle 403s by executing PY file + on403: ~/bin/on403.py # hint; default is unset + + # recompile handlers on each request -- expensive but convenient when hacking on stuff + hot-handlers + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// event hooks (see --help-hooks) \\0000000000000000000000000000000000000000000000000000000\ + + # REPEATABLE: execute CMD before a file upload starts + xbu: ~/bin/execute-before-upload.py # hint; default is unset + + # REPEATABLE: execute CMD after a file upload finishes + xau: ~/bin/execute-after-upload.py # hint; default is unset + + # REPEATABLE: execute CMD after all uploads finish and volume is idle + xiu: ~/bin/execute-idle-upload.py # hint; default is unset + + # REPEATABLE: execute CMD before a file copy + xbc: ~/bin/execute-before-copy.py # hint; default is unset + + # REPEATABLE: execute CMD after a file copy + xac: ~/bin/execute-after-copy.py # hint; default is unset + + # REPEATABLE: execute CMD before a file move/rename + xbr: ~/bin/execute-before-rename.py # hint; default is unset + + # REPEATABLE: execute CMD after a file move/rename + xar: ~/bin/execute-after-rename.py # hint; default is unset + + # REPEATABLE: execute CMD before a file delete + xbd: ~/bin/execute-before-delete.py # hint; default is unset + + # REPEATABLE: execute CMD after a file delete + xad: ~/bin/execute-after-delete.py # hint; default is unset + + # REPEATABLE: execute CMD on message + xm: ~/bin/execute-on-message.py # hint; default is unset + + # REPEATABLE: execute CMD if someone gets banned (pw/404/403/url) + xban: ~/bin/execute-on-ban.py # hint; default is unset + + # verbose hooks + hook-v + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// grafana/prometheus metrics endpoint \\00000000000000000000000000000000000000000000000000\ + + # enable openmetrics at /.cpr/metrics for admin accounts + stats + + # disable disk-space metrics (used/free space) + nos-hdd + + # disable volume size metrics (num files, total bytes, vmaxb/vmaxn) + nos-vol + + # disable volume state metrics (indexing, analyzing, activity) + nos-vst + + # disable dupe-files metrics (good idea; very slow) + nos-dup + + # disable unfinished-uploads metrics + nos-unf + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// yolo options \\0000000000000000000000000000000000000000000000000000000000000000000000000\ + + # disable csrf protections; let other domains/sites impersonate you through cross-site requests + allow-csrf + + # allow cookies from other domains (if you follow a link from another website into your server, you will arrive logged-in); this reduces protection against CSRF + cookie-lax + + # permit ?move=[...] and ?delete as GET + getmod + + # allow users with write-only access to upload logues and readmes without adding the _wo_ filename prefix + # 📂 also available as volflag "wo_up_readme" + wo-up-readme + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// opt-outs \\00000000000000000000000000000000000000000000000000000000000000000000000000000\ + + # never write anything to disk (debug/benchmark) + nw + + # do not disable quick-edit-mode on windows (it is disabled to avoid accidental text selection in the terminal window, as this would pause execution) + keep-qem + + # disable webdav support + no-dav + + # disable delete operations + no-del + + # disable move/rename operations + no-mv + + # disable copy operations + no-cp + + # no title hostname; don't show --name in + nth + + # no info hostname -- don't show in UI + nih + + # no info disk-usage -- don't show in UI + nid + + # no powered-by-copyparty branding in UI + nb + + # reject download-as-zip if more than N files in total; optionally takes a unit suffix: [256], [9K], [4G] + # 📂 also available as volflag "zipmaxn" + zipmaxn: N # placeholder; default is unset + + # reject download-as-zip if total download size exceeds SZ bytes; optionally takes a unit suffix: [256M], [4G], [2T] + # 📂 also available as volflag "zipmaxs" + zipmaxs: SZ # placeholder; default is unset + + # custom errormessage when download size exceeds max + # 📂 also available as volflag "zipmaxt" + zipmaxt: TXT # placeholder; default is unset + + # authenticated users bypass the zip size limit + # 📂 also available as volflag "zipmaxu" + zipmaxu + + # who can download as zip/tar? [0]=nobody, [1]=admins, [2]=authenticated-with-read-access, [3]=everyone-with-read-access (volflag=zip_who) + # └─WARNING: if a nested volume has a more restrictive value than a parent volume, then this will be ignored if the download is initiated from the parent, more lenient volume + zip-who: 3 # default + + # regex of user-agents to reject from download-as-zip/tar; disable with [no] or blank + ua-nozip: Barkrowler|bingbot|BLEXBot|Googlebot|GoogleOther|GPTBot|PetalBot|SeekportBot|SemrushBot|YandexBot # default + + # disable download as zip/tar; same as --zip-who=0 + no-zip + + # disable download as compressed tar (?tar=gz, ?tar=bz2, ?tar=xz, ?tar=gz:9, ...) + no-tarcmp + + # do not allow clients (or server config) to schedule an upload to be deleted after a given time + no-lifetime + + # disable race-the-beam (lockstep download of files which are currently being uploaded) + # 📂 also available as volflag "nopipe" + no-pipe + + # disable streaming a growing files with ?tail + # 📂 also available as volflag "notail" + no-tail + + # do not write uploader-IP into the database; will also disable unpost, you may want --forget-ip instead + # 📂 also available as volflag "no_db_ip" + no-db-ip + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// safety options \\00000000000000000000000000000000000000000000000000000000000000000000000\ + + # increase safety: Disable thumbnails / potentially dangerous software (ffmpeg/pillow/vips), hide partial uploads, avoid crawlers. + # └─Alias of --dotpart --no-thumb --no-mtag-ff --no-robots --force-js + s + + # further increase safety: Prevent js-injection, accidental move/delete, broken symlinks, webdav, 404 on 403, ban on excessive 404s. + # └─Alias of -s --unpost=0 --no-del --no-mv --hardlink --vague-403 -nih + ss + + # further increase safety: Enable logging to disk, scan for dangerous symlinks. + # └─Alias of -ss --no-dav --no-logues --no-readme -lo=cpp-%Y-%m%d-%H%M%S.txt.xz --ls=**,*,ln,p,r + sss + + # do a sanity/safety check of all volumes on startup; arguments USER,VOL,FLAGS (see --help-ls); example [**,*,ln,p,r] + ls: **,*,ln,p,r # hint; default is unset + + # never follow symlinks leaving the volume root, unless the link is into another volume where the user has similar access + # 📂 also available as volflag "xvol" + xvol + + # stay within the filesystem of the volume root; do not descend into other devices (symlink or bind-mount to another HDD, ...) + # 📂 also available as volflag "xdev" + xdev + + # disallow moving dotfiles; makes it impossible to move folders containing dotfiles + no-dot-mv + + # disallow renaming dotfiles; makes it impossible to turn something into a dotfile + no-dot-ren + + # disable rendering .prologue/.epilogue.html into directory listings + no-logues + + # disable rendering readme/preadme.md into directory listings + no-readme + + # send 404 instead of 403 (security through ambiguity, very enterprise) + vague-403 + + # don't send folder listings as HTML, force clients to use the embedded json instead -- slight protection against misbehaving search engines which ignore --no-robots + force-js + + # adds http and html headers asking search engines to not index anything + # 📂 also available as volflag "norobots" + no-robots + + # logout clients after H hours of inactivity; [0.0028]=10sec, [0.1]=6min, [24]=day, [168]=week, [720]=month, [8760]=year) + logout: 8086 # default + + # more than N wrong passwords in W minutes = ban for B minutes; disable with [no] + ban-pw: 9,60,1440 # default + + # more than N password-changes in W minutes = ban for B minutes; disable with [no] + ban-pwc: 5,60,1440 # default + + # hitting more than N 404's in W minutes = ban for B minutes; only affects users who cannot see directory listings because their access is either g/G/h + ban-404: 50,60,1440 # default + + # hitting more than N 403's in W minutes = ban for B minutes; [1440]=day, [10080]=week, [43200]=month + ban-403: 9,2,1440 # default + + # hitting more than N 422's in W minutes = ban for B minutes (invalid requests, attempted exploits ++) + ban-422: 9,2,1440 # default + + # hitting more than N sus URL's in W minutes = ban for B minutes; applies only to permissions g/G/h (decent replacement for --ban-404 if that can't be used) + ban-url: 9,2,1440 # default + + # URLs which are considered sus / eligible for banning; disable with blank or [no] + sus-urls: \.php$|(^|/)wp-(admin|content|includes)/ # default + + # harmless URLs ignored from 404-bans; disable with blank or [no] + nonsus-urls: ^(favicon\.ico|robots\.txt)$|^apple-touch-icon|^\.well-known # default + + # if a client is banned, reject its connection as soon as possible; not a good idea to enable when proxied behind cloudflare since it could ban your reverse-proxy + early-ban + + # if a client maxes out the server connection limit, downgrade it from connection:keep-alive to connection:close for MIN minutes (and also kill its active connections) -- disable with 0 + aclose: 10 # default + + # if a client maxes out the server connection limit without sending headers, ban it for B minutes; disable with [0] + loris: 60 # default + + # Access-Control-Allow-Origin; list of origins (domains/IPs without port) to accept requests from; [https://1.2.3.4]. Default [*] allows requests from all sites but removes cookies and http-auth; only ?pw=hunter2 survives + acao: * # default + + # Access-Control-Allow-Methods; list of methods to accept from offsite ('*' behaves like --acao's description) + acam: GET,HEAD # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// salting options \\0000000000000000000000000000000000000000000000000000000000000000000000\ + + # account-pw hashing algorithm; one of these, best to worst: argon2 scrypt sha2 none (each optionally followed by alg-specific comma-sep. config) + ah-alg: argon2 # hint; default is unset + + # account-pw salt; ignored if --ah-alg is none (default) + ah-salt: shangalabangala # hint; default is autogenerated and stored in $XDG_CONFIG_PATH/.copyparty/ + + # generate hashed password for PW, or read passwords from STDIN if PW is [-] + ah-gen: PW # placeholder + + # launch an interactive shell which hashes passwords without ever storing or displaying the original passwords + ah-cli + + # per-file accesskey salt; used to generate unpredictable URLs for hidden files + fk-salt: shangalabangala # hint; default is autogenerated and stored in $XDG_CONFIG_PATH/.copyparty/ + + # per-directory accesskey salt; used to generate unpredictable URLs to share folders with users who only have the 'get' permission + dk-salt: shangalabangala # hint; default is autogenerated and stored in $XDG_CONFIG_PATH/.copyparty/ + + # up2k file-hash salt; serves no purpose, no reason to change this (but delete all databases if you do) + warksalt: hunter2 # default + + # on startup, print the effective value of --ah-salt (the autogenerated value in $XDG_CONFIG_HOME unless otherwise specified) + show-ah-salt + + # on startup, print the effective value of --fk-salt (the autogenerated value in $XDG_CONFIG_HOME unless otherwise specified) + show-fk-salt + + # on startup, print the effective value of --dk-salt (the autogenerated value in $XDG_CONFIG_HOME unless otherwise specified) + show-dk-salt + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// shutdown options \\000000000000000000000000000000000000000000000000000000000000000000000\ + + # continue running even if it's impossible to listen on some of the requested endpoints + ign-ebind + + # continue running even if it's impossible to receive connections at all + ign-ebind-all + + # shutdown after WHEN has finished; [cfg] config parsing, [idx] volscan + multimedia indexing + exit: WHEN # placeholder; default is unset + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// logging options \\0000000000000000000000000000000000000000000000000000000000000000000000\ + + # quiet; disable most STDOUT messages + q + + # logfile, example: cpp-%Y-%m%d-%H%M%S.txt.xz (NB: some errors may appear on STDOUT only) + lo: PATH # placeholder + + # disable colors; same as environment-variable NO_COLOR + no-ansi + + # force colors; overrides environment-variable NO_COLOR + ansi + + # don't flush the logfile after each write; tiny bit faster + no-logflush + + # do not list volumes and permissions on startup + no-voldump + + # do not use local timezone; assume the TZ env-var is UTC (tiny bit faster) + log-utc + + # timestamp resolution / number of timestamp decimals + log-tdec: 3 # default + + # log failed login attempt passwords: 0=terse, 1=plaintext, 2=hashed + log-badpwd: 1 # default + + # debug: print tcp-server msgs + log-conn + + # debug: print http-server threadpool scaling + log-htp + + # print request HEADER; [*]=all + ihead: user-agent # hint; default is unset + + # print response HEADER; [*]=all + ohead: set-cooke # hint; default is unset + + # dont log URLs matching regex RE + lf-url: ^/\.cpr/|[?&]th=[wjp]|/\.(_|ql_|DS_Store$|localized$) # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// admin panel options \\000000000000000000000000000000000000000000000000000000000000000000\ + + # disable ?reload=cfg (reload users/volumes/volflags from config file) + no-reload + + # disable ?scan (volume reindexing) + no-rescan + + # disable ?stack (list all stacks) + no-stack + + # disable ?ru (list of recent uploads) + no-ups-page + + # don't show list of incoming files in controlpanel + no-up-list + + # who can see active downloads in the controlpanel? [0]=nobody, [1]=admins, [2]=everyone + dl-list: 2 # default + + # who can see recent uploads on the ?ru page? [0]=nobody, [1]=admins, [2]=everyone + # 📂 also available as volflag "ups_who" + ups-who: 2 # default + + # let everyone see upload timestamps on the ?ru page, not just admins + ups-when + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// thumbnail options \\00000000000000000000000000000000000000000000000000000000000000000000\ + + # disable all thumbnails + # 📂 also available as volflag "dthumb" + no-thumb + + # disable video thumbnails + # 📂 also available as volflag "dvthumb" + no-vthumb + + # disable audio thumbnails (spectrograms) + # 📂 also available as volflag "dathumb" + no-athumb + + # thumbnail res + # 📂 also available as volflag "thsize" + th-size: 320x256 # default + + # num cpu cores to use for generating thumbnails + th-mt: 4 # hint; default is autodetect + + # conversion timeout in seconds + # 📂 also available as volflag "convt" + th-convt: 60 # default + + # max memory usage (GiB) permitted by thumbnailer; not very accurate + th-ram-max: 3 # hint; default is 60% of free ram at startup (some conditions apply) + + # crop thumbnails to 4:3 or keep dynamic height; client can override in UI unless force. [y]=crop, [n]=nocrop, [fy]=force-y, [fn]=force-n + # 📂 also available as volflag "crop" + th-crop: y # default + + # show thumbs at 3x resolution; client can override in UI unless force. [y]=yes, [n]=no, [fy]=force-yes, [fn]=force-no + # 📂 also available as volflag "th3x" + th-x3: n # default + + # image decoders, in order of preference + th-dec: vips,pil,ff # default + + # disable jpg output + th-no-jpg + + # disable webp output + th-no-webp + + # force jpg output for video thumbs (avoids issues on some FFmpeg builds) + th-ff-jpg + + # use swresample instead of soxr for audio thumbs (faster, lower accuracy, avoids issues on some FFmpeg builds) + th-ff-swr + + # activity labeling cooldown -- avoids doing keepalive pokes (updating the mtime) on thumbnail folders more often than SEC seconds + th-poke: 300 # default + + # cleanup interval; 0=disabled + th-clean: 43200 # default + + # max folder age -- folders which haven't been poked for longer than --th-poke seconds will get deleted every --th-clean seconds + th-maxage: 604800 # default + + # folder thumbnails to stat/look for; enabling -e2d will make these case-insensitive, and try them as dotfiles (.folder.jpg), and also automatically select thumbnails for all folders that contain pics, even if none match this pattern + th-covers: folder.png,folder.jpg,cover.png,cover.jpg # default + + # image formats to decode using pillow + th-r-pil: a,very,long,list,of,file,extensions # hint + + # image formats to decode using pyvips + th-r-vips: a,very,long,list,of,file,extensions # hint + + # image formats to decode using ffmpeg + th-r-ffi: a,very,long,list,of,file,extensions # hint + + # video formats to decode using ffmpeg + th-r-ffv: a,very,long,list,of,file,extensions # hint + + # audio formats to decode using ffmpeg + th-r-ffa: a,very,long,list,of,file,extensions # hint + + # audio formats which provoke https://trac.ffmpeg.org/ticket/10797 (huge ram usage for s3xmodit spectrograms) + th-spec-cnv: a,very,long,list,of,file,extensions # hint + + # audio/image formats to decompress before passing to ffmpeg + au-unpk: mdz=mod.zip, mdgz=mod.gz, mdxz=mod.xz, and so on + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// transcoding options \\000000000000000000000000000000000000000000000000000000000000000000\ + + # target bitrate for transcoding to opus; set 0 to disable + q-opus: 128 # default + + # target quality for transcoding to mp3, for example [192k] (CBR) or [q0] (CQ/CRF, q0=maxquality, q9=smallest); set 0 to disable + q-mp3: q2 # default + + # allow transcoding to wav (lossless, uncompressed) + allow-wav + + # allow transcoding to flac (lossless, compressed) + allow-flac + + # disable transcoding to caf-opus (affects iOS v12~v17), will use mp3 instead + no-caf + + # disable transcoding to webm-opus (iOS v18 and later), will use mp3 instead + no-owa + + # disable audio transcoding + no-acode + + # disable batch audio transcoding by folder download (zip/tar) + no-bacode + + # delete cached transcode output after SEC seconds + ac-maxage: 86400 # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// tailing options (realtime streaming of a growing file) \\0000000000000000000000000000000\ + + # who can tail? [0]=nobody, [1]=admins, [2]=authenticated-with-read-access, [3]=everyone-with-read-access + # 📂 also available as volflag "tail_who" + tail-who: 2 # default + + # do not allow starting a new tail if more than N active downloads + tail-cmax: 64 # default + + # terminate connection after SEC seconds; [0]=never + # 📂 also available as volflag "tail_tmax" + tail-tmax: 0 # default + + # check for new data every SEC seconds + # 📂 also available as volflag "tail_rate" + tail-rate: 0.2 # default + + # send a zerobyte if connection is idle for SEC seconds to prevent disconnect + tail-ka: 3 # default + + # check if file was replaced (new fd) if idle for SEC seconds + # 📂 also available as volflag "tail_fd" + tail-fd: 1 # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// RSS options \\00000000000000000000000000000000000000000000000000000000000000000000000000\ + + # enable RSS output (experimental) + # 📂 also available as volflag "rss" + rss + + # default number of files to return (url-param 'nf') + rss-nf: 250 # default + + # default list of file extensions to include (url-param 'fext'); blank=all + rss-fext: mp3,opus # hint; default is unset + + # default sort order (url-param 'sort'); [m]=last-modified [u]=upload-time [n]=filename [s]=filesize; Uppercase=oldest-first. Note that upload-time is 0 for non-uploaded files + rss-sort: m # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// general db options \\0000000000000000000000000000000000000000000000000000000000000000000\ + + # enable up2k database; this enables file search, upload-undo, improves deduplication + e2d + + # scan writable folders for new files on startup; sets -e2d + e2ds + + # scans all folders on startup; sets -e2ds + e2dsa + + # verify file integrity; rehash all files and compare with db + e2v + + # on hash mismatch: update the database with the new hash + e2vu + + # on hash mismatch: panic and quit copyparty + e2vp + + # where to store volume data (db, thumbs); default is a folder named ".hist" inside each volume + # 📂 also available as volflag "hist" + hist: PATH # placeholder; default is unset + + # override where the volume databases are to be placed; default is the same as --hist + # 📂 also available as volflag "dbpath" + dbpath: PATH # placeholder; default is unset + + # regex: disable hashing of matching absolute-filesystem-paths during e2ds folder scans + # 📂 also available as volflag "nohash" + no-hash: ^/mnt/nas/linux-isos/knoppix/ + + # regex: disable indexing of matching absolute-filesystem-paths during e2ds folder scans + # 📂 also available as volflag "noidx" + no-idx: ^/mnt/nas/logs/ + + # do not show total recursive size of folders in listings, show inode size instead; slightly faster + # 📂 also available as volflag "nodirsz" + no-dirsz + + # if the directory-sizes in the UI are bonkers, use this along with -e2dsa to rebuild the index from scratch + re-dirsz + + # disable rescan acceleration; do full database integrity check -- makes the db ~5% smaller and bootup/rescans 3~10x slower + no-dhash + + # force a cache rebuild on startup; enable this once if it gets out of sync (should never be necessary) + re-dhash + + # never forget indexed files, even when deleted from disk -- makes it impossible to ever upload the same file twice -- only useful for offloading uploads to a cloud service or something + # 📂 also available as volflag "noforget" + no-forget + + # remove uploader-IP from database (and make unpost impossible) MIN minutes after upload, for GDPR reasons. Default [0] is never-forget. [1440]=day, [10080]=week, [43200]=month. + # 📂 also available as volflag "forget_ip" + forget-ip: 0 # default (disabled) + + # database durability profile; sets the tradeoff between robustness and speed, see --help-dbd + # 📂 also available as volflag "dbd" + dbd: wal # default + + # on upload: check all volumes for dupes, not just the target volume (probably buggy, not recommended) + # 📂 also available as volflag "xlink" + xlink + + # num cpu cores to use for file hashing; set 0 or 1 for single-core hashing + hash-mt: 4 # default is autodetect but max 5 + + # rescan filesystem for changes every SEC seconds; 0=off + # 📂 also available as volflag "scan" + re-maxage: 0 # default + + # defer any scheduled volume reindexing until SEC seconds after last db write (uploads, renames, ...) + db-act: 10 # default + + # search deadline -- terminate searches running for more than SEC seconds + srch-time: 45 # default + + # max search results to allow clients to fetch; 125 results will be shown initially + srch-hits: 7999 # default + + # regex: exclude files from search results if the file-URL matches PTN (case-sensitive) + # 📂 also available as volflag "srch_excl" + srch-excl: password|logs/[0-9] # hint (any URL containing 'password' or 'logs/DIGIT'), default is unset + + # show dotfiles in search results (volflags: dotsrch | nodotsrch) + dotsrch + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// metadata db options \\000000000000000000000000000000000000000000000000000000000000000000\ + + # enable metadata indexing; makes it possible to search for artist/title/codec/resolution/... + e2t + + # scan newly discovered files for metadata on startup; sets -e2t + e2ts + + # delete all metadata from DB and do a full rescan; sets -e2ts + e2tsr + + # use FFprobe for tags instead; will detect more tags + no-mutagen + + # never use FFprobe as tag reader; is probably safer + no-mtag-ff + + # timeout for FFprobe tag-scan + mtag-to: 60 # default + + # num cpu cores to use for tag scanning + mtag-mt: 4 # hint; default is autodetect + + # verbose tag scanning; print errors from mtp subprocesses and such + mtag-v + + # debug mtp settings and mutagen/FFprobe parsers + mtag-vv + + # REPEATABLE: add/replace metadata mapping + mtm: qux=foo,bar # hint (clone metadata-key foo to qux with bar as fallback); default is unset + + # tags to index/display (comma-sep.); either an entire replacement list, or add/remove stuff on the default-list with +foo or /bar" + mte: .files,circle,album,.tn,artist,title,.bpm,key,.dur,.q,.vq,.aq,vc,ac,fmt,res,.fps,ahash,vhash # default + + # tags to hide by default (comma-sep.); assign/add/remove same as -mte" + mth: .vq,.aq,vc,ac,fmt,res,.fps # default + + # REPEATABLE: read tag M using program BIN to parse the file + mtp: .bpm=~/bin/audio-bpm.py # hint; default is unset + mtp: key=f,t5,~/bin/audio-key.py + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// textfile options \\000000000000000000000000000000000000000000000000000000000000000000000\ + + # where to store old version of markdown files; [s]=subfolder, [v]=volume-histpath, [n]=nope/disabled + # 📂 also available as volflag "md_hist" + md-hist: s # default + + # the textfile editor will check for serverside changes every SEC seconds + mcr: 60 # default + + # enable markdown plugins -- neat but dangerous, big XSS risk + emp + + # enable textfile expansion -- replace {{self.ip}} and such; see --help-exp + # 📂 also available as volflag "exp" + exp + + # comma/space-separated list of placeholders to expand in markdown files; add/remove stuff on the default list with +hdr_foo or /vf.scan + # 📂 also available as volflag "exp_md" + exp-md: self.ip self.ua self.uname self.host cfg.name cfg.logout vf.scan vf.thsize hdr.cf_ipcountry srv.itime srv.htime # default + + # comma/space-separated list of placeholders to expand in prologue/epilogue files + # 📂 also available as volflag "exp_lg" + exp-lg: self.ip self.ua self.uname self.host cfg.name cfg.logout vf.scan vf.thsize hdr.cf_ipcountry srv.itime srv.htime # default + + # regex of user-agents to reject from viewing documents through ?doc=[...]; disable with [no] or blank + ua-nodoc: Barkrowler|bingbot|BLEXBot|Googlebot|GoogleOther|GPTBot|PetalBot|SeekportBot|SemrushBot|YandexBot # default + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// og / open graph / discord-embed options \\0000000000000000000000000000000000000000000000\ + + # disable hotlinking and return an html document instead; this is required by open-graph, but can also be useful on its own + # 📂 also available as volflag "og" + og + + # only disable hotlinking / engage OG behavior if the useragent matches regex RE + # 📂 also available as volflag "og_ua" + og-ua: (Discord|Twitter|Slack)bot # hint; default is unset + + # do not return the regular copyparty html, but instead load the jinja2 template at PATH (if path contains 'EXT' then EXT will be replaced with the requested file's extension) + # 📂 also available as volflag "og_tpl" + og-tpl: /var/copyparty/opengraph-EXT.j2 # hint; default is unset + + # do not automatically add OG entries into <head> (useful if you're doing this yourself in a template or such) + # 📂 also available as volflag "og_no_head" + og-no-head + + # thumbnail format; j=jpeg, jf=jpeg-uncropped, jf3=jpeg-uncropped-large, w=webm, ... + # 📂 also available as volflag "og_th" + og-th: jf3 # default + + # fallback title if there is nothing in the -e2t database + # 📂 also available as volflag "og_title" + og-title: HEY CHECK THIS OUT # hint; default is blank + + # audio title format; takes any metadata key + # 📂 also available as volflag "og_title_a" + og-title-a: 🎵 {{ artist }} - {{ title }} # default + + # video title format; takes any metadata key + # 📂 also available as volflag "og_title_v" + og-title-v: {{ title }} # default + + # image title format; takes any metadata key + # 📂 also available as volflag "og_title_i" + og-title-i: {{ title }} # default + + # force default title; do not read from tags + # 📂 also available as volflag "og_s_title" + og-s-title + + # description text; same for all files, disable with [-] + # 📂 also available as volflag "og_desc" + og-desc: bottom text # hint; default is blank + + # sitename; defaults to --name, disable with [-] + # 📂 also available as volflag "og_site" + og-site: dank memes # hint; default is blank which means it'll copy the `name` global-option + + # accent color (3 or 6 hex digits); may also affect safari and/or android-chrome + # 📂 also available as volflag "tcolor" + tcolor: 333 # default (copyparty gray) + + # query-string parceling; translate a request for /foo/.uqe/BASE64 into /foo?TEXT, or /foo/?TEXT if the first character in TEXT is a slash. Automatically enabled for --og + uqe + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// ui options \\000000000000000000000000000000000000000000000000000000000000000000000000000\ + + # show grid/thumbnails by default + # 📂 also available as volflag "grid" + grid + + # select files in grid by ctrl-click + # 📂 also available as volflag "gsel" + gsel + + # default to local timezone instead of UTC + localtime + + # language; one of the following: eng nor chi + lang: eng # default + + # default theme to use (0..7) + theme: 0 # default + + # number of themes installed + themes: 8 # default + + # default audio/video volume percent + au-vol: 50 # default; anything 0-100 + + # default sort order, comma-separated column IDs (see header tooltips), prefix with '-' for descending. Examples: href -href ext sz ts tags/Album tags/.tn + # 📂 also available as volflag "sort" + sort: href # default + + # default-enable natural sort of filenames with leading numbers + # 📂 also available as volflag "nsort" + nsort + + # number of sorting rules to include in media URLs by default + # 📂 also available as volflag "hsortn" + hsortn: 2 # default + + # default-enable seeing dotfiles; only takes effect if user has the necessary permissions + see-dots + + # number of confirmations to show when deleting files (2/1/0) + qdel: 2 # default + + # don't show files/folders matching REGEX in file list. WARNING: Purely cosmetic! Does not affect API calls, just the browser. Example: [\.(js|css)$] + # 📂 also available as volflag "unlist" + unlist: REGEX # placeholder + + # favicon-text [ foreground [ background ] ], set blank to disable + favico: c 000 none # default + + # REPEATABLE: use thumbnail-image VP for file-extension E, example: [exe=/.res/exe.png] + # 📂 also available as volflag "ext_th" + ext-th: 7z=/.icons/archive.png # hint; default is unset + ext-th: exe=/.icons/glass-of-red-liquid.png + + # emoji or emoji,css Example: [🥖,padding:0] + spinner: 🌲 # default + + # URL to additional CSS to include in the filebrowser html + css-browser: /.res/rice.css # hint; default is unset + + # URL to additional JS to include in the filebrowser html + js-browser: /.res/mousetrail.js # hint; default is unset + + # URL to additional JS to include in all other pages + js-other: /.res/snowflakes.jpg # hint; default is unset + + # text to append to the <head> of all HTML pages (except for basic-browser); can be @PATH to send the contents of a file at PATH, and/or begin with % to render as jinja2 template + # 📂 also available as volflag "html_head" + html-head: <script>alert(1);</script> # hint; default is unset + + # if a folder contains index.html, show that instead of the directory listing by default (can be changed in the client settings UI, or add ?v to URL for override) + ih + + # file extensions to present as plaintext + textfiles: txt,nfo,diz,cue,readme # default + + # max size of embedded textfiles on ?doc= (anything bigger will be lazy-loaded by JS) + txt-max: 64 # default + + # title / service-name to show in html documents + doctitle: copyparty @ --name # default (--name will copy from global-option `name`) + + # server name (displayed in filebrowser document title) + bname: --name # default (copy global-option `name`) + + # powered-by link; disable with -nb + pb-url: https://github.com/9001/copyparty # default + + # show version on the control panel (incompatible with -nb) + ver + + # configure the option to enable/disable k304 on the controlpanel (workaround for buggy reverse-proxies); [0] = hidden and default-off, [1] = visible and default-off, [2] = visible and default-on + k304: 0 # default + + # configure the option to enable/disable no304 on the controlpanel (workaround for buggy caching in browsers); [0] = hidden and default-off, [1] = visible and default-off, [2] = visible and default-on + no304: 0 # default + + # list of capabilities to allow in the iframe 'sandbox' attribute for README.md docs (volflag=md_sbf); see https://developer.mozilla.org/en-US/docs/Web/HTML/Element/iframe#sandbox + md-sbf: downloads forms popups scripts top-navigation-by-user-activation # default + + # list of capabilities to allow in the iframe 'sandbox' attribute for prologue/epilogue docs + # 📂 also available as volflag "lg_sbf" + lg-sbf: downloads forms popups scripts top-navigation-by-user-activation # default + + # the value of the iframe 'allow' attribute for README.md docs, for example [fullscreen] + # 📂 also available as volflag "md_sba" + md-sba: fullscreen # hint; default is blank + + # the value of the iframe 'allow' attribute for prologue/epilogue docs (volflag=lg_sba); see https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Permissions-Policy#iframes + lg-sba: TXT # placeholder + + # don't sandbox README/PREADME.md documents (volflags: no_sb_md | sb_md) + no-sb-md + + # don't sandbox prologue/epilogue docs (volflags: no_sb_lg | sb_lg); enables non-js support + no-sb-lg + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// debug options \\000000000000000000000000000000000000000000000000000000000000000000000000\ + + # verbose config file parser (explain config) + vc + + # generate config file from current config (best-effort; probably buggy) + cgen + + # list information about detected optional dependencies + deps + + # kernel-bug workaround: disable poll; use select instead (limits max num clients to ~700) + no-poll + + # kernel-bug workaround: disable sendfile; do a safe and slow read-send-loop instead + no-sendfile + + # kernel-bug workaround: disable scandir; do a listdir + stat on each file instead + no-scandir + + # wait for initial filesystem indexing before accepting client requests + no-fastboot + + # disable httpserver threadpool, create threads as-needed instead + no-htp + + # when listening on unix-sockets, do a basic delete+bind instead of the default atomic bind + rm-sck + + # explain search processing, and do some extra expensive sanity checks + srch-dbg + + # use mdns-domain instead of server-ip on /?hc + rclone-mdns + + # write stacktrace to Path every S second, for example --stackmon=./st/%Y-%m/%d/%H%M.xz,60 + stackmon: P,S # placeholder + + # list active threads every SEC + log-thrs: 0 # default + + # log filekey params for files where path matches REGEX; [.] (a single dot) = all files + log-fk: /mnt/a-problematic-fuse/this-folder-breaks-filekeys/ # hint; default is unset + + # [up2k] if a client uploads a bitflipped/corrupted chunk, store a copy according to --bf-nc and --bf-dir + bak-flips + + # bak-flips: stop if there's more than NUM files at --kf-dir already; default: 6.3 GiB max (200*32M) + bf-nc: 200 # default + + # bak-flips: store corrupted chunks at PATH; default: folder named 'bf' wherever copyparty was started + bf-dir: /srv/bitflips/ # hint; default = bf + + # bak-flips: log corruption info to a textfile at PATH + bf-log: /srv/bitflips/the-history.txt # hint; default is unset + + ############################################################################################### + ############################################################################################### + ############################################################################################### + ############################################################################################### + ############################################################################################### + ############################################################################################### + #####/ + ##### This is the end of the [global] config section + ####/ + +[accounts] + foo: bar # username foo, password bar + +[groups] + g1: u1, u2, u3 # group "g1" with users u1, u2, and u3 + +[/the/url/to/share/this/volume/on/] + /the/actual/filesystem/path/ + accs: + r: username_who_gets_Read_access + w: username_who_gets_Write_access + m: username_who_gets_Move_access + d: username_who_gets_Delete_access + .: username_who_can_see_Dotfiles + g: username_who_gets_Get_access + G: username_who_gets_upGet_access + h: username_who_gets_html_access + a: username_who_gets_admin_access + A: username_who_gets_ReadWriteMoveDeleteDotfileAdmin_access + rwm: @g1 # the group "g1" gets Read+Write+Move + flags: + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// uploads, general \\000000000000000000000000000000000000000000000000000000000000000000000\ + + # enable symlink-based file deduplication + dedup + + # enable hardlink-based file deduplication, with fallback on symlinks when that is impossible + hardlink + + # dedup with hardlink only, never symlink; make a full copy if hardlink is impossible + hardlinkonly + + # enable reflink-based file deduplication, with fallback on full copy when that is impossible + reflink + + # verify on-disk data before using it for dedup + safededup + + # take dupe data from clients, even if available on HDD + noclone + + # rejects existing files (instead of linking/cloning them) + nodupe + + # unix-permission for new dirs/folders + chmod_d: 755 + + # unix-permission for new files + chmod_f: 644 + + # change owner of new files/folders to unix-user 573 + uid: 573 + + # change owner of new files/folders to unix-group 999 + gid: 999 + + # force use of sparse files, mainly for s3-backed storage + sparse + + # deny use of sparse files, mainly for slow storage + nosparse + + # enable full WebDAV write support (dangerous); PUT-operations will now OVERWRITE existing files + daw + + # forces all uploads into the top folder of the vfs + nosub + + # enables filetype detection for nameless uploads + magic + + # fallback filename for nameless uploads + put_name + + # default checksum-hasher for PUT/WebDAV uploads + put_ck + + # default checksum-hasher for bup/basic uploads + bup_ck + + # allows server-side gzip compression of uploads with ?gz + gz + + # allows server-side lzma compression of uploads with ?xz + xz + + # forces server-side compression, optional arg: xz,9 + pk + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// upload rules \\0000000000000000000000000000000000000000000000000000000000000000000000000\ + + # max 250 uploads over 15min + maxn: 250,600 + + # max 1 GiB over 5min (suffixes: b, k, m, g, t) + maxb: 1g,300 + + # total volume size max 1 GiB (suffixes: b, k, m, g, t) + vmaxb: 1g + + # max 4096 files in volume (suffixes: b, k, m, g, t) + vmaxn: 4k + + # return medialinks for non-up2k uploads (not hotlinks) + medialinks + + # write-only users can upload logues without getting renamed + wo_up_readme + + # force randomized filenames, 9 chars long by default + rand + + # randomized filenames are N chars long + nrand: N + + # overwrite existing files? 0=no 1=if-older 2=always + u2ow: N + + # [f]orce [c]lient-last-modified or [u]pload-time + u2ts: fc + + # allow aborting unfinished uploads? 0=no 1=strict 2=ip-chk 3=acct-chk + u2abort: 1 + + # allow filesizes between 1 KiB and 3MiB + sz: 1k-3m + + # ensure 1 GiB free disk space + df: 1g + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// upload rotation -- (moves all uploads into the specified folder structure) \\00000000000\ + + # 3 levels of subfolders with 100 entries in each + rotn: 100,3 + + # date-formatted organizing + rotf: %Y-%m/%d-%H + + # uploads are deleted after 1 hour + lifetime: 3600 + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// database, general \\00000000000000000000000000000000000000000000000000000000000000000000\ + + # enable database; makes files searchable + enables upload-undo + e2d + + # scan writable folders for new files on startup; also sets -e2d + e2ds + + # scans all folders for new files on startup; also sets -e2d + e2dsa + + # enable multimedia indexing; makes it possible to search for tags + e2t + + # scan existing files for tags on startup; also sets -e2t + e2ts + + # delete all metadata from DB (full rescan); also sets -e2ts + e2tsr + + # disables metadata collection for existing files + d2ts + + # verify integrity on startup by hashing files and comparing to db + e2v + + # when e2v fails, update the db (assume on-disk files are good) + e2vu + + # when e2v fails, panic and quit copyparty + e2vp + + # disables onboot indexing, overrides -e2ds* + d2ds + + # disables metadata collection, overrides -e2t* + d2t + + # disables file verification, overrides -e2v* + d2v + + # disables all database stuff, overrides -e2* + d2d + + # puts thumbnails and indexes at that location + hist: /tmp/cdb + + # puts indexes at that location + dbpath: /tmp/cdb + + # disable db if file foo doesn't exist + landmark: foo + + # scan for new files every 60sec, same as --re-maxage + scan: 60 + + # skips hashing file contents if path matches *.iso + nohash: \.iso$ + + # fully ignores the contents at paths matching *.iso + noidx: \.iso$ + + # don't forget files when deleted from disk + noforget + + # forget uploader-IP after 30 days (GDPR) + forget_ip: 43200 + + # never store uploader-IP in the db; disables unpost + no_db_ip + + # avoid excessive reindexing on android sdcardfs + fat32 + + # database speed-durability tradeoff + dbd: [acid|swal|wal|yolo] + + # cross-volume dupe detection / linking (dangerous) + xlink + + # do not descend into other filesystems + xdev + + # do not follow symlinks leaving the volume root + xvol + + # show dotfiles in search results + dotsrch + + # hide dotfiles in search results (default) + nodotsrch + + # exclude search results with URL matching this regex + srch_excl + + # media-tags to index/display + mte: artist,title + + # media-tags to hide by default + mth: fmt,res,ac + + # uses the "audio-bpm.py" program to generate ".bpm" tags from uploads (f = overwrite tags) + mtp: .bpm=f,audio-bpm.py + + # collects two tags at once + mtp: ahash,vhash=media-hash.py + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// thumbnails \\000000000000000000000000000000000000000000000000000000000000000000000000000\ + + # disables all thumbnails + dthumb + + # disables video thumbnails + dvthumb + + # disables audio thumbnails (spectrograms) + dathumb + + # disables image thumbnails + dithumb + + # compress audio waveforms 33% better + pngquant + + # thumbnail res; WxH + thsize + + # center-cropping (y/n/fy/fn) + crop + + # 3x resolution (y/n/fy/fn) + th3x + + # conversion timeout in seconds + convt + + # use /b.png as thumbnail for file-extension s + ext_th: s=/b.png + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// handlers -- (better explained in --help-handlers) \\000000000000000000000000000000000000\ + + # handle 404s by executing PY file + on404: ~/bin/hook.py + + # handle 403s by executing PY file + on403: ~/bin/hook.py + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// event hooks -- (better explained in --help-hooks) \\000000000000000000000000000000000000\ + + # execute CMD before a file upload starts + xbu: ~/bin/hook.py + + # execute CMD after a file upload finishes + xau: ~/bin/hook.py + + # execute CMD after all uploads finish and volume is idle + xiu: ~/bin/hook.py + + # execute CMD before a file copy + xbc: ~/bin/hook.py + + # execute CMD after a file copy + xac: ~/bin/hook.py + + # execute CMD before a file rename/move + xbr: ~/bin/hook.py + + # execute CMD after a file rename/move + xar: ~/bin/hook.py + + # execute CMD before a file delete + xbd: ~/bin/hook.py + + # execute CMD after a file delete + xad: ~/bin/hook.py + + # execute CMD on message + xm: ~/bin/hook.py + + # execute CMD if someone gets banned + xban: ~/bin/hook.py + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// client and ux \\000000000000000000000000000000000000000000000000000000000000000000000000\ + + # show grid/thumbnails by default + grid + + # select files in grid by ctrl-click + gsel + + # default sort order + sort + + # natural-sort of leading digits in filenames + nsort + + # number of sort-rules to add to media URLs + hsortn + + # dont list files matching REGEX + unlist + + # includes TXT in the <head>, or @PATH for file at PATH + html_head: <script>alert(1);</script> + + # theme color (a hint for webbrowsers, discord, etc.) + tcolor: #fc0 + + # don't show total folder size + nodirsz + + # allows indexing by search engines (default) + robots + + # kindly asks search engines to leave + norobots + + # don't list read-access in controlpanel + unlistcr + + # don't list write-access in controlpanel + unlistcw + + # disable js sandbox for markdown files + no_sb_md + + # disable js sandbox for prologue/epilogue + no_sb_lg + + # enable js sandbox for markdown files (default) + sb_md + + # enable js sandbox for prologue/epilogue (default) + sb_lg + + # list of markdown-sandbox safeguards to disable + md_sbf + + # list of *logue-sandbox safeguards to disable + lg_sbf + + # value of iframe allow-prop for markdown-sandbox + md_sba + + # value of iframe allow-prop for *logue-sandbox + lg_sba + + # return html and markdown as text/html + nohtml + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// opengraph (discord embeds) \\00000000000000000000000000000000000000000000000000000000000\ + + # enable OG (disables hotlinking) + og + + # sitename; defaults to --name, disable with '-' + og_site + + # description text for all files; disable with '-' + og_desc + + # thumbnail format; j / jf / jf3 / w / w3 / ... + og_th: jf + + # audio title format; default: {{ artist }} - {{ title }} + og_title_a + + # video title format; default: {{ title }} + og_title_v + + # image title format; default: {{ title }} + og_title_i + + # fallback title if there's nothing in the db + og_title: foo + + # force default title; do not read from tags + og_s_title + + # custom html; see --og-tpl in --help + og_tpl + + # you want to add tags manually with og_tpl + og_no_head + + # if defined: only send OG html if useragent matches this regex + og_ua + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// textfiles \\0000000000000000000000000000000000000000000000000000000000000000000000000000\ + + # where to put markdown backups; s=subfolder, v=volHist, n=nope + md_hist + + # enable textfile expansion; see --help-exp + exp + + # placeholders to expand in markdown files; see --help + exp_md + + # placeholders to expand in prologue/epilogue; see --help + exp_lg + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// tailing \\000000000000000000000000000000000000000000000000000000000000000000000000000000\ + + # disable ?tail (download a growing file continuously) + notail + + # check if file was replaced (new fd) every 1 sec + tail_fd: 1 + + # check for new data every 0.2 sec + tail_rate: 0.2 + + # kill connection after 30 sec + tail_tmax: 30 + + # restrict ?tail access (1=admins,2=authed,3=everyone) + tail_who: 2 + + ###000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000\ + ###// others \\0000000000000000000000000000000000000000000000000000000000000000000000000000000\ + + # allow all users with read-access to enable the option to show dotfiles in listings + dots + + # generates per-file accesskeys, which are then required at the "g" permission; keys are invalidated if filesize or inode changes + fk: 8 + + # generates slightly weaker per-file accesskeys, which are then required at the "g" permission; not affected by filesize or inode numbers + fka: 8 + + # generates per-directory accesskeys, which are then required at the "g" permission; keys are invalidated if filesize or inode changes + dk: 8 + + # per-directory accesskeys allow browsing into subdirs + dks + + # allow seeing files (not folders) inside a specific folder with "g" perm, and does not require a valid dirkey to do so + dky + + # allow '?rss' URL suffix (experimental) + rss + + # expensive analysis for mimetype accuracy + rmagic + + # restrict viewing the list of recent uploads + ups_who: 2 + + # restrict access to download-as-zip/tar + zip_who: 2 + + # reject download-as-zip if more than 9000 files + zipmaxn: 9k + + # reject download-as-zip if size over 2 GiB + zipmaxs: 2g + + # reply with 'no' if download-as-zip exceeds max + zipmaxt: no + + # zip-size-limit does not apply to authenticated users + zipmaxu + + # disable race-the-beam (download unfinished uploads) + nopipe + + # ms-windows: timeout for renaming busy files + mv_retry + + # ms-windows: timeout for deleting busy files + rm_retry + + # ask webdav clients to login for all folders + davauth + + # show lastmod time of symlink destination, not the link itself (note: this option is always enabled for recursive listings) + davrt From 00cb1f74e22dea5041c74b2389ac3087a4444041 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Wed, 6 Aug 2025 20:23:39 +0000 Subject: [PATCH 091/174] golf --- copyparty/web/browser.js | 4 ++-- copyparty/web/md2.js | 6 +++--- copyparty/web/mde.js | 6 +++--- copyparty/web/shares.js | 8 +++----- copyparty/web/up2k.js | 2 +- copyparty/web/util.js | 10 +++++----- 6 files changed, 17 insertions(+), 19 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index d7982c78..df630251 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -11957,7 +11957,7 @@ function ev_load_m3u(e) { function () { load_m3u(url); }, function () { if (has(perms, 'write') && has(perms, 'delete')) - window.location = url + '?edit'; + location = url + '?edit'; else showfile.show(url); } @@ -12556,7 +12556,7 @@ var treectl = (function () { r.reqls = function (url, hpush, back, hydrate) { if (IE && !history.pushState) - return window.location = url; + return location = url; var xhr = new XHR(), m = /[?&](k=[^&#]+)/.exec(url), diff --git a/copyparty/web/md2.js b/copyparty/web/md2.js index f7d0b59c..26b1e14b 100644 --- a/copyparty/web/md2.js +++ b/copyparty/web/md2.js @@ -255,7 +255,7 @@ function Modpoll() { } console.log('modpoll...'); - var url = (document.location + '').split('?')[0] + '?_=' + Date.now(); + var url = (location + '').split('?')[0] + '?_=' + Date.now(); var xhr = new XHR(); xhr.open('GET', url, true); xhr.responseType = 'text'; @@ -346,7 +346,7 @@ function save(e) { fd.append("lastmod", (force ? -1 : last_modified)); fd.append("body", txt); - var url = (document.location + '').split('?')[0]; + var url = (location + '').split('?')[0]; var xhr = new XHR(); xhr.open('POST', url, true); xhr.responseType = 'text'; @@ -404,7 +404,7 @@ function save_cb() { function run_savechk(lastmod, txt, btn, ntry) { // download the saved doc from the server and compare - var url = (document.location + '').split('?')[0] + '?_=' + Date.now(); + var url = (location + '').split('?')[0] + '?_=' + Date.now(); var xhr = new XHR(); xhr.open('GET', url, true); xhr.responseType = 'text'; diff --git a/copyparty/web/mde.js b/copyparty/web/mde.js index 5c2872df..058825c9 100644 --- a/copyparty/web/mde.js +++ b/copyparty/web/mde.js @@ -6,7 +6,7 @@ var dom_doc = ebi('m'); var dom_md = ebi('mt'); (function () { - var n = document.location + ''; + var n = location + ''; n = (n.slice(n.indexOf('//') + 2).split('?')[0] + '?v').split('/'); n[0] = 'top'; var loc = []; @@ -113,7 +113,7 @@ function save(mde) { fd.append("lastmod", (force ? -1 : last_modified)); fd.append("body", txt); - var url = (document.location + '').split('?')[0]; + var url = (location + '').split('?')[0]; var xhr = new XHR(); xhr.open('POST', url, true); xhr.responseType = 'text'; @@ -166,7 +166,7 @@ function save_cb() { //alert('save OK -- wrote ' + r.size + ' bytes.\n\nsha512: ' + r.sha512); // download the saved doc from the server and compare - var url = (document.location + '').split('?')[0] + '?_=' + Date.now(); + var url = (location + '').split('?')[0] + '?_=' + Date.now(); var xhr = new XHR(); xhr.open('GET', url, true); xhr.responseType = 'text'; diff --git a/copyparty/web/shares.js b/copyparty/web/shares.js index d14dfeb7..281de673 100644 --- a/copyparty/web/shares.js +++ b/copyparty/web/shares.js @@ -1,11 +1,9 @@ -var SRS = SR.trimEnd('/') + '/'; - var t = QSA('a[k]'); for (var a = 0; a < t.length; a++) t[a].onclick = rm; function rm() { - var u = SRS + '?eshare=rm&skey=' + uricom_enc(this.getAttribute('k')), + var u = SR + '/?eshare=rm&skey=' + uricom_enc(this.getAttribute('k')), xhr = new XHR(); xhr.open('POST', u, true); @@ -15,7 +13,7 @@ function rm() { function bump() { var k = this.closest('tr').getElementsByTagName('a')[2].getAttribute('k'), - u = SRS + '?skey=' + uricom_enc(k) + '&eshare=' + this.value, + u = SR + '/?skey=' + uricom_enc(k) + '&eshare=' + this.value, xhr = new XHR(); xhr.open('POST', u, true); @@ -27,7 +25,7 @@ function cb() { if (this.status !== 200) return modal.alert('<h6>server error</h6>' + esc(unpre(this.responseText))); - document.location = '?shares'; + location = '?shares'; } function qr(e) { diff --git a/copyparty/web/up2k.js b/copyparty/web/up2k.js index 31ed1d6a..1e49fc4e 100644 --- a/copyparty/web/up2k.js +++ b/copyparty/web/up2k.js @@ -1572,7 +1572,7 @@ function up2k_init(subtle) { function linklist() { var ret = [], - base = document.location.origin.replace(/\/$/, ''); + base = location.origin.replace(/\/$/, ''); for (var a = 0; a < st.files.length; a++) { var t = st.files[a], diff --git a/copyparty/web/util.js b/copyparty/web/util.js index 929316e5..e58e7cca 100644 --- a/copyparty/web/util.js +++ b/copyparty/web/util.js @@ -120,7 +120,7 @@ function esc(txt) { function basenames(txt) { return (txt + '').replace(/https?:\/\/[^ \/]+\//g, '/').replace(/js\?_=[a-zA-Z]{4}/g, 'js'); } -if ((document.location + '').indexOf(',rej,') + 1) +if ((location + '').indexOf(',rej,') + 1) window.onunhandledrejection = function (e) { var err = e.reason; try { @@ -741,7 +741,7 @@ function assert_vp(path) { if (path.indexOf('//') + 1) throw 'nonlocal1: ' + path; - var o = window.location.origin; + var o = location.origin; if (have_URL && (new URL(path, o)).origin != o) throw 'nonlocal2: ' + path; } @@ -893,7 +893,7 @@ function uricom_adec(arr, li) { function get_evpath() { - var ret = document.location.pathname; + var ret = location.pathname; if (ret.indexOf('/') !== 0) ret = '/' + ret; @@ -1249,10 +1249,10 @@ function hist_replace(url) { function sethash(hv) { if (window.history && history.replaceState) { - hist_replace(document.location.pathname + document.location.search + '#' + hv); + hist_replace(location.pathname + location.search + '#' + hv); } else { - document.location.hash = hv; + location.hash = hv; } } From 0f55a1ae86675490fad2297770aa7c0f063bc031 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Wed, 6 Aug 2025 20:24:31 +0000 Subject: [PATCH 092/174] fix js-crash for url `//`; closes #487 --- copyparty/web/browser.js | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index df630251..de66887e 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -6742,6 +6742,10 @@ var ACtx = !IPHONE && (window.AudioContext || window.webkitAudioContext), dk, mp; +if (location.pathname.indexOf('//') === 0) + hist_replace(location.pathname.replace(/^\/+/, '/')); + + if (window.og_fn) { hash0 = 1; hist_replace(vsplit(get_evpath())[0]); From a57f7cc2f859a4d1ed09f5ce928728f11ec3a1b8 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Wed, 6 Aug 2025 20:25:29 +0000 Subject: [PATCH 093/174] ie9: recent-uploads --- copyparty/httpcli.py | 4 ++++ copyparty/web/rups.html | 9 +-------- copyparty/web/rups.js | 8 +++++--- 3 files changed, 10 insertions(+), 11 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index d562485b..274316d5 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -5447,6 +5447,8 @@ class HttpCli(object): elif nfi == 3: if not vp.endswith(vfi): continue + else: + continue n -= 1 if not n: @@ -5571,6 +5573,8 @@ class HttpCli(object): elif nfi == 3: if not vp.endswith(vfi): continue + else: + continue if not dots and "/." in vp: continue diff --git a/copyparty/web/rups.html b/copyparty/web/rups.html index af90eefe..2cc0443a 100644 --- a/copyparty/web/rups.html +++ b/copyparty/web/rups.html @@ -19,14 +19,7 @@ <a href="{{ r }}/?h">control-panel</a>   Filter: <input type="text" id="filter" size="20" placeholder="documents/passwords" />   <span id="hits"></span> - <table id="tab"><thead><tr> - <th>size</th> - <th>who</th> - <th>when</th> - <th>age</th> - <th>dir</th> - <th>file</th> - </tr></thead><tbody id="tb"></tbody></table> + <div id="tw"></div> </div> <a href="#" id="repl">π</a> <script> diff --git a/copyparty/web/rups.js b/copyparty/web/rups.js index 9559be8c..17a44556 100644 --- a/copyparty/web/rups.js +++ b/copyparty/web/rups.js @@ -1,5 +1,6 @@ function render() { - var ups = V.ups, now = V.now, html = []; + var html = ['<table id="tab"><thead><tr><th>size</th><th>who</th><th>when</th><th>age</th><th>dir</th><th>file</th></tr></thead><tbody>']; + var ups = V.ups, now = V.now; ebi('filter').value = V.filter; ebi('hits').innerHTML = 'showing ' + ups.length + ' files'; @@ -26,7 +27,8 @@ function render() { var t = V.filter ? ' matching the filter' : ''; html = ['<tr><td colspan="6">there are no uploads' + t + '</td></tr>']; } - ebi('tb').innerHTML = html.join(''); + html.push('</tbody></table>'); + ebi('tw').innerHTML = html.join('\n'); } render(); @@ -46,7 +48,7 @@ function ask(e) { V = JSON.parse(this.responseText) } catch (ex) { - ebi('tb').innerHTML = '<tr><td colspan="6">failed to decode server response as json: <pre>' + esc(this.responseText) + '</pre></td></tr>'; + ebi('tw').innerHTML = 'failed to decode server response as json: <pre>' + esc(this.responseText) + '</pre>'; return; } render(); From 7ae84dea1a105483d5131b167ce7d5da981e16c3 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Wed, 6 Aug 2025 20:30:01 +0000 Subject: [PATCH 094/174] autorefresh controlpanel --- copyparty/__main__.py | 1 + copyparty/httpcli.py | 8 ++++++++ copyparty/web/splash.html | 2 +- copyparty/web/splash.js | 3 +++ 4 files changed, 13 insertions(+), 1 deletion(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 3bb8c69e..3f07bfc1 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1583,6 +1583,7 @@ def add_ui(ap, retry): ap2.add_argument("--ver", action="store_true", help="show version on the control panel (incompatible with \033[33m-nb\033[0m)") ap2.add_argument("--k304", metavar="NUM", type=int, default=0, help="configure the option to enable/disable k304 on the controlpanel (workaround for buggy reverse-proxies); [\033[32m0\033[0m] = hidden and default-off, [\033[32m1\033[0m] = visible and default-off, [\033[32m2\033[0m] = visible and default-on") ap2.add_argument("--no304", metavar="NUM", type=int, default=0, help="configure the option to enable/disable no304 on the controlpanel (workaround for buggy caching in browsers); [\033[32m0\033[0m] = hidden and default-off, [\033[32m1\033[0m] = visible and default-off, [\033[32m2\033[0m] = visible and default-on") + ap2.add_argument("--ctl-re", metavar="SEC", type=int, default=1, help="the controlpanel Refresh-button will autorefresh every SEC; [\033[32m0\033[0m] = just once") ap2.add_argument("--md-sbf", metavar="FLAGS", type=u, default="downloads forms popups scripts top-navigation-by-user-activation", help="list of capabilities to allow in the iframe 'sandbox' attribute for README.md docs (volflag=md_sbf); see https://developer.mozilla.org/en-US/docs/Web/HTML/Element/iframe#sandbox") ap2.add_argument("--lg-sbf", metavar="FLAGS", type=u, default="downloads forms popups scripts top-navigation-by-user-activation", help="list of capabilities to allow in the iframe 'sandbox' attribute for prologue/epilogue docs (volflag=lg_sbf)") ap2.add_argument("--md-sba", metavar="TXT", type=u, default="", help="the value of the iframe 'allow' attribute for README.md docs, for example [\033[32mfullscreen\033[0m] (volflag=md_sba)") diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 274316d5..b335fe4a 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -5071,6 +5071,13 @@ class HttpCli(object): self.reply(zb, mime="text/plain; charset=utf-8") return True + re_btn = "" + nre = self.args.ctl_re + if "re" in self.uparam: + self.out_headers["Refresh"] = str(nre) + elif nre: + re_btn = "&re=%s" % (nre,) + html = self.j2s( "splash", this=self, @@ -5088,6 +5095,7 @@ class HttpCli(object): mtpq=vs["mtpq"], dbwt=vs["dbwt"], url_suf=suf, + re=re_btn, k304=self.k304(), no304=self.no304(), k304vis=self.args.k304 > 0, diff --git a/copyparty/web/splash.html b/copyparty/web/splash.html index 2fdc264f..8dd56bab 100644 --- a/copyparty/web/splash.html +++ b/copyparty/web/splash.html @@ -15,7 +15,7 @@ <body> <div id="wrap"> {%- if not in_shr %} - <a id="a" href="{{ r }}/?h" class="af">refresh</a> + <a id="a" href="{{ r }}/?h{{ re }}" class="af">refresh</a> <a id="v" href="{{ r }}/?hc" class="af">connect</a> {%- if this.uname == '*' %} diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index bad3615f..d3d395af 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -385,6 +385,9 @@ if (o && /[0-9]+$/.exec(o.innerHTML)) ebi('uhash').value = '' + location.hash; +if (/\&re=/.test('' + location)) + ebi('a').className = 'af g'; + (function() { if (!ebi('x')) return; From 50f46187f112bb62b33af94539a1c3e61b98fe52 Mon Sep 17 00:00:00 2001 From: chamdim <94919340+chamdim@users.noreply.github.com> Date: Thu, 7 Aug 2025 00:00:42 +0300 Subject: [PATCH 095/174] Greek translation (#468) * Greek translation * Update browser.js * greek: fixes before merge --------- Signed-off-by: chamdim <94919340+chamdim@users.noreply.github.com> Signed-off-by: ed <s@ocv.me> Co-authored-by: ed <s@ocv.me> --- copyparty/web/browser.js | 631 ++++++++++++++++++++++++++++++++++++++- 1 file changed, 630 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index de66887e..60b4a349 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3152,6 +3152,635 @@ var Ls = { "lang_set": "ladataanko sivu uudestaan kielen vaihtamiseksi?", }, + "grc": { + "tt": "Ελληνικά", + + "cols": { + "c": "κουμπιά ενεργειών", + "dur": "διάρκεια", + "q": "ποιότητα / bitrate", + "Ac": "κωδικοποιητής ήχου", + "Vc": "κωδικοποιητής βίντεο", + "Fmt": "μορφή / container", + "Ahash": "checksum ήχου", + "Vhash": "checksum βίντεο", + "Res": "ανάλυση", + "T": "τύπος αρχείου", + "aq": "ποιότητα ήχου / bitrate", + "vq": "ποιότητα βίντεο / bitrate", + "pixfmt": "subsampling / δομή εικονοστοιχείων", + "resw": "οριζόντια ανάλυση", + "resh": "κάθετη ανάλυση", + "chs": "κανάλια ήχου", + "hz": "συχνότητα δειγματοληψίας" + }, + + "hks": [ + [ + "διάφορα", + ["ESC", "κλείσιμο διαφόρων λειτουργιών"], + + "διαχειριστής αρχείων", + ["G", "εναλλαγή λίστας / πλέγματος"], + ["T", "εναλλαγή μικρογραφιών / εικονιδίων"], + ["⇧ A/D", "μέγεθος μικρογραφιών"], + ["ctrl-K", "διαγραφή επιλεγμένων"], + ["ctrl-X", "αποκοπή επιλογής στο πρόχειρο"], + ["ctrl-C", "αντιγραφή επιλογής στο πρόχειρο"], + ["ctrl-V", "επικόλληση (μετακίνηση/αντιγραφή) εδώ"], + ["Y", "λήψη επιλεγμένων"], + ["F2", "μετονομασία επιλεγμένων"], + + "λίστα αρχείων", + ["space", "εναλλαγή επιλογής αρχείου"], + ["↑/↓", "μετακίνηση δείκτη επιλογής"], + ["ctrl ↑/↓", "μετακίνηση δείκτη και προβολής"], + ["⇧ ↑/↓", "επιλογή προηγούμενου/επόμενου αρχείου"], + ["ctrl-A", "επιλογή όλων των αρχείων / φακέλων"] + ], [ + "πλοήγηση", + ["B", "εναλλαγή σε καρτέλες διαδρομών / δέντρο διαδρομών"], + ["I/K", "προηγούμενος/επόμενος φάκελος"], + ["M", "γονικός φάκελος (ή σμίκρυνση τρέχοντος)"], + ["V", "εναλλαγή φακέλων / δέντρο αρχείων κειμένου"], + ["A/D", "μέγεθος πίνακα πλοήγησης"] + ], [ + "μουσική", + ["J/L", "προηγούμενο/επόμενο τραγούδι"], + ["U/O", "μετάβαση 10δευτ πίσω/μπροστά"], + ["0..9", "μετάβαση στο 0%..90%"], + ["P", "αναπαραγωγή/παύση (ξεκινάει κιόλας)"], + ["S", "επιλογή αναπαραγόμενου τραγουδιού"], + ["Y", "λήψη τραγουδιού"] + ], [ + "εικόνες", + ["J/L, ←/→", "προηγούμενη/επόμενη εικόνα"], + ["Home/End", "πρώτη/τελευταία εικόνα"], + ["F", "πλήρης οθόνη"], + ["R", "περιστροφή δεξιόστροφα"], + ["⇧ R", "περιστροφή αριστερόστροφα"], + ["S", "επιλογή εικόνας"], + ["Y", "λήψη εικόνας"] + ], [ + "βίντεο", + ["U/O", "μετάβαση 10δευτ πίσω/μπροστά"], + ["P/K/Space", "αναπαραγωγή/παύση"], + ["C", "συνέχεια στο επόμενο"], + ["V", "επανάληψη"], + ["M", "σίγαση"], + ["[ και ]", "ορισμός διαστήματος επανάληψης"] + ], [ + "αρχεία κειμένου", + ["I/K", "προηγούμενο/επόμενο αρχείο"], + ["M", "κλείσιμο αρχείου"], + ["E", "επεξεργασία αρχείου"], + ["S", "επιλογή αρχείου (για αποκοπή/αντιγραφή/μετονομασία)"] + ] + ], + + "m_ok": "Εντάξει", + "m_ng": "Άκυρο", + + "enable": "Ενεργοποίηση", + "danger": "ΚΙΝΔΥΝΟΣ", + "clipped": "αντιγράφηκε στο πρόχειρο", + + "ht_s1": "δευτερόλεπτο", + "ht_s2": "δευτερόλεπτα", + "ht_m1": "λεπτό", + "ht_m2": "λεπτά", + "ht_h1": "ώρα", + "ht_h2": "ώρες", + "ht_d1": "μέρα", + "ht_d2": "μέρες", + "ht_and": " και ", + + "goh": "πίνακας ελέγχου", + "gop": 'προηγούμενος φάκελος στο ίδιο επίπεδο">προηγούμενο', + "gou": 'γονικός φάκελος">πάνω', + "gon": 'επόμενος φάκελος">επόμενο', + "logout": "Αποσύνδεση ", + "access": " πρόσβαση", + "ot_close": "κλείσιμο υπομενού", + "ot_search": "αναζήτηση αρχείων με βάση χαρακτηριστικά, διαδρομή / όνομα, μουσικά tags ή οποιονδήποτε συνδυασμό$N$N<code>foo bar</code> = πρέπει να περιέχει και τα «foo» και «bar»,$N<code>foo -bar</code> = πρέπει να περιέχει το «foo» αλλά όχι το «bar»,$N<code>^yana .opus$</code> = να ξεκινά με «yana» και να είναι αρχείο «opus»$N<code>"try unite"</code> = να περιέχει ακριβώς «try unite»$N$Nη μορφή ημερομηνίας είναι iso-8601, όπως$N<code>2009-12-31</code> ή <code>2020-09-12 23:30:00</code>", + "ot_unpost": "unpost: διαγραφή πρόσφατων μεταφορτώσεων ή ακύρωση ανολοκλήρωτων", + "ot_bup": "bup: βασικός uploader, υποστηρίζει μέχρι και netscape 4.0", + "ot_mkdir": "mkdir: δημιουργία νέου φακέλου", + "ot_md": "new-md: δημιουργία νέου markdown εγγράφου", + "ot_msg": "msg: αποστολή μηνύματος στο server log", + "ot_mp": "επιλογές media player", + "ot_cfg": "επιλογές ρυθμίσεων", + "ot_u2i": 'up2k: ανέβασε αρχεία (αν έχεις δικαίωμα εγγραφής) ή ενεργοποίησε τη λειτουργία αναζήτησης για να δεις αν υπάρχουν ήδη στο server$N$Nοι μεταφορτώσεις συνεχίζονται αν διακοπούν, είναι πολυνηματικές και διατηρούν τις χρονοσφραγίδες, αλλά καταναλώνουν περισσότερο CPU από τον [🎈]  (βασικός uploader)<br /><br />κατά τη διάρκεια της μεταφόρτωσης, αυτό το εικονίδιο δείχνει την πρόοδό της!', + "ot_u2w": 'up2k: ανέβασε αρχεία με υποστήριξη συνέχισης (κλείσε τον browser και ρίξε τα ίδια αρχεία ξανά μετά)$N$Nπολυνηματικό, διατηρεί τις χρονοσφραγίδες, αλλά καταναλώνει περισσότερο CPU από τον [🎈]  (βασικός uploader)<br /><br />κατά τη διάρκεια της μεταφόρτωσης, αυτό το εικονίδιο δείχνει την πρόοδό της!', + "ot_noie": 'Χρησιμοποίησε Chrome / Firefox / Edge', + + "ab_mkdir": "δημιουργία φακέλου", + "ab_mkdoc": "νέο markdown έγγραφο", + "ab_msg": "στείλε μήνυμα στο server log", + + "ay_path": "πήγαινε σε φακέλους", + "ay_files": "πήγαινε σε αρχεία", + + "wt_ren": "μετονομασία επιλεγμένων$NΣυντόμευση: F2", + "wt_del": "διαγραφή επιλεγμένων$NΣυντόμευση: ctrl-K", + "wt_cut": "αποκοπή επιλεγμένων <small>(και επικόλληση αλλού)</small>$NΣυντόμευση: ctrl-X", + "wt_cpy": "αντιγραφή επιλεγμένων στο πρόχειρο$N(για επικόλληση αλλού)$NΣυντόμευση: ctrl-C", + "wt_pst": "επικόλληση αποκομμένων / αντεγραμμένων$NΣυντόμευση: ctrl-V", + "wt_selall": "επιλογή όλων$NΣυντόμευση: ctrl-A (με το αρχείο επιλεγμένο)", + "wt_selinv": "αντιστροφή επιλογής", + "wt_zip1": "κατέβασμα φακέλου ως συμπιεσμένο αρχείο", + "wt_selzip": "κατέβασμα επιλογής ως συμπιεσμένο αρχείο", + "wt_seldl": "κατέβασμα επιλογής ως μεμονωμένα αρχεία$NΣυντόμευση: Y", + "wt_npirc": "αντιγραφή πληροφοριών τραγουδιού σε μορφή irc", + "wt_nptxt": "αντιγραφή πληροφοριών τραγουδιού ως κείμενο", + "wt_m3ua": "προσθήκη σε m3u λίστα αναπαραγωγής (μετά πάτησε <code>📻αντιγραφή</code>)", + "wt_m3uc": "αντιγραφή m3u λίστας αναπαραγωγής στο πρόχειρο", + "wt_grid": "εναλλαγή πλέγματος / λίστας$NΣυντόμευση: G", + "wt_prev": "προηγούμενο κομμάτι$NΣυντόμευση: J", + "wt_play": "αναπαραγωγή / παύση$NΣυντόμευση: P", + "wt_next": "επόμενο κομμάτι$NΣυντόμευση: L", + + "ul_par": "παράλληλες μεταφορτώσεις:", + "ut_rand": "τυχαιοποίηση ονομάτων αρχείων", + "ut_u2ts": "αντιγραφή της τελευταίας τροποποιημένης χρονοσφραγίδας αλλαγής$Nαπό το σύστημά σου στον server\">📅", + "ut_ow": "αντικατάσταση σε ήδη υπάρχοντα αρχεία του server?$N🛡️: ποτέ (θα δημιουργηθεί νέο όνομα)$N🕒: αν το αρχείο του server είναι παλαιότερο$N♻️: πάντα να αντικαθίστανται αν διαφέρουν", + "ut_mt": "συνέχιση υπολογισμού hash για άλλα αρχεία κατά τη μεταφόρτωση$N$Nαπενεργοποίησέ το αν η CPU ή ο δίσκος σου ζορίζονται", + "ut_ask": 'επιβεβαίωση πριν ξεκινήσει η μεταφόρτωση">💭', + "ut_pot": "βελτίωση ταχύτητας μεταφόρτωσης σε αργές συσκευές$Nμε απλοποίηση του UI", + "ut_srch": "μην ανεβάζεις, έλεγξε αν τα αρχεία$Nυπάρχουν ήδη στον server (ψάχνει σε όλους τους φακέλους που έχεις πρόσβαση)", + "ut_par": "κάνε παύση στις μεταφορτώσεις βάζοντάς το 0$N$Nαύξησε το αν έχεις αργή/μεγάλη καθυστέρηση σύνδεσης$N$Nκράτα το 1 σε LAN ή αν ο server έχει αργό δίσκο", + "ul_btn": "ρίξε αρχεία / φακέλους<br>εδώ (ή κάνε κλικ σε μένα)", + "ul_btnu": "Μ Ε Τ Α Φ Ο Ρ Τ Ω Σ Η", + "ul_btns": "Α Ν Α Ζ Η Τ Η Σ Η", + + "ul_hash": "υπολογισμός hash", + "ul_send": "αποστολή", + "ul_done": "ολοκληρώθηκε", + "ul_idle1": "καμία μεταφόρτωση στην ουρά για την ώρα", + "ut_etah": "μέση ταχύτητα <em>υπολογισμού hash</em> και εκτίμηση χρόνου μέχρι την ολοκλήρωση", + "ut_etau": "μέση ταχύτητα <em>μεταφόρτωσης</em> και εκτίμηση χρόνου μέχρι την ολοκλήρωση", + "ut_etat": "μέση <em>συνολική</em> ταχύτητα και εκτίμηση χρόνου μέχρι την ολοκλήρωση", + + "uct_ok": "ολοκληρώθηκε επιτυχώς", + "uct_ng": "no-good: απέτυχε / απορρίφθηκε / δεν βρέθηκε", + "uct_done": "ολοκληρωμένα και αποτυχημένα", + "uct_bz": "κάνει hash ή μεταφορτώνει", + "uct_q": "σε αναμονή, εκκρεμεί", + + "utl_name": "όνομα αρχείου", + "utl_ulist": "λίστα", + "utl_ucopy": "αντιγραφή", + "utl_links": "σύνδεσμοι", + "utl_stat": "κατάσταση", + "utl_prog": "πρόοδος", + + // keep short: + "utl_404": "404", + "utl_err": "ΣΦΑΛΜΑ", + "utl_oserr": "ΣΦ-ΛΕ", + "utl_found": "βρέθηκε", + "utl_defer": "αναβολή", + "utl_yolo": "YOLO", + "utl_done": "έγινε", + + "ul_flagblk": "τα αρχεία προστέθηκαν στην ουρά</b><br>αλλά υπάρχει άλλη ενεργή μεταφόρτωση σε άλλη καρτέλα,<br>οπότε περίμενε να τελειώσει αυτό πρώτα", + "ul_btnlk": "ο διακομιστής έχει κλειδώσει αυτήν την επιλογή σε αυτήν την κατάσταση", + + "udt_up": "Μεταφόρτωση", + "udt_srch": "Αναζήτηση", + "udt_drop": "ρίξ' το εδώ", + + "u_nav_m": '<h6>οκ, τι έχουμε εδώ;</h6><code>Enter</code> = Αρχεία (ένα ή περισσότερα)\n<code>ESC</code> = Ένας φάκελος (μαζί με υποφακέλους)', + "u_nav_b": '<a href="#" id="modal-ok">Αρχεία</a><a href="#" id="modal-ng">Ένας φάκελος</a>', + + "cl_opts": "διακόπτες", + "cl_themes": "θέμα", + "cl_langs": "γλώσσα", + "cl_ziptype": "λήψη φακέλου", + "cl_uopts": "διακόπτες μεταφόρτωσης", + "cl_favico": "favicon", + "cl_bigdir": "μεγάλοι φάκελοι", + "cl_hsort": "#ταξινόμηση", + "cl_keytype": "σημείωση πλήκτρων", + "cl_hiddenc": "κρυφές στήλες", + "cl_hidec": "κρύψε", + "cl_reset": "επανεκκίνηση", + "cl_hpick": "πάτησε στις κεφαλίδες στηλών για να τις κρύψεις στον πίνακα παρακάτω", + "cl_hcancel": "η απόκρυψη στηλών ακυρώθηκε", + + "ct_grid": '田 το πλέγμα', + "ct_ttips": '◔ ◡ ◔">ℹ️ συμβουλές εργαλείων', + "ct_thumb": 'σε προβολή πλέγματος, εναλλαγή εικονιδίων ή μικρογραφιών$NΠλήκτρο συντόμευσης: T">🖼️ μικρογραφίες', + "ct_csel": 'χρησιμοποίησε CTRL και SHIFT για επιλογή αρχείων σε προβολή πλέγματος">επιλογή', + "ct_ihop": 'όταν η προβολή εικόνων κλείνει, κάνε scroll στο τελευταίο προβολόμενο αρχείο">g⮯', + "ct_dots": 'εμφάνιση κρυφών αρχείων (αν το επιτρέπει ο server)">dotfiles', + "ct_qdel": 'όταν διαγράφεις αρχεία, ζήτα επιβεβαίωση μόνο μία φορά">γρήγορη διαγραφή', + "ct_dir1st": 'ταξινόμηση φακέλων πριν από τα αρχεία">📁 πρώτα', + "ct_nsort": 'φυσική ταξινόμηση (για ονόματα αρχείων με αριθμούς στην αρχή)">φυσική ταξινόμηση', + "ct_utc": 'εμφάνιση όλων των ημερομηνιών σε UTC">UTC', + "ct_readme": 'εμφάνιση README.md στις λίστες φακέλων">📜 πληροφορίες', + "ct_idxh": 'εμφάνιση index.html αντί για λίστα φακέλων">html', + "ct_sbars": 'εμφάνιση μπαρών κύλισης">⟊', + + "cut_umod": "αν το αρχείο υπάρχει ήδη στον server, ενημέρωσέ το με την τελευταία χρονοσφραγίδα τροποποίησης για να ταιριάζει με το τοπικό αρχείο (απαιτεί δικαιώματα εγγραφής+διαγραφής)\">re📅", + + "cut_turbo": "το κουμπί yolo, πιθανόν να ΜΗΝ ΘΕΛΕΙΣ να το ενεργοποιήσεις:$N$Nχρησιμοποίησέ το αν μεταφορτώνεις πολλά αρχεία και χρειάστηκε να ξαναρχίσεις, και θες να συνεχίσεις τη μεταφότρωση όσο το δυνατόν πιο γρήγορα$N$Nαντικαθιστά τον έλεγχο hash με απλό <em>"έχει το ίδιο μέγεθος αρχείου στον server?"</em> οπότε αν το περιεχόμενο είναι διαφορετικό, ΔΕΝ θα ανέβει$N$Nπρέπει να το κλείσεις όταν τελειώσει η μεταφόρτωση και μετά να "μεταφορτώσεις" πάλι τα ίδια αρχεία για να τα επιβεβαιώσει το τοπικό σου πρόγραμμα\">turbo", + + "cut_datechk": "δεν επηρεάζει τίποτα εκτός αν το turbo είναι ενεργοποιημένο$N$Nμειώνει λίγο τον παράγοντα yolo; ελέγχει αν οι χρονοσφραγίδες στο διακομιστή ταιριάζουν με τα δικά σου$N$Nπιάνει <em>θεωρητικά</em> τις περισσότερες μισοτελειωμένες/κατεστραμμένες μεταφορτώσεις, αλλά δεν αντικαθιστά τον έλεγχο με το turbo απενεργοποιημένο μετέπειτα\">έλεγχος ημερομηνίας", + + "cut_u2sz": "μέγεθος (σε MiB) κάθε κομματιού μεταφόρτωσης; μεγάλες τιμές λειτουργούν καλύτερα σε μεγαλύτερες αποστάσεις διακομιστή-πελάτη. Δοκίμασε μικρές τιμές σε πολύ άστατες συνδέσεις", + + "cut_flag": "εξασφαλίζει ότι μόνο μία καρτέλα μεταφορτώνει κάθε φορά $N -- οι άλλες καρτέλες πρέπει να το έχουν κι αυτές ενεργό $N -- επηρεάζει μόνο τις καρτέλες που βρίσκονται στο ίδιο διεύθυνση", + + "cut_az": "μεταφόρτωσε τα αρχεία αλφαβητικά, αντί για το μικρότερο αρχείο, πρώτα$N$Nη αλφαβητική σειρά βοηθά να καταλάβεις αν κάτι χάλασε στο διακομιστή αλλά κάνει το ανέβασμα λίγο πιο αργό σε fiber / LAN", + + "cut_nag": "ειδοποίηση λειτουργικού συστήματος όταν τελειώσει η μεταφόρτωση$N(μόνο αν ο browser ή η καρτέλα δεν είναι ενεργά)", + "cut_sfx": "ηχητική ειδοποίηση όταν τελειώσει η μεταφόρτωση$N(μόνο αν ο browser ή η καρτέλα δεν είναι ενεργά)", + + "cut_mt": "χρησιμοποίησε multithreading για να επιταχύνεις το hashing των αρχείων$N$Nχρησιμοποιεί web-workers και χρειάζεται$Nπερισσότερη RAM (μέχρι 512 MiB επιπλέον)$N$Nκάνει το https 30% πιο γρήγορο, το http 4.5x πιο γρήγορο\">mt", + + "cut_wasm": "χρησιμοποίησε wasm αντί για τον ενσωματωμένο hasher του browser; βελτιώνει την ταχύτητα σε chrome-based browsers αλλά αυξάνει το φορτίο της CPU, και παλιές εκδόσεις chrome έχουν bugs που κάνουν το browser να τρώει όλη τη RAM και να κρασάρει αν ενεργοποιηθεί\">wasm", + + "cft_text": "κείμενο favicon (κενό και ανανέωση για απενεργοποίηση)", + "cft_fg": "χρώμα προσκηνίου", + "cft_bg": "χρώμα παρασκηνίου", + + "cdt_lim": "μέγιστος αριθμός αρχείων προς εμφάνιση σε ένα φάκελο", + "cdt_ask": "όταν φτάνεις στο τέλος,$Nαντί να φορτώσει περισσότερα αρχεία,$Nρώτα τι να κάνει", + "cdt_hsort": "πόσους κανόνες ταξινόμησης (<code>,sorthref</code>) να συμπεριλάβει σε URLs πολυμέσων. Αν το βάλεις 0 αγνοεί και κανόνες ταξινόμησης στους συνδέσμους πολυμέσων", + + "tt_entree": "εμφάνιση navpane (δέντρο διαδρομών)$NΠλήκτρο συντόμευσης: B", + "tt_detree": "εμφάνιση breadcrumbs (καρτέλες διαδρομών)$NΠλήκτρο συντόμευσης: B", + "tt_visdir": "κύλιση στον επιλεγμένο φάκελο", + "tt_ftree": "εναλλαγή δέντρου διαδρομών / αρχείων κειμένου$NΠλήκτρο συντόμευσης: V", + "tt_pdock": "εμφάνιση γονικών φακέλων σε σταθερή μπάρα επάνω", + "tt_dynt": "αυτόματη επέκταση καθώς επεκτείνεται το δέντρο διαδρομών", + "tt_wrap": "αναδίπλωση λέξεων", + "tt_hover": "αποκάλυψη των γραμμών που ξεπερνούν το πλάτος με το ποντίκι πάνω τους$N( σπάει το scroll εκτός αν το ποντίκι $N  είναι στην αριστερή στήλη )", + + "ml_pmode": "στο τέλος του φακέλου...", + "ml_btns": "εντολές", + "ml_tcode": "μετακωδικοποίηση", + "ml_tcode2": "μετακωδικοποίηση σε", + "ml_tint": "φίλτρο χρώματος", + "ml_eq": "ισοσταθμιστής ήχου", + "ml_drc": "συμπιεστής δυναμικής εμβέλειας", + + "mt_loop": "επανάληψη ενός τραγουδιού\">🔁", + "mt_one": "σταμάτα μετά από ένα τραγούδι\">1️⃣", + "mt_shuf": "τυχαία σειρά τραγουδιών σε κάθε φάκελο\">🔀", + "mt_aplay": "αυτόματη αναπαραγωγή αν υπάρχει song-ID στη διεύθυνση που μπήκες στο διακομιστή$N$Nη απενεργοποίηση αυτού, σταματά το URL από το να ενημερώνεται με τα song-ID ενώ παίζει η μουσική για να αποτραπεί η αυτόματη αναπαραγωγή αν χαθούν αυτές οι ρυθμίσεις αλλά το URL παραμείνει το ίδιο\">a▶", + "mt_preload": "ξεκίνα τη φόρτωση του επόμενου τραγουδιού κοντά στο τέλος για συνεχόμενη ακρόαση\">προφόρτωση", + "mt_prescan": "πήγαινε στον επόμενο φάκελο πριν τελειώσει το τελευταίο τραγούδι$Nγια να μη σταματήσει το πρόγραμμα περιήγησης να παίζει μουσική\">nav", + "mt_fullpre": "προσπάθησε να προφορτώσεις ολόκληρο το τραγούδι;$N✅ ενεργό σε <b>αναξιόπιστες</b> συνδέσεις,$N❌ πιθανότατα απενεργοποιημένο σε αργές συνδέσεις\">πλήρες", + "mt_fau": "σε κινητά, πρόλαβε να μην σταματήσει η μουσική αν το επόμενο τραγούδι δεν προφορτώθηκε γρήγορα (μπορεί να προκαλέσει πρόβλημα στην εμφάνιση των ετικετών)\">☕️", + "mt_waves": "γραμμή αναζήτησης κυματομορφής:$Nεμφάνιση έντασης ήχου στην μπάρα αναζήτησης\">~s", + "mt_npclip": "εμφάνισε κουμπιά για αντιγραφή του τρέχοντος τραγουδιού\">/np", + "mt_m3u_c": "εμφάνισε κουμπιά για αντιγραφή των$Nεπιλεγμένων τραγουδιών ως καταχωρήσεις λίστας m3u8\">📻", + "mt_octl": "ενσωμάτωση στο λειτουργικό σύστημα (σηντομεύσεις πλήκτρων πολυμέσων / osd)\">έλεγχος-OS", + "mt_oseek": "επιτρέπει την αναζήτηση μέσω ενσωμάτωσης του λειτουργικού συστήματος$N$Nσημείωση: σε μερικές συσκευές (iPhones),$Nαντικαθιστά το κουμπί επόμενου τραγουδιού\">αναζήτηση", + "mt_oscv": "εμφάνιση εξωφύλλου άλμπουμ σε osd\">εξώφυλλο", + "mt_follow": "κρατά το τρέχον κομμάτι ορατό κατά την κύλιση\">🎯", + "mt_compact": "συμπαγή κουμπιά ελέγχου\">⟎", + "mt_uncache": "καθάρισε την προσωρινή μνήμη  (δοκίμασε αυτό αν ο browser έχει αποθηκεύσει$Nχαλασμένο αντίγραφο τραγουδιού και αρνείται να παίξει)\">εκκαθάριση", + "mt_mloop": "τυχαία αναπαραγωγή στον ανοικτό φάκελο\">🔁 τυχαία αναπαραγωγή", + "mt_mnext": "φόρτωση επόμενου φακέλου και συνέχιση\">📂 επόμενο", + "mt_mstop": "σταμάτησε την αναπαραγωγή\">⏸ σταμάτημα", + "mt_cflac": "μετατροπή flac / wav σε opus\">flac", + "mt_caac": "μετατροπή aac / m4a σε opus\">aac", + "mt_coth": "μετατροπή όλων των άλλων (εκτός των mp3) σε opus\">άλλο", + "mt_c2opus": "καλύτερη επιλογή για desktop, laptop, android\">opus", + "mt_c2owa": "opus-weba, για iOS 17.5 και νεότερα\">owa", + "mt_c2caf": "opus-caf, για iOS 11 έως 17\">caf", + "mt_c2mp3": "χρησιμοποίησε αυτό σε πολύ παλιές συσκευές\">mp3", + "mt_c2flac": "βέλτιστη ποιότητα ήχου αλλά τεράστιο αρχείο για μεταφόρτωση\">flac", + "mt_c2wav": "ασυμπίεστη αναπαραγωγή (ακόμα μεγαλύτερο αρχείο)\">wav", + "mt_c2ok": "μια χαρά, σοφή επιλογή", + "mt_c2nd": "δεν είναι η προτεινόμενη μορφή εξόδου για τη συσκευή σου, αλλά αυτό είναι ok", + "mt_c2ng": "η συσκευή σου φαίνεται να μην υποστηρίζει αυτήν τη μορφή εξόδου, αλλά ας το δοκιμάσουμε ούτως ή άλλως", + "mt_xowa": "υπάρχουν bugs σε iOS που εμποδίζουν την αναπαραγωγή στο παρασκήνιο με αυτήν τη μορφή· χρησιμοποίησε caf ή mp3 αντ’ αυτού", + "mt_tint": "επίπεδο φόντου (0-100) στην μπάρα αναζήτησης$Nγια να κάνεις το buffering λιγότερο ενοχλητικό", + "mt_eq": "ενεργοποιεί τον ισοσταθμιστή και τον έλεγχο ενίσχυσης;$N$Nενίσχυση <code>0</code> = στάνταρ 100% ένταση (απαράλλαχτη)$N$Nεύρος <code>1  </code> = στάνταρ στερεοφωνικό (απαράλλαχτο)$Nεύρος <code>0.5</code> = 50% αριστερά-δεξιά μίξη ήχου$Nεύρος <code>0  </code> = μονοφωνικό$N$Nενίσχυση <code>-0.8</code> & εύρος <code>10</code> = αφαίρεση φωνής :^)$N$Nη ενεργοποίηση του ισοσταθμιστή κάνει τα άλμπουμ χωρίς κενά, να παίζουν χωρίς καθόλου κενά, οπότε άφησέ το ενεργό με όλες τις τιμές στο μηδέν (εκτός από εύρος = 1) αν σε νοιάζει", + "mt_drc": "ενεργοποιεί τον συμπιεστή δυναμικής εμβέλειας (εξομάλυνση έντασης / ακραία συμπίεση έντασης); θα ενεργοποιήσει και τον ισοσταθμιστή για να ισορροπήσει τον ήχο, οπότε βάλε όλα τα πεδία ισοσταθμιστή εκτός από το 'εύρος' στο 0 αν δεν το θες$N$Nχαμηλώνει την ένταση του ήχου πάνω από το όριο (THRESHOLD) dB; για κάθε RATIO dB πέρα από το όριο υπάρχει 1 dB εξόδου, οπότε οι προεπιλεγμένες τιμές όριο -24 και 'λόγος' 12 σημαίνουν ότι δεν θα ξεπεράσει ποτέ τα -22 dB και είναι ασφαλές να αυξήσεις την ενίσχυση ισοσταθμιστή σε 0.8, ή και 1.8 με ATK 0 και μεγάλο RLS όπως 90 (δουλεύει μόνο σε firefox· το RLS είναι max 1 σε άλλους browsers)$N$N(δες wikipedia, το εξηγούν καλύτερα)", + + "mb_play": "παίξε", + "mm_hashplay": "να παίξω αυτό το αρχείο ήχου;", + "mm_m3u": "πάτα <code>Enter/OK</code> για Αναπαραγωγή\nπάτα <code>ESC/Άκυρο</code> για Επεξεργασία", + "mp_breq": "χρειάζεται firefox 82+ ή chrome 73+ ή iOS 15+", + "mm_bload": "φορτώνει...", + "mm_bconv": "μετατροπή σε {0}, περίμενε...", + "mm_opusen": "ο browser σου δεν παίζει αρχεία aac / m4a;\nη μετατροπή σε opus είναι τώρα ενεργή", + "mm_playerr": "η αναπαραγωγή, απέτυχε: ", + "mm_eabrt": "Η προσπάθεια αναπαραγωγής ακυρώθηκε", + "mm_enet": "Η σύνδεση του ίντερνέτ σου είναι χάλια", + "mm_edec": "Το αρχείο αυτό είναι μάλλον κατεστραμμένο;;", + "mm_esupp": "Ο browser σου δεν καταλαβαίνει αυτή τη μορφή ήχου", + "mm_eunk": "Άγνωστο σφάλμα", + "mm_e404": "Αδύνατη η αναπαραγωγή ήχου; σφάλμα 404: Το αρχείο δεν βρέθηκε.", + "mm_e403": "Αδύνατη η αναπαραγωγή ήχου; σφάλμα 403: Άρνηση πρόσβασης.\n\nΔοκίμασε F5 για επαναφόρτωση, ίσως να έχεις αποσυνδεθεί", + "mm_e500": "Αδύνατη η αναπαραγωγή ήχου; σφάλμα 500: Έλεγξε τα logs του διακομιστή.", + "mm_e5xx": "Αδύνατη η αναπαραγωγή ήχου; σφάλμα διακομιστή", + "mm_nof": "δεν βρέθηκαν άλλα αρχεία ήχου τριγύρω", + "mm_prescan": "Αναζήτηση μουσικής για επόμενο τραγούδι...", + "mm_scank": "Βρέθηκε το επόμενο τραγούδι:", + "mm_uncache": "κρυφή μνήμη καθαρίστηκε· όλα τα τραγούδια θα ξανακατεβούν στην επόμενη αναπαραγωγή", + "mm_hnf": "το τραγούδι αυτό πλέον δεν υπάρχει", + + "im_hnf": "η εικόνα αυτή πλέον δεν υπάρχει", + + "f_empty": "αυτός ο φάκελος είναι άδειος", + "f_chide": "αυτό θα κρύψει τη στήλη «{0}»\n\nμπορείς να εμφανίσεις τις στήλες από τις ρυθμίσεις", + "f_bigtxt": "αυτό το αρχείο είναι {0} MiB σε μέγεθος — σίγουρα θέλεις να το δεις ως κείμενο;", + "f_bigtxt2": "να δεις μόνο το τέλος του αρχείου αντί για όλο; αυτό ενεργοποιεί και το following/tailing, που δείχνει νέες γραμμές που προστίθενται ζωντανά", + "fbd_more": '<div id="blazy">εμφανίζονται <code>{0}</code> από <code>{1}</code> αρχεία; <a href="#" id="bd_more">δείξε {2}</a> ή <a href="#" id="bd_all">δείξε τα όλα</a></div>', + "fbd_all": '<div id="blazy">εμφανίζονται <code>{0}</code> από <code>{1}</code> αρχεία; <a href="#" id="bd_all">δείξε όλα</a></div>', + "f_anota": "μόνο {0} από τα {1} αντικείμενα επιλέχθηκαν;\nγια να επιλέξεις ολόκληρο το φάκελο, κύλησε πρώτα μέχρι κάτω", + + "f_dls": 'οι σύνδεσμοι αρχείων στον τρέχοντα φάκελο έχουν\nμετατραπεί σε συνδέσμους λήψης', + + "f_partial": "Για να κατεβάσεις με ασφάλεια ένα αρχείο που ανεβαίνει, κλίκαρε το αρχείο με το ίδιο όνομα, αλλά χωρίς την κατάληξη <code>.PARTIAL</code>. Πάτα Άκυρο ή Escape για να σταματήσεις.\n\nΠάτα OK / Enter αν αγνοείς την προειδοποίηση και κατέβασε το <code>.PARTIAL</code> αρχείο, που σχεδόν σίγουρα θα είναι κατεστραμμένο.", + + "ft_paste": "επικόλλησε {0} αντικείμενα$NΠλήκτρο συντόμευσης: ctrl-V", + "fr_eperm": 'δεν μπορεί να μετονομαστεί:\nδεν έχεις δικαίωμα “μετακίνησης” σε αυτόν το φάκελο', + "fd_eperm": 'δεν μπορεί να διαγραφεί:\nδεν έχεις δικαίωμα “διαγραφής” σε αυτόν το φάκελο', + "fc_eperm": 'δεν μπορεί να κοπεί:\nδεν έχεις δικαίωμα “μετακίνησης” σε αυτόν το φάκελο', + "fp_eperm": 'δεν μπορεί να επικολληθεί:\nδεν έχεις δικαίωμα “εγγραφής” σε αυτόν το φάκελο', + "fr_emore": "επίλεξε τουλάχιστον ένα αντικείμενο για μετονομασία", + "fd_emore": "επίλεξε τουλάχιστον ένα αντικείμενο για διαγραφή", + "fc_emore": "επίλεξε τουλάχιστον ένα αντικείμενο για αποκοπή", + "fcp_emore": "επίλεξε τουλάχιστον ένα αντικείμενο για αντιγραφή στο πρόχειρο", + + "fs_sc": "μοιράσου το φάκελο που βρίσκεσαι", + "fs_ss": "μοιράσου τα επιλεγμένα αρχεία", + "fs_just1d": "δεν μπορείς να επιλέξεις περισσότερους από έναν φακέλους,\nή να αναμείξεις αρχεία και φακέλους στην ίδια επιλογή", + "fs_abrt": "❌ ακύρωση", + "fs_rand": "🎲 τυχαίο όνομα", + "fs_go": "✅ δημιούργησε κοινή χρήση", + "fs_name": "όνομα", + "fs_src": "πηγή", + "fs_pwd": "κωδικός", + "fs_exp": "λήξη", + "fs_tmin": "λεπτά", + "fs_thrs": "ώρες", + "fs_tdays": "ημέρες", + "fs_never": "αιώνιο", + "fs_pname": "προαιρετικό όνομα συνδέσμου; αν είναι κενό, θα είναι τυχαίο", + "fs_tsrc": "το αρχείο ή ο φάκελος προς κοινή χρήση", + "fs_ppwd": "προαιρετικός κωδικός", + "fs_w8": "δημιουργία κοινής χρήσης...", + "fs_ok": "πάτα <code>Enter/OK</code> για Πρόχειρο\nπάτα <code>ESC/Άκυρο</code> για Κλείσιμο", + + "frt_dec": "μπορεί να διορθώσει μερικές περιπτώσεις κατεστραμμένων ονομάτων αρχείων\">αποκωδικοποίηση url", + "frt_rst": "επανέφερε τα ονόματα αρχείων στα αρχικά τους\">↺ επαναφορά", + "frt_abrt": "ακύρωσε και κλείσε αυτό το παράθυρο\">❌ ακύρωση", + "frb_apply": "ΕΦΑΡΜΟΓΗ ΜΕΤΟΝΟΜΑΣΙΑΣ", + "fr_adv": "μαζική / μεταδεδομένα / μετονομασία με πρότυπα\">προχωρημένη", + "fr_case": "regex με διάκριση πεζών/κεφαλαίων\">case", + "fr_win": "ασφαλή ονόματα για windows; αντικαθιστά <code><>:"\\|?*</code> με ιαπωνικούς χαρακτήρες πλήρους πλάτους\">win", + "fr_slash": "αντικαθίσταται <code>/</code> με χαρακτήρα που δεν δημιουργεί νέους φακέλους\">όχι /", + "fr_re": "μοτίβα αναζήτησης (regex) για αναζήτηση στα αρχικά ονόματα; τα καταγραφόμενα groups μπορούν να χρησιμοποιηθούν στο πεδίο μορφοποίησης παρακάτω όπως <code>(1)</code> και <code>(2)</code> και ούτω καθεξής", + "fr_fmt": "εμπνευσμένο από foobar2000:$N<code>(title)</code> αντικαθίσταται από τίτλο τραγουδιού,$N<code>[(artist) - ](title)</code> παραλείπει το [this] αν το artist είναι κενό$N<code>$lpad((tn),2,0)</code> γεμίζει τον αριθμό κομματιού σε 2 ψηφία", + "fr_pdel": "διαγραφή", + "fr_pnew": "αποθήκευση ως", + "fr_pname": "δώσε όνομα για τη νέα προεπιλογή", + "fr_aborted": "ακυρώθηκε", + "fr_lold": "παλιό όνομα", + "fr_lnew": "νέο όνομα", + "fr_tags": "ετικέτες για τα επιλεγμένα αρχεία (μόνο για ανάγνωση):", + "fr_busy": "μετονομασία {0} αντικειμένων...\n\n{1}", + "fr_efail": "αποτυχία μετονομασίας:\n", + "fr_nchg": "{0} από τα νέα ονόματα άλλαξαν λόγω <code>win</code> και/ή <code>όχι /</code>\n\nΕίναι ΟΚ να συνεχίσουμε με αυτά τα ονόματα;", + + "fd_ok": "διαγραφή OK", + "fd_err": "αποτυχία διαγραφής:\n", + "fd_none": "δεν διαγράφηκε τίποτα; ίσως μπλοκαρισμένο από τις ρυθμίσεις του διακομιστή (xbd);", + "fd_busy": "διαγραφή {0} αντικειμένων...\n\n{1}", + "fd_warn1": "ΔΙΑΓΡΑΦΗ αυτών των {0} αντικειμένων;", + "fd_warn2": "<b>Τελευταία ευκαιρία!</b> Δεν υπάρχει αναίρεση. Διαγραφή;", + + "fc_ok": "αποκοπή {0} αντικειμένων", + "fc_warn": 'αποκοπή {0} αντικειμένων\n\nαλλά: μόνο <b>αυτή</b> η καρτέλα browser μπορεί να τα επικολλήσει\n(λόγω πολύ μεγάλης επιλογής)', + + "fcc_ok": "αντιγράφηκαν {0} αντικείμενα στο πρόχειρο", + "fcc_warn": "αντιγράφηκαν {0} αντικείμενα στο πρόχειρο\n\nαλλά: μόνο <b>αυτή</b> η καρτέλα browser μπορεί να τα επικολλήσει\n(λόγω πολύ μεγάλης επιλογής)", + + "fp_apply": "χρησιμοποίησε αυτά τα ονόματα", + "fp_ecut": "πρώτα κάνε αποκοπή ή αντιγραφή κάποιων αρχείων / φακέλων για επικόλληση / μετακίνηση\n\nσημείωση: μπορείς να αποκόπτεις / επικολλάς ανάμεσα σε διαφορετικές καρτέλες browser", + "fp_ename": "τα {0} αντικείμενα δεν μπορούν να μετακινηθούν εδώ γιατί τα ονόματα υπάρχουν ήδη. Δώσε νέα ονόματα παρακάτω για να συνεχίσεις, ή άφησε κενό για να τα αγνοήσεις:", + "fcp_ename": "τα {0} αντικείμενα δεν μπορούν να αντιγραφούν εδώ γιατί τα ονόματα υπάρχουν ήδη. Δώσε νέα ονόματα παρακάτω για να συνεχίσεις, ή άφησε κενό για να τα αγνοήσεις:", + "fp_emore": "υπάρχουν ακόμα συγκρούσεις ονομάτων που πρέπει να διορθωθούν", + "fp_ok": "μετακίνηση OK", + "fcp_ok": "αντιγραφή OK", + "fp_busy": "μετακίνηση {0} αντικειμένων...\n\n{1}", + "fcp_busy": "αντιγραφή {0} αντικειμένων...\n\n{1}", + "fp_err": "αποτυχία μετακίνησης:\n", + "fcp_err": "αποτυχία αντιγραφής:\n", + "fp_confirm": "να μετακινηθούν αυτά τα {0} αντικείμενα εδώ;", + "fcp_confirm": "να αντιγραφούν αυτά τα {0} αντικείμενα εδώ;", + "fp_etab": "αποτυχία ανάγνωσης πρόχειρου από άλλη καρτέλα browser", + "fp_name": "μεταφόρτωση αρχείου από τη συσκευή σου. Δώσε του όνομα:", + "fp_both_m": '<h6>διάλεξε τι θα επικολλήσεις</h6><code>Enter</code> = Μετακίνηση {0} αρχείων από «{1}»\n<code>ESC</code> = Μεταφόρτωση {2} αρχείων από τη συσκευή σου', + "fcp_both_m": '<h6>διάλεξε τι θα επικολλήσεις</h6><code>Enter</code> = Αντιγραφή {0} αρχείων από «{1}»\n<code>ESC</code> = Μεταφόρτωση {2} αρχείων από τη συσκευή σου', + "fp_both_b": '<a href="#" id="modal-ok">Μετακίνηση</a><a href="#" id="modal-ng">Μεταφόρτωση</a>', + "fcp_both_b": '<a href="#" id="modal-ok">Αντιγραφή</a><a href="#" id="modal-ng">Μεταφόρτωση</a>', + + "mk_noname": "γράψε ένα όνομα στο πεδίο κειμένου αριστερά πριν το κάνεις :p", + + "tv_load": "Φόρτωση αρχείου κειμένου:\n\n{0}\n\n{1}% ({2} από {3} MiB φορτωμένα)", + "tv_xe1": "αδυναμία φόρτωσης αρχείου κειμένου:\n\nσφάλμα ", + "tv_xe2": "404, αρχείο δεν βρέθηκε", + "tv_lst": "λίστα αρχείων κειμένου σε", + "tvt_close": "επιστροφή στην προβολή φακέλου$NΣυντόμευση: M (ή Esc)\">❌ κλείσιμο", + "tvt_dl": "κατέβασε αυτό το αρχείο$NΣυντόμευση: Y\">💾 λήψη", + "tvt_prev": "προβολή προηγούμενου εγγράφου$NΣυντόμευση: i\">⬆ προηγούμενο", + "tvt_next": "προβολή επόμενου εγγράφου$NΣυντόμευση: K\">⬇ επόμενο", + "tvt_sel": "επέλεξε αρχείο   (για αποκοπή / αντιγραφή / διαγραφή / ...)$NΣυντόμευση: S\">επιλογή", + "tvt_edit": "άνοιγμα αρχείου στον επεξεργαστή κειμένου$NΣυντόμευση: E\">✏️ επεξεργασία", + "tvt_tail": "παρακολούθηση αρχείου για αλλαγές; εμφάνιση νέων γραμμών σε πραγματικό χρόνο\">📡 παρακολούθηση", + "tvt_wrap": "αναδίπλωση λέξεων\">↵", + "tvt_atail": "κλείδωμα κύλισης στο κάτω μέρος\">⚓", + "tvt_ctail": "αποκωδικοποίηση χρωμάτων τερματικού (ansi escape codes)\">🌈", + "tvt_ntail": "όριο κύλισης (πόσα bytes κειμένου να κρατούνται φορτωμένα)", + + "m3u_add1": "το τραγούδι προστέθηκε στη λίστα m3u", + "m3u_addn": "προστέθηκαν {0} τραγούδια στη λίστα m3u", + "m3u_clip": "η λίστα m3u αντιγράφηκε στο πρόχειρο\n\nπρέπει να φτιάξεις ένα νέο αρχείο κειμένου με όνομα η_λίστα_μου.m3u και να επικολλήσεις τη λίστα μέσα· αυτό θα το καταστήσει αναπαράξιμο", + + "gt_vau": "μην δείχνεις το βίντεο, παίξε μόνο τον ήχο\">🎧", + "gt_msel": "ενεργοποίηση επιλογής αρχείων; ctrl-κλικ σε αρχείο για παράκαμψη$N$N<em>όταν είναι ενεργό: διπλό κλικ σε αρχείο / φάκελο το ανοίγει</em>$N$NΣυντόμευση: S\">πολλαπλή επιλογή", + "gt_crop": "κεντραρισμένη περικοπή μικρογραφιών\">περικοπή", + "gt_3x": "μικρογραφίες υψηλής ανάλυσης\">3x", + "gt_zoom": "ζουμ", + "gt_chop": "κόψε", + "gt_sort": "ταξινόμηση κατά", + "gt_name": "όνομα", + "gt_sz": "μέγεθος", + "gt_ts": "ημερομηνία", + "gt_ext": "τύπος", + "gt_c1": "μεγαλύτερη περικοπή ονομάτων αρχείων (δείξε λιγότερα)", + "gt_c2": "μικρότερη περικοπή ονομάτων αρχείων (δείξε περισσότερα)", + + "sm_w8": "αναζήτηση...", + "sm_prev": "τα παρακάτω αποτελέσματα αναζήτησης προέρχονται από προηγούμενη αναζήτηση:\n ", + "sl_close": "κλείσιμο αποτελεσμάτων αναζήτησης", + "sl_hits": "εμφανίζονται {0} αποτελέσματα", + "sl_moar": "φόρτωσε περισσότερα", + + "s_sz": "μέγεθος", + "s_dt": "ημερομηνία", + "s_rd": "μονοπάτι", + "s_fn": "όνομα", + "s_ta": "ετικέτες", + "s_ua": "ανέβηκε@", + "s_ad": "προχωρ.", + "s_s1": "ελάχιστο σε MiB", + "s_s2": "μέγιστο σε MiB", + "s_d1": "ελάχιστο iso8601", + "s_d2": "μέγιστο iso8601", + "s_u1": "μεταφορτώθηκε αργότερα", + "s_u2": "και/ή πριν", + "s_r1": "το μονοπάτι περιέχει   (χωρισμένα με κενό)", + "s_f1": "το όνομα περιέχει   (άρνηση με -nope)", + "s_t1": "οι ετικέτες περιέχουν   (^=αρχή, τέλος=$)", + "s_a1": "συγκεκριμένες ιδιότητες μεταδεδομένων", + + "md_eshow": "δεν μπορεί να εμφανιστεί ", + "md_off": "[📜<em>readme</em>] απενεργοποιημένο στο [⚙️] -- κρυμμένο έγγραφο", + + "badreply": "Αποτυχία ανάλυσης απάντησης από το διακομιστή", + + "xhr403": "403: Πρόσβαση αρνήθηκε\n\nδοκίμασε το F5, ίσως αποσυνδέθηκες", + "xhr0": "άγνωστο (πιθανόν αποσύνδεση από το διακομιστή ή ο διακομιστής είναι εκτός σύνδεσης)", + "cf_ok": "συγγνώμη γι' αυτό -- η προστασία DD" + wah + "oS ενεργοποιήθηκε\n\nοι διαδικασίες θα συνεχιστούν σε περίπου 30 δευτερόλεπτα\n\nαν δεν γίνει τίποτα, πάτα F5 για επαναφόρτωση", + "tl_xe1": "αδύνατη η λίστα υποφακέλων:\n\nσφάλμα ", + "tl_xe2": "404: Ο φάκελος δεν βρέθηκε", + "fl_xe1": "αδύνατη η λίστα αρχείων σε φάκελο:\n\nσφάλμα ", + "fl_xe2": "404: Ο φάκελος δεν βρέθηκε", + "fd_xe1": "αδύνατη η δημιουργία υποφακέλου:\n\nσφάλμα ", + "fd_xe2": "404: Ο γονικός φάκελος δεν βρέθηκε", + "fsm_xe1": "αδύνατη η αποστολή μηνύματος:\n\nσφάλμα ", + "fsm_xe2": "404: Ο γονικός φάκελος δεν βρέθηκε", + "fu_xe1": "αποτυχία φόρτωσης λίστας unpost από το διακομιστή:\n\nσφάλμα ", + "fu_xe2": "404: Το αρχείο δεν βρέθηκε??", + + "fz_tar": "μη συμπιεσμένο αρχείο gnu-tar (linux / mac)", + "fz_pax": "μη συμπιεσμένο pax-format tar (πιο αργό)", + "fz_targz": "gnu-tar με συμπίεση gzip επίπεδο 3$N$Nσυνήθως πολύ αργό, οπότε$Nχρησιμοποίησε καλύτερα μη συμπιεσμένο tar", + "fz_tarxz": "gnu-tar με συμπίεση xz επίπεδο 1$N$Nσυνήθως πολύ αργό, οπότε$Nχρησιμοποίησε καλύτερα μη συμπιεσμένο tar", + "fz_zip8": "zip με ονόματα αρχείων utf8 (ίσως να κολλάει σε windows 7 και παλιότερα)", + "fz_zipd": "zip με παραδοσιακά ονόματα cp437, για πολύ παλιό λογισμικό", + "fz_zipc": "cp437 με crc32 υπολογισμένο νωρίτερα,$Nγια MS-DOS PKZIP v2.04g (οκτώβριος 1993)$N(παίρνει παραπάνω χρόνο πριν ξεκινήσει η μεταφόρτωση)", + + "un_m1": "μπορείς να διαγράψεις τα πρόσφατα αρχεία που μεταφόρτωσες (ή να ακυρώσεις τα μισοτελειωμένα) παρακάτω", + "un_upd": "ανανέωση", + "un_m4": "ή μοιράσου τα αρχεία που βλέπεις παρακάτω:", + "un_ulist": "εμφάνιση", + "un_ucopy": "αντιγραφή", + "un_flt": "προαιρετικό φίλτρο:  η διεύθυνση πρέπει να περιέχει", + "un_fclr": "καθαρισμός φίλτρου", + "un_derr": "αποτυχία διαγραφής unpost:\n", + "un_f5": "κάτι χάλασε, δοκίμασε την ανανέωση ή πάτα F5", + "un_uf5": "συγγνώμη αλλά πρέπει να ανανεώσεις τη σελίδα (πχ με F5 ή CTRL-R) πριν ακυρώσεις αυτήν την αποστολή", + "un_nou": "<b>προσοχή:</b> ο διακομιστής είναι πολύ φορτωμένος για να δείξει μισοτελειωμένες αποστολές· πάτα την ανανέωση, σε λίγο", + "un_noc": "<b>προσοχή:</b> το unpost των ολοκληρωμένων αρχείων δεν επιτρέπεται από τη ρύθμιση του διακομιστή", + "un_max": "εμφανίζονται τα πρώτα 2000 αρχεία (χρησιμοποίησε φίλτρο)", + "un_avail": "μπορείς να διαγράψεις {0} πρόσφατα αρχεία<br />μπορείς να ακυρώσεις {1} μισοτελειωμένες αποστολές", + "un_m2": "ταξινομημένα κατά χρόνο μεταφόρτωσης; τα πιο πρόσφατα πρώτα:", + "un_no1": "άκυρο! καμία μεταφόρτωση δεν είναι αρκετά πρόσφατη", + "un_no2": "άκυρο! καμία μεταφόρτωση με αυτό το φίλτρο δεν είναι αρκετά πρόσφατη", + "un_next": "διάγραψε τα επόμενα {0} αρχεία παρακάτω", + "un_abrt": "άκυρο", + "un_del": "διαγραφή", + "un_m3": "φορτώνω τις πρόσφατες μεταφορτώσεις σου...", + "un_busy": "διαγράφω {0} αρχεία...", + "un_clip": "αντιγράφηκαν {0} σύνδεσμοι στο πρόχειρο", + + "u_https1": "πρέπει", + "u_https2": "μετάβαση σε https", + "u_https3": "για καλύτερη απόδοση", + "u_ancient": 'ο browser σου είναι εντυπωσιακά απαρχαιωμένος — ίσως να <a href="#" onclick="goto(\'bup\')">χρησιμοποιήσεις το bup αντί γι\' αυτό</a>', + "u_nowork": "χρειάζεται firefox 53+ ή chrome 57+ ή iOS 11+", + "tail_2old": "χρειάζεται firefox 105+ ή chrome 71+ ή iOS 14.5+", + "u_nodrop": "ο browser σου είναι πολύ παλιός για drag&drop μεταφορτώσεις", + "u_notdir": "αυτός δεν είναι φάκελος!\n\nο browser σου είναι πολύ παλιός,\nδοκίμασε drag&drop αντ' αυτού", + "u_uri": "για να κάνεις drag&drop εικόνων από άλλα παράθυρα browser,\nρίξ' τες πάνω στο μεγάλο κουμπί μεταφόρτωσης", + "u_enpot": 'άλλαξε στο <a href="#">potato UI</a> (ίσως ανεβάζει πιο γρήγορα)', + "u_depot": 'άλλαξε στο <a href="#">fancy UI</a> (ίσως ανεβάζει πιο αργά)', + "u_gotpot": "αλλάζω στο potato UI για πιο γρήγορη μεταφόρτωση,\n\nμπορείς να το αλλάξεις πάλι αν θες!", + "u_pott": "<p>αρχεία:   <b>{0}</b> ολοκληρωμένα,   <b>{1}</b> αποτυχημένα,   <b>{2}</b> σε εξέλιξη,   <b>{3}</b> σε ουρά</p>", + "u_ever": "αυτός είναι ο βασικός uploader; το up2k θέλει τουλάχιστον<br>chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'αυτός είναι ο βασικός uploader; το <a href="#" id="u2yea">up2k</a> είναι καλύτερο', + "u_uput": "βελτιστοποίηση για ταχύτητα (παράλειψη ελέγχου ακεραιότητας)", + "u_ewrite": "δεν έχεις δικαίωμα εγγραφής σε αυτόν τον φάκελο", + "u_eread": "δεν έχεις δικαίωμα ανάγνωσης σε αυτόν τον φάκελο", + "u_enoi": "η αναζήτηση αρχείων δεν είναι ενεργοποιημένη στο αρχείο ρυθμίσεων του διακομιστή", + "u_enoow": "δεν μπορείς να κάνεις αντικατάσταση εδώ· χρειάζεται δικαίωμα Διαγραφής", + "u_badf": "Αυτά τα {0} αρχεία (από {1} συνολικά) παραλείφθηκαν, πιθανώς λόγω δικαιωμάτων συστήματος αρχείων:\n\n", + "u_blankf": "Αυτά τα {0} αρχεία (από {1} συνολικά) είναι άδεια / κενά· να τα μεταφορτώσω έτσι κι αλλιώς;\n\n", + "u_applef": "Αυτά τα {0} αρχεία (από {1} συνολικά) πιθανώς δεν είναι επιθυμητά;\nΠάτα <code>OK/Enter</code> για ΝΑ ΑΓΝΟΗΘΟΥΝ τα παρακάτω αρχεία,\nΠάτα <code>Cancel/ESC</code> για ΝΑ ΜΗΝ ΑΠΟΚΛΕΙΣΤΟΥΝ και να ΜΕΤΑΦΟΡΤΩΘΟΎΝ κι αυτά:\n\n", + "u_just1": "\nΊσως δουλέψει καλύτερα αν επιλέξεις μόνο ένα αρχείο", + "u_ff_many": "αν χρησιμοποιείς <b>Linux / MacOS / Android,</b> τότε τόσα αρχεία <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>μπορεί</em> να κατάρρευση του Firefox!</a>\nαν γίνει αυτό, δοκίμασε ξανά (ή χρησιμοποίησε τον Chrome).", + "u_up_life": "Αυτή η μεταφόρτωση θα διαγραφεί από το διακομιστή\n{0} μετά την ολοκλήρωσή της", + "u_asku": "μεταφόρτωση αυτών των {0} αρχείων στο <code>{1}</code>", + "u_unpt": "μπορείς να αναιρέσεις / διαγράψεις αυτήν τη μεταφόρτωση χρησιμοποιώντας το 🧯, πάνω αριστερά", + "u_bigtab": "θα εμφανιστούν {0} αρχεία\n\nαυτό μπορεί να κάνει τον browser σου να κολλήσει, είσαι σίγουρος;", + "u_scan": "Σάρωση αρχείων...", + "u_dirstuck": "ο επεξεργαστής φακέλων κόλλησε προσπαθώντας να προσπελάσει τα εξής {0} αντικείμενα· θα τα παραλείψει:", + "u_etadone": "Ολοκληρώθηκε ({0}, {1} αρχεία)", + "u_etaprep": "(προετοιμασία για μεταφόρτωση)", + "u_hashdone": "το hashing ολοκληρώθηκε", + "u_hashing": "υπολογισμός hash", + "u_hs": "handshaking...", + "u_started": "τα αρχεία ανεβαίνουν τώρα· δες τα στο [🚀]", + "u_dupdefer": "διπλότυπο; θα επεξεργαστεί μετά από όλα τα άλλα αρχεία", + "u_actx": "πάτα αυτό το κείμενο για να μην χάσεις<br />απόδοση όταν αλλάζεις παράθυρα/καρτέλες", + "u_fixed": "ΟΚ!  Το διόρθωσα 👍", + "u_cuerr": "αποτυχία μεταφόρτωσης τμήματοςς {0} από {1};\nπιθανώς ακίνδυνο, συνεχίζω\n\nαρχείο: {2}", + "u_cuerr2": "ο διακομιστής απέρριψε τη μεταφόρτωση (τμήμα {0} από {1});\nθα ξαναδοκιμάσει αργότερα\n\nαρχείο: {2}\n\nσφάλμα ", + "u_ehstmp": "θα ξαναδοκιμάσει; δες κάτω δεξιά", + "u_ehsfin": "ο διακομιστής απέρριψε το αίτημα ολοκλήρωσης της μεταφόρτωσης; ξαναδοκιμάζει...", + "u_ehssrch": "ο διακομιστής απέρριψε το αίτημα αναζήτησης; ξαναδοκιμάζει...", + "u_ehsinit": "ο διακομιστής απέρριψε το αίτημα για εκκίνηση μεταφόρτωσης; ξαναδοκιμάζει...", + "u_eneths": "σφάλμα δικτύου κατά το handshake μεταφόρτωσης; ξαναδοκιμάζει...", + "u_enethd": "σφάλμα δικτύου κατά τον έλεγχο ύπαρξης στόχου; ξαναδοκιμάζει...", + "u_cbusy": "ο διακομιστής περιμένει να μας εμπιστευτεί ξανά μετά από πρόβλημα δικτύου...", + "u_ehsdf": "ο διακομιστής έμεινε από χώρο στο δίσκο!\n\nθα συνεχίσει να ξαναδοκιμάζει,\nσε περίπτωση που κάποιος\nελευθερώσει αρκετό χώρο για συνέχεια", + "u_emtleak1": "φαίνεται πως ο browser σου έχει διαρροή μνήμης;\nπαρακαλώ", + "u_emtleak2": ' <a href="{0}">αλλαγή σε https (συνιστάται)</a> ή ', + "u_emtleak3": ' ', + "u_emtleakc": 'δοκίμασε τα εξής:\n<ul><li>πάτα <code>F5</code> για ανανέωση σελίδας</li><li>μετά απενεργοποίησε το  <code>mt</code>  κουμπί στις  <code>⚙️ ρυθμίσεις</code></li><li>και δοκίμασε ξανά τη μεταφόρτωση</li></ul>Οι μεταφορτώσιες θα είναι λίγο πιο αργές, αλλά ok.\nΣυγγνώμη για την ταλαιπωρία!\n\nPS: το chrome v107 <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">έχει διόρθωση γι\' αυτό</a>', + "u_emtleakf": 'δοκίμασε τα εξής:\n<ul><li>πάτα <code>F5</code> για ανανέωση σελίδας</li><li>μετά άνοιξε το <code>🥔</code> (potato) στο UI μεταφόρτωσης<li>και δοκίμασε ξανά τη μεταφόρτωση</li></ul>\nPS: ο firefox <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">ελπίζει να φτιάξει αυτό το bug</a> κάποια στιγμή', + "u_s404": "δεν βρέθηκε στο διακομιστή", + "u_expl": "επεξήγηση", + "u_maxconn": "οι περισσότεροι browser το περιορίζουν στα 6, αλλά ο firefox σου επιτρέπει να το αυξήσεις με <code>connections-per-server</code> στο <code>about:config</code>", + "u_tu": '<p class="warn">ΠΡΟΕΙΔΟΠΟΙΗΣΗ: το turbo είναι ενεργοποιημένο, <span> το πρόγραμμα πελάτη ίσως να μην ανιχνεύσει και να μην ξαναεκκινήσει μισοτελειωμένες μεταφορτώσεις; δες τα tooltip του κουμπιού turbo</span></p>', + "u_ts": '<p class="warn">ΠΡΟΕΙΔΟΠΟΙΗΣΗ: το turbo είναι ενεργοποιημένο, <span> τα αποτελέσματα αναζήτησης μπορεί να είναι λάθος; δες τα tooltip του κουμπιού turbo</span></p>', + "u_turbo_c": "το turbo είναι απενεργοποιημένο στο αρχείο ρυθμίσεων του διακομιστή", + "u_turbo_g": "απενεργοποιώ το turbo επειδή δεν έχεις δικαίωμα\nγια τη λίστα φακέλων σε αυτόν τον τόμο", + "u_life_cfg": 'αυτόματη διαγραφή μετά από <input id="lifem" p="60" /> λεπτά (ή <input id="lifeh" p="3600" /> ώρες)', + "u_life_est": 'η μεταφόρτωση θα διαγραφεί <span id="lifew" tt="τοπική ώρα">---</span>', + "u_life_max": 'αυτός ο φάκελος επιβάλλει\nμέγιστη διάρκεια ζωής {0}', + "u_unp_ok": "επιτρέπεται το unpost για {0}", + "u_unp_ng": "δεν επιτρέπεται το unpost", + "ue_ro": "έχεις μόνο δικαίωμα ανάγνωσης σε αυτόν το φάκελο\n\n", + "ue_nl": "δεν είσαι συνδεδεμένος τώρα", + "ue_la": 'είσαι συνδεδεμένος ως "{0}"', + "ue_sr": "είσαι σε λειτουργία αναζήτησης αρχείων\n\nπήγαινε σε λειτουργία μεταφόρτωσης πατώντας το 🔎 (δίπλα στο μεγάλο κουμπί ΑΝΑΖΗΤΗΣΗΣ) και δοκίμασε πάλι\n\nσυγγνώμη", + "ue_ta": "δοκίμασε να μεταφορτώσεις εκ νέου, θα πρέπει να δουλέψει τώρα", + "ue_ab": "αυτό το αρχείο ανεβαίνει σε άλλο φάκελο και η μεταφόρτωση πρέπει να ολοκληρωθεί πριν ανέβει αλλού.\n\nΜπορείς να ακυρώσεις και να ξεχάσεις την αρχική μεταφόρτωση με το κουμπί 🧯 πάνω αριστερά", + "ur_1uo": "ΟΚ: Το αρχείο ανέβηκε επιτυχώς", + "ur_auo": "ΟΚ: Και τα {0} αρχεία ανέβηκαν επιτυχώς", + "ur_1so": "ΟΚ: Το αρχείο βρέθηκε στο διακομιστή", + "ur_aso": "ΟΚ: Και τα {0} αρχεία βρέθηκαν στο διακομιστή", + "ur_1un": "Η μεταφόρτωση απέτυχε, συγγνώμη", + "ur_aun": "Και οι {0} μεταφορτώσεις απέτυχαν, συγγνώμη", + "ur_1sn": "Το αρχείο ΔΕΝ βρέθηκε στο διακομιστή", + "ur_asn": "Τα {0} αρχεία ΔΕΝ βρέθηκαν στο διακομιστή", + "ur_um": "Ολοκληρώθηκε;\n{0} μεταφορτώσεις είναι OK,\n{1} μεταφορτώσεις απέτυχαν, συγγνώμη", + "ur_sm": "Ολοκληρώθηκε;\n{0} αρχεία βρέθηκαν στο διακομιστή,\n{1} αρχεία ΔΕΝ βρέθηκαν στο διακομιστή", + + "lang_set": "ανανέωση σελίδας για εφαρμογή της αλλαγής;" + }, "ita": { "tt": "Italiano", @@ -6298,7 +6927,7 @@ var Ls = { }, }; -var LANGS = ["eng", "nor", "chi", "deu", "fin", "ita", "nld", "rus", "spa", "ukr"]; +var LANGS = ["eng", "nor", "chi", "deu", "fin", "ita", "nld", "rus", "spa", "ukr", "grc"]; if (window.langmod) langmod(); From 392abd0675bfb13f6b74cb7918a39842bfe95475 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Wed, 6 Aug 2025 21:02:25 +0000 Subject: [PATCH 096/174] add greek splash tl; closes #493 --- copyparty/web/splash.js | 41 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 41 insertions(+) diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index d3d395af..14865781 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -177,6 +177,47 @@ var Ls = { "af1": "näytä viimeaikaiset lataukset", "ag1": "näytä tunnetut IdP-käyttäjät", }, + "grc": { + "a1": "ανανέωση", + "b1": "γεια σου ξένε!   <small>(δεν είσαι συνδεδεμένος)</small>", + "c1": "αποσύνδεση", + "d1": "σωρός απορριμμάτων", + "d2": "εμφανίζει την κατάσταση όλων των ενεργών διεργασιών", + "e1": "επαναφόρτωση του cfg", + "e2": "φορτώνει ξανά τα αρχεία ρυθμίσεων (λογαριασμοί/τόμοι/volflags),$Nκαι κάνει επανεξέταση όλων των τόμων e2ds$N$Nσημείωση: οποιαδήποτε αλλαγή στις καθολικές ρυθμίσεις$Nαπαιτεί πλήρη επανεκκίνηση για να εφαρμοστεί", + "f1": "μπορείς να περιηγηθείς:", + "g1": "μπορείς να εκτελέσεις μεταφόρτωση σε:", + "cc1": "άλλα πράγματα:", + "h1": "απενεργοποίση k304", + "i1": "ενεργοποίηση k304", + "j1": "η ενεργοποίηση του k304 θα αποσυνδέσει το πρόγραμμα πελάτη σου σε κάθε HTTP 304, κάτι που μπορεί να αποτρέψει κάποια προβληματικά proxies από το να κολλάνε (να μην φορτώνουν ξαφνικά σελίδες), <em>αλλά</em> θα κάνει τα πράγματα, γενικά πιο αργά", + "k1": "επαναφορά ρυθμίσεων στο πρόγραμμα πελάτη", + "l1": "συνδέσου για περισσότερα:", + "m1": "καλώς ήρθες,", + "n1": "404 δεν βρέθηκε  ┐( ´ -`)┌", + "o1": '´η μήπως δεν έχεις πρόσβαση -- δοκίμασε έναν κωδικό <a href="' + SR + '/?h">πήγαινε στην αρχική</a>', + "p1": "403 απαγορευμένο  ~┻━┻", + "q1": 'δοκίμασε έναν κωδικό <a href="' + SR + '/?h">πήγαινε στην αρχική</a>', + "r1": "πίσω στην αρχική", + ".s1": "επανάληψη σάρωσης", + "t1": "ενέργεια", + "u2": "χρόνος από την τελευταία εγγραφή του διακομιστή$N( μεταφόρτωση / μετονομασία / ... )$N$N17d = 17 days$N1ω23 = 1 ώρα 23 λεπτά$N4λ56 = 4 λεπτά 56 δευτερόλεπτα", + "v1": "σύνδεση", + "v2": "χρησιμοποίησε αυτόν το διακομιστή σαν τοπικό δίσκο", + "w1": "εναλλαγή σε https", + "x1": "αλλαγή κωδικού", + "y1": "επεξεργασία κοινόχρηστων φακέλων", + "z1": "ξεκλείδωμα αυτού του κοινόχρηστου φακέλου:", + "ta1": "συμπλήρωσε πρώτα το νέο σου κωδικό", + "ta2": "επανέλαβε για να επιβεβαιώσεις το νέο κωδικό:", + "ta3": "βρέθηκε τυπογραφικό λάθος· δοκίμασε ξανά", + "aa1": "εισερχόμενα αρχεία:", + "ab1": "απενεργοποίηση no304", + "ac1": "ενεργοποίηση no304", + "ad1": "η ενεργοποίηση του no304 θα απενεργοποιήσει όλη την προσωρινή αποθήκευση· δοκίμασέ το αν το k304 δεν ήταν αρκετό. Προσοχή, θα σπαταλήσει τεράστιο όγκο δικτυακής κίνησης!", + "ae1": "ενεργές μεταφορτώσεις:", + "af1": "προβολή πρόσφατων μεταφορτώσεων", + }, "ita": { "a1": "aggiorna", "b1": "ciao   <small>(non sei connesso)</small>", From 3b26884c691dc23f473e09107ad7f41a4a248fbe Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Wed, 6 Aug 2025 21:08:11 +0000 Subject: [PATCH 097/174] tl cleanup --- copyparty/web/browser.js | 2 +- copyparty/web/splash.js | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 60b4a349..9c20c5d8 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -6927,7 +6927,7 @@ var Ls = { }, }; -var LANGS = ["eng", "nor", "chi", "deu", "fin", "ita", "nld", "rus", "spa", "ukr", "grc"]; +var LANGS = ["eng", "nor", "chi", "deu", "fin", "grc", "ita", "nld", "rus", "spa", "ukr"]; if (window.langmod) langmod(); diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 14865781..1a24beb2 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -174,7 +174,7 @@ var Ls = { "ac1": "ota no304 käyttöön", "ad1": "no304:n lopettaa välimuistin käytön kokonaan; kokeile tätä jos k304 ei riittänyt. Tuhlaa valtavan määrän verkkoliikennettä!", "ae1": "lähtevät:", - "af1": "näytä viimeaikaiset lataukset", + "af1": "näytä viimeaikaiset lataukset", "ag1": "näytä tunnetut IdP-käyttäjät", }, "grc": { @@ -259,7 +259,7 @@ var Ls = { "ae1": "in uscita:", "af1": "mostra i file caricati di recente", "ag1": "mostra utenti IdP conosciuti" -}, + }, "nld": { "a1": "Update", "b1": "Hallo, hoe gaat het met jou?   <small>(Je bent niet ingelogd)</small>", From 29a4e54799564b5124d1a84e62a11dc71739954f Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Wed, 6 Aug 2025 21:10:15 +0000 Subject: [PATCH 098/174] tl cleanup --- copyparty/web/browser.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 9c20c5d8..ea4c3f1f 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -2439,7 +2439,7 @@ var Ls = { "u_https1": "für bessere Performance solltest du", "u_https2": "auf HTTPS wechseln", - "u_https3": "", + "u_https3": " ", "u_ancient": 'Dein Browser ist verdammt antik -- vielleicht solltest du <a href="#" onclick="goto(\'bup\')">stattdessen bup benutzen</a>', "u_nowork": "Benötigt Firefox 53+ oder Chrome 57+ oder iOS 11+", "tail_2old": "Benötigt Firefox 105+ oder Chrome 71+ oder iOS 14.5+", @@ -2489,7 +2489,7 @@ var Ls = { "u_ehsdf": "Server hat kein Speicherplatz mehr!\n\nwerde es erneut versuchen, falls jemand\ngenug Platz schafft um fortzufahren", "u_emtleak1": "scheint, als ob dein Browser ein Memory Leak hätte;\nbitte", "u_emtleak2": ' <a href="{0}">wechsle auf HTTPS (empfohlen)</a> oder ', - "u_emtleak3": '', + "u_emtleak3": ' ', "u_emtleakc": 'versuche folgendes:\n<ul><li>drücke <code>F5</code> um die Seite neu zu laden</li><li>deaktivere dann den  <code>mt</code>  Button in den  <code>⚙️ Einstellungen</code></li><li>und versuche den Upload nochmal.</li></ul>Uploads werden etwas langsamer sein, aber man kann ja nicht alles haben.\nSorry für die Umstände !\n\nPS: Chrome v107 <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">hat ein Bugfix</a> dafür', "u_emtleakf": 'versuche folgendes:\n<ul><li>drücke <code>F5</code> um die Seite neu zu laden</li><li>aktivere dann <code>🥔</code> (potato) im Upload UI<li>und versuche den Upload nochmal</li></ul>\nPS: Firefox <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">hat hoffentlich irgendwann ein Bugfix</a>', "u_s404": "nicht auf dem Server gefunden", @@ -2857,7 +2857,7 @@ var Ls = { "mm_uncache": "välimuisti tyhjennetty; kaikki kappaleet ladataan uudelleen seuraavalla toistolla", "mm_hnf": "tuota kappaletta ei enää ole olemassa", -" im_hnf": "tuota kuvaa ei enää ole olemassa", + "im_hnf": "tuota kuvaa ei enää ole olemassa", "f_empty": 'tämä hakemisto on tyhjä', "f_chide": 'tämä piilottaa sarakkeen «{0}»\n\nvoit palauttaa sarakkeet asetuksista', From a9ee4f24d50286079788fc4f203b486ff5871f19 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Wed, 6 Aug 2025 21:53:20 +0000 Subject: [PATCH 099/174] dropdowns for languages, themes, key-notation --- copyparty/web/browser.css | 5 +++ copyparty/web/browser.js | 73 +++++++++++++++++---------------------- 2 files changed, 37 insertions(+), 41 deletions(-) diff --git a/copyparty/web/browser.css b/copyparty/web/browser.css index 1f5a7da6..b5506f39 100644 --- a/copyparty/web/browser.css +++ b/copyparty/web/browser.css @@ -1374,6 +1374,7 @@ html.y #ops svg circle { #op_cfg input[type=text] { top: -.3em; } +.opview select, .opview input[type=text] { color: var(--fg); background: var(--txt-bg); @@ -1384,6 +1385,10 @@ html.y #ops svg circle { border-radius: .2em; padding: .2em .3em; } +.opview select { + padding: .3em; + margin: .2em .4em; +} .opview input.err { color: var(--err-fg); background: var(--err-bg); diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index ea4c3f1f..b84c39ee 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -7151,13 +7151,12 @@ ebi('op_cfg').innerHTML = ( '</div>\n' + '<div>\n' + ' <h3>' + L.cl_themes + '</h3>\n' + - ' <div id="themes">\n' + + ' <div><select id="themes"></select></div>\n' + ' </div>\n' + '</div>\n' + '<div>\n' + ' <h3>' + L.cl_langs + '</h3>\n' + - ' <div id="langs">\n' + - ' </div>\n' + + ' <div><select id="langs"></select></div>\n' + '</div>\n' + (have_zip ? ( '<div><h3>' + L.cl_ziptype + '</h3><div id="arc_fmt"></div></div>\n' @@ -7204,7 +7203,7 @@ ebi('op_cfg').innerHTML = ( ' </td>\n' + ' </div>\n' + '</div>\n' + - '<div><h3>' + L.cl_keytype + '</h3><div id="key_notation"></div></div>\n' + + '<div><h3>' + L.cl_keytype + '</h3><div><select id="key_notation"></select></div></div>\n' + '<div><h3>' + L.cl_hiddenc + '  ' + (MOBILE ? '<a href="#" id="hcolsh">' + L.cl_hidec + '</a> / ' : '') + '<a href="#" id="hcolsr">' + L.cl_reset + '</a></h3><div id="hcols"></div></div>' ); @@ -14109,25 +14108,21 @@ var mukey = (function () { defnot = 'rekobo_alnum'; var map = {}, - html = []; + html = [], + cb = ebi('key_notation'); for (var k in maps) { if (!maps.hasOwnProperty(k)) continue; - html.push( - '<span><input type="radio" name="keytype" value="' + k + '" id="key_' + k + '">' + - '<label for="key_' + k + '">' + k + '</label></span>'); - + html.push('<option value="{0}">{0}</option>'.format(k)); for (var a = 0; a < 24; a++) maps[k][a] = maps[k][a].trim(); } - ebi('key_notation').innerHTML = html.join('\n'); + cb.innerHTML = html.join(''); - function set_key_notation(e) { - ev(e); - var notation = this.getAttribute('value'); - load_notation(notation); + function set_key_notation() { + load_notation(cb.value); try_render(); } @@ -14184,14 +14179,10 @@ var mukey = (function () { if (!maps[notation]) notation = defnot; - ebi('key_' + notation).checked = true; + cb.value = notation; + cb.onchange = set_key_notation; load_notation(notation); - var o = QSA('#key_notation input'); - for (var a = 0; a < o.length; a++) { - o[a].onchange = set_key_notation; - } - return { "render": try_render }; @@ -14230,17 +14221,17 @@ var settheme = (function () { showfile.setstyle(); bchrome(); - var html = [], itheme = ax.indexOf(theme[0]) * 2 + (light ? 1 : 0), + var html = [], + cb = ebi('themes'), + itheme = ax.indexOf(theme[0]) * 2 + (light ? 1 : 0), names = ['classic dark', 'classic light', 'pm-monokai', 'flat light', 'vice', 'hotdog stand', 'hacker', 'hi-con']; for (var a = 0; a < themes; a++) - html.push('<a href="#" class="btn tgl' + (a == itheme ? ' on' : '') + - '" tt="' + (names[a] || 'custom') + '">' + a + '</a>'); + html.push('<option value="{0}">{0} ┃ {1}</option>'.format(a, names[a] || 'custom')); ebi('themes').innerHTML = html.join(''); - var btns = QSA('#themes a'); - for (var a = 0; a < themes; a++) - btns[a].onclick = r.go; + cb.value = itheme; + cb.onchange = r.onsel; if (chldr) { var x = r.ldr[itheme] || [tre]; @@ -14249,12 +14240,13 @@ var settheme = (function () { } bcfg_set('light', light); - tt.att(ebi('themes')); } - r.go = function (e) { - var i = e; - try { ev(e); i = e.target.textContent; } catch (ex) { } + r.onsel = function () { + r.go(parseInt(ebi('themes').value)); + }; + + r.go = function (i) { light = i % 2 == 1; var c = ax[Math.floor(i / 2)], l = light ? 'y' : 'z'; @@ -14262,7 +14254,7 @@ var settheme = (function () { themen = c + l; swrite('cpp_thm', theme); freshen(); - } + }; freshen(); return r; @@ -14272,26 +14264,25 @@ var settheme = (function () { (function () { function freshen() { lang = sread("cpp_lang", LANGS) || lang; - var k, html = []; + var k, cb = ebi('langs'), html = []; for (var a = 0; a < LANGS.length; a++) { k = LANGS[a]; - html.push('<a href="#" class="btn tgl' + (k == lang ? ' on' : '') + - '" tt="' + Ls[k].tt + '">' + k + '</a>'); + html.push('<option value="{0}">{0} ┃ {1}</option>'.format(k, Ls[k].tt)); } - ebi('langs').innerHTML = html.join(''); - var btns = QSA('#langs a'); - for (var a = 0, aa = btns.length; a < aa; a++) - btns[a].onclick = setlang; + cb.innerHTML = html.join(''); + cb.onchange = setlang; + cb.value = lang; } function setlang(e) { ev(e); var t = L.lang_set; - L = Ls[this.textContent]; - swrite("cpp_lang", this.textContent); + lang = ebi('langs').value; + L = Ls[lang]; + swrite("cpp_lang", lang); freshen(); modal.confirm(L.lang_set + "\n\n" + t, location.reload.bind(location), null); - }; + } freshen(); })(); From 9b9d2a92ca128289b9e6943ad24e3d6f5f3df5ce Mon Sep 17 00:00:00 2001 From: mati1210 <49544174+mati1210@users.noreply.github.com> Date: Thu, 7 Aug 2025 12:00:53 -0300 Subject: [PATCH 100/174] support systemd socket activation (fd passing) (#515) * add support for socket passing * slight tweaks before merge --------- Signed-off-by: mat <matheuz1210@gmail.com> Signed-off-by: ed <s@ocv.me> Co-authored-by: ed <s@ocv.me> --- copyparty/__main__.py | 6 ++++-- copyparty/ftpd.py | 2 +- copyparty/svchub.py | 4 ++-- copyparty/tcpsrv.py | 21 ++++++++++++++++----- copyparty/tftpd.py | 2 +- 5 files changed, 24 insertions(+), 11 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 3f07bfc1..86b6b031 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -536,7 +536,7 @@ def get_sects(): dedent( """ \033[33m-i\033[0m takes a comma-separated list of interfaces to listen on; - IP-addresses and/or unix-sockets (Unix Domain Sockets) + IP-addresses, unix-sockets and/or open file descriptors the default (\033[32m-i ::\033[0m) means all IPv4 and IPv6 addresses @@ -562,7 +562,9 @@ def get_sects(): \033[32m-i unix:\033[33m/dev/shm/party.sock\033[0m keeps umask-defined permission (usually \033[33m0600\033[0m) and the same user/group as copyparty - \033[33m-p\033[0m (tcp ports) is ignored for unix sockets + \033[32m-i fd:\033[33m3\033[0m uses the socket passed to copyparty on file descriptor 3 + + \033[33m-p\033[0m (tcp ports) is ignored for non ip-addresses """ ), ], diff --git a/copyparty/ftpd.py b/copyparty/ftpd.py index 2f45c3f4..bace7d28 100644 --- a/copyparty/ftpd.py +++ b/copyparty/ftpd.py @@ -607,7 +607,7 @@ class Ftpd(object): if "::" in ips: ips.append("0.0.0.0") - ips = [x for x in ips if "unix:" not in x] + ips = [x for x in ips if not x.startswith(("unix:", "fd:"))] if self.args.ftp4: ips = [x for x in ips if ":" not in x] diff --git a/copyparty/svchub.py b/copyparty/svchub.py index bd7562e4..9a00f890 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -868,7 +868,7 @@ class SvcHub(object): have_tcp = False for zs in al.i: - if not zs.startswith("unix:"): + if not zs.startswith(("unix:", "fd:")): have_tcp = True if not have_tcp: zb = False @@ -878,7 +878,7 @@ class SvcHub(object): setattr(al, zs, False) zb = True if zb: - t = "only listening on unix-sockets; cannot enable zeroconf/mdns/ssdp as requested" + t = "no ip addresses provided; cannot enable zeroconf/mdns/ssdp as requested" self.log("root", t, 3) if not self.args.no_dav: diff --git a/copyparty/tcpsrv.py b/copyparty/tcpsrv.py index 53388c5e..6c61fe61 100644 --- a/copyparty/tcpsrv.py +++ b/copyparty/tcpsrv.py @@ -25,8 +25,8 @@ from .util import ( termsize, ) -if True: - from typing import Generator, Union +if True: # pylint: disable=using-constant-test + from typing import Generator, Optional, Union if TYPE_CHECKING: from .svchub import SvcHub @@ -245,8 +245,10 @@ class TcpSrv(object): def _listen(self, ip: str, port: int) -> None: uds_perm = uds_gid = -1 + bound: Optional[socket.socket] = None + tcp = False + if "unix:" in ip: - tcp = False ipv = socket.AF_UNIX uds = ip.split(":") ip = uds[-1] @@ -259,7 +261,12 @@ class TcpSrv(object): import grp uds_gid = grp.getgrnam(uds[2]).gr_gid + elif "fd:" in ip: + fd = ip[3:] + bound = socket.socket(fileno=int(fd)) + tcp = bound.proto == socket.IPPROTO_TCP + ipv = bound.family elif ":" in ip: tcp = True ipv = socket.AF_INET6 @@ -267,7 +274,7 @@ class TcpSrv(object): tcp = True ipv = socket.AF_INET - srv = socket.socket(ipv, socket.SOCK_STREAM) + srv = bound or socket.socket(ipv, socket.SOCK_STREAM) if not ANYWIN or self.args.reuseaddr: srv.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) @@ -285,6 +292,10 @@ class TcpSrv(object): if getattr(self.args, "freebind", False): srv.setsockopt(socket.SOL_IP, socket.IP_FREEBIND, 1) + if bound: + self.srv.append(srv) + return + try: if tcp: srv.bind((ip, port)) @@ -437,7 +448,7 @@ class TcpSrv(object): def detect_interfaces(self, listen_ips: list[str]) -> dict[str, Netdev]: from .stolen.ifaddr import get_adapters - listen_ips = [x for x in listen_ips if "unix:" not in x] + listen_ips = [x for x in listen_ips if not x.startswith(("unix:", "fd:"))] nics = get_adapters(True) eps: dict[str, Netdev] = {} diff --git a/copyparty/tftpd.py b/copyparty/tftpd.py index 6f5726b3..4ef01103 100644 --- a/copyparty/tftpd.py +++ b/copyparty/tftpd.py @@ -179,7 +179,7 @@ class Tftpd(object): if "::" in ips: ips.append("0.0.0.0") - ips = [x for x in ips if "unix:" not in x] + ips = [x for x in ips if not x.startswith(("unix:", "fd:"))] if self.args.tftp4: ips = [x for x in ips if ":" not in x] From 54caf63f6aab9198d0e2038da38b2d7b0622bffc Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 15:18:59 +0000 Subject: [PATCH 101/174] optimize --- copyparty/__main__.py | 4 ++-- copyparty/httpcli.py | 13 +++++-------- copyparty/multicast.py | 6 +----- copyparty/svchub.py | 2 +- 4 files changed, 9 insertions(+), 16 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 86b6b031..eb44ef4f 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -536,7 +536,7 @@ def get_sects(): dedent( """ \033[33m-i\033[0m takes a comma-separated list of interfaces to listen on; - IP-addresses, unix-sockets and/or open file descriptors + IP-addresses, unix-sockets, and/or open file descriptors the default (\033[32m-i ::\033[0m) means all IPv4 and IPv6 addresses @@ -564,7 +564,7 @@ def get_sects(): \033[32m-i fd:\033[33m3\033[0m uses the socket passed to copyparty on file descriptor 3 - \033[33m-p\033[0m (tcp ports) is ignored for non ip-addresses + \033[33m-p\033[0m (tcp ports) is ignored for unix-sockets and FDs """ ), ], diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index b335fe4a..c982a812 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -262,7 +262,8 @@ class HttpCli(object): def _assert_safe_rem(self, rem: str) -> None: # sanity check to prevent any disasters - if rem.startswith("/") or rem.startswith("../") or "/../" in rem: + # (this function hopefully serves no purpose; validation has already happened at this point, this only exists as a last-ditch effort just in case) + if rem.startswith(("/", "../")) or "/../" in rem: raise Exception("that was close") def _gen_fk(self, alg: int, salt: str, fspath: str, fsize: int, inode: int) -> str: @@ -5031,7 +5032,7 @@ class HttpCli(object): wvol = [x for x in wvol if "unlistcw" not in allvols[x[1:-1]].flags] fmt = self.uparam.get("ls", "") - if not fmt and (self.ua.startswith("curl/") or self.ua.startswith("fetch")): + if not fmt and self.ua.startswith(("curl/", "fetch")): fmt = "v" if fmt in ["v", "t", "txt"]: @@ -5141,7 +5142,7 @@ class HttpCli(object): t = '<h1 id="n">404 not found  ┐( ´ -`)┌</h1><p><a id="r" href="{}/?h">go home</a></p>' pt = "404 not found ┐( ´ -`)┌" - if self.ua.startswith("curl/") or self.ua.startswith("fetch"): + if self.ua.startswith(("curl/", "fetch")): pt = "# acct: %s\n%s\n" % (self.uname, pt) self.reply(pt.encode("utf-8"), status=rc) return True @@ -6267,11 +6268,7 @@ class HttpCli(object): is_ls = "ls" in self.uparam is_js = self.args.force_js or self.cookies.get("js") == "y" - if ( - not is_ls - and not add_og - and (self.ua.startswith("curl/") or self.ua.startswith("fetch")) - ): + if not is_ls and not add_og and self.ua.startswith(("curl/", "fetch")): self.uparam["ls"] = "v" is_ls = True diff --git a/copyparty/multicast.py b/copyparty/multicast.py index 6d6507f8..766dc15d 100644 --- a/copyparty/multicast.py +++ b/copyparty/multicast.py @@ -183,11 +183,7 @@ class MCast(object): srv.ips[oth_ip.split("/")[0]] = ipaddress.ip_network(oth_ip, False) # gvfs breaks if a linklocal ip appears in a dns reply - ll = { - k: v - for k, v in srv.ips.items() - if k.startswith("169.254") or k.startswith("fe80") - } + ll = {k: v for k, v in srv.ips.items() if k.startswith(("169.254", "fe80"))} rt = {k: v for k, v in srv.ips.items() if k not in ll} if self.args.ll or not rt: diff --git a/copyparty/svchub.py b/copyparty/svchub.py index 9a00f890..07d85d83 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -878,7 +878,7 @@ class SvcHub(object): setattr(al, zs, False) zb = True if zb: - t = "no ip addresses provided; cannot enable zeroconf/mdns/ssdp as requested" + t = "not listening on any ip-addresses (only unix-sockets and/or FDs); cannot enable zeroconf/mdns/ssdp as requested" self.log("root", t, 3) if not self.args.no_dav: From 8a0746c6af6d72cac2c8f10dfa383c76b93562f8 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 15:24:15 +0000 Subject: [PATCH 102/174] fix viewing .MD files --- copyparty/httpcli.py | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index c982a812..79003bcc 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -3584,7 +3584,7 @@ class HttpCli(object): rem = "{}/{}".format(rp, fn).strip("/") dbv, vrem = vfs.get_dbv(rem) - if not rem.endswith(".md") and not self.can_delete: + if not rem.lower().endswith(".md") and not self.can_delete: raise Pebkac(400, "only markdown pls") if nullwrite: @@ -6196,7 +6196,8 @@ class HttpCli(object): if not use_filekey: return self.tx_404(True) - if add_og and not abspath.lower().endswith(".md"): + is_md = abspath.lower().endswith(".md") + if add_og and not is_md: if og_ua or self.host not in self.headers.get("referer", ""): self.vpath, og_fn = vsplit(self.vpath) vpath = self.vpath @@ -6208,10 +6209,10 @@ class HttpCli(object): vpnodes.pop() if ( - (abspath.endswith(".md") or self.can_delete) + (is_md or self.can_delete) and "nohtml" not in vn.flags and ( - ("v" in self.uparam and abspath.endswith(".md")) + (is_md and "v" in self.uparam) or "edit" in self.uparam or "edit2" in self.uparam ) From 4a0435681449ada06c87c9cb7cff37b92d4d3bb4 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 15:57:51 +0000 Subject: [PATCH 103/174] fix non-e2d dupe finalizing; when the up2k database is not enabled, only the 38400 most recent uploads are kept in memory serverside the webui did not anticipate this, expecting the server to finalize all dupes with just a single pass of brief handshakes fix this by doing as many passes as necessary, only stopping if a pass does not make any progress (filesystem-issues or some such) --- copyparty/web/up2k.js | 36 ++++++++++++++++++++++++++++-------- 1 file changed, 28 insertions(+), 8 deletions(-) diff --git a/copyparty/web/up2k.js b/copyparty/web/up2k.js index 1e49fc4e..2743ea36 100644 --- a/copyparty/web/up2k.js +++ b/copyparty/web/up2k.js @@ -964,6 +964,7 @@ function up2k_init(subtle) { "t": 0 }, "car": 0, + "nre": 0, "slow_io": null, "oserr": false, "modn": 0, @@ -1783,8 +1784,7 @@ function up2k_init(subtle) { } var tasker = (function () { - var running = false, - was_busy = false; + var running = false; var defer = function () { running = false; @@ -1801,7 +1801,17 @@ function up2k_init(subtle) { while (true) { var now = Date.now(), blocktime = now - r.tact, - is_busy = st.car < st.files.length; + was_busy = st.is_busy, + is_busy = !!( // gzip take the wheel + st.car < st.files.length || + st.busy.hash.length || + st.todo.hash.length || + st.busy.handshake.length || + st.todo.handshake.length || + st.busy.upload.length || + st.todo.upload.length || + st.busy.head.length || + st.todo.head.length); if (blocktime > 2500) console.log('main thread blocked for ' + blocktime); @@ -1809,7 +1819,16 @@ function up2k_init(subtle) { r.tact = now; if (was_busy && !is_busy) { - for (var a = 0; a < st.files.length; a++) { + var nre = 0, nf = 0; + for (var a = 0; a < st.files.length; a++) + if (st.files[a].want_recheck) + nre++; + console.log('nre', nre, 'st', st.nre); + if (st.nre != nre) { + st.nre = nre; + nf = st.files.length; + } + for (var a = 0; a < nf; a++) { var t = st.files[a]; if (t.want_recheck) { t.rechecks++; @@ -1817,7 +1836,7 @@ function up2k_init(subtle) { push_t(st.todo.handshake, t); } } - is_busy = st.todo.handshake.length; + is_busy = !!st.todo.handshake.length; try { if (!is_busy && !uc.fsearch && !msel.getsel().length && (!mp.au || mp.au.paused)) treectl.goto(); @@ -1826,7 +1845,7 @@ function up2k_init(subtle) { } if (was_busy != is_busy) { - st.is_busy = was_busy = is_busy; + st.is_busy = is_busy; window[(is_busy ? "add" : "remove") + "EventListener"]("beforeunload", warn_uploader_busy); @@ -1947,7 +1966,7 @@ function up2k_init(subtle) { for (var a = 0; a < st.files.length; a++) { var t = st.files[a]; - if (t.want_recheck && !t.rechecks) + if (t.want_recheck && t.rechecks < 999) return; } @@ -2693,8 +2712,9 @@ function up2k_init(subtle) { if (ofs !== -1) { err = err.slice(0, ofs + 1) + linksplit(err.slice(ofs + 2).trimEnd()).join(' / '); } - if (!t.rechecks && (err_pend || err_srcb)) { + if (!t.rechecks) t.rechecks = 0; + if (t.rechecks < 999 && (err_pend || err_srcb)) { t.want_recheck = true; if (st.busy.upload.length || st.busy.handshake.length || st.bytes.uploaded) { err = L.u_dupdefer; From 3f8cb7e877e9ef8dffd1cdf910c36ef1ac5aad34 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 17:00:42 +0000 Subject: [PATCH 104/174] xff: require explicit configuration of --rproxy; try to avoid dangerous misconfiguration of how to determine the client's IP by more aggressively asking for the correct config; if the --xff-hdr (default: x-forwarded-for) appears in a request then it will now be ignored unless --rproxy says which IP to use --- README.md | 8 ++++++-- copyparty/__main__.py | 2 +- copyparty/httpcli.py | 17 ++++++++++++++--- 3 files changed, 21 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 9746581b..2afc4a81 100644 --- a/README.md +++ b/README.md @@ -2058,7 +2058,11 @@ you can either: * or do location-based proxying, using `--rp-loc=/stuff` to tell copyparty where it is mounted -- has a slight performance cost and higher chance of bugs * if copyparty says `incorrect --rp-loc or webserver config; expected vpath starting with [...]` it's likely because the webserver is stripping away the proxy location from the request URLs -- see the `ProxyPass` in the apache example below -when running behind a reverse-proxy (this includes services like cloudflare), it is important to configure real-ip correctly, as many features rely on knowing the client's IP. Look out for red and yellow log messages which explain how to do this. But basically, set `--xff-hdr` to the name of the http header to read the IP from (usually `x-forwarded-for`, but cloudflare uses `cf-connecting-ip`), and then `--xff-src` to the IP of the reverse-proxy so copyparty will trust the xff-hdr. Note that `--rp-loc` in particular will not work at all unless you do this +when running behind a reverse-proxy (this includes services like cloudflare), it is important to configure real-ip correctly, as many features rely on knowing the client's IP. The best/safest approach is to configure your reverse-proxy so it gives copyparty a header which only contains the client's true/real IP-address, and then setting `--xff-hdr theHeaderName --rproxy 1` but alternatively, if you want/need to let copyparty handle this, look out for red and yellow log messages which explain how to do that. Basically, the log will say this: + +> set `--xff-hdr` to the name of the http-header to read the IP from (usually `x-forwarded-for`, but cloudflare uses `cf-connecting-ip`), and then `--xff-src` to the IP of the reverse-proxy so copyparty will trust the xff-hdr. You will also need to configure `--rproxy` to `1` if the header only contains one IP (the correct one) or to a *negative value* if it contains multiple; `-1` being the rightmost and most trusted IP (the nearest proxy, so usually not the correct one), `-2` being the second-closest hop, and so on + +Note that `--rp-loc` in particular will not work at all unless you configure the above correctly some reverse proxies (such as [Caddy](https://caddyserver.com/)) can automatically obtain a valid https/tls certificate for you, and some support HTTP/2 and QUIC which *could* be a nice speed boost, depending on a lot of factors * **warning:** nginx-QUIC (HTTP/3) is still experimental and can make uploads much slower, so HTTP/1.1 is recommended for now @@ -2615,7 +2619,7 @@ there is a [discord server](https://discord.gg/25J8CdTT6G) with an `@everyone` some notes on hardening -* set `--rproxy 0` if your copyparty is directly facing the internet (not through a reverse-proxy) +* set `--rproxy 0` *if and only if* your copyparty is directly facing the internet (not through a reverse-proxy) * cors doesn't work right otherwise * if you allow anonymous uploads or otherwise don't trust the contents of a volume, you can prevent XSS with volflag `nohtml` * this returns html documents as plaintext, and also disables markdown rendering diff --git a/copyparty/__main__.py b/copyparty/__main__.py index eb44ef4f..81476e7f 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1091,7 +1091,7 @@ def add_network(ap): ap2.add_argument("-i", metavar="IP", type=u, default="::", help="IPs and/or unix-sockets to listen on (comma-separated list; see \033[33m--help-bind\033[0m). Default: all IPv4 and IPv6") ap2.add_argument("-p", metavar="PORT", type=u, default="3923", help="ports to listen on (comma/range); ignored for unix-sockets") ap2.add_argument("--ll", action="store_true", help="include link-local IPv4/IPv6 in mDNS replies, even if the NIC has routable IPs (breaks some mDNS clients)") - ap2.add_argument("--rproxy", metavar="DEPTH", type=int, default=1, help="which ip to associate clients with; [\033[32m0\033[0m]=tcp, [\033[32m1\033[0m]=origin (first x-fwd, unsafe), [\033[32m2\033[0m]=outermost-proxy, [\033[32m3\033[0m]=second-proxy, [\033[32m-1\033[0m]=closest-proxy") + ap2.add_argument("--rproxy", metavar="DEPTH", type=int, default=9999999, help="which ip to associate clients with; [\033[32m0\033[0m]=tcp, [\033[32m1\033[0m]=origin (first x-fwd, unsafe), [\033[32m-1\033[0m]=closest-proxy, [\033[32m-2\033[0m]=second-hop, [\033[32m-3\033[0m]=third-hop") ap2.add_argument("--xff-hdr", metavar="NAME", type=u, default="x-forwarded-for", help="if reverse-proxied, which http header to read the client's real ip from") ap2.add_argument("--xff-src", metavar="CIDR", type=u, default="127.0.0.0/8, ::1/128", help="list of trusted reverse-proxy CIDRs (comma-separated); only accept the real-ip header (\033[33m--xff-hdr\033[0m) and IdP headers if the incoming connection is from an IP within either of these subnets. Specify [\033[32mlan\033[0m] to allow all LAN / private / non-internet IPs. Can be disabled with [\033[32many\033[0m] if you are behind cloudflare (or similar) and are using \033[32m--xff-hdr=cf-connecting-ip\033[0m (or similar)") ap2.add_argument("--ipa", metavar="CIDR", type=u, default="", help="only accept connections from IP-addresses inside \033[33mCIDR\033[0m (comma-separated); examples: [\033[32mlan\033[0m] or [\033[32m10.89.0.0/16, 192.168.33.0/24\033[0m]") diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 79003bcc..2986275f 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -384,9 +384,20 @@ class HttpCli(object): try: cli_ip = zsl[n].strip() except: - cli_ip = zsl[0].strip() - t = "rproxy={} oob x-fwd {}" - self.log(t.format(self.args.rproxy, zso), c=3) + cli_ip = self.ip + self.bad_xff = True + if self.args.rproxy != 9999999: + t = "global-option --rproxy %d could not be used (out-of-bounds) for the received header [%s]" + self.log(t % (self.args.rproxy, zso), c=3) + else: + zsl = [ + " rproxy: %d if this client's IP-address is [%s]" + % (-1 - zd, zs.strip()) + for zd, zs in enumerate(zsl) + ] + t = 'could not determine the client\'s IP-address because the global-option --rproxy has not been configured, so the request-header [%s] specified by global-option --xff-hdr cannot be used safely! Please see the "reverse-proxy" section in the readme. The best approach is to configure your reverse-proxy to give copyparty the exact IP-address to assume (perhaps in another header), but you may also try the following:' + t = t % (self.args.xff_hdr,) + self.log("%s\n\n%s\n" % (t, "\n".join(zsl)), 3) pip = self.conn.addr[0] xffs = self.conn.xff_nm From 9e980bb55203c2a73a0a9efc7792c7e0fd379ab2 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 17:57:10 +0000 Subject: [PATCH 105/174] try to detect proxies with misbehaving caches (#488) --- copyparty/__main__.py | 1 + copyparty/httpcli.py | 6 ++++++ copyparty/web/browser.js | 4 ++++ tests/util.py | 4 ++-- 4 files changed, 13 insertions(+), 2 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 81476e7f..0715af28 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1300,6 +1300,7 @@ def add_yolo(ap): ap2 = ap.add_argument_group("yolo options") ap2.add_argument("--allow-csrf", action="store_true", help="disable csrf protections; let other domains/sites impersonate you through cross-site requests") ap2.add_argument("--cookie-lax", action="store_true", help="allow cookies from other domains (if you follow a link from another website into your server, you will arrive logged-in); this reduces protection against CSRF") + ap2.add_argument("--no-fnugg", action="store_true", help="disable the smoketest for caching-related issues in the web-UI") ap2.add_argument("--getmod", action="store_true", help="permit ?move=[...] and ?delete as GET") ap2.add_argument("--wo-up-readme", action="store_true", help="allow users with write-only access to upload logues and readmes without adding the _wo_ filename prefix (volflag=wo_up_readme)") diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 2986275f..3da299c7 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -6025,6 +6025,12 @@ class HttpCli(object): else: [x.pop(k) for k in ["name", "dt"] for y in [dirs, files] for x in y] + # nonce (tlnote: norwegian for flake as in snowflake) + if self.args.no_fnugg: + ls["fnugg"] = "nei" + elif "fnugg" in self.headers: + ls["fnugg"] = self.headers["fnugg"] + ret = json.dumps(ls) mime = "application/json" diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index b84c39ee..0563d8ee 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -13204,6 +13204,7 @@ var treectl = (function () { xhr.hydrate = hydrate; xhr.ts = Date.now(); xhr.open('GET', xhr.top + '?ls' + uq, true); + xhr.setRequestHeader('Fnugg', '' + xhr.ts); xhr.onload = xhr.onerror = recvls; xhr.send(); @@ -13269,6 +13270,9 @@ var treectl = (function () { if (r.chk_index_html(this.top, res)) return; + if (this.ts != res.fnugg && res.fnugg != 'nei' && sread('no_fnugg') !== '1') + toast.warn(60, "WARNING: A proxy/CDN between your webbrowser and the server is misbehaving, and caching responses it shouldn't. As a result, you are now seeing stale directory listings. There will be many issues.\n\nIf you need to ignore this and stop these messages, you can set the global-option 'no-fnugg' on the server, or click <code>π</code> and run this: <code>STG.no_fnugg=1</code>"); + for (var a = 0; a < res.files.length; a++) if (res.files[a].tags === undefined) res.files[a].tags = {}; diff --git a/tests/util.py b/tests/util.py index ea9bfeb9..7b4ad2aa 100644 --- a/tests/util.py +++ b/tests/util.py @@ -143,7 +143,7 @@ class Cfg(Namespace): def __init__(self, a=None, v=None, c=None, **ka0): ka = {} - ex = "allow_flac allow_wav chpw cookie_lax daw dav_auth dav_mac dav_rt e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp early_ban ed emp exp force_js getmod grid gsel hardlink hardlink_only ih ihead localtime magic nid nih no_acode no_athumb no_bauth no_clone no_cp no_dav no_db_ip no_del no_dirsz no_dupe no_lifetime no_logues no_mv no_pipe no_poll no_readme no_robots no_sb_md no_sb_lg no_scandir no_tail no_tarcmp no_thumb no_vthumb no_zip nrand nsort nw og og_no_head og_s_title ohead q rand re_dirsz reflink rmagic rss smb srch_dbg srch_excl stats uqe vague_403 vc ver wo_up_readme write_uplog xdev xlink xvol zipmaxu zs" + ex = "allow_flac allow_wav chpw cookie_lax daw dav_auth dav_mac dav_rt e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp early_ban ed emp exp force_js getmod grid gsel hardlink hardlink_only ih ihead localtime magic nid nih no_acode no_athumb no_bauth no_clone no_cp no_dav no_db_ip no_del no_dirsz no_dupe no_fnugg no_lifetime no_logues no_mv no_pipe no_poll no_readme no_robots no_sb_md no_sb_lg no_scandir no_tail no_tarcmp no_thumb no_vthumb no_zip nrand nsort nw og og_no_head og_s_title ohead q rand re_dirsz reflink rmagic rss smb srch_dbg srch_excl stats uqe vague_403 vc ver wo_up_readme write_uplog xdev xlink xvol zipmaxu zs" ka.update(**{k: False for k in ex.split()}) ex = "dav_inf dedup dotpart dotsrch hook_v no_dhash no_fastboot no_fpool no_htp no_rescan no_sendfile no_ses no_snap no_up_list no_voldump re_dhash see_dots plain_ip" @@ -161,7 +161,7 @@ class Cfg(Namespace): ex = "au_vol dl_list mtab_age reg_cap s_thead s_tbody tail_tmax tail_who th_convt ups_who zip_who" ka.update(**{k: 9 for k in ex.split()}) - ex = "db_act forget_ip idp_cookie idp_store k304 loris no304 nosubtle re_maxage rproxy rsp_jtr rsp_slp s_wr_slp snap_wri theme themes turbo u2ow zipmaxn zipmaxs" + ex = "ctl_re db_act forget_ip idp_cookie idp_store k304 loris no304 nosubtle re_maxage rproxy rsp_jtr rsp_slp s_wr_slp snap_wri theme themes turbo u2ow zipmaxn zipmaxs" ka.update(**{k: 0 for k in ex.split()}) ex = "ah_alg bname chmod_f chpw_db doctitle df exit favico idp_h_usr ipa html_head lg_sba lg_sbf log_fk md_sba md_sbf name og_desc og_site og_th og_title og_title_a og_title_v og_title_i shr tcolor textfiles unlist vname xff_src zipmaxt R RS SR" From 34d98e9980c40450d9375ff48e9bdc31110d05c1 Mon Sep 17 00:00:00 2001 From: exci <76759714+icxes@users.noreply.github.com> Date: Thu, 7 Aug 2025 21:10:26 +0300 Subject: [PATCH 106/174] add systemd user services and templated services (#502) * move service files from contrib/package/arch/ to /contrib/systemd/ * add simpler default copyparty.conf that puts users in jail * remove warning about .conf files in ~/.config/copyparty/ * update PKGBUILD with changes * add links to configuration examples in index.md * fix link to the example config * update README.md arch instructions --------- Co-authored-by: icxes <icxes@dev.null@need.moe> Co-authored-by: ed <s@ocv.me> --- README.md | 6 +- contrib/package/arch/PKGBUILD | 58 ++++++++----------- contrib/package/arch/copyparty.conf | 7 --- contrib/package/arch/copyparty.service | 32 ---------- contrib/package/arch/index.md | 3 - contrib/systemd/copyparty-user.service | 26 +++++++++ contrib/systemd/copyparty.conf | 45 +++----------- contrib/systemd/copyparty.example.conf | 42 ++++++++++++++ contrib/systemd/copyparty@.service | 30 ++++++++++ contrib/systemd/index.md | 10 ++++ contrib/systemd/prisonparty-user.service | 33 +++++++++++ .../prisonparty@.service} | 25 ++++---- copyparty/svchub.py | 9 --- 13 files changed, 191 insertions(+), 135 deletions(-) delete mode 100644 contrib/package/arch/copyparty.conf delete mode 100644 contrib/package/arch/copyparty.service delete mode 100644 contrib/package/arch/index.md create mode 100644 contrib/systemd/copyparty-user.service create mode 100644 contrib/systemd/copyparty.example.conf create mode 100644 contrib/systemd/copyparty@.service create mode 100644 contrib/systemd/index.md create mode 100644 contrib/systemd/prisonparty-user.service rename contrib/{package/arch/prisonparty.service => systemd/prisonparty@.service} (51%) diff --git a/README.md b/README.md index 2afc4a81..39bd932f 100644 --- a/README.md +++ b/README.md @@ -2281,11 +2281,9 @@ if your distro/OS is not mentioned below, there might be some hints in the [«on `pacman -S copyparty` (in [arch linux extra](https://archlinux.org/packages/extra/any/copyparty/)) -it comes with a [systemd service](./contrib/package/arch/copyparty.service) and expects to find one or more [config files](./docs/example.conf) in `/etc/copyparty.d/` +it comes with a [systemd service](./contrib/systemd/copyparty@.service) as well as a [user service](./contrib/systemd/copyparty-user.service), and expects to find a [config file](./contrib/systemd/copyparty.example.conf) in `/etc/copyparty/copyparty.conf` or `~/.config/copyparty/copyparty.conf` -after installing it, you may want to `cp /usr/lib/systemd/system/copyparty.service /etc/systemd/system/` and then `vim /etc/systemd/system/copyparty.service` to change what user/group it is running as (you only need to do this once) - -NOTE: there used to be an aur package; this evaporated when copyparty was adopted by the official archlinux repos. If you're still using the aur package, please move +after installing, start either the system service or the user service and navigate to http://127.0.0.1:3923 for further instructions (unless you already edited the config files, in which case you are good to go, probably) ## fedora package diff --git a/contrib/package/arch/PKGBUILD b/contrib/package/arch/PKGBUILD index 9efb5c41..c6dc4687 100644 --- a/contrib/package/arch/PKGBUILD +++ b/contrib/package/arch/PKGBUILD @@ -1,4 +1,7 @@ # Maintainer: icxes <dev.null@need.moe> +# Contributor: Morgan Adamiec <morganamilo@archlinux.org> +# NOTE: You generally shouldn't use this PKGBUILD on Arch, as it is mainly for testing purposes. Install copyparty using pacman instead. + pkgname=copyparty pkgver="1.18.10" pkgrel=1 @@ -6,52 +9,41 @@ pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFT arch=("any") url="https://github.com/9001/${pkgname}" license=('MIT') -depends=("python" "lsof" "python-jinja") +depends=("bash" "python" "lsof" "python-jinja") makedepends=("python-wheel" "python-setuptools" "python-build" "python-installer" "make" "pigz") optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tags" - "cfssl: generate TLS certificates on startup (pointless when reverse-proxied)" - "python-mutagen: music tags (alternative)" - "python-pillow: thumbnails for images" - "python-pyvips: thumbnails for images (higher quality, faster, uses more ram)" - "libkeyfinder-git: detection of musical keys" - "qm-vamp-plugins: BPM detection" - "python-pyopenssl: ftps functionality" - "python-pyzmq: send zeromq messages from event-hooks" - "python-argon2-cffi: hashed passwords in config" - "python-impacket-git: smb support (bad idea)" + "cfssl: generate TLS certificates on startup" + "python-mutagen: music tags (alternative)" + "python-pillow: thumbnails for images" + "python-pyvips: thumbnails for images (higher quality, faster, uses more ram)" + "libkeyfinder: detection of musical keys" + "python-pyopenssl: ftps functionality" + "python-pyzmq: send zeromq messages from event-hooks" + "python-argon2-cffi: hashed passwords in config" ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") -backup=("etc/${pkgname}.d/init" ) +backup=("etc/${pkgname}/copyparty.conf" ) sha256sums=("49c5fedf7619437bc0af125cb4e8360d9bda9d87ef45d6314d7acf163ab4cf99") build() { + cd "${srcdir}/${pkgname}-${pkgver}/copyparty/web" + make + cd "${srcdir}/${pkgname}-${pkgver}" - - pushd copyparty/web - make -j$(nproc) - rm Makefile - popd - - python3 -m build -wn + python -m build --wheel --no-isolation } package() { cd "${srcdir}/${pkgname}-${pkgver}" - python3 -m installer -d "$pkgdir" dist/*.whl + python -m installer --destdir="$pkgdir" dist/*.whl - install -dm755 "${pkgdir}/etc/${pkgname}.d" + install -dm755 "${pkgdir}/etc/${pkgname}" install -Dm755 "bin/prisonparty.sh" "${pkgdir}/usr/bin/prisonparty" - install -Dm644 "contrib/package/arch/${pkgname}.conf" "${pkgdir}/etc/${pkgname}.d/init" - install -Dm644 "contrib/package/arch/${pkgname}.service" "${pkgdir}/usr/lib/systemd/system/${pkgname}.service" - install -Dm644 "contrib/package/arch/prisonparty.service" "${pkgdir}/usr/lib/systemd/system/prisonparty.service" - install -Dm644 "contrib/package/arch/index.md" "${pkgdir}/var/lib/${pkgname}-jail/README.md" + install -Dm644 "contrib/systemd/${pkgname}.conf" "${pkgdir}/etc/${pkgname}/copyparty.conf" + install -Dm644 "contrib/systemd/${pkgname}@.service" "${pkgdir}/usr/lib/systemd/system/${pkgname}@.service" + install -Dm644 "contrib/systemd/${pkgname}-user.service" "${pkgdir}/usr/lib/systemd/user/${pkgname}.service" + install -Dm644 "contrib/systemd/prisonparty@.service" "${pkgdir}/usr/lib/systemd/system/prisonparty@.service" + install -Dm644 "contrib/systemd/prisonparty-user.service" "${pkgdir}/usr/lib/systemd/user/prisonparty.service" + install -Dm644 "contrib/systemd/index.md" "${pkgdir}/var/lib/${pkgname}-jail/README.md" install -Dm644 "LICENSE" "${pkgdir}/usr/share/licenses/${pkgname}/LICENSE" - - find /etc/${pkgname}.d -iname '*.conf' 2>/dev/null | grep -qE . && return - echo "┏━━━━━━━━━━━━━━━──-" - echo "┃ Configure ${pkgname} by adding .conf files into /etc/${pkgname}.d/" - echo "┃ and maybe copy+edit one of the following to /etc/systemd/system/:" - echo "┣━♦ /usr/lib/systemd/system/${pkgname}.service (standard)" - echo "┣━♦ /usr/lib/systemd/system/prisonparty.service (chroot)" - echo "┗━━━━━━━━━━━━━━━──-" } diff --git a/contrib/package/arch/copyparty.conf b/contrib/package/arch/copyparty.conf deleted file mode 100644 index 1d90d772..00000000 --- a/contrib/package/arch/copyparty.conf +++ /dev/null @@ -1,7 +0,0 @@ -## import all *.conf files from the current folder (/etc/copyparty.d) -% ./ - -# add additional .conf files to this folder; -# see example config files for reference: -# https://github.com/9001/copyparty/blob/hovudstraum/docs/example.conf -# https://github.com/9001/copyparty/tree/hovudstraum/docs/copyparty.d diff --git a/contrib/package/arch/copyparty.service b/contrib/package/arch/copyparty.service deleted file mode 100644 index 22dac3d6..00000000 --- a/contrib/package/arch/copyparty.service +++ /dev/null @@ -1,32 +0,0 @@ -# this will start `/usr/bin/copyparty-sfx.py` -# and read config from `/etc/copyparty.d/*.conf` -# -# you probably want to: -# change "User=cpp" and "/home/cpp/" to another user -# -# unless you add -q to disable logging, you may want to remove the -# following line to allow buffering (slightly better performance): -# Environment=PYTHONUNBUFFERED=x - -[Unit] -Description=copyparty file server - -[Service] -Type=notify -SyslogIdentifier=copyparty -Environment=PYTHONUNBUFFERED=x -WorkingDirectory=/var/lib/copyparty-jail -ExecReload=/bin/kill -s USR1 $MAINPID - -# user to run as + where the TLS certificate is (if any) -User=cpp -Environment=XDG_CONFIG_HOME=/home/cpp/.config - -# stop systemd-tmpfiles-clean.timer from deleting copyparty while it's running -ExecStartPre=+/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf' - -# run copyparty -ExecStart=/usr/bin/python3 /usr/bin/copyparty -c /etc/copyparty.d/init - -[Install] -WantedBy=multi-user.target diff --git a/contrib/package/arch/index.md b/contrib/package/arch/index.md deleted file mode 100644 index 016c0b0a..00000000 --- a/contrib/package/arch/index.md +++ /dev/null @@ -1,3 +0,0 @@ -this is `/var/lib/copyparty-jail`, the fallback webroot when copyparty has not yet been configured - -please add some `*.conf` files to `/etc/copyparty.d/` diff --git a/contrib/systemd/copyparty-user.service b/contrib/systemd/copyparty-user.service new file mode 100644 index 00000000..c6f61000 --- /dev/null +++ b/contrib/systemd/copyparty-user.service @@ -0,0 +1,26 @@ +# this will start `/usr/bin/copyparty` +# and read config from `$HOME/.config/copyparty.conf` +# +# unless you add -q to disable logging, you may want to remove the +# following line to allow buffering (slightly better performance): +# Environment=PYTHONUNBUFFERED=x + +[Unit] +Description=copyparty file server + +[Service] +Type=notify +SyslogIdentifier=copyparty +WorkingDirectory=/var/lib/copyparty-jail +Environment=PYTHONUNBUFFERED=x +Environment=PRTY_CONFIG=%h/.config/copyparty/copyparty.conf +ExecReload=/bin/kill -s USR1 $MAINPID + +# ensure there is a config +ExecStartPre=/bin/bash -c 'if [[ ! -f %h/.config/copyparty/copyparty.conf ]]; then mkdir -p %h/.config/copyparty; cp /etc/copyparty/copyparty.conf %h/.config/copyparty/copyparty.conf; fi' + +# run copyparty +ExecStart=/usr/bin/python3 /usr/bin/copyparty + +[Install] +WantedBy=default.target diff --git a/contrib/systemd/copyparty.conf b/contrib/systemd/copyparty.conf index 79560d0d..5d9bdbe3 100644 --- a/contrib/systemd/copyparty.conf +++ b/contrib/systemd/copyparty.conf @@ -1,42 +1,13 @@ -# not actually YAML but lets pretend: -# -*- mode: yaml -*- -# vim: ft=yaml: - - -# put this file in /etc/ - - [global] - e2dsa # enable file indexing and filesystem scanning - e2ts # and enable multimedia indexing - ansi # and colors in log messages - - # disable logging to stdout/journalctl and log to a file instead; - # $LOGS_DIRECTORY is usually /var/log/copyparty (comes from systemd) - # and copyparty replaces %Y-%m%d with Year-MonthDay, so the - # full path will be something like /var/log/copyparty/2023-1130.txt - # (note: enable compression by adding .xz at the end) - q, lo: $LOGS_DIRECTORY/%Y-%m%d.log - - # p: 80,443,3923 # listen on 80/443 as well (requires CAP_NET_BIND_SERVICE) - # i: 127.0.0.1 # only allow connections from localhost (reverse-proxies) - # ftp: 3921 # enable ftp server on port 3921 - # p: 3939 # listen on another port - # df: 16 # stop accepting uploads if less than 16 GB free disk space - # ver # show copyparty version in the controlpanel - # grid # show thumbnails/grid-view by default - # theme: 2 # monokai - # name: datasaver # change the server-name that's displayed in the browser - # stats, nos-dup # enable the prometheus endpoint, but disable the dupes counter (too slow) - # no-robots, force-js # make it harder for search engines to read your server - + i: 127.0.0.1 [accounts] - ed: wark # username: password + user: password - -[/] # create a volume at "/" (the webroot), which will - /mnt # share the contents of the "/mnt" folder +[/] + /var/lib/copyparty-jail accs: - rw: * # everyone gets read-write access, but - rwmda: ed # the user "ed" gets read-write-move-delete-admin + r: * + rwdma: user + flags: + grid \ No newline at end of file diff --git a/contrib/systemd/copyparty.example.conf b/contrib/systemd/copyparty.example.conf new file mode 100644 index 00000000..79560d0d --- /dev/null +++ b/contrib/systemd/copyparty.example.conf @@ -0,0 +1,42 @@ +# not actually YAML but lets pretend: +# -*- mode: yaml -*- +# vim: ft=yaml: + + +# put this file in /etc/ + + +[global] + e2dsa # enable file indexing and filesystem scanning + e2ts # and enable multimedia indexing + ansi # and colors in log messages + + # disable logging to stdout/journalctl and log to a file instead; + # $LOGS_DIRECTORY is usually /var/log/copyparty (comes from systemd) + # and copyparty replaces %Y-%m%d with Year-MonthDay, so the + # full path will be something like /var/log/copyparty/2023-1130.txt + # (note: enable compression by adding .xz at the end) + q, lo: $LOGS_DIRECTORY/%Y-%m%d.log + + # p: 80,443,3923 # listen on 80/443 as well (requires CAP_NET_BIND_SERVICE) + # i: 127.0.0.1 # only allow connections from localhost (reverse-proxies) + # ftp: 3921 # enable ftp server on port 3921 + # p: 3939 # listen on another port + # df: 16 # stop accepting uploads if less than 16 GB free disk space + # ver # show copyparty version in the controlpanel + # grid # show thumbnails/grid-view by default + # theme: 2 # monokai + # name: datasaver # change the server-name that's displayed in the browser + # stats, nos-dup # enable the prometheus endpoint, but disable the dupes counter (too slow) + # no-robots, force-js # make it harder for search engines to read your server + + +[accounts] + ed: wark # username: password + + +[/] # create a volume at "/" (the webroot), which will + /mnt # share the contents of the "/mnt" folder + accs: + rw: * # everyone gets read-write access, but + rwmda: ed # the user "ed" gets read-write-move-delete-admin diff --git a/contrib/systemd/copyparty@.service b/contrib/systemd/copyparty@.service new file mode 100644 index 00000000..cd891eab --- /dev/null +++ b/contrib/systemd/copyparty@.service @@ -0,0 +1,30 @@ +# this will start `/usr/bin/copyparty` +# and read config from `/etc/copyparty/copyparty.conf` +# +# the %i refers to whatever you put after the copyparty@ +# so with copyparty@foo.service, %i == foo +# +# unless you add -q to disable logging, you may want to remove the +# following line to allow buffering (slightly better performance): +# Environment=PYTHONUNBUFFERED=x + +[Unit] +Description=copyparty file server + +[Service] +Type=notify +SyslogIdentifier=copyparty +WorkingDirectory=/var/lib/copyparty-jail +Environment=PYTHONUNBUFFERED=x +Environment=PRTY_CONFIG=/etc/copyparty/copyparty.conf +ExecReload=/bin/kill -s USR1 $MAINPID + +# user to run as + where the TLS certificate is (if any) +User=%i +Environment=XDG_CONFIG_HOME=/home/%i/.config + +# run copyparty +ExecStart=/usr/bin/python3 /usr/bin/copyparty + +[Install] +WantedBy=multi-user.target diff --git a/contrib/systemd/index.md b/contrib/systemd/index.md new file mode 100644 index 00000000..cfd1b715 --- /dev/null +++ b/contrib/systemd/index.md @@ -0,0 +1,10 @@ +this is `/var/lib/copyparty-jail`, the fallback webroot when copyparty has not yet been configured + +please edit `/etc/copyparty/copyparty.conf` (if running as a system service) +or `$HOME/.config/copyparty/copyparty.conf` if running as a user service + +a basic configuration example is available at https://github.com/9001/copyparty/blob/hovudstraum/contrib/systemd/copyparty.example.conf +a configuration example that explains most flags is available at https://github.com/9001/copyparty/blob/hovudstraum/docs/chungus.conf + +the full list of configuration options can be seen at https://ocv.me/copyparty/helptext.html +or by running `copyparty --help` diff --git a/contrib/systemd/prisonparty-user.service b/contrib/systemd/prisonparty-user.service new file mode 100644 index 00000000..e0aaeaf8 --- /dev/null +++ b/contrib/systemd/prisonparty-user.service @@ -0,0 +1,33 @@ +# this will start `/usr/bin/copyparty` +# in a chroot, preventing accidental access elsewhere, +# and read copyparty config from `$HOME/.config/copyparty.conf` +# +# expose additional filesystem locations to copyparty +# by listing them between the last `%u` and `--` +# +# unless you add -q to disable logging, you may want to remove the +# following line to allow buffering (slightly better performance): +# Environment=PYTHONUNBUFFERED=x + +[Unit] +Description=copyparty file server + +[Service] +Type=notify +SyslogIdentifier=prisonparty +WorkingDirectory=/var/lib/copyparty-jail +Environment=PYTHONUNBUFFERED=x +Environment=PRTY_CONFIG=%h/.config/copyparty/copyparty.conf +ExecReload=/bin/kill -s USR1 $MAINPID + +# ensure there is a config +ExecStartPre=/bin/bash -c 'if [[ ! -f %h/.config/copyparty/copyparty.conf ]]; then mkdir -p %h/.config/copyparty; cp /etc/copyparty/copyparty.conf %h/.config/copyparty/copyparty.conf; fi' + +# run copyparty +ExecStart=/bin/bash /usr/bin/prisonparty /var/lib/copyparty-jail %u %u \ + %h/.config/copyparty \ + -- \ + /usr/bin/python3 /usr/bin/copyparty + +[Install] +WantedBy=default.target diff --git a/contrib/package/arch/prisonparty.service b/contrib/systemd/prisonparty@.service similarity index 51% rename from contrib/package/arch/prisonparty.service rename to contrib/systemd/prisonparty@.service index cd35ba99..d30010ae 100644 --- a/contrib/package/arch/prisonparty.service +++ b/contrib/systemd/prisonparty@.service @@ -1,11 +1,13 @@ -# this will start `/usr/bin/copyparty-sfx.py` +# this will start `/usr/bin/copyparty` # in a chroot, preventing accidental access elsewhere, -# and read copyparty config from `/etc/copyparty.d/*.conf` +# and read copyparty config from `/etc/copyparty/copyparty.conf` # # expose additional filesystem locations to copyparty -# by listing them between the last `cpp` and `--` +# by listing them between the last `%i` and `--` # -# `cpp cpp` = user/group to run copyparty as; can be IDs (1000 1000) +# `%i %i` = user/group to run copyparty as; can be IDs (1000 1000) +# the %i refers to whatever you put after the prisonparty@ +# so with prisonparty@foo.service, %i == foo # # unless you add -q to disable logging, you may want to remove the # following line to allow buffering (slightly better performance): @@ -15,19 +17,22 @@ Description=copyparty file server [Service] +Type=notify SyslogIdentifier=prisonparty -Environment=PYTHONUNBUFFERED=x WorkingDirectory=/var/lib/copyparty-jail +Environment=PYTHONUNBUFFERED=x +Environment=PRTY_CONFIG=/etc/copyparty/copyparty.conf ExecReload=/bin/kill -s USR1 $MAINPID -# stop systemd-tmpfiles-clean.timer from deleting copyparty while it's running -ExecStartPre=+/bin/bash -c 'mkdir -p /run/tmpfiles.d/ && echo "x /tmp/pe-copyparty*" > /run/tmpfiles.d/copyparty.conf' +# user to run as + where the TLS certificate is (if any) +User=%i +Environment=XDG_CONFIG_HOME=/home/%i/.config # run copyparty -ExecStart=/bin/bash /usr/bin/prisonparty /var/lib/copyparty-jail cpp cpp \ - /etc/copyparty.d \ +ExecStart=/bin/bash /usr/bin/prisonparty /var/lib/copyparty-jail %i %i \ + /etc/copyparty \ -- \ - /usr/bin/python3 /usr/bin/copyparty -c /etc/copyparty.d/init + /usr/bin/python3 /usr/bin/copyparty [Install] WantedBy=multi-user.target diff --git a/copyparty/svchub.py b/copyparty/svchub.py index 07d85d83..1af13276 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -850,15 +850,6 @@ class SvcHub(object): def _check_env(self) -> None: al = self.args - try: - files = os.listdir(E.cfg) - except: - files = [] - - hits = [x for x in files if x.lower().endswith(".conf")] - if hits: - t = "WARNING: found config files in [%s]: %s\n config files are not expected here, and will NOT be loaded (unless your setup is intentionally hella funky)" - self.log("root", t % (E.cfg, ", ".join(hits)), 3) if self.args.no_bauth: t = "WARNING: --no-bauth disables support for the Android app; you may want to use --bauth-last instead" From 2943c7f2d51c4eb51706e246e67d72624e6d7f45 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 18:30:40 +0000 Subject: [PATCH 107/174] move the docker config smoketest over here --- copyparty/authsrv.py | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 44629e62..9c52f046 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -1700,6 +1700,7 @@ class AuthSrv(object): if not mount and not self.args.idp_h_usr: # -h says our defaults are CWD at root and read/write for everyone axs = AXS(["*"], ["*"], None, None) + ehint = "" if self.is_lxc: t = "Read-access has been disabled due to failsafe: Docker detected, but %s. This failsafe is to prevent unintended access if this is due to accidental loss of config. You can override this safeguard and allow read/write to all of /w/ by adding the following arguments to the docker container: -v .::rw" if len(cfg_files_loaded) == 1: @@ -1709,10 +1710,21 @@ class AuthSrv(object): else: self.log(t % ("the config does not define any volumes",), 1) axs = AXS() + ehint = "; please try moving them up one level, into the parent folder:" elif self.args.c: t = "Read-access has been disabled due to failsafe: No volumes were defined by the config-file. This failsafe is to prevent unintended access if this is due to accidental loss of config. You can override this safeguard and allow read/write to the working-directory by adding the following arguments: -v .::rw" self.log(t, 1) axs = AXS() + ehint = ":" + if ehint: + try: + files = os.listdir(E.cfg) + except: + files = [] + hits = [x for x in files if x.lower().endswith(".conf")] + if hits: + t = "Hint: Found some config files in [%s], but these were not automatically loaded because they are in the wrong place%s %s\n" + self.log(t % (E.cfg, ehint, ", ".join(hits)), 3) zvf = {"tcolor": self.args.tcolor} vfs = VFS(self.log_func, absreal("."), "", "", axs, zvf) if not axs.uread: From bcc3b1568e2ceb46a40bff6661fd72970ebf3544 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 18:39:22 +0000 Subject: [PATCH 108/174] add qrcode to connectpage; closes #523 --- copyparty/web/svcs.html | 1 + copyparty/web/svcs.js | 11 +++++++++++ 2 files changed, 12 insertions(+) diff --git a/copyparty/web/svcs.html b/copyparty/web/svcs.html index a4b57746..26fd5267 100644 --- a/copyparty/web/svcs.html +++ b/copyparty/web/svcs.html @@ -37,6 +37,7 @@ {% if accs %}<code><b id="pw0">{{ pw }}</b></code>=password, {% endif %}<code><b>mp</b></code>=mountpoint </span> {% if accs %}<a href="#" id="setpw">use real password</a>{% endif %} + <a href="#" id="qr">show qr</a> </p> diff --git a/copyparty/web/svcs.js b/copyparty/web/svcs.js index 0de94523..dad7faab 100644 --- a/copyparty/web/svcs.js +++ b/copyparty/web/svcs.js @@ -49,12 +49,14 @@ function setos(os) { setos(WINDOWS ? 'win' : LINUX ? 'lin' : MACOS ? 'mac' : 'idk'); +var pw = ''; function setpw(e) { ev(e); modal.prompt('password:', '', function (v) { if (!v) return; + pw = v; var pw0 = ebi('pw0').innerHTML, oa = QSA('b'); @@ -67,3 +69,12 @@ function setpw(e) { } if (ebi('setpw')) ebi('setpw').onclick = setpw; + + +ebi('qr').onclick = function () { + var url = ('' + location).split('?')[0]; + if (pw) + url += '?pw=' + pw; + var txt = esc(url) + '<img class="b64" width="100" height="100" src="' + addq(url, 'qr') + '" />'; + modal.alert(txt); +}; From 8b31ed8816506776950cdc44ef7fc4a5ff681fa9 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 19:11:28 +0000 Subject: [PATCH 109/174] text-editor: optional EOL conversion; closes #513 --- copyparty/__main__.py | 1 + copyparty/cfg.py | 2 ++ copyparty/httpcli.py | 4 ++++ copyparty/util.py | 12 ++++++++++++ tests/util.py | 2 +- 5 files changed, 20 insertions(+), 1 deletion(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 0715af28..d0fc325e 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1528,6 +1528,7 @@ def add_db_metadata(ap): def add_txt(ap): ap2 = ap.add_argument_group("textfile options") ap2.add_argument("--md-hist", metavar="TXT", type=u, default="s", help="where to store old version of markdown files; [\033[32ms\033[0m]=subfolder, [\033[32mv\033[0m]=volume-histpath, [\033[32mn\033[0m]=nope/disabled (volflag=md_hist)") + ap2.add_argument("--txt-eol", metavar="TYPE", type=u, default="", help="enable EOL conversion when writing documents; supported: CRLF, LF (volflag=txt_eol)") ap2.add_argument("-mcr", metavar="SEC", type=int, default=60, help="the textfile editor will check for serverside changes every \033[33mSEC\033[0m seconds") ap2.add_argument("-emp", action="store_true", help="enable markdown plugins -- neat but dangerous, big XSS risk") ap2.add_argument("--exp", action="store_true", help="enable textfile expansion -- replace {{self.ip}} and such; see \033[33m--help-exp\033[0m (volflag=exp)") diff --git a/copyparty/cfg.py b/copyparty/cfg.py index 2f75ab28..8a07f956 100644 --- a/copyparty/cfg.py +++ b/copyparty/cfg.py @@ -111,6 +111,7 @@ def vf_vmap() -> dict[str, str]: "tail_tmax", "tail_who", "tcolor", + "txt_eol", "unlist", "u2abort", "u2ts", @@ -322,6 +323,7 @@ flagcats = { "exp": "enable textfile expansion; see --help-exp", "exp_md": "placeholders to expand in markdown files; see --help", "exp_lg": "placeholders to expand in prologue/epilogue; see --help", + "txt_eol=lf": "enable EOL conversion when writing docs (LF or CRLF)", }, "tailing": { "notail": "disable ?tail (download a growing file continuously)", diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 3da299c7..e1c91c51 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -62,6 +62,7 @@ from .util import ( alltrace, atomic_move, b64dec, + eol_conv, exclude_dotfiles, formatdate, fsenc, @@ -3679,6 +3680,9 @@ class HttpCli(object): if p_field != "body": raise Pebkac(400, "expected body, got {}".format(p_field)) + if "txt_eol" in vfs.flags: + p_data = eol_conv(p_data, vfs.flags["txt_eol"]) + xbu = vfs.flags.get("xbu") if xbu: if not runhook( diff --git a/copyparty/util.py b/copyparty/util.py index 15b1924b..588028ab 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -2982,6 +2982,18 @@ def justcopy( return tlen, "checksum-disabled", "checksum-disabled" +def eol_conv( + fin: Generator[bytes, None, None], + conv: str +) -> Generator[bytes, None, None]: + crlf = conv.lower() == "crlf" + for buf in fin: + buf = buf.replace(b"\r", b"") + if crlf: + buf = buf.replace(b"\n", b"\r\n") + yield buf + + def hashcopy( fin: Generator[bytes, None, None], fout: Union[typing.BinaryIO, typing.IO[Any]], diff --git a/tests/util.py b/tests/util.py index 7b4ad2aa..5968215a 100644 --- a/tests/util.py +++ b/tests/util.py @@ -164,7 +164,7 @@ class Cfg(Namespace): ex = "ctl_re db_act forget_ip idp_cookie idp_store k304 loris no304 nosubtle re_maxage rproxy rsp_jtr rsp_slp s_wr_slp snap_wri theme themes turbo u2ow zipmaxn zipmaxs" ka.update(**{k: 0 for k in ex.split()}) - ex = "ah_alg bname chmod_f chpw_db doctitle df exit favico idp_h_usr ipa html_head lg_sba lg_sbf log_fk md_sba md_sbf name og_desc og_site og_th og_title og_title_a og_title_v og_title_i shr tcolor textfiles unlist vname xff_src zipmaxt R RS SR" + ex = "ah_alg bname chmod_f chpw_db doctitle df exit favico idp_h_usr ipa html_head lg_sba lg_sbf log_fk md_sba md_sbf name og_desc og_site og_th og_title og_title_a og_title_v og_title_i shr tcolor textfiles txt_eol unlist vname xff_src zipmaxt R RS SR" ka.update(**{k: "" for k in ex.split()}) ex = "ban_403 ban_404 ban_422 ban_pw ban_pwc ban_url spinner" From 13499d2846eb48b40fe6de93d4c7a7a90545db03 Mon Sep 17 00:00:00 2001 From: icxes <dev.null@need.moe> Date: Thu, 7 Aug 2025 22:42:33 +0300 Subject: [PATCH 110/174] remove prisonparty-user.service there is no real point to a prisonparty user service, as chroot requires root --- contrib/systemd/prisonparty-user.service | 33 ------------------------ 1 file changed, 33 deletions(-) delete mode 100644 contrib/systemd/prisonparty-user.service diff --git a/contrib/systemd/prisonparty-user.service b/contrib/systemd/prisonparty-user.service deleted file mode 100644 index e0aaeaf8..00000000 --- a/contrib/systemd/prisonparty-user.service +++ /dev/null @@ -1,33 +0,0 @@ -# this will start `/usr/bin/copyparty` -# in a chroot, preventing accidental access elsewhere, -# and read copyparty config from `$HOME/.config/copyparty.conf` -# -# expose additional filesystem locations to copyparty -# by listing them between the last `%u` and `--` -# -# unless you add -q to disable logging, you may want to remove the -# following line to allow buffering (slightly better performance): -# Environment=PYTHONUNBUFFERED=x - -[Unit] -Description=copyparty file server - -[Service] -Type=notify -SyslogIdentifier=prisonparty -WorkingDirectory=/var/lib/copyparty-jail -Environment=PYTHONUNBUFFERED=x -Environment=PRTY_CONFIG=%h/.config/copyparty/copyparty.conf -ExecReload=/bin/kill -s USR1 $MAINPID - -# ensure there is a config -ExecStartPre=/bin/bash -c 'if [[ ! -f %h/.config/copyparty/copyparty.conf ]]; then mkdir -p %h/.config/copyparty; cp /etc/copyparty/copyparty.conf %h/.config/copyparty/copyparty.conf; fi' - -# run copyparty -ExecStart=/bin/bash /usr/bin/prisonparty /var/lib/copyparty-jail %u %u \ - %h/.config/copyparty \ - -- \ - /usr/bin/python3 /usr/bin/copyparty - -[Install] -WantedBy=default.target From 3c42a34f7b43bdb17ba40432c3a493aedbaf5a88 Mon Sep 17 00:00:00 2001 From: icxes <dev.null@need.moe> Date: Thu, 7 Aug 2025 22:47:46 +0300 Subject: [PATCH 111/174] update PKGBUILD; remove prisonparty user service --- contrib/package/arch/PKGBUILD | 1 - 1 file changed, 1 deletion(-) diff --git a/contrib/package/arch/PKGBUILD b/contrib/package/arch/PKGBUILD index c6dc4687..2673ec2a 100644 --- a/contrib/package/arch/PKGBUILD +++ b/contrib/package/arch/PKGBUILD @@ -43,7 +43,6 @@ package() { install -Dm644 "contrib/systemd/${pkgname}@.service" "${pkgdir}/usr/lib/systemd/system/${pkgname}@.service" install -Dm644 "contrib/systemd/${pkgname}-user.service" "${pkgdir}/usr/lib/systemd/user/${pkgname}.service" install -Dm644 "contrib/systemd/prisonparty@.service" "${pkgdir}/usr/lib/systemd/system/prisonparty@.service" - install -Dm644 "contrib/systemd/prisonparty-user.service" "${pkgdir}/usr/lib/systemd/user/prisonparty.service" install -Dm644 "contrib/systemd/index.md" "${pkgdir}/var/lib/${pkgname}-jail/README.md" install -Dm644 "LICENSE" "${pkgdir}/usr/share/licenses/${pkgname}/LICENSE" } From 346515ccf17086392ef0440cad686a4065b6a54c Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 20:29:44 +0000 Subject: [PATCH 112/174] add optional username login; closes #511 --- README.md | 10 ++++++++++ copyparty/__main__.py | 6 +++++- copyparty/authsrv.py | 11 +++++++++++ copyparty/ftpd.py | 7 ++++++- copyparty/httpcli.py | 14 ++++++++++++++ copyparty/pwhash.py | 4 ++++ copyparty/util.py | 3 +-- copyparty/web/splash.html | 5 +++++ copyparty/web/splash.js | 2 +- tests/util.py | 2 +- 10 files changed, 58 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 39bd932f..e063e4af 100644 --- a/README.md +++ b/README.md @@ -437,6 +437,7 @@ upgrade notes * can I link someone to a password-protected volume/file by including the password in the URL? * yes, by adding `?pw=hunter2` to the end; replace `?` with `&` if there are parameters in the URL already, meaning it contains a `?` near the end + * if you have enabled `--accounts` then do `?pw=username:password` instead * how do I stop `.hist` folders from appearing everywhere on my HDD? * by default, a `.hist` folder is created inside each volume for the filesystem index, thumbnails, audio transcodes, and markdown document history. Use the `--hist` global-option or the `hist` volflag to move it somewhere else; see [database location](#database-location) @@ -1016,6 +1017,7 @@ a feed example: https://cd.ocv.me/a/d2/d22/?rss&fext=mp3 url parameters: * `pw=hunter2` for password auth + * if you enabled `--usernames` then do `pw=username:password` instead * `recursive` to also include subfolders * `title=foo` changes the feed title (default: folder name) * `fext=mp3,opus` only include mp3 and opus files (default: all) @@ -1301,6 +1303,7 @@ an FTP server can be started using `--ftp 3921`, and/or `--ftps` for explicit T * if you enable both `ftp` and `ftps`, the port-range will be divided in half * some older software (filezilla on debian-stable) cannot passive-mode with TLS * login with any username + your password, or put your password in the username field + * unless you enabled `--usernames` some recommended FTP / FTPS clients; `wark` = example password: * https://winscp.net/eng/download.php @@ -1318,6 +1321,7 @@ click the [connect](http://127.0.0.1:3923/?hc) button in the control-panel to se general usage: * login with any username + your password, or put your password in the username field (password field can be empty/whatever) + * unless you enabled `--usernames` on macos, connect from finder: * [Go] -> [Connect to Server...] -> http://192.168.123.1:3923/ @@ -1333,6 +1337,7 @@ using the GUI (winXP or later): * rightclick [my computer] -> [map network drive] -> Folder: `http://192.168.123.1:3923/` * on winXP only, click the `Sign up for online storage` hyperlink instead and put the URL there * providing your password as the username is recommended; the password field can be anything or empty + * unless you enabled `--usernames` the webdav client that's built into windows has the following list of bugs; you can avoid all of these by connecting with rclone instead: * win7+ doesn't actually send the password to the server when reauthenticating after a reboot unless you first try to login with an incorrect password and then switch to the correct password @@ -1390,6 +1395,7 @@ some **BIG WARNINGS** specific to SMB/CIFS, in decreasing importance: * the smb backend is not fully integrated with vfs, meaning there could be security issues (path traversal). Please use `--smb-port` (see below) and [prisonparty](./bin/prisonparty.sh) or [bubbleparty](./bin/bubbleparty.sh) * account passwords work per-volume as expected, and so does account permissions (read/write/move/delete), but `--smbw` must be given to allow write-access from smb * [shadowing](#shadowing) probably works as expected but no guarantees +* not compatible with pw-hashing or `--usernames` and some minor issues, * clients only see the first ~400 files in big folders; @@ -2506,6 +2512,8 @@ you can provide passwords using header `PW: hunter2`, cookie `cppwd=hunter2`, ur > for basic-authentication, all of the following are accepted: `password` / `whatever:password` / `password:whatever` (the username is ignored) +* unless you've enabled `--usernames`, then it's `PW: usr:pwd`, cookie `cppwd=usr:pwd`, url-param `?pw=usr:pwd` + NOTE: curl will not send the original filename if you use `-T` combined with url-params! Also, make sure to always leave a trailing slash in URLs unless you want to override the filename @@ -2721,6 +2729,8 @@ when generating hashes using `--ah-cli` for docker or systemd services, make sur * inspecting the generated salt using `--show-ah-salt` in copyparty service configuration * setting the same `--ah-salt` in both environments +> ⚠️ if you have enabled `--usernames` then provide the password as `username:password` when hashing it, for example `ed:hunter2` + ## https diff --git a/copyparty/__main__.py b/copyparty/__main__.py index d0fc325e..2b55b058 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -918,6 +918,9 @@ def get_sects(): copyparty will also hash and print any passwords that are non-hashed (password which do not start with '+') and then terminate afterwards + if you have enabled --accounts then the password + must be provided as username:password for hashing + \033[36m--ah-alg\033[0m specifies the hashing algorithm and a list of optional comma-separated arguments: @@ -1002,6 +1005,7 @@ def add_general(ap, nc, srvname): ap2.add_argument("-a", metavar="ACCT", type=u, action="append", help="\033[34mREPEATABLE:\033[0m add account, \033[33mUSER\033[0m:\033[33mPASS\033[0m; example [\033[32med:wark\033[0m]") ap2.add_argument("-v", metavar="VOL", type=u, action="append", help="\033[34mREPEATABLE:\033[0m add volume, \033[33mSRC\033[0m:\033[33mDST\033[0m:\033[33mFLAG\033[0m; examples [\033[32m.::r\033[0m], [\033[32m/mnt/nas/music:/music:r:aed\033[0m], see --help-accounts") ap2.add_argument("--grp", metavar="G:N,N", type=u, action="append", help="\033[34mREPEATABLE:\033[0m add group, \033[33mNAME\033[0m:\033[33mUSER1\033[0m,\033[33mUSER2\033[0m,\033[33m...\033[0m; example [\033[32madmins:ed,foo,bar\033[0m]") + ap2.add_argument("--usernames", action="store_true", help="require username and password for login; default is just password") ap2.add_argument("-ed", action="store_true", help="enable the ?dots url parameter / client option which allows clients to see dotfiles / hidden files (volflag=dots)") ap2.add_argument("--urlform", metavar="MODE", type=u, default="print,xm", help="how to handle url-form POSTs; see \033[33m--help-urlform\033[0m") ap2.add_argument("--wintitle", metavar="TXT", type=u, default="cpp @ $pub", help="server terminal title, for example [\033[32m$ip-10.1.2.\033[0m] or [\033[32m$ip-]") @@ -1393,7 +1397,7 @@ def add_logging(ap): ap2.add_argument("--no-voldump", action="store_true", help="do not list volumes and permissions on startup") ap2.add_argument("--log-utc", action="store_true", help="do not use local timezone; assume the TZ env-var is UTC (tiny bit faster)") ap2.add_argument("--log-tdec", metavar="N", type=int, default=3, help="timestamp resolution / number of timestamp decimals") - ap2.add_argument("--log-badpwd", metavar="N", type=int, default=1, help="log failed login attempt passwords: 0=terse, 1=plaintext, 2=hashed") + ap2.add_argument("--log-badpwd", metavar="N", type=int, default=2, help="log failed login attempt passwords: 0=terse, 1=plaintext, 2=hashed") ap2.add_argument("--log-conn", action="store_true", help="debug: print tcp-server msgs") ap2.add_argument("--log-htp", action="store_true", help="debug: print http-server threadpool scaling") ap2.add_argument("--ihead", metavar="HEADER", type=u, action='append', help="print request \033[33mHEADER\033[0m; [\033[32m*\033[0m]=all") diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 9c52f046..0f1c94f9 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -2642,6 +2642,8 @@ class AuthSrv(object): self.re_pwd = None pwds = [re.escape(x) for x in self.iacct.keys()] pwds.extend(list(self.sesa)) + if self.args.usernames: + pwds.extend([x.split(":", 1)[1] for x in pwds if ":" in x]) if pwds: if self.ah.on: zs = r"(\[H\] pw:.*|[?&]pw=)([^&]+)" @@ -2942,6 +2944,9 @@ class AuthSrv(object): t = "minimum password length: %d characters" return False, t % (self.args.chpw_len,) + if self.args.usernames: + pw = "%s:%s" % (uname, pw) + hpw = self.ah.hash(pw) if self.ah.on else pw if hpw == self.acct[uname]: @@ -3033,6 +3038,12 @@ class AuthSrv(object): self.log("chpw: " + msg, 6) def setup_pwhash(self, acct: dict[str, str]) -> None: + if self.args.usernames: + for uname, pw in list(acct.items())[:]: + if pw.startswith("+") and len(pw) == 33: + continue + acct[uname] = "%s:%s" % (uname, pw) + self.ah = PWHash(self.args) if not self.ah.on: if self.args.ah_cli or self.args.ah_gen: diff --git a/copyparty/ftpd.py b/copyparty/ftpd.py index bace7d28..004fb492 100644 --- a/copyparty/ftpd.py +++ b/copyparty/ftpd.py @@ -83,7 +83,12 @@ class FtpAuth(DummyAuthorizer): uname = "*" if username != "anonymous": uname = "" - for zs in (password, username): + if args.usernames: + alts = ["%s:%s" % (username, password)] + else: + alts = password, username + + for zs in alts: zs = asrv.iacct.get(asrv.ah.hash(zs), "") if zs: uname = zs diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index e1c91c51..cb061ac2 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -2937,12 +2937,16 @@ class HttpCli(object): def handle_chpw(self) -> bool: assert self.parser # !rm + if self.args.usernames: + self.parser.require("uname", 64) pwd = self.parser.require("pw", 64) self.parser.drop() ok, msg = self.asrv.chpw(self.conn.hsrv.broker, self.uname, pwd) if ok: self.cbonk(self.conn.hsrv.gpwc, pwd, "pw", "too many password changes") + if self.args.usernames: + pwd = "%s:%s" % (self.uname, pwd) ok, msg = self.get_pwd_cookie(pwd) if ok: msg = "new password OK" @@ -2955,6 +2959,13 @@ class HttpCli(object): def handle_login(self) -> bool: assert self.parser # !rm + if self.args.usernames: + try: + un = self.parser.require("uname", 256) + except: + un = "" + else: + un = "" pwd = self.parser.require("cppwd", 64) try: uhash = self.parser.require("uhash", 256) @@ -2965,6 +2976,9 @@ class HttpCli(object): if not pwd: raise Pebkac(422, "password cannot be blank") + if un: + pwd = "%s:%s" % (un, pwd) + dst = self.args.SRS if self.vpath: dst += quotep(self.vpaths) diff --git a/copyparty/pwhash.py b/copyparty/pwhash.py index ebbd78f3..d4455ca8 100644 --- a/copyparty/pwhash.py +++ b/copyparty/pwhash.py @@ -147,6 +147,10 @@ class PWHash(object): def cli(self) -> None: import getpass + if self.args.usernames: + t = "since you have enabled --usernames, please provide username:password" + print(t) + while True: try: p1 = getpass.getpass("password> ") diff --git a/copyparty/util.py b/copyparty/util.py index 588028ab..b5a2c45e 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -2983,8 +2983,7 @@ def justcopy( def eol_conv( - fin: Generator[bytes, None, None], - conv: str + fin: Generator[bytes, None, None], conv: str ) -> Generator[bytes, None, None]: crlf = conv.lower() == "crlf" for buf in fin: diff --git a/copyparty/web/splash.html b/copyparty/web/splash.html index 8dd56bab..4363efd2 100644 --- a/copyparty/web/splash.html +++ b/copyparty/web/splash.html @@ -120,7 +120,12 @@ <div> <form id="lf" method="post" enctype="multipart/form-data" action="{{ r }}/{{ qvpath }}"> <input type="hidden" id="la" name="act" value="login" /> + {% if this.args.usernames %} + <input type="text" id="lu" name="uname" placeholder=" username" size="12" /> + <input type="password" id="lp" name="cppwd" placeholder=" password" size="12" /> + {% else %} <input type="password" id="lp" name="cppwd" placeholder=" password" /> + {% endif %} <input type="hidden" name="uhash" id="uhash" value="x" /> <input type="submit" id="ls" value="login" /> {% if chpw %} diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 1a24beb2..52e666f3 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -416,7 +416,7 @@ try { catch (ex) { } tt.init(); -var o = QS('input[name="cppwd"]'); +var o = QS('input[name="uname"]') || QS('input[name="cppwd"]'); if (!ebi('c') && o.offsetTop + o.offsetHeight < window.innerHeight) o.focus(); diff --git a/tests/util.py b/tests/util.py index 5968215a..d3884ca9 100644 --- a/tests/util.py +++ b/tests/util.py @@ -143,7 +143,7 @@ class Cfg(Namespace): def __init__(self, a=None, v=None, c=None, **ka0): ka = {} - ex = "allow_flac allow_wav chpw cookie_lax daw dav_auth dav_mac dav_rt e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp early_ban ed emp exp force_js getmod grid gsel hardlink hardlink_only ih ihead localtime magic nid nih no_acode no_athumb no_bauth no_clone no_cp no_dav no_db_ip no_del no_dirsz no_dupe no_fnugg no_lifetime no_logues no_mv no_pipe no_poll no_readme no_robots no_sb_md no_sb_lg no_scandir no_tail no_tarcmp no_thumb no_vthumb no_zip nrand nsort nw og og_no_head og_s_title ohead q rand re_dirsz reflink rmagic rss smb srch_dbg srch_excl stats uqe vague_403 vc ver wo_up_readme write_uplog xdev xlink xvol zipmaxu zs" + ex = "allow_flac allow_wav chpw cookie_lax daw dav_auth dav_mac dav_rt e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp early_ban ed emp exp force_js getmod grid gsel hardlink hardlink_only ih ihead localtime magic nid nih no_acode no_athumb no_bauth no_clone no_cp no_dav no_db_ip no_del no_dirsz no_dupe no_fnugg no_lifetime no_logues no_mv no_pipe no_poll no_readme no_robots no_sb_md no_sb_lg no_scandir no_tail no_tarcmp no_thumb no_vthumb no_zip nrand nsort nw og og_no_head og_s_title ohead q rand re_dirsz reflink rmagic rss smb srch_dbg srch_excl stats uqe usernames vague_403 vc ver wo_up_readme write_uplog xdev xlink xvol zipmaxu zs" ka.update(**{k: False for k in ex.split()}) ex = "dav_inf dedup dotpart dotsrch hook_v no_dhash no_fastboot no_fpool no_htp no_rescan no_sendfile no_ses no_snap no_up_list no_voldump re_dhash see_dots plain_ip" From 1b71294aab0bfa4be10a71abacc84a397a6d78dc Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 20:30:09 +0000 Subject: [PATCH 113/174] close #387 --- copyparty/up2k.py | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/copyparty/up2k.py b/copyparty/up2k.py index 785d0a4f..b233cecb 100644 --- a/copyparty/up2k.py +++ b/copyparty/up2k.py @@ -375,11 +375,12 @@ class Up2k(object): if ineed == ihash or not ineed: continue + poke = job["poke"] zt = ( ineed / ihash, job["size"], - int(job["t0c"]), - int(job["poke"]), + int(job.get("t0c", poke)), + int(poke), djoin(vtop, job["prel"], job["name"]), ) ret.append(zt) From d9e3f998d1f5b508f7b57782d44ce43f9dd75e55 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 20:53:25 +0000 Subject: [PATCH 114/174] fix zipmaxu (it did nothing) --- copyparty/httpcli.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index cb061ac2..73af0df7 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -4667,7 +4667,9 @@ class HttpCli(object): else: fn = self.host.split(":")[0] - if vn.flags.get("zipmax") and (not self.uname or not "zipmaxu" in vn.flags): + if vn.flags.get("zipmax") and not ( + vn.flags.get("zipmaxu") and self.uname != "*" + ): maxs = vn.flags.get("zipmaxs_v") or 0 maxn = vn.flags.get("zipmaxn_v") or 0 nf = 0 From 1bf23fabc6838c79bd7a95b6c97661e16504e57d Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 20:59:41 +0000 Subject: [PATCH 115/174] unames: fix shares --- copyparty/httpcli.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 73af0df7..402d623f 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -2959,9 +2959,11 @@ class HttpCli(object): def handle_login(self) -> bool: assert self.parser # !rm - if self.args.usernames: + if self.args.usernames and not ( + self.args.shr and self.vpath.startswith(self.args.shr1) + ): try: - un = self.parser.require("uname", 256) + un = self.parser.require("uname", 64) except: un = "" else: From 36ab323d084d78491814ead1c71af5996ae249d4 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 21:26:59 +0000 Subject: [PATCH 116/174] sfx: simplify (remove bruteforce packing) --- scripts/make-sfx.sh | 21 ++++++--------------- scripts/rls.sh | 29 +---------------------------- 2 files changed, 7 insertions(+), 43 deletions(-) diff --git a/scripts/make-sfx.sh b/scripts/make-sfx.sh index deb7bee0..a1ee1a06 100755 --- a/scripts/make-sfx.sh +++ b/scripts/make-sfx.sh @@ -74,7 +74,6 @@ gtar=$(command -v gtar || command -v gnutar) || true sed() { gsed "$@"; } find() { gfind "$@"; } sort() { gsort "$@"; } - shuf() { gshuf "$@"; } nproc() { gnproc; } sha1sum() { shasum "$@"; } unexpand() { gunexpand "$@"; } @@ -157,9 +156,9 @@ stamp=$( done | sort | tail -n 1 | sha1sum | cut -c-16 ) -rm -rf sfx$CSN/* -mkdir -p sfx$CSN build -cd sfx$CSN +rm -rf sfx/* +mkdir -p sfx build +cd sfx tmpdir="$( printf '%s\n' "$TMPDIR" /tmp | @@ -395,7 +394,7 @@ ts=$(date -u +%s) hts=$(date -u +%Y-%m%d-%H%M%S) # --date=@$ts (thx osx) mkdir -p ../dist -sfx_out=../dist/copyparty-sfx$CSN +sfx_out=../dist/copyparty-sfx echo cleanup find -name '*.pyc' -delete @@ -554,7 +553,7 @@ gzres() { } -zdir="$tmpdir/cpp-mksfx$CSN" +zdir="$tmpdir/cpp-mksfx" [ -e "$zdir/$stamp" ] || rm -rf "$zdir" mkdir -p "$zdir" echo a > "$zdir/$stamp" @@ -583,15 +582,7 @@ echo gen tarlist for d in copyparty partftpy magic j2 py2 py37 ftp; do find $d -type f || true; done | # strip_hints sed -r 's/(.*)\.(.*)/\2 \1/' | LC_ALL=C sort | sed -r 's/([^ ]*) (.*)/\2.\1/' | grep -vE '/list1?$' > list1 - -for n in {1..50}; do - (grep -vE '\.gz$' list1; grep -E '\.gz$' list1 | (shuf||gshuf) ) >list || true - s=$( (sha1sum||shasum) < list | cut -c-16) - grep -q $s "$zdir/h" 2>/dev/null && continue - echo $s >> "$zdir/h" - break -done -[ $n -eq 50 ] && exit +(grep -vE '\.gz$' list1; grep -E '\.gz$' list1) >list echo creating tar tar -cf tar "${targs[@]}" --numeric-owner -T list diff --git a/scripts/rls.sh b/scripts/rls.sh index 613b53e0..e72eef0d 100755 --- a/scripts/rls.sh +++ b/scripts/rls.sh @@ -32,11 +32,7 @@ v=$1 rm -f ../dist/copyparty-sfx* shift ./make-sfx.sh "$@" -f=../dist/copyparty-sfx -[ -e $f.py ] && s= || s=-gz -# TODO: the -gz suffix is gone, can drop all the $s stuff probably - -$f$s.py --version >/dev/null +../dist/copyparty-sfx.py --version >/dev/null while [ "$1" ]; do case "$1" in @@ -46,27 +42,4 @@ while [ "$1" ]; do shift done -[ $parallel -gt 1 ] && { - printf '\033[%s' s 2r H "0;1;37;44mbruteforcing sfx size -- press enter to terminate" K u "7m $* " K $'27m\n' - trap "rm -f .sfx-run; printf '\033[%s' s r u" INT TERM EXIT - touch .sfx-run - min=99999999 - for ((a=0; a<$parallel; a++)); do - while [ -e .sfx-run ]; do - CSN=$a ./make-sfx.sh re "$@" - sz=$(wc -c <$f$a$s.py | awk '{print$1}') - [ $sz -ge $min ] && continue - mv $f$a$s.py $f$s.py.$sz - min=$sz - done & - done - read - exit -} - -while true; do - mv $f$s.py $f$s.$(wc -c <$f$s.py | awk '{print$1}').py - ./make-sfx.sh re "$@" -done - # git tag -d v$v; git push --delete origin v$v From 33497e6b11b41c82c1fc5694a46d80f0c0b9b417 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 21:28:56 +0000 Subject: [PATCH 117/174] sfx: add english-only --- README.md | 2 ++ scripts/rls.sh | 5 +++++ 2 files changed, 7 insertions(+) diff --git a/README.md b/README.md index e063e4af..476d8ddc 100644 --- a/README.md +++ b/README.md @@ -2848,6 +2848,8 @@ these are standalone programs and will never be imported / evaluated by copypart the self-contained "binary" (recommended!) [copyparty-sfx.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-sfx.py) will unpack itself and run copyparty, assuming you have python installed of course +if you only need english, [copyparty-en.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-en.py) is the same thing but smaller + you can reduce the sfx size by repacking it; see [./docs/devnotes.md#sfx-repack](./docs/devnotes.md#sfx-repack) diff --git a/scripts/rls.sh b/scripts/rls.sh index e72eef0d..767ca2de 100755 --- a/scripts/rls.sh +++ b/scripts/rls.sh @@ -33,6 +33,7 @@ rm -f ../dist/copyparty-sfx* shift ./make-sfx.sh "$@" ../dist/copyparty-sfx.py --version >/dev/null +mv ../dist/copyparty-{sfx,int}.py while [ "$1" ]; do case "$1" in @@ -42,4 +43,8 @@ while [ "$1" ]; do shift done +./make-sfx.sh re lang eng "$@" +mv ../dist/copyparty-{sfx,en}.py +mv ../dist/copyparty-{int,sfx}.py + # git tag -d v$v; git push --delete origin v$v From a98360f2134362b1c377f10545216bcc023d5917 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 21:30:55 +0000 Subject: [PATCH 118/174] copyparty.exe: update to python 3.13.6 --- scripts/pyinstaller/deps.sha512 | 2 +- scripts/pyinstaller/notes.txt | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/scripts/pyinstaller/deps.sha512 b/scripts/pyinstaller/deps.sha512 index b698b95a..1d324c0e 100644 --- a/scripts/pyinstaller/deps.sha512 +++ b/scripts/pyinstaller/deps.sha512 @@ -30,5 +30,5 @@ a726fb46cce24f781fc8b55a3e6dea0a884ebc3b2b400ea74aa02333699f4955a5dc1e2ec5927ac7 3e39ea6e16b502d99a2e6544579095d0f7c6097761cd85135d5e929b9dec1b32e80669a846f94ee8c2cca9be2f5fe728625d09453988864c04e16bb8445c3f91 pillow-11.3.0-cp313-cp313-win_amd64.whl 59fbbcae044f4ee73d203ac74b553b27bfad3e6b2f3fb290fd3f8774753c6b545176b6b3399c240b092d131d152290ce732750accd962dc1e48e930be85f5e53 pyinstaller-6.14.1-py3-none-win_amd64.whl fc6f3e144c5f5b662412de07cb8bf0c2eb3b3be21d19ec448aef3c4244d779b9ab8027fd67a4871e6e13823b248ea0f5a7a9241a53aef30f3b51a6d3cb5bdb3f pyinstaller_hooks_contrib-2025.5-py3-none-any.whl -2c7a52e223b8186c21009d3fa5ed6a856d8eb4ef3b98f5d24c378c6a1afbfa1378bd7a51d6addc500e263d7989efb544c862bf920055e740f137c702dfd9d18b python-3.13.5-amd64.exe +3c37ea72ab062f65116a5faaaccbaa960a971797c78dbe6a504f41e562b018e7f28924647b5577fdb4fedfa61ffe0f1153842a8585f93b0332ed4d97905f6609 python-3.13.6-amd64.exe 2a0420f7faaa33d2132b82895a8282688030e939db0225ad8abb95a47bdb87b45318f10985fc3cee271a9121441c1526caa363d7f2e4a4b18b1a674068766e87 setuptools-80.9.0-py3-none-any.whl diff --git a/scripts/pyinstaller/notes.txt b/scripts/pyinstaller/notes.txt index 61f76fa4..44cebd10 100644 --- a/scripts/pyinstaller/notes.txt +++ b/scripts/pyinstaller/notes.txt @@ -40,7 +40,7 @@ fns=( pillow-11.3.0-cp313-cp313-win_amd64.whl pyinstaller-6.14.1-py3-none-win_amd64.whl pyinstaller_hooks_contrib-2025.5-py3-none-any.whl - python-3.13.5-amd64.exe + python-3.13.6-amd64.exe setuptools-80.9.0-py3-none-any.whl ) [ $w7 ] && fns+=( From c955658332d12b29525b4e392b9467a66832132c Mon Sep 17 00:00:00 2001 From: Jakub Pelc <jakub.pelc@seznam.cz> Date: Thu, 7 Aug 2025 23:48:32 +0200 Subject: [PATCH 119/174] Czech translation (#471) * added czech translation * add czech translations to splash --------- Signed-off-by: Jakub Pelc <jakub.pelc@seznam.cz> Signed-off-by: ed <s@ocv.me> Co-authored-by: ed <s@ocv.me> --- copyparty/web/browser.js | 635 ++++++++++++++++++++++++++++++++++++++- copyparty/web/splash.js | 41 +++ 2 files changed, 675 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 0563d8ee..1b95ced2 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -1894,6 +1894,639 @@ var Ls = { "lang_set": "刷新以使更改生效?", }, + "cze": { + "tt": "Čeština", + + "cols": { + "c": "tlačítka akcí", + "dur": "doba trvání", + "q": "kvalita / bitrate", + "Ac": "audio kodek", + "Vc": "video kodek", + "Fmt": "formát / kontejner", + "Ahash": "kontrolní součet audia", + "Vhash": "kontrolní součet videa", + "Res": "rozlišení", + "T": "typ souboru", + "aq": "kvalita zvuku / bitrate", + "vq": "kvalita videa / bitrate", + "pixfmt": "podvzorkování / struktura pixelů", + "resw": "horizontální rozlišení", + "resh": "vertikální rozlišení", + "chs": "audio kanály", + "hz": "vzorkovací frekvence" + }, + + "hks": [ + [ + "různé", + ["ESC", "zavřít různé věci"], + + "správce souborů", + ["G", "přepnout seznam / zobrazení mřížky"], + ["T", "přepnout náhledy / ikony"], + ["⇧ A/D", "velikost náhledů"], + ["ctrl-K", "smazat vybrané"], + ["ctrl-X", "vyjmout výběr do schránky"], + ["ctrl-C", "kopírovat výběr do schránky"], + ["ctrl-V", "vložit (přesunout/kopírovat) zde"], + ["Y", "stáhnout vybrané"], + ["F2", "přejmenovat vybrané"], + + "výběr souborů", + ["space", "přepnout výběr souboru"], + ["↑/↓", "posunout kurzor výběru"], + ["ctrl ↑/↓", "posunout kurzor a zobrazení"], + ["⇧ ↑/↓", "vybrat předchozí/následující soubor"], + ["ctrl-A", "vybrat všechny soubory / složky"], + ], [ + "navigace", + ["B", "přepnout drobečkovou navigaci / navigační panel"], + ["I/K", "předchozí/následující složka"], + ["M", "nadřazená složka (nebo sbalit aktuální)"], + ["V", "přepnout složky / textové soubory v navigačním panelu"], + ["A/D", "velikost navigačního panelu"], + ], [ + "audio přehrávač", + ["J/L", "předchozí/následující skladba"], + ["U/O", "přeskočit 10 sekund zpět/vpřed"], + ["0..9", "přejít na 0%..90%"], + ["P", "přehrát/pozastavit (také spustit)"], + ["S", "vybrat přehrávanou skladbu"], + ["Y", "stáhnout skladbu"], + ], [ + "prohlížeč obrázků", + ["J/L, ←/→", "předchozí/následující obrázek"], + ["Home/End", "první/poslední obrázek"], + ["F", "celá obrazovka"], + ["R", "otočit po směru hodinových ručiček"], + ["⇧ R", "otočit proti směru hodinových ručiček"], + ["S", "vybrat obrázek"], + ["Y", "stáhnout obrázek"], + ], [ + "video přehrávač", + ["U/O", "přeskočit 10 sekund zpět/vpřed"], + ["P/K/Space", "přehrát/pozastavit"], + ["C", "pokračovat přehráváním následující"], + ["V", "smyčka"], + ["M", "ztlumit"], + ["[ and ]", "nastavit interval smyčky"], + ], [ + "prohlížeč textových souborů", + ["I/K", "předchozí/následující soubor"], + ["M", "zavřít textový soubor"], + ["E", "upravit textový soubor"], + ["S", "vybrat soubor (pro vyjmutí/kopírování/přejmenování)"], + ] + ], + + "m_ok": "OK", + "m_ng": "Zrušit", + + "enable": "Povolit", + "danger": "NEBEZPEČÍ", + "clipped": "zkopírováno do schránky", + + "ht_s1": "sekunda", + "ht_s2": "sekundy", + "ht_s5": "sekund", + "ht_m1": "minuta", + "ht_m2": "minuty", + "ht_m5": "minut", + "ht_h1": "hodina", + "ht_h2": "hodiny", + "ht_h5": "hodin", + "ht_d1": "den", + "ht_d2": "dny", + "ht_d5": "dní", + "ht_and": " a ", + + "goh": "ovládací panel", + "gop": 'předchozí sourozenec">předchozí', + "gou": 'nadřazená složka">nahoru', + "gon": 'následující složka">následující', + "logout": "Odhlásit ", + "access": " přístup", + "ot_close": "zavřít podnabídku", + "ot_search": "hledat soubory podle atributů, cesty / názvu, hudebních tagů nebo jejich kombinace$N$N<code>foo bar</code> = musí obsahovat jak «foo» tak «bar»,$N<code>foo -bar</code> = musí obsahovat «foo» ale ne «bar»,$N<code>^yana .opus$</code> = začíná na «yana» a je to «opus» soubor$N<code>"try unite"</code> = obsahuje přesně «try unite»$N$Nformát data je iso-8601, jako$N<code>2009-12-31</code> nebo <code>2020-09-12 23:30:00</code>", + "ot_unpost": "unpost: smazat vaše nedávné nahrání nebo zrušit nedokončené", + "ot_bup": "bup: základní nahrávač, podporuje i netscape 4.0", + "ot_mkdir": "mkdir: vytvořit nový adresář", + "ot_md": "new-md: vytvořit nový markdown dokument", + "ot_msg": "msg: poslat zprávu do logu serveru", + "ot_mp": "možnosti přehrávače médií", + "ot_cfg": "možnosti konfigurace", + "ot_u2i": 'up2k: nahrát soubory (pokud máte oprávnění k zápisu) nebo přepnout do vyhledávacího režimu a podívat se, zda existují někde na serveru$N$Nnahrávání je obnovitelné, vícevláknové a časové značky souborů jsou zachovány, ale používá více CPU než [🎈]  (základní nahrávač)<br /><br />během nahrávání se tato ikona stává indikátorem průběhu!', + "ot_u2w": 'up2k: nahrát soubory s podporou obnovení (zavřete prohlížeč a stejné soubory přetáhněte později)$N$Nvícevláknové a časové značky souborů jsou zachovány, ale používá více CPU než [🎈]  (základní nahrávač)<br /><br />během nahrávání se tato ikona stává indikátorem průběhu!', + "ot_noie": 'Prosím použijte Chrome / Firefox / Edge', + + "ab_mkdir": "vytvořit adresář", + "ab_mkdoc": "nový markdown dokument", + "ab_msg": "poslat zprávu do logu serveru", + + "ay_path": "přejít na složky", + "ay_files": "přejít na soubory", + + "wt_ren": "přejmenovat vybrané položky$NKlávesová zkratka: F2", + "wt_del": "smazat vybrané položky$NKlávesová zkratka: ctrl-K", + "wt_cut": "vyjmout vybrané položky <small>(pak je vložit někam jinam)</small>$NKlávesová zkratka: ctrl-X", + "wt_cpy": "kopírovat vybrané položky do schránky$N(pro vložení někam jinam)$NKlávesová zkratka: ctrl-C", + "wt_pst": "vložit dříve vyjmutý / zkopírovaný výběr$NKlávesová zkratka: ctrl-V", + "wt_selall": "vybrat všechny soubory$NKlávesová zkratka: ctrl-A (když je zaměřen soubor)", + "wt_selinv": "invertovat výběr", + "wt_zip1": "stáhnout tuto složku jako archiv", + "wt_selzip": "stáhnout výběr jako archiv", + "wt_seldl": "stáhnout výběr jako samostatné soubory$NKlávesová zkratka: Y", + "wt_npirc": "kopírovat informace o stopě ve formátu irc", + "wt_nptxt": "kopírovat informace o stopě v prostém textu", + "wt_m3ua": "přidat do m3u playlistu (klikněte později na <code>📻kopírovat</code>)", + "wt_m3uc": "kopírovat m3u playlist do schránky", + "wt_grid": "přepnout zobrazení mřížky / seznamu$NKlávesová zkratka: G", + "wt_prev": "předchozí stopa$NKlávesová zkratka: J", + "wt_play": "přehrát / pozastavit$NKlávesová zkratka: P", + "wt_next": "následující stopa$NKlávesová zkratka: L", + + "ul_par": "paralelní nahrávání:", + "ut_rand": "náhodné názvy souborů", + "ut_u2ts": "kopírovat časovou značku poslední změny$Nz vašeho souborového systému na server\">📅", + "ut_ow": "přepsat existující soubory na serveru?$N🛡️: nikdy (místo toho vytvoří nový název souboru)$N🕒: přepsat pokud je soubor na serveru starší než váš$N♻️: vždy přepsat pokud se soubory liší", + "ut_mt": "pokračovat v hashování ostatních souborů během nahrávání$N$Nmožná zakázat pokud je vaše CPU nebo HDD bottleneckem", + "ut_ask": 'požádat o potvrzení před zahájením nahrávání">💭', + "ut_pot": "zlepšit rychlost nahrávání na pomalých zařízeních$Nzjednodušením UI", + "ut_srch": "skutečně nenahrávat, místo toho zkontrolovat zda soubory již $N existují na serveru (prohledá všechny složky které můžete číst)", + "ut_par": "pozastavit nahrávání nastavením na 0$N$Nzvýšit pokud je vaše připojení pomalé / vysoká latence$N$Nponechat na 1 v LAN nebo pokud je HDD serveru bottleneckem", + "ul_btn": "přetáhněte soubory / složky<br>sem (nebo sem klikněte)", + "ul_btnu": "N A H R Á T", + "ul_btns": "H L E D A T", + + "ul_hash": "hash", + "ul_send": "odeslat", + "ul_done": "hotovo", + "ul_idle1": "zatím nejsou zařazena žádná nahrávání", + "ut_etah": "průměrná rychlost <em>hashování</em> a odhadovaný čas do dokončení", + "ut_etau": "průměrná rychlost <em>nahrávání</em> a odhadovaný čas do dokončení", + "ut_etat": "průměrná <em>celková</em> rychlost a odhadovaný čas do dokončení", + + "uct_ok": "úspěšně dokončeno", + "uct_ng": "nedobré: selhalo / odmítnuto / nenalezeno", + "uct_done": "celkem", + "uct_bz": "hashování nebo nahrávání", + "uct_q": "nečinné, čekající", + + "utl_name": "název souboru", + "utl_ulist": "seznam", + "utl_ucopy": "kopírovat", + "utl_links": "odkazy", + "utl_stat": "stav", + "utl_prog": "průběh", + + // keep short: + "utl_404": "404", + "utl_err": "CHYBA", + "utl_oserr": "chyba OS", + "utl_found": "nalezeno", + "utl_defer": "odložit", + "utl_yolo": "YOLO", + "utl_done": "hotovo", + + "ul_flagblk": "soubory byly přidány do fronty</b><br>nicméně v jiné kartě prohlížeče běží up2k,<br>takže čekáme až skončí", + "ul_btnlk": "konfigurace serveru uzamkla tento přepínač v tomto stavu", + + "udt_up": "Nahrávání", + "udt_srch": "Hledání", + "udt_drop": "přetáhněte to sem", + + "u_nav_m": '<h6>jasnačka, co máte?</h6><code>Enter</code> = Soubory (jeden nebo více)\n<code>ESC</code> = Jednu složku (včetně podsložek)', + "u_nav_b": '<a href="#" id="modal-ok">Soubory</a><a href="#" id="modal-ng">Jedna složka</a>', + + "cl_opts": "přepínače", + "cl_themes": "téma", + "cl_langs": "jazyk", + "cl_ziptype": "stahování složky", + "cl_uopts": "up2k přepínače", + "cl_favico": "favicon", + "cl_bigdir": "velké adresáře", + "cl_hsort": "#řazení", + "cl_keytype": "notace kláves", + "cl_hiddenc": "skryté sloupce", + "cl_hidec": "skrýt", + "cl_reset": "resetovat", + "cl_hpick": "klepněte na záhlaví sloupců pro skrytí v tabulce níže", + "cl_hcancel": "skrývání sloupců zrušeno", + + "ct_grid": '田 mřížka', + "ct_ttips": '◔ ◡ ◔">ℹ️ nápovědy', + "ct_thumb": 'v zobrazení mřížky přepnout ikony nebo náhledy$NKlávesová zkratka: T">🖼️ náhledy', + "ct_csel": 'použít CTRL a SHIFT pro výběr souborů v zobrazení mřížky">výběr', + "ct_ihop": 'když se zavře prohlížeč obrázků, posunout dolů k naposledy zobrazenému souboru">g⮯', + "ct_dots": 'zobrazit skryté soubory (pokud to server povoluje)">dotfiles', + "ct_qdel": 'při mazání souborů požádat o potvrzení jen jednou">rychlé mazání', + "ct_dir1st": 'řadit složky před soubory">📁 první', + "ct_nsort": 'přirozené řazení (pro názvy souborů s úvodními číslicemi)">přirozené řazení', + "ct_utc": 'zobrazit všechny časy v UTC">UTC', + "ct_readme": 'zobrazit README.md v seznamech složek">📜 readme', + "ct_idxh": 'zobrazit index.html místo seznamu složky">htm', + "ct_sbars": 'zobrazit posuvníky">⟊', + + "cut_umod": "pokud soubor na serveru již existuje, aktualizovat časovou značku posledního změny serveru tak, aby odpovídala vašemu lokálnímu souboru (vyžaduje oprávnění k zápisu+mazání)\">re📅", + + "cut_turbo": "yolo tlačítko, pravděpodobně to NECHCETE povolit:$N$Npoužijte to pokud jste nahrávali obrovské množství souborů a museli jste restartovat z nějakého důvodu a chcete pokračovat v nahrávání ASAP$N$Ntoto nahradí hash-kontrolu jednoduchým <em>"má to stejnou velikost souboru na serveru?"</em> takže pokud se obsah souborů liší, NEBUDE nahrán$N$Nměli byste to vypnout když nahrávání skončí a pak znovu "nahrát" stejné soubory aby je klient ověřil\">turbo", + + "cut_datechk": "nemá žádný efekt pokud není povoleno turbo tlačítko$N$Nsnižuje yolo faktor o trochu; kontroluje zda časové značky souborů na serveru odpovídají vašim$N$Nměl by <em>teoreticky</em> zachytit většinu nedokončených / poškozených nahrávání, ale není náhradou za ověřovací průchod s turbem vypnutým poté\">kontrola data", + + "cut_u2sz": "velikost (v MiB) každého kusu nahrávání; velké hodnoty lépe létají přes atlantik. Zkuste nízké hodnoty na velmi nespolehlivých připojeních", + + "cut_flag": "zajistit aby nahrávala jen jedna karta najednou $N -- ostatní karty to musí mít také povoleno $N -- ovlivňuje jen karty na stejné doméně", + + "cut_az": "nahrávat soubory v abecedním pořadí, spíše než nejmenší-soubor-první$N$Nabecední pořadí může usnadnit kontrolu zda se něco pokazilo na serveru, ale činí nahrávání mírně pomalejší na optice / LAN", + + "cut_nag": "notifikace OS když nahrávání skončí$N(jen pokud prohlížeč nebo karta není aktivní)", + "cut_sfx": "zvukové upozornění když nahrávání skončí$N(jen pokud prohlížeč nebo karta není aktivní)", + + "cut_mt": "použít vícevláknové zpracování pro zrychlení hashování souborů$N$Ntoto používá web-workers a vyžaduje$Nvíce RAM (až 512 MiB navíc)$N$Ndělá https o 30% rychlejší a http 4,5x rychlejší\">mt", + + "cut_wasm": "použijte wasm místo vestavěného hashování prohlížeče; zlepšuje rychlost na prohlížečích založených na chrome ale zvyšuje zátěž CPU, mnoho starších verzí chrome má chyby které způsobují že prohlížeč spotřebuje veškerou RAM a spadne pokud je toto povoleno\">wasm", + + "cft_text": "text favicon (prázdné a obnovte pro zakázání)", + "cft_fg": "barva popředí", + "cft_bg": "barva pozadí", + + "cdt_lim": "maximální počet souborů k zobrazení ve složce", + "cdt_ask": "při posunování na konec,$Nmísto načítání více souborů,$N se zeptat co dělat", + "cdt_hsort": "kolik pravidel řazení (<code>,sorthref</code>) zahrnout do media-URL. Nastavení na 0 bude také ignorovat pravidla řazení zahrnutá v media odkazech při kliknutí na ně", + + "tt_entree": "zobrazit navigační panel (postranní strom adresářů)$NKlávesová zkratka: B", + "tt_detree": "zobrazit drobečkovou navigaci$NKlávesová zkratka: B", + "tt_visdir": "posunout k vybrané složce", + "tt_ftree": "přepnout strom složek / textové soubory$NKlávesová zkratka: V", + "tt_pdock": "zobrazit nadřazené složky v ukotveném panelu nahoře", + "tt_dynt": "automaticky rozrůstat jak se strom rozšiřuje", + "tt_wrap": "zalomení řádků", + "tt_hover": "odhalit přetékající řádky při najetí$N( ruší posun pokud kurzor myši $N  není v levém okraji )", + + "ml_pmode": "na konci složky...", + "ml_btns": "příkazy", + "ml_tcode": "transkódovat", + "ml_tcode2": "transkódovat na", + "ml_tint": "odstín", + "ml_eq": "audio ekvalizér", + "ml_drc": "kompresor dynamického rozsahu", + + "mt_loop": "smyčka/opakovat jednu skladbu\">🔁", + "mt_one": "zastavit po jedné skladbě\">1️⃣", + "mt_shuf": "zamíchat skladby v každé složce\">🔀", + "mt_aplay": "automatické přehrávání pokud je ID skladby v odkazu kterým jste přišli na server$N$Nzakázání toho také zastaví aktualizaci URL stránky s ID skladby při přehrávání hudby, aby se zabránilo automatickému přehrávání pokud se tato nastavení ztratí ale URL zůstane\">a▶", + "mt_preload": "začít načítat následující skladbu před koncem pro plynulé přehrávání\">přednahrání", + "mt_prescan": "přejít do následující složky před tím než$Nskončí poslední skladba, aby byl webprohlížeč$Nspokojen aby nezastavil přehrávání\">nav", + "mt_fullpre": "zkusit přednahrát celou skladbu;$N✅ povolit na <b>nespolehlivých</b> připojeních,$N❌ <b>zakázat</b> na pomalých připojeních pravděpodobně\">úplné", + "mt_fau": "na telefonech zabránit zastavení hudby, pokud se další píseň nenahraje dostatečně rychle (může způsobit chybné zobrazení tagů)\">☕️", + "mt_waves": "vlnový posuvník:$Nzobrazit amplitudu zvuku v posuvníku\">~s", + "mt_npclip": "zobrazit tlačítka pro kopírování aktuálně přehrávané písně do schránky\">/np", + "mt_m3u_c": "zobrazit tlačítka pro kopírování$Nvybraných písní jako položky m3u8 playlistu\">📻", + "mt_octl": "integrace s OS (mediální klávesy / osd)\">os-ctl", + "mt_oseek": "povolit posunování přes integraci s OS$N$Npoznámka: na některých zařízeních (iPhone),$Nto nahradí tlačítko další písně\">seek", + "mt_oscv": "zobrazit obal alba v osd\">art", + "mt_follow": "udržet přehrávanou stopu v zobrazení\">🎯", + "mt_compact": "kompaktní ovládání\">⟎", + "mt_uncache": "vymazat cache  (zkuste to, pokud váš prohlížeč uložil$Nporušenou kopii písně a odmítá ji přehrát)\">uncache", + "mt_mloop": "opakovat otevřenou složku\">🔁 loop", + "mt_mnext": "načíst další složku a pokračovat\">📂 next", + "mt_mstop": "zastavit přehrávání\">⏸ stop", + "mt_cflac": "převést flac / wav na opus\">flac", + "mt_caac": "převést aac / m4a na opus\">aac", + "mt_coth": "převést všechny ostatní (ne mp3) na opus\">oth", + "mt_c2opus": "nejlepší volba pro desktopy, laptopy, android\">opus", + "mt_c2owa": "opus-weba, pro iOS 17.5 a novější\">owa", + "mt_c2caf": "opus-caf, pro iOS 11 až 17\">caf", + "mt_c2mp3": "použijte na velmi starých zařízeních\">mp3", + "mt_c2flac": "nejlepší kvalita zvuku, ale obrovské stahování\">flac", + "mt_c2wav": "nekomprimované přehrávání (ještě větší)\">wav", + "mt_c2ok": "výborně, dobrá volba", + "mt_c2nd": "to není doporučený výstupní formát pro vaše zařízení, ale v pořádku", + "mt_c2ng": "vaše zařízení, zdá se, nepodporuje tento výstupní formát, ale zkusíme to", + "mt_xowa": "v iOS jsou chyby bránící přehrávání na pozadí s tímto formátem; použijte prosím caf nebo mp3", + "mt_tint": "úroveň pozadí (0-100) na posuvníku$Nabyste učinili ukládání do vyrovnávací paměti méně rušivým", + "mt_eq": "povoluje ekvalizér a ovládání zisku;$N$Nboost <code>0</code> = standardní 100% hlasitost (nezměněno)$N$Nwidth <code>1  </code> = standardní stereo (nezměněno)$Nwidth <code>0.5</code> = 50% levý-pravý crossfeed$Nwidth <code>0  </code> = mono$N$Nboost <code>-0.8</code> & width <code>10</code> = odstranění vokálů :^)$N$Npovolení ekvalizéru činí alba bez mezer zcela bez mezer, takže to nechte zapnuté se všemi hodnotami na nule (kromě width = 1), pokud vám na tom záleží", + "mt_drc": "povoluje kompresor dynamického rozsahu (vyrovnávač hlasitosti / brickwaller); také povolí EQ pro vyvážení špaget, takže nastavte všechna EQ pole kromě 'width' na 0, pokud to nechcete$N$Nsnižuje hlasitost zvuku nad THRESHOLD dB; pro každý RATIO dB za THRESHOLD je 1 dB výstupu, takže výchozí hodnoty tresh -24 a ratio 12 znamenají, že by nikdy nemělo být hlasitější než -22 dB a je bezpečné zvýšit boost ekvalizéru na 0.8, nebo dokonce 1.8 s ATK 0 a obrovským RLS jako 90 (funguje pouze ve firefoxu; RLS je max 1 v jiných prohlížečích)$N$N(viz wikipedia, vysvětlují to mnohem lépe)", + + "mb_play": "přehrát", + "mm_hashplay": "přehrát tento audio soubor?", + "mm_m3u": "stiskněte <code>Enter/OK</code> pro Přehrání\nstiskněte <code>ESC/Cancel</code> pro Úpravu", + "mp_breq": "potřebujete firefox 82+ nebo chrome 73+ nebo iOS 15+", + "mm_bload": "nyní se načítá...", + "mm_bconv": "převádí se na {0}, čekejte prosím...", + "mm_opusen": "váš prohlížeč nemůže přehrát aac / m4a soubory;\ntranscoding na opus je nyní povolen", + "mm_playerr": "přehrávání selhalo: ", + "mm_eabrt": "Pokus o přehrávání byl zrušen", + "mm_enet": "Vaše internetové připojení je nestabilní", + "mm_edec": "Tento soubor je údajně poškozený??", + "mm_esupp": "Váš prohlížeč nerozumí tomuto audio formátu", + "mm_eunk": "Neznámá chyba", + "mm_e404": "Nelze přehrát audio; chyba 404: Soubor nenalezen.", + "mm_e403": "Nelze přehrát audio; chyba 403: Přístup odepřen.\n\nZkuste stisknout F5 pro obnovení, možná jste se odhlásili", + "mm_e500": "Nelze přehrát audio; chyba 500: Zkontrolujte logy serveru.", + "mm_e5xx": "Nelze přehrát audio; chyba serveru ", + "mm_nof": "žádné další audio soubory v okolí nenalezeny", + "mm_prescan": "Hledám hudbu k dalšímu přehrání...", + "mm_scank": "Další píseň nalezena:", + "mm_uncache": "cache vymazána; všechny písně se znovu stáhnou při dalším přehrávání", + "mm_hnf": "tato píseň již neexistuje", + + "im_hnf": "tento obrázek již neexistuje", + + "f_empty": 'tato složka je prázdná', + "f_chide": 'toto skryje sloupec «{0}»\n\nmůžete odkrýt sloupce v záložce nastavení', + "f_bigtxt": "tento soubor má {0} MiB -- opravdu zobrazit jako text?", + "f_bigtxt2": "zobrazit pouze konec souboru? to také povolí sledování/tailing, zobrazí nově přidané řádky textu v reálném čase", + "fbd_more": '<div id="blazy">zobrazuji <code>{0}</code> z <code>{1}</code> souborů; <a href="#" id="bd_more">zobraz {2}</a> nebo <a href="#" id="bd_all">zobraz všechny</a></div>', + "fbd_all": '<div id="blazy">zobrazuji <code>{0}</code> z <code>{1}</code> souborů; <a href="#" id="bd_all">zobraz všechny</a></div>', + "f_anota": "pouze {0} z {1} položek bylo vybráno;\npro výběr celé složky nejprve přejděte na konec", + + "f_dls": 'odkazy na soubory v aktuální složce byly\nzměněny na odkazy ke stažení', + + "f_partial": "Pro bezpečné stažení souboru, který se aktuálně nahrává, klikněte prosím na soubor se stejným názvem, ale bez přípony <code>.PARTIAL</code>. Stiskněte prosím CANCEL nebo Escape.\n\nStisknutím OK / Enter ignorujete toto varování a pokračujete ve stahování <code>.PARTIAL</code> dočasného souboru, což téměř jistě vyústí jako poškozená data.", + + "ft_paste": "vložit {0} položek$NKlávesová zkratka: ctrl-V", + "fr_eperm": 'nelze přejmenovat:\nnemáte oprávnění "přesunout" v této složce', + "fd_eperm": 'nelze smazat:\nnemáte oprávnění "smazat" v této složce', + "fc_eperm": 'nelze vyjmout:\nnemáte oprávnění "přesunout" v této složce', + "fp_eperm": 'nelze vložit:\nnemáte oprávnění "zapisovat" v této složce', + "fr_emore": "vyberte alespoň jednu položku k přejmenování", + "fd_emore": "vyberte alespoň jednu položku ke smazání", + "fc_emore": "vyberte alespoň jednu položku k vyjmutí", + "fcp_emore": "vyberte alespoň jednu položku k zkopírování do schránky", + + "fs_sc": "sdílet složku, ve které se nacházíte", + "fs_ss": "sdílet vybrané soubory", + "fs_just1d": "nelze vybrat více než jednu složku,\nnebo míchat soubory a složky v jednom výběru", + "fs_abrt": "❌ zrušit", + "fs_rand": "🎲 náhodný.název", + "fs_go": "✅ vytvořit sdílení", + "fs_name": "název", + "fs_src": "zdroj", + "fs_pwd": "heslo", + "fs_exp": "vypršení", + "fs_tmin": "min", + "fs_thrs": "hodin", + "fs_tdays": "dní", + "fs_never": "navždy", + "fs_pname": "volitelný název odkazu; bude náhodný, pokud je prázdný", + "fs_tsrc": "soubor nebo složka ke sdílení", + "fs_ppwd": "volitelné heslo", + "fs_w8": "vytváření sdílení...", + "fs_ok": "stiskněte <code>Enter/OK</code> pro zkopírování do schránky\nstiskněte <code>ESC/Zrušit</code> pro zavření", + + "frt_dec": "může opravit některé případy porušených názvů souborů\">url-decode", + "frt_rst": "resetovat změněné názvy souborů zpět na původní\">↺ reset", + "frt_abrt": "zrušit a zavřít toto okno\">❌ cancel", + "frb_apply": "PŘEJMENOVAT", + "fr_adv": "dávkové / metadata / přejmenování podle vzoru\">pokročilé", + "fr_case": "regex citlivý na velikost písmen\">velikost", + "fr_win": "názvy bezpečné pro windows; nahradit <code><>:"\\|?*</code> japonskými plnošířkovými znaky\">win", + "fr_slash": "nahradit <code>/</code> znakem který nezpůsobí vytvoření nových složek\">žádné /", + "fr_re": "vzor regex hledání k aplikaci na původní názvy souborů; zachycené skupiny mohou být odkazovány v poli formátu níže jako <code>(1)</code> a <code>(2)</code> atd.", + "fr_fmt": "inspirováno foobar2000:$N<code>(title)</code> je nahrazeno názvem skladby,$N<code>[(artist) - ](title)</code> přeskočí [tuto] část pokud je umělec prázdný$N<code>$lpad((tn),2,0)</code> doplní číslo stopy na 2 číslice", + "fr_pdel": "smazat", + "fr_pnew": "uložit jako", + "fr_pname": "zadejte název pro vaše nové přednastavení", + "fr_aborted": "zrušeno", + "fr_lold": "starý název", + "fr_lnew": "nový název", + "fr_tags": "tagy pro vybrané soubory (pouze pro čtení, jen pro referenci):", + "fr_busy": "přejmenovávám {0} položek...\n\n{1}", + "fr_efail": "přejmenování selhalo:\n", + "fr_nchg": "{0} z nových názvů bylo změněno kvůli <code>win</code> a/nebo <code>žádné /</code>\n\nPokračovat s těmito změněnými novými názvy?", + + "fd_ok": "mazání OK", + "fd_err": "mazání selhalo:\n", + "fd_none": "nic nebylo smazáno; možná blokováno konfigurací serveru (xbd)?", + "fd_busy": "mažu {0} položek...\n\n{1}", + "fd_warn1": "SMAZAT těchto {0} položek?", + "fd_warn2": "<b>Poslední šance!</b> Nelze vrátit zpět. Smazat?", + + "fc_ok": "vyjmout {0} položek", + "fc_warn": 'vyjmout {0} položek\n\nale: pouze <b>tato</b> karta prohlížeče je může vložit\n(protože výběr je tak absolutně masivní)', + + "fcc_ok": "zkopírováno {0} položek do schránky", + "fcc_warn": 'zkopírováno {0} položek do schránky\n\nale: pouze <b>tato</b> karta prohlížeče je může vložit\n(protože výběr je tak absolutně masivní)', + + "fp_apply": "použít tyto názvy", + "fp_ecut": "nejprve vyjměte nebo zkopírujte nějaké soubory / složky k vložení / přesunutí\n\npoznámka: můžete vyjmout / vložit přes různé karty prohlížeče", + "fp_ename": "{0} položek sem nelze přesunout protože názvy jsou již obsazené. Dejte jim nové názvy níže pro pokračování, nebo název nechte prázdný pro přeskočení:", + "fcp_ename": "{0} položek sem nelze zkopírovat protože názvy jsou již obsazené. Dejte jim nové názvy níže pro pokračování, nebo název nechte prázdný pro přeskočení:", + "fp_emore": "stále jsou některé kolize názvů souborů k opravě", + "fp_ok": "přesun OK", + "fcp_ok": "kopírování OK", + "fp_busy": "přesouvám {0} položek...\n\n{1}", + "fcp_busy": "kopíruji {0} položek...\n\n{1}", + "fp_err": "přesun selhal:\n", + "fcp_err": "kopírování selhalo:\n", + "fp_confirm": "přesunout těchto {0} položek sem?", + "fcp_confirm": "zkopírovat těchto {0} položek sem?", + "fp_etab": 'selhalo čtení schránky z jiné karty prohlížeče', + "fp_name": "nahrávání souboru z vašeho zařízení. Dejte mu název:", + "fp_both_m": '<h6>vyberte co vložit</h6><code>Enter</code> = Přesunout {0} souborů z «{1}»\n<code>ESC</code> = Nahrát {2} souborů z vašeho zařízení', + "fcp_both_m": '<h6>vyberte co vložit</h6><code>Enter</code> = Kopírovat {0} souborů z «{1}»\n<code>ESC</code> = Nahrát {2} souborů z vašeho zařízení', + "fp_both_b": '<a href="#" id="modal-ok">Přesunout</a><a href="#" id="modal-ng">Nahrát</a>', + "fcp_both_b": '<a href="#" id="modal-ok">Kopírovat</a><a href="#" id="modal-ng">Nahrát</a>', + + "mk_noname": "napište název do textového pole vlevo předtím než to uděláte :p", + + "tv_load": "Načítání textového dokumentu:\n\n{0}\n\n{1}% ({2} z {3} MiB načteno)", + "tv_xe1": "nelze načíst textový soubor:\n\nchyba ", + "tv_xe2": "404, soubor nenalezen", + "tv_lst": "seznam textových souborů v", + "tvt_close": "návrat do zobrazení složky$NKlávesová zkratka: M (nebo Esc)\">❌ zavřít", + "tvt_dl": "stáhnout tento soubor$NKlávesová zkratka: Y\">💾 stáhnout", + "tvt_prev": "zobrazit předchozí dokument$NKlávesová zkratka: i\">⬆ předchozí", + "tvt_next": "zobrazit následující dokument$NKlávesová zkratka: K\">⬇ další", + "tvt_sel": "vybrat soubor   ( pro vyjmutí / kopírování / mazání / ... )$NKlávesová zkratka: S\">výběr", + "tvt_edit": "otevřít soubor v textovém editoru$NKlávesová zkratka: E\">✏️ upravit", + "tvt_tail": "sledovat soubor pro změny; zobrazit nové řádky v reálném čase\">📡 sledovat", + "tvt_wrap": "zalamování slov\">↵", + "tvt_atail": "zamknout posun na konec stránky\">⚓", + "tvt_ctail": "dekódovat barvy terminálu (ansi escape kódy)\">🌈", + "tvt_ntail": "limit zpětného posouvání (kolik bajtů textu ponechat načtených)", + + "m3u_add1": "skladba přidána do m3u playlistu", + "m3u_addn": "{0} skladeb přidáno do m3u playlistu", + "m3u_clip": "m3u playlist nyní zkopírován do schránky\n\nměli byste vytvořit nový textový soubor pojmenovaný něco.m3u a vložit playlist do tohoto dokumentu; toto ho učiní přehratelným", + + "gt_vau": "nezobrazovat videa, jen přehrát zvuk\">🎧", + "gt_msel": "povolit výběr souborů; ctrl-klik na soubor pro přepsání$N$N<em>když aktivní: dvojklik na soubor / složku pro otevření</em>$N$NKlávesová zkratka: S\">výběr více", + "gt_crop": "ořez náhledů na střed\">ořez", + "gt_3x": "náhledy s vysokým rozlišením\">3x", + "gt_zoom": "zoom", + "gt_chop": "rozdělit", + "gt_sort": "řadit podle", + "gt_name": "název", + "gt_sz": "velikost", + "gt_ts": "datum", + "gt_ext": "typ", + "gt_c1": "více zkrátit názvy souborů (zobrazit méně)", + "gt_c2": "méně zkrátit názvy souborů (zobrazit více)", + + "sm_w8": "hledám...", + "sm_prev": "výsledky hledání níže jsou z předchozího dotazu:\n ", + "sl_close": "zavřít výsledky hledání", + "sl_hits": "zobrazuji {0} zásahů", + "sl_moar": "načíst více", + + "s_sz": "velikost", + "s_dt": "datum", + "s_rd": "cesta", + "s_fn": "název", + "s_ta": "tagy", + "s_ua": "nahráno@", + "s_ad": "pokročilé", + "s_s1": "minimum MiB", + "s_s2": "maximum MiB", + "s_d1": "min. iso8601", + "s_d2": "max. iso8601", + "s_u1": "nahráno po", + "s_u2": "a/nebo před", + "s_r1": "cesta obsahuje   (oddělené mezerami)", + "s_f1": "název obsahuje   (negace s -ne)", + "s_t1": "tagy obsahují   (^=začátek, konec=$)", + "s_a1": "specifické vlastnosti metadat", + + "md_eshow": "nelze vykreslit ", + "md_off": "[📜<em>readme</em>] zakázáno v [⚙️] -- dokument skryt", + + "badreply": "Selhalo parsování odpovědi ze serveru", + + "xhr403": "403: Přístup odepřen\n\nzkuste stisknout F5, možná jste se odhlásili", + "xhr0": "neznámý (pravděpodobně ztraceno spojení se serverem, nebo server je offline)", + "cf_ok": "omlouváme se za to -- DD" + wah + "oS ochrana se aktivovala\n\nvěci by se měly obnovit asi za 30 sekund\n\npokud se nic nestane, stiskněte F5 pro obnovení stránky", + "tl_xe1": "nelze vypsat podsložky:\n\nchyba ", + "tl_xe2": "404: Složka nenalezena", + "fl_xe1": "nelze vypsat soubory ve složce:\n\nchyba ", + "fl_xe2": "404: Složka nenalezena", + "fd_xe1": "nelze vytvořit podsložku:\n\nchyba ", + "fd_xe2": "404: Nadřazená složka nenalezena", + "fsm_xe1": "nelze odeslat zprávu:\n\nchyba ", + "fsm_xe2": "404: Nadřazená složka nenalezena", + "fu_xe1": "selhalo načtení unpost seznamu ze serveru:\n\nchyba ", + "fu_xe2": "404: Soubor nenalezen??", + + "fz_tar": "nekomprimovaný gnu-tar soubor (linux / mac)", + "fz_pax": "nekomprimovaný tar formátu pax (pomalejší)", + "fz_targz": "gnu-tar s gzip kompresí úrovně 3$N$Nto je obvykle velmi pomalé, takže$Npoužijte místo toho nekomprimovaný tar", + "fz_tarxz": "gnu-tar s xz kompresí úrovně 1$N$Nto je obvykle velmi pomalé, takže$Npoužijte místo toho nekomprimovaný tar", + "fz_zip8": "zip s utf8 názvy souborů (možná problematické na windows 7 a starších)", + "fz_zipd": "zip s tradičními cp437 názvy souborů, pro opravdu starý software", + "fz_zipc": "cp437 s crc32 vypočítaným brzy,$Npro MS-DOS PKZIP v2.04g (říjen 1993)$N(trvá déle zpracovat před začátkem stahování)", + + "un_m1": "můžete smazat vaše nedávné nahrání (nebo zrušit nedokončené) níže", + "un_upd": "obnovit", + "un_m4": "nebo sdílet soubory viditelné níže:", + "un_ulist": "zobrazit", + "un_ucopy": "kopírovat", + "un_flt": "volitelný filtr:  URL musí obsahovat", + "un_fclr": "vymazat filtr", + "un_derr": 'unpost-delete selhalo:\n', + "un_f5": 'něco se pokazilo, zkuste prosím obnovit, nebo stiskněte F5', + "un_uf5": "omlouváme se ale musíte obnovit stránku (například stisknutím F5 nebo CTRL-R) předtím než toto nahrávání může být zrušeno", + "un_nou": '<b>varování:</b> server je příliš zaneprázdněn pro zobrazení nedokončených nahrávání; za chvíli klikněte na odkaz "obnovit"', + "un_noc": '<b>varování:</b> unpost plně nahraných souborů není povoleno/dovoleno v konfiguraci serveru', + "un_max": "zobrazuji prvních 2000 souborů (použijte filtr)", + "un_avail": "{0} nedávných nahrávání může být smazáno<br />{1} nedokončených může být zrušeno", + "un_m2": "řazeno podle času nahrávání; nejnovější první:", + "un_no1": "počkej! žádná nahrávání nejsou dostatečně nedávná", + "un_no2": "počkej! žádná nahrávání odpovídající tomuto filtru nejsou dostatečně nedávná", + "un_next": "smazat dalších {0} souborů níže", + "un_abrt": "zrušit", + "un_del": "smazat", + "un_m3": "načítám vaše nedávné nahrání...", + "un_busy": "mažu {0} souborů...", + "un_clip": "{0} odkazů zkopírováno do schránky", + + "u_https1": "měli byste", + "u_https2": "přejít na https", + "u_https3": "pro lepší výkon", + "u_ancient": "váš prohlížeč je úctyhodně starý -- možná byste měli <a href=\"#\" onclick=\"goto('bup')\">použít bup</a>", + "u_nowork": "vyžadován firefox 53+ nebo chrome 57+ nebo iOS 11+", + "tail_2old": "vyžadován firefox 105+ nebo chrome 71+ nebo iOS 14.5+", + "u_nodrop": "váš prohlížeč je příliš starý pro nahrávání přetažením (drag-and-drop)", + "u_notdir": "toto není složka!\n\nváš prohlížeč je příliš starý,\nzkuste prosím soubory přetáhnout", + "u_uri": "pro přetažení obrázků z jiných oken prohlížeče,\nje prosím přetáhněte na velké tlačítko pro nahrávání", + "u_enpot": "přepnout na <a href=\"#\">potato UI</a> (může zrychlit nahrávání)", + "u_depot": "přepnout na <a href=\"#\">fancy UI</a> (může zpomalit nahrávání)", + "u_gotpot": "přepínám na potato UI pro zrychlení nahrávání,\n\npokud nesouhlasíte, klidně jej přepněte zpět!", + "u_pott": "<p>soubory:   <b>{0}</b> dokončeno,   <b>{1}</b> selhalo,   <b>{2}</b> nahrává se,   <b>{3}</b> ve frontě</p>", + "u_ever": "toto je základní nahrávání; up2k vyžaduje alespoň<br>chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": "toto je základní nahrávání; <a href=\"#\" id=\"u2yea\">up2k</a> je lepší", + "u_uput": "optimalizovat pro rychlost (přeskočit kontrolní součet)", + "u_ewrite": "nemáte oprávnění k zápisu do této složky", + "u_eread": "nemáte oprávnění ke čtení této složky", + "u_enoi": "vyhledávání souborů není povoleno v konfiguraci serveru", + "u_enoow": "přepsání zde nebude fungovat; je vyžadováno oprávnění k mazání", + "u_badf": "Těchto {0} souborů (z celkem {1}) bylo přeskočeno, pravděpodobně kvůli oprávněním v souborovém systému:\n\n", + "u_blankf": "Těchto {0} souborů (z celkem {1}) je prázdných; přesto je nahrát?\n\n", + "u_applef": "Těchto {0} souborů (z celkem {1}) je pravděpodobně nežádoucích;\nStiskněte <code>OK/Enter</code> pro PŘESKOČENÍ následujících souborů,\nStiskněte <code>Cancel/ESC</code> pro Zahrnutí a NAHRÁNÍ i těchto souborů:\n\n", + "u_just1": "\nMožná to bude fungovat lépe, když vyberete pouze jeden soubor", + "u_ff_many": "pokud používáte <b>Linux / MacOS / Android,</b> takové množství souborů <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>může</em> shodit Firefox!</a>\npokud se to stane, zkuste to prosím znovu (nebo použijte Chrome).", + "u_up_life": "Tento upload bude smazán ze serveru\n{0} po jeho dokončení", + "u_asku": "Nahrát {0} souborů do <code>{1}</code>", + "u_unpt": "toto nahrávání můžete vrátit zpět / smazat pomocí 🧯 vlevo nahoře", + "u_bigtab": "chystám se zobrazit {0} souborů\n\nto může shodit váš prohlížeč, jste si jisti?", + "u_scan": "Skenuji soubory...", + "u_dirstuck": "procházení adresáře se zaseklo při pokusu o přístup k následujícím {0} položkám; budou přeskočeny:", + "u_etadone": "Hotovo ({0}, {1} souborů)", + "u_etaprep": "(příprava na nahrávání)", + "u_hashdone": "hashování dokončeno", + "u_hashing": "hashování", + "u_hs": "navazuji spojení...", + "u_started": "soubory se nyní nahrávají; viz [🚀]", + "u_dupdefer": "duplikát; bude zpracován po všech ostatních souborech", + "u_actx": "klikněte na tento text, abyste zabránili ztrátě<br />výkonu při přepínání do jiných oken/záložek", + "u_fixed": "OK!  Opraveno 👍", + "u_cuerr": "nepodařilo se nahrát část {0} z {1};\npatrně neškodné, pokračuji\n\nsoubor: {2}", + "u_cuerr2": "server odmítl nahrání (část {0} z {1});\nzopakuji později\n\nsoubor: {2}\n\nchyba ", + "u_ehstmp": "zopakuji pokus; viz vpravo dole", + "u_ehsfin": "server odmítl požadavek na dokončení nahrávání; opakuji pokus...", + "u_ehssrch": "server odmítl požadavek na vyhledávání; opakuji pokus...", + "u_ehsinit": "server odmítl požadavek na zahájení nahrávání; opakuji pokus...", + "u_eneths": "síťová chyba při navazování spojení pro nahrávání; opakuji pokus...", + "u_enethd": "síťová chyba při ověřování existence cíle; opakuji pokus...", + "u_cbusy": "čekám, až nám server po síťovém problému začne znovu důvěřovat...", + "u_ehsdf": "na serveru došlo místo na disku!\n\nbudu to zkoušet dál, pro případ, že někdo\nuvolní dostatek místa pro pokračování", + "u_emtleak1": "vypadá to, že váš webový prohlížeč může mít únik paměti (memory leak);\nprosím", + "u_emtleak2": " <a href=\"{0}\">přejděte na https (doporučeno)</a> nebo ", + "u_emtleak3": " ", + "u_emtleakc": "zkuste následující:\n<ul><li>stiskněte <code>F5</code> pro obnovení stránky</li><li>poté vypněte tlačítko  <code>mt</code>  v  <code>⚙️ nastavení</code></li><li>a zkuste nahrávání znovu</li></ul>Nahrávání bude o něco pomalejší, ale co se dá dělat.\nOmlouváme se za potíže!\n\nPS: chrome v107 <a href=\"https://bugs.chromium.org/p/chromium/issues/detail?id=1354816\" target=\"_blank\">obsahuje opravu</a> pro tento problém", + "u_emtleakf": "zkuste následující:\n<ul><li>stiskněte <code>F5</code> pro obnovení stránky</li><li>poté zapněte <code>🥔</code> (potato) v rozhraní nahrávání<li>a zkuste nahrávání znovu</li></ul>\nPS: firefox snad <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\">bude mít opravu</a> v některé z příštích verzí", + "u_s404": "nenalezeno na serveru", + "u_expl": "vysvětlit", + "u_maxconn": "většina prohlížečů omezuje počet na 6, ale firefox umožňuje toto navýšit pomocí <code>connections-per-server</code> v <code>about:config</code>", + "u_tu": "<p class=\"warn\">VAROVÁNÍ: turbo zapnuto, <span> klient nemusí detekovat a obnovit nedokončené nahrávání; viz nápovědu u tlačítka turbo</span></p>", + "u_ts": "<p class=\"warn\">VAROVÁNÍ: turbo zapnuto, <span> výsledky vyhledávání mohou být nesprávné; viz nápovědu u tlačítka turbo</span></p>", + "u_turbo_c": "turbo je vypnuto v konfiguraci serveru", + "u_turbo_g": "vypínám turbo, protože nemáte oprávnění\nk výpisu adresářů na tomto svazku", + "u_life_cfg": 'automatické smazání po <input id="lifem" p="60" /> min (nebo <input id="lifeh" p="3600" /> hodinách)', + "u_life_est": 'nahrání bude smazáno <span id="lifew" tt="local time">---</span>', + "u_life_max": 'tato složka vynucuje\nmax. životnost {0}', + "u_unp_ok": 'unpost je povoleno pro {0}', + "u_unp_ng": 'unpost NEBUDE povoleno', + "ue_ro": 'váš přístup k této složce je pouze pro čtení\n\n', + "ue_nl": 'momentálně nejste přihlášeni', + "ue_la": 'momentálně jste přihlášeni jako "{0}"', + "ue_sr": 'momentálně jste v režimu vyhledávání souborů\n\npřepněte do režimu nahrávání kliknutím na lupu 🔎 (vedle velkého tlačítka HLEDAT) a zkuste nahrávání znovu\n\nomlouváme se', + "ue_ta": 'zkuste nahrávání znovu, nyní by to mělo fungovat', + "ue_ab": "tento soubor se již nahrává do jiné složky a toto nahrávání musí být dokončeno předtím, než může být soubor nahrán jinam.\n\nMůžete zrušit a zapomenout na původní nahrávání pomocí levého horního 🧯", + "ur_1uo": "OK: Soubor úspěšně nahrán", + "ur_auo": "OK: Všech {0} souborů úspěšně nahráno", + "ur_1so": "OK: Soubor nalezen na serveru", + "ur_aso": "OK: Všech {0} souborů nalezeno na serveru", + "ur_1un": "Nahrání selhalo, omlouváme se", + "ur_aun": "Všech {0} nahrání selhalo, omlouváme se", + "ur_1sn": "Soubor NEBYL nalezen na serveru", + "ur_asn": "{0} souborů NEBYLO nalezeno na serveru", + "ur_um": "Dokončeno;\n{0} nahrání OK,\n{1} nahrání selhalo, omlouváme se", + "ur_sm": "Dokončeno;\n{0} souborů nalezeno na serveru,\n{1} souborů NENALEZENO na serveru", + + "lang_set": "obnovit stránku, aby se změna projevila?", + }, "deu": { "tt": "Deutsch", @@ -6927,7 +7560,7 @@ var Ls = { }, }; -var LANGS = ["eng", "nor", "chi", "deu", "fin", "grc", "ita", "nld", "rus", "spa", "ukr"]; +var LANGS = ["eng", "nor", "chi", "cze", "deu", "fin", "grc", "ita", "nld", "rus", "spa", "ukr"]; if (window.langmod) langmod(); diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 52e666f3..54eafc71 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -94,6 +94,47 @@ var Ls = { "af1": "显示最近上传的文件", //m "ag1": "查看已知 IdP 用户", //m }, + "cze": { + "a1": "obnovit", + "b1": "ahoj cizinče   <small>(nejsi přihlášen)</small>", + "c1": "odhlásit se", + "d1": "vypsat zásobníku", // TLNote: "d2" is the tooltip for this button + "d2": "zobrazit stav všech aktivních vláken", + "e1": "znovu načíst konfiguraci", + "e2": "znovu načíst konfigurační soubory (accounts/volumes/volflags),$Na prohledat všechny e2ds úložiště$N$Npoznámka: všechny změny globálních nastavení$Nvyžadují úplné restartování, aby se projevily", + "f1": "můžeš procházet:", + "g1": "můžeš nahrávat do:", + "cc1": "další věci:", + "h1": "zakázat k304", // TLNote: "j1" explains what k304 is + "i1": "povolit k304", + "j1": "povolení k304 odpojí vašeho klienta při každém HTTP 304, což může zabránit některým chybovým proxy serverům, aby se zasekly (náhle nenačítaly stránky), <em>ale</em> také to obecně zpomalí věci", + "k1": "resetovat nastavení klienta", + "l1": "přihlaste se pro více:", + "m1": "vítej zpět,", // TLNote: "welcome back, USERNAME" + "n1": "404 nenalezeno  ┐( ´ -`)┌", + "o1": 'nebo možná nemáš přístup -- zkus heslo nebo <a href="' + SR + '/?h">jdi domů</a>', + "p1": "403 zakázáno  ~┻━┻", + "q1": 'použij heslo nebo <a href="' + SR + '/?h">jdi domů</a>', + "r1": "jdi domů", + ".s1": "znovu prohledat", + "t1": "akce", // TLNote: this is the header above the "rescan" buttons + "u2": "čas od posledního zápisu na server$N( upload / rename / ... )$N$N17d = 17 dní$N1h23 = 1 hodina 23 minut$N4m56 = 4 minuty 56 sekund", + "v1": "připojit", + "v2": "použít tento server jako místní HDD", + "w1": "přepnout na https", + "x1": "změnit heslo", + "y1": "upravit sdílení", // TLNote: shows the list of folders that the user has decided to share + "z1": "odblokovat toto sdílení:", // TLNote: the password prompt to see a hidden share + "ta1": "nejprve vyplňte své nové heslo", + "ta2": "zopakujte pro potvrzení nového hesla:", + "ta3": "nalezen překlep; zkuste to prosím znovu", + "aa1": "příchozí soubory:", + "ab1": "deaktivovat no304", + "ac1": "povolit no304", + "ad1": "povolení no304 deaktivuje veškeré mezipaměti; zkuste to, pokud k304 nestačilo. To ovšem zapříčíní obrovské množství síťového provozu!", + "ae1": "aktivní stahování:", + "af1": "zobrazit nedávné nahrávání", + }, "deu": { "a1": "Neu laden", "b1": "Tach, wie geht's?   <small>(Du bist nicht angemeldet)</small>", From f0b1c82b44b1ef522840060dc391f73280ecfb0c Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 22:09:11 +0000 Subject: [PATCH 120/174] i18n: support czech declensions --- copyparty/web/browser.js | 10 +++++----- copyparty/web/util.js | 10 +++++++--- 2 files changed, 12 insertions(+), 8 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 1b95ced2..986feb78 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -2247,10 +2247,10 @@ var Ls = { "f_partial": "Pro bezpečné stažení souboru, který se aktuálně nahrává, klikněte prosím na soubor se stejným názvem, ale bez přípony <code>.PARTIAL</code>. Stiskněte prosím CANCEL nebo Escape.\n\nStisknutím OK / Enter ignorujete toto varování a pokračujete ve stahování <code>.PARTIAL</code> dočasného souboru, což téměř jistě vyústí jako poškozená data.", "ft_paste": "vložit {0} položek$NKlávesová zkratka: ctrl-V", - "fr_eperm": 'nelze přejmenovat:\nnemáte oprávnění "přesunout" v této složce', - "fd_eperm": 'nelze smazat:\nnemáte oprávnění "smazat" v této složce', - "fc_eperm": 'nelze vyjmout:\nnemáte oprávnění "přesunout" v této složce', - "fp_eperm": 'nelze vložit:\nnemáte oprávnění "zapisovat" v této složce', + "fr_eperm": 'nelze přejmenovat:\nnemáte oprávnění “přesunout” v této složce', + "fd_eperm": 'nelze smazat:\nnemáte oprávnění “smazat” v této složce', + "fc_eperm": 'nelze vyjmout:\nnemáte oprávnění “přesunout” v této složce', + "fp_eperm": 'nelze vložit:\nnemáte oprávnění “zapisovat” v této složce', "fr_emore": "vyberte alespoň jednu položku k přejmenování", "fd_emore": "vyberte alespoň jednu položku ke smazání", "fc_emore": "vyberte alespoň jednu položku k vyjmutí", @@ -7580,7 +7580,7 @@ for (var a = 0; a < LANGS.length; a++) { t2 = Ls[LANGS[i2]]; for (var k in t1) - if (!t2[k]) { + if (!t2[k] && !/^ht_.5$/.test(k)) { console.log("E missing TL", LANGS[i2], k); t2[k] = t1[k]; } diff --git a/copyparty/web/util.js b/copyparty/web/util.js index e58e7cca..489cd416 100644 --- a/copyparty/web/util.js +++ b/copyparty/web/util.js @@ -1020,9 +1020,13 @@ function lhumantime(v) { if (!L || tp.length < 2 || tp[1].indexOf('$') + 1) return t; - var ret = ''; - for (var a = 0; a < tp.length; a += 2) - ret += tp[a] + ' ' + L['ht_' + tp[a + 1] + (tp[a]==1?1:2)] + L.ht_and; + var u, n, ret = ''; + for (var a = 0; a < tp.length; a += 2) { + n = tp[a]; + u = L.ht_h5 ? (n==1 ? 1 : (n>1&&n<5) ? 2 : 5) : + (n==1 ? 1 : 2); + ret += tp[a] + ' ' + L['ht_' + tp[a + 1] + u] + L.ht_and; + } return ret.slice(0, -L.ht_and.length); } From e9ecb2edc5378c9406064912772c6fb49c59eda7 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 22:13:52 +0000 Subject: [PATCH 121/174] v1.19.0 --- README.md | 2 +- copyparty/__main__.py | 2 +- copyparty/__version__.py | 6 +++--- docs/changelog.md | 36 ++++++++++++++++++++++++++++++++++++ 4 files changed, 41 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index 476d8ddc..8f470a50 100644 --- a/README.md +++ b/README.md @@ -437,7 +437,7 @@ upgrade notes * can I link someone to a password-protected volume/file by including the password in the URL? * yes, by adding `?pw=hunter2` to the end; replace `?` with `&` if there are parameters in the URL already, meaning it contains a `?` near the end - * if you have enabled `--accounts` then do `?pw=username:password` instead + * if you have enabled `--usernames` then do `?pw=username:password` instead * how do I stop `.hist` folders from appearing everywhere on my HDD? * by default, a `.hist` folder is created inside each volume for the filesystem index, thumbnails, audio transcodes, and markdown document history. Use the `--hist` global-option or the `hist` volflag to move it somewhere else; see [database location](#database-location) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 2b55b058..e87fe487 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -918,7 +918,7 @@ def get_sects(): copyparty will also hash and print any passwords that are non-hashed (password which do not start with '+') and then terminate afterwards - if you have enabled --accounts then the password + if you have enabled --usernames then the password must be provided as username:password for hashing \033[36m--ah-alg\033[0m specifies the hashing algorithm and a diff --git a/copyparty/__version__.py b/copyparty/__version__.py index 30bd3fd6..3350f413 100644 --- a/copyparty/__version__.py +++ b/copyparty/__version__.py @@ -1,8 +1,8 @@ # coding: utf-8 -VERSION = (1, 18, 10) -CODENAME = "logtail" -BUILD_DT = (2025, 8, 4) +VERSION = (1, 19, 0) +CODENAME = "usernames" +BUILD_DT = (2025, 8, 7) S_VERSION = ".".join(map(str, VERSION)) S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) diff --git a/docs/changelog.md b/docs/changelog.md index 3cc8bdf1..a3c0aaae 100644 --- a/docs/changelog.md +++ b/docs/changelog.md @@ -1,3 +1,39 @@ +▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀ +# 2025-0804-0013 `v1.18.10` idp speedboost + +## 🧪 new features + +* #426 add Dutch translation (thx @DeStilleGast!) 3798e19a +* #458 add Italian translation (thx @AOTREVAI!) a38e6e65 +* #456 transcode to flac/wav (thx @missaustraliana!) b469db3c b2d48c64 0d09fb68 +* #439 config-file can be provided through `PRTY_CONFIG` (thx @icxes!) 971360e9 +* #459 videos can become folder thumbnails 16bbcce5 +* add `--idp-cookie`, session-tickets for IdP auth (performance boost) f9502c3d + * useful when the IdP-server becomes a bottleneck + +## 🩹 bugfixes + +* #412 fix PUT-uploads into volumes with `nosub` volflag 47fa4a92 +* #435 ignore spurious exceptions from browser extensions 39e55824 +* #449 IPv6 QR-Code didn't include port 66a5bf36 +* #295 do not force `d2d` in blank vfs (introduced in v1.18.3) 848315c0 + +## 🔧 other changes + +* #440 improved finnish translation (thx @icxes!) a68d5b03 +* point to the `-nc` option in the "at max connections" warning 153d240d +* the play-button now indicates "play-as-audio" for video-files 40d56bb3 +* docs: + * #411 improve password-hashing instructions (thx @chinponya!) c69c7c8a + * #429 improve `--cert` helptext (thx @kzshantonu!) 7e3825f8 + * #413 copyparty is Wii Internet Channel compatible! (thx @techflashYT!) 50f16293 + * #461 how to use groups without IdP e85a7107 + * mention that WebDAV and OpenGraph are incompatible by default (and how to fix that) 0bc1b8f7 + * #345 short explanation about the sfx in quickstart ae5eefc5 +* #398 pypi-package now has extra-group `all` 6eaf8af1 + + + ▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀ # 2025-0801-2056 `v1.18.9` fix Denial-of-Service From 4eca4885f3d224c3c4b2c4e6e70e8923cbeaf147 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 7 Aug 2025 22:43:08 +0000 Subject: [PATCH 122/174] update pkgs to 1.19.0 --- contrib/package/arch/PKGBUILD | 4 ++-- contrib/package/makedeb-mpr/PKGBUILD | 4 ++-- contrib/package/nix/copyparty/pin.json | 6 +++--- 3 files changed, 7 insertions(+), 7 deletions(-) diff --git a/contrib/package/arch/PKGBUILD b/contrib/package/arch/PKGBUILD index 2673ec2a..e5896fd7 100644 --- a/contrib/package/arch/PKGBUILD +++ b/contrib/package/arch/PKGBUILD @@ -3,7 +3,7 @@ # NOTE: You generally shouldn't use this PKGBUILD on Arch, as it is mainly for testing purposes. Install copyparty using pacman instead. pkgname=copyparty -pkgver="1.18.10" +pkgver="1.19.0" pkgrel=1 pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" arch=("any") @@ -23,7 +23,7 @@ optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tag ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") backup=("etc/${pkgname}/copyparty.conf" ) -sha256sums=("49c5fedf7619437bc0af125cb4e8360d9bda9d87ef45d6314d7acf163ab4cf99") +sha256sums=("179b027d51e4fe7ebdab2b18c07475d52c57e2ce69256292b157a8efacd82118") build() { cd "${srcdir}/${pkgname}-${pkgver}/copyparty/web" diff --git a/contrib/package/makedeb-mpr/PKGBUILD b/contrib/package/makedeb-mpr/PKGBUILD index f16e5459..645623c0 100644 --- a/contrib/package/makedeb-mpr/PKGBUILD +++ b/contrib/package/makedeb-mpr/PKGBUILD @@ -2,7 +2,7 @@ pkgname=copyparty -pkgver=1.18.10 +pkgver=1.19.0 pkgrel=1 pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" arch=("any") @@ -20,7 +20,7 @@ optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tag ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") backup=("/etc/${pkgname}.d/init" ) -sha256sums=("49c5fedf7619437bc0af125cb4e8360d9bda9d87ef45d6314d7acf163ab4cf99") +sha256sums=("179b027d51e4fe7ebdab2b18c07475d52c57e2ce69256292b157a8efacd82118") build() { cd "${srcdir}/${pkgname}-${pkgver}/copyparty/web" diff --git a/contrib/package/nix/copyparty/pin.json b/contrib/package/nix/copyparty/pin.json index 70657f2c..c0d8c68a 100644 --- a/contrib/package/nix/copyparty/pin.json +++ b/contrib/package/nix/copyparty/pin.json @@ -1,5 +1,5 @@ { - "url": "https://github.com/9001/copyparty/releases/download/v1.18.10/copyparty-sfx.py", - "version": "1.18.10", - "hash": "sha256-2FdQ5aCwNdZ5Jy9mn8rr8g41+QHT5tsEc+GeAKvhGeg=" + "url": "https://github.com/9001/copyparty/releases/download/v1.19.0/copyparty-sfx.py", + "version": "1.19.0", + "hash": "sha256-9A+zPtkVtUuGHB/JJV3fhVtJderLUGxHqvuJQz0/1+Q=" } \ No newline at end of file From 0dd5987250274cc7006e9a1a355e388130d0e46f Mon Sep 17 00:00:00 2001 From: Vlad <vladislav.burzakovskij@satoshilabs.com> Date: Fri, 8 Aug 2025 09:53:50 +0200 Subject: [PATCH 123/174] add Ukrainian translations for control panel (#525) --- copyparty/web/browser.js | 2 +- copyparty/web/splash.js | 42 ++++++++++++++++++++++++++++++++++++++++ 2 files changed, 43 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 986feb78..54029e9a 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -7087,7 +7087,7 @@ var Ls = { "ut_srch": "не завантажувати, а перевірити, чи файли вже $N існують на сервері (сканує всі папки, які ви можете читати)", "ut_par": "призупинити завантаження, встановивши 0$N$Nзбільшіть, якщо ваше з'єднання повільне / висока затримка$N$Nзалишіть 1 в локальній мережі або якщо HDD сервера є вузьким місцем", "ul_btn": "перетягніть файли / папки<br> (або клацніть сюди)", - "ul_btnu": "З А В А Н Т А Ж И Т И", + "ul_btnu": "ЗАВАНТАЖИТИ", "ul_btns": "П О Ш У К", "ul_hash": "хеш", diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 54eafc71..16d21185 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -385,6 +385,48 @@ var Ls = { "af1": "mostrar subidas recientes", "ag1": "mostrar usuarios IdP conocidos" }, + "ukr": { + "a1": "оновити", + "b1": "привітик, незнайомцю   <small>(ви не авторизовані)</small>", + "c1": "вийти", + "d1": "трасування стека", + "d2": "показує стан усіх активних потоків", + "e1": "перезавантажити конфіг", + "e2": "перезавантажити файли конфігурації (облікові записи/томи/прапорці),$Nта пересканувати всі томи e2ds$N$Nувага: будь-які зміни глобальних налаштувань$Nвимагають повного перезапуску", + "f1": "ви можете бачити:", + "g1": "ви можете завантажувати файли в:", + "cc1": "всяка всячина:", + "h1": "вимкнути k304", + "i1": "увімкнути k304", + "j1": "увімкнення k304 буде відключати ваш клієнт при кожному HTTP 304, що може запобігти зависанню деяких глючних проксі (раптово перестають завантажувати сторінки), <em>але</em> це також зробить усе повільнішим загалом", + "k1": "скинути налаштування клієнта", + "l1": "авторизуйтесь для інших опцій:", + "m1": "з поверненням,", + "n1": "404 не знайдено  ┐( ´ -`)┌", + "o1": 'або у вас немає доступу -- спробуйте авторизуватися або <a href="' + SR + '/?h">повернутися на головну</a>', + "p1": "403 доступ заборонений  ~┻━┻", + "q1": 'авторизуйтесь або <a href="' + SR + '/?h">поверніться на головну</a>', + "r1": "повернутися на головну", + ".s1": "пересканувати", + "t1": "дія", + "u2": "час з останнього запису сервера$N( завантаження / перейменування / ... )$N$N17d = 17 днів$N1h23 = 1 година 23 хвилини$N4m56 = 4 хвилини 56 секунд", + "v1": "підключити", + "v2": "використовувати цей сервер як локальний HDD", + "w1": "перейти на https", + "x1": "змінити пароль", + "y1": "керування доступом", + "z1": "розблокувати:", + "ta1": "спочатку заповніть ваш новий пароль", + "ta2": "повторіть для підтвердження нового пароля:", + "ta3": "описка; спробуйте знову", + "aa1": "вхідні файли:", + "ab1": "вимкнути no304", + "ac1": "увімкнути no304", + "ad1": "увімкнення no304 вимкне все кешування; спробуйте це, якщо k304 було недостатньо. Це витратить величезну кількість мережевого трафіку!", + "ae1": "активні завантаження:", + "af1": "показати нещодавні завантаження", + "ag1": "показати відомих IdP-користувачів" + }, "rus": { "a1": "обновить", "b1": "приветик, незнакомец   <small>(вы не авторизованы)</small>", From f8a7c02f23996a1f14fc538d4cfcb04a44a86110 Mon Sep 17 00:00:00 2001 From: chamdim <94919340+chamdim@users.noreply.github.com> Date: Fri, 8 Aug 2025 10:54:11 +0300 Subject: [PATCH 124/174] Greek: fix typo (#528) Signed-off-by: chamdim <94919340+chamdim@users.noreply.github.com> --- copyparty/web/splash.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 16d21185..dc540fb4 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -222,7 +222,7 @@ var Ls = { "a1": "ανανέωση", "b1": "γεια σου ξένε!   <small>(δεν είσαι συνδεδεμένος)</small>", "c1": "αποσύνδεση", - "d1": "σωρός απορριμμάτων", + "d1": "λίστα διεργασιών", "d2": "εμφανίζει την κατάσταση όλων των ενεργών διεργασιών", "e1": "επαναφόρτωση του cfg", "e2": "φορτώνει ξανά τα αρχεία ρυθμίσεων (λογαριασμοί/τόμοι/volflags),$Nκαι κάνει επανεξέταση όλων των τόμων e2ds$N$Nσημείωση: οποιαδήποτε αλλαγή στις καθολικές ρυθμίσεις$Nαπαιτεί πλήρη επανεκκίνηση για να εφαρμοστεί", From bd514f066606d71e51a1914c3f07d070b7249ab3 Mon Sep 17 00:00:00 2001 From: chamdim <94919340+chamdim@users.noreply.github.com> Date: Fri, 8 Aug 2025 10:54:27 +0300 Subject: [PATCH 125/174] Greek: fix typos (#529) Signed-off-by: chamdim <94919340+chamdim@users.noreply.github.com> --- copyparty/web/browser.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 54029e9a..c9f9e70b 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -4005,7 +4005,7 @@ var Ls = { "ct_ttips": '◔ ◡ ◔">ℹ️ συμβουλές εργαλείων', "ct_thumb": 'σε προβολή πλέγματος, εναλλαγή εικονιδίων ή μικρογραφιών$NΠλήκτρο συντόμευσης: T">🖼️ μικρογραφίες', "ct_csel": 'χρησιμοποίησε CTRL και SHIFT για επιλογή αρχείων σε προβολή πλέγματος">επιλογή', - "ct_ihop": 'όταν η προβολή εικόνων κλείνει, κάνε scroll στο τελευταίο προβολόμενο αρχείο">g⮯', + "ct_ihop": 'όταν η προβολή εικόνων κλείνει, κάνε scroll στο τελευταίο προβαλλόμενο αρχείο">g⮯', "ct_dots": 'εμφάνιση κρυφών αρχείων (αν το επιτρέπει ο server)">dotfiles', "ct_qdel": 'όταν διαγράφεις αρχεία, ζήτα επιβεβαίωση μόνο μία φορά">γρήγορη διαγραφή', "ct_dir1st": 'ταξινόμηση φακέλων πριν από τα αρχεία">📁 πρώτα', @@ -4039,7 +4039,7 @@ var Ls = { "cft_bg": "χρώμα παρασκηνίου", "cdt_lim": "μέγιστος αριθμός αρχείων προς εμφάνιση σε ένα φάκελο", - "cdt_ask": "όταν φτάνεις στο τέλος,$Nαντί να φορτώσει περισσότερα αρχεία,$Nρώτα τι να κάνει", + "cdt_ask": "όταν φτάνεις στο τέλος,$Nαντί να φορτώσει περισσότερα αρχεία,$Nρωτά τι να κάνει", "cdt_hsort": "πόσους κανόνες ταξινόμησης (<code>,sorthref</code>) να συμπεριλάβει σε URLs πολυμέσων. Αν το βάλεις 0 αγνοεί και κανόνες ταξινόμησης στους συνδέσμους πολυμέσων", "tt_entree": "εμφάνιση navpane (δέντρο διαδρομών)$NΠλήκτρο συντόμευσης: B", From 3931bc27791658449f6dc3fc24e3a53d43d62a49 Mon Sep 17 00:00:00 2001 From: Kent Daleng <kent.daleng@nav.no> Date: Fri, 8 Aug 2025 22:54:24 +0200 Subject: [PATCH 126/174] =?UTF-8?q?legg=20=C3=A5t=20nynorskoversetjing=20(?= =?UTF-8?q?#537)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * legg åt nynorskoversetjing, og fikser et par typos på bokmål også :) * add nno to splash, fix a few more stray typos * more til -> åt --------- Signed-off-by: Kent D <lolexplode@gmail.com> --- copyparty/web/browser.js | 640 ++++++++++++++++++++++++++++++++++++++- copyparty/web/splash.js | 42 +++ 2 files changed, 677 insertions(+), 5 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index c9f9e70b..70ee0acf 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -853,7 +853,7 @@ var Ls = { "cl_hcancel": "kolonne-skjuling avbrutt", "ct_grid": '田 ikoner', - "ct_ttips": 'hvis hjelpetekst ved å holde musen over ting">ℹ️ tips', + "ct_ttips": 'vis hjelpetekst ved å holde musen over ting">ℹ️ tips', "ct_thumb": 'vis miniatyrbilder istedenfor ikoner$NSnarvei: T">🖼️ bilder', "ct_csel": 'bruk tastene CTRL og SHIFT for markering av filer i ikonvisning">merk', "ct_ihop": 'bla ned til sist viste bilde når bildeviseren lukkes">g⮯', @@ -878,8 +878,8 @@ var Ls = { "cut_az": "last opp filer i alfabetisk rekkefølge, istedenfor minste-fil-først$N$Nalfabetisk kan gjøre det lettere å anslå om alt gikk bra, men er bittelitt tregere på fiber / LAN", - "cut_nag": "meldingsvarsel når opplastning er ferdig$N(kun on nettleserfanen ikke er synlig)", - "cut_sfx": "lydvarsel når opplastning er ferdig$N(kun on nettleserfanen ikke er synlig)", + "cut_nag": "meldingsvarsel når opplastning er ferdig$N(kun om nettleserfanen ikke er synlig)", + "cut_sfx": "lydvarsel når opplastning er ferdig$N(kun om nettleserfanen ikke er synlig)", "cut_mt": "raskere befaring ved å bruke hele CPU'en$N$Ndenne funksjonen anvender web-workers$Nog krever mer RAM (opptil 512 MiB ekstra)$N$Ngjør https 30% raskere, http 4.5x raskere\">mt", @@ -913,7 +913,7 @@ var Ls = { "mt_loop": "spill den samme sangen om og om igjen\">🔁", "mt_one": "spill kun én sang\">1️⃣", "mt_shuf": "sangene i hver mappe$Nspilles i tilfeldig rekkefølge\">🔀", - "mt_aplay": "forsøk å starte avspilling hvis linken du klikket på for å åpne nettsiden inneholder en sang-ID$N$Nhvis denne deaktiveres så vil heller ikke nettside-URLen bli oppdatert med sang-ID'er når musikk spilles, i tilfelle innstillingene skulle gå tapt og nettsiden lastes på ny\">a▶", + "mt_aplay": "forsøk å starte avspilling hvis linken du klikket på for å åpne nettsiden inneholder en sang-ID$N$Nhvis denne deaktiveres så vil heller ikke nettside-URL'en bli oppdatert med sang-ID'er når musikk spilles, i tilfelle innstillingene skulle gå tapt og nettsiden lastes på ny\">a▶", "mt_preload": "hent ned litt av neste sang i forkant,$Nslik at pausen i overgangen blir mindre\">forles", "mt_prescan": "ved behov, bla til neste mappe$Nslik at nettleseren lar oss$Nfortsette å spille musikk\">bla", "mt_fullpre": "hent ned hele neste sang, ikke bare litt:$N✅ skru på hvis nettet ditt er <b>ustabilt</b>,$N❌ skru av hvis nettet ditt er <b>tregt</b>\">full", @@ -5672,6 +5672,636 @@ var Ls = { "lang_set": "Vernieuw de pagina om de wijziging door te voeren?", }, + "nno": { + "tt": "Nynorsk", + + "cols": { + "c": "handlingsknappar", + "dur": "varigheit", + "q": "kvalitet / bitrate", + "Ac": "lydformat", + "Vc": "videoformat", + "Fmt": "format / innpakning", + "Ahash": "lydkontrollsum", + "Vhash": "videokontrollsum", + "Res": "oppløysing", + "T": "filtype", + "aq": "lydkvalitet / bitrate", + "vq": "videokvalitet / bitrate", + "pixfmt": "fargekoding / detaljnivå", + "resw": "horisontal oppløysing", + "resh": "vertikal oppløysing", + "chs": "lydkanaler", + "hz": "lydoppløsing" + }, + + "hks": [ + [ + "ymse", + ["ESC", "lukk saker og ting"], + + "filbehandlar", + ["G", "listevisning eller ikon"], + ["T", "miniatyrbilder på/av"], + ["⇧ A/D", "ikonstorleik"], + ["ctrl-K", "slett valde"], + ["ctrl-X", "klipp ut valde"], + ["ctrl-C", "kopiér åt utklippstavle"], + ["ctrl-V", "lim inn (flytt/kopiér)"], + ["Y", "last ned valde"], + ["F2", "endre namn på valde"], + + "filmarkering", + ["space", "markér fil"], + ["↑/↓", "flytt markør"], + ["ctrl ↑/↓", "flytt markør og scroll"], + ["⇧ ↑/↓", "velg forr./neste fil"], + ["ctrl-A", "velg alle filer / mapper"], + ], [ + "navigering", + ["B", "mappehierarki eller filsti"], + ["I/K", "forr./neste mappe"], + ["M", "eitt nivå opp (eller lukk)"], + ["V", "vis mapper eller tekstfiler"], + ["A/D", "panelstorleik"], + ], [ + "musikkspelar", + ["J/L", "forr./neste song"], + ["U/O", "hopp 10sek bak/fram"], + ["0..9", "hopp åt 0%..90%"], + ["P", "pause, eller start / fortsett"], + ["S", "marker spelande song"], + ["Y", "last ned song"], + ], [ + "bildevisar", + ["J/L, ←/→", "forr./neste bilde"], + ["Home/End", "første/siste bilde"], + ["F", "fullskjermvisning"], + ["R", "rotér åt høyre"], + ["⇧ R", "rotér åt venstre"], + ["S", "markér bilde"], + ["Y", "last ned bilde"], + ], [ + "videospelar", + ["U/O", "hopp 10sek bak/fram"], + ["P/K/Space", "pause / fortsett"], + ["C", "fortsett åt neste fil"], + ["V", "gjenta avspeling"], + ["M", "lyd av/på"], + ["[ og ]", "gjentaksintervall"], + ], [ + "dokumentvisar", + ["I/K", "forr./neste fil"], + ["M", "lukk tekstdokument"], + ["E", "redigér tekstdokument"], + ["S", "markér fil (for F2/ctrl-x/...)"], + ["Y", "last ned tekstfil"], + ] + ], + + "m_ok": "OK", + "m_ng": "Avbryt", + + "enable": "Aktiv", + "danger": "VARSKU", + "clipped": "kopiert åt utklippstavla", + + "ht_s1": "sekund", + "ht_s2": "sekund", + "ht_m1": "minutt", + "ht_m2": "minutt", + "ht_h1": "time", + "ht_h2": "timar", + "ht_d1": "dag", + "ht_d2": "dagar", + "ht_and": " og ", + + "goh": "kontrollpanel", + "gop": 'navigér åt mappa før den her">forr.', + "gou": 'navigér eitt nivå opp">opp', + "gon": 'navigér åt mappa etter den her">neste', + "logout": "Logg ut ", + "access": " åtgang", + "ot_close": "lukk reiskap", + "ot_search": "søk etter filer ved å angje filnamn, mappenamn, tid, storleik, eller metadata som songtittel / artist / osv.$N$N<code>foo bar</code> = inneheld båe «foo» og «bar»,$N<code>foo -bar</code> = innehold «foo» men ikkje «bar»,$N<code>^yana .opus$</code> = startar med «yana», filtype «opus»$N<code>"try unite"</code> = «try unite» eksakt$N$Ndatoformat er iso-8601, så f.eks.$N<code>2009-12-31</code> eller <code>2020-09-12 23:30:00</code>", + "ot_unpost": "unpost: slett filer som du nyleg har lastet opp; «angre-knappen»", + "ot_bup": "bup: tradisjonell / primitiv filopplasting,$N$Nfungerar i om lag samtlege nettlesarar", + "ot_mkdir": "mkdir: lag ei ny mappe", + "ot_md": "new-md: lag eit nytt markdown-dokument", + "ot_msg": "msg: send ein beskjed åt serverloggen", + "ot_mp": "musikkspelarinstillinger", + "ot_cfg": "andre innstillinger", + "ot_u2i": 'up2k: last opp filer (viss du har skriveåtgang) eller bytt åt søkemodus for å sjekke om filene finnast ein eller annan plass på serveren$N$Nopplastinger kan startast opp att etter avbrot, skjer stykkevis for potensielt høgare ytelse, og ivaretek datostempling -- men bruker litt meir prosessorkraft enn [🎈]  (den primitive opplastaren "bup")<br /><br />mens opplastinger føregår så visast framdrifta her oppe!', + "ot_u2w": 'up2k: filopplasting med støtte for å starte opp att avbrotne opplastinger -- steng ned nettlesaren og drage dei same filene inn i nettlesaren igjen for å plukke opp att der du slapp$N$Nopplastinger skjer stykkevis for potensielt høgare ytelse, og ivaretek datostempling -- men bruker litt meir prosessorkraft enn [🎈]  (den primitive opplastaren "bup")<br /><br />mens opplastinger føregår så visast framdrifta her oppe!', + "ot_noie": 'Fungerer mye betre i Chrome / Firefox / Edge', + + "ab_mkdir": "lag mappe", + "ab_mkdoc": "nytt dokument", + "ab_msg": "send melding", + + "ay_path": "gå videre åt mapper", + "ay_files": "gå videre åt filer", + + "wt_ren": "gje nye namn åt dei valde filene$NSnarvei: F2", + "wt_del": "slett dei valde filene$NSnarvei: ctrl-K", + "wt_cut": "klipp ut dei valde filene <small>(for å lime inn ein annan plass)</small>$NSnarvei: ctrl-X", + "wt_cpy": "kopiér dei valde filene åt utklippstavla$N(for å lime inn ein annan plass)$NSnarvei: ctrl-C", + "wt_pst": "lim inn filer (som tidligare blei klipt ut / kopiert ein annan plass)$NSnarvei: ctrl-V", + "wt_selall": "velg alle filer$NSnarvei: ctrl-A (mens fokus er på ei fil)", + "wt_selinv": "invertér utval", + "wt_zip1": "last ned denne mappa som eit arkiv", + "wt_selzip": "last ned dei valde filene som eit arkiv", + "wt_seldl": "last ned dei valde filene$NSnarvei: Y", + "wt_npirc": "kopiér songinfo (irc-formatert)", + "wt_nptxt": "kopiér songinfo", + "wt_m3ua": "legg song åt i m3u-speleliste$N(husk å klikk på <code>📻copy</code> senere)", + "wt_m3uc": "kopiér m3u-spelelista åt utklippstavla", + "wt_grid": "bytt mellom ikon og listevising$NSnarvei: G", + "wt_prev": "førre song$NSnarvei: J", + "wt_play": "play / pause$NSnarvei: P", + "wt_next": "neste song$NSnarvei: L", + + "ul_par": "samtidige handl.:", + "ut_rand": "finn opp nye tilfeldige filnamn", + "ut_u2ts": "gje fila på serveren same$Ntidsstempel som lokalt hos deg\">📅", + "ut_ow": "overskrive eksisterande filer på serveren?$N🛡️: aldri (finn på eit nytt filnamn i staden for)$N🕒: overskriv viss fila åt serveren er eldre$N♻️: alltid, gitt at innholdet er annleis", + "ut_mt": "fortsett å synfare køa mens opplasting føregår$N$Nskru denne av dersom du har ein$Ntreig prosessor eller harddisk", + "ut_ask": 'bekreft filutvalg før opplasting startar">💭', + "ut_pot": "forbetre ytinga på treige einheiter ved å$Nforenkle brukergrensesnittet", + "ut_srch": "gjer eit søk i staden for å laste opp --$Nleitar gjennom alle mappane du har lov åt å sjå", + "ut_par": "sett åt 0 for å midlertidig stoppe opplasting$N$Nhøge verdier (4 eller 8) kan gje betre yting,$Nspesielt på treige internettlinjer$N$Nbør ikkje vere høgare enn 1 på LAN$Neller viss serveren sin harddisk er treig", + "ul_btn": "slepp filer / mapper<br>her (eller klikk meg)", + "ul_btnu": "L A S T   O P P", + "ul_btns": "F I L S Ø K", + + "ul_hash": "synfar", + "ul_send": " send", + "ul_done": "total", + "ul_idle1": "ingen handlinger i køen", + "ut_etah": "snitthastigheit for <em>synfaring</em> samt gjenståande tid", + "ut_etau": "snitthastigheit for <em>opplasting</em> samt gjenståande tid", + "ut_etat": "<em>total</em> snitthastigheit og gjenståande tid", + + "uct_ok": "fullført uten problem", + "uct_ng": "fullført under tvil (duplikat, ikkje funne, ...)", + "uct_done": "fullført (enten <em>ok</em> eller <em>ng</em>)", + "uct_bz": "aktive handlinger (synfaring / opplasting)", + "uct_q": "køa", + + "utl_name": "filnamn", + "utl_ulist": "vis", + "utl_ucopy": "kopiér", + "utl_links": "lenker", + "utl_stat": "status", + "utl_prog": "fremdrift", + + // må vere korte: + "utl_404": "404", + "utl_err": "FEIL!", + "utl_oserr": "OS-feil", + "utl_found": "funnet", + "utl_defer": "seinare", + "utl_yolo": "YOLO", + "utl_done": "ferdig", + + "ul_flagblk": "filene har blitt lagd i køa</b><br>men det er ein anna nettlesarfane som held på med synfaring eller opplasting akkurat no,<br>så venter åt den er ferdig først", + "ul_btnlk": "brytaren har blitt låst åt denne tilstanden i serverens konfigurasjon", + + "udt_up": "Last opp", + "udt_srch": "Søk", + "udt_drop": "Slepp filene her", + + "u_nav_m": '<h6>kva har du?</h6><code>Enter</code> = Filer (éin eller fleire)\n<code>ESC</code> = Éi mappe (inkludert undermapper)', + "u_nav_b": '<a href="#" id="modal-ok">Filer</a><a href="#" id="modal-ng">Éi mappe</a>', + + "cl_opts": "brytarar", + "cl_themes": "utsjånad", + "cl_langs": "språk", + "cl_ziptype": "nedlasting av mapper", + "cl_uopts": "up2k-brytarar", + "cl_favico": "favicon", + "cl_bigdir": "store mapper", + "cl_hsort": "#sort", + "cl_keytype": "notasjon for musikalsk dur", + "cl_hiddenc": "skjulte kolonner", + "cl_hidec": "skjul", + "cl_reset": "nullstill", + "cl_hpick": "klikk på overskrifta åt kolonnene du ønskjer å skjule i tabellen nedanfor", + "cl_hcancel": "kolonne-skjuling avbrote", + + "ct_grid": '田 ikon', + "ct_ttips": 'vis hjelpetekst ved å holde musa over ting">ℹ️ tips', + "ct_thumb": 'vis miniatyrbilder i staden for ikon$NSnarvei: T">🖼️ bilder', + "ct_csel": 'bruk tastane CTRL og SHIFT for markering av filer i ikonvising">merk', + "ct_ihop": 'bla ned åt sist viste bilde når bildevisaren lukkast">g⮯', + "ct_dots": 'vis skjulte filer (gitt at serveren tillèt det)">.synlig', + "ct_qdel": 'sletteknappen spør berre éin gong om stadfesting">hurtig🗑️', + "ct_dir1st": 'sortér slik at mapper kjem framanfor filer">📁 først', + "ct_nsort": 'naturlig sortering (skjønar tal i filnamn)">nsort', + "ct_utc": 'bruk UTC for alle klokkeslett">UTC', + "ct_readme": 'vis README.md nedanfor filene">📜 readme', + "ct_idxh": 'vis index.html i staden for filliste">htm', + "ct_sbars": 'vis rullgardiner / skrollefelt">⟊', + + "cut_umod": 'i tilfelle ei fil du lastar opp alt finnast på serveren, så skal tidsstempelet åt serveren oppdaterast slik at det stemmer overeins med din lokale fil (krev rettigheitene write+delete)">re📅', + + "cut_turbo": "forenkla synfaring ved opplasting; bør etter alt å døme <em>ikkje</em> skruast på:$N$Nnyttig dersom du var midt i ei svær opplasting som måtte startast på nytt av ein eller annan grunn, og du vil komme i gang igjen så raskt som i det heile mulig.$N$Nnår denne er skrudd på så forenklast synfaringa kraftig; i staden for å utføre ein trygg sjekk på om filene finnast på serveren i god stand, så sjekkast det kun om <em>filstorleiken</em> stemmer. Så dersom ein korrupt fil vere på serveren allerede, på same plass, med same storleik og namn, så blir det <em>ikkje oppdaga</em>.$N$Ndet anbefalast å kun benytte denne funksjonen for å komme seg raskt gjennom sjølve opplastinga, for så å skru den av, og åt slutt "laste opp" dei same filene éin gong åt -- slik at integriteten kan verifiserast\">turbo", + + "cut_datechk": "har ingen effekt dersom turbo er skrudd av$N$Ngjer turbo bittelitt tryggare ved å sjekke datostemplinga på filene (i tillegg åt filstorleik)$N$N<em>burde</em> oppdage og gjenoppta dei fleste ufullstendige opplastinger, men er <em>ikkje</em> ein fullverdig erstatning for å deaktivere turbo og gjere ein skikkeleg sjekk\">date-chk", + + "cut_u2sz": "storleik i megabyte for kvart bruddstykke for opplasting. Store verdiar flyg betre over atlanteren. Små verdiar kan vere betre på flettande ustabile samband", + + "cut_flag": "samkøyrer nettlesarfaner slik at berre éin $N kan holde på med synfaring / opplasting $N -- andre faner må óg ha denne skrudd på $N -- fungerar kun innanom same domene", + + "cut_az": "last opp filer i alfabetisk rekkefølge, i staden for minste-fil-først$N$Nalfabetisk kan gjere det lettare å anslå om alt gjekk bra, men er bittelitt treigare på fiber / LAN", + + "cut_nag": "meldingsvarsel når opplasting er ferdig$N(kun om nettlesarfana ikkje er synlig)", + "cut_sfx": "lydvarsel når opplasting er ferdig$N(kun om nettlesarfanen ikkje er synlig)", + + "cut_mt": "raskere synfaring ved å bruke heile CPU'en$N$Ndenne funksjonen nytter web-workers$Nog krev meir RAM (opptil 512 MiB ekstra)$N$Ngjer https 30% raskare, http 4.5x raskare\">mt", + + "cut_wasm": "bruk wasm i staden for nettlesaren sin sha512-funksjon; gjev betre yting på chrome-baserte nettlesarar, men brukar meir CPU, og eldre versjoner av chrome toler det ikkje (et opp all RAM og kræsjer)\">wasm", + + "cft_text": "ikontekst (blank ut og last siden på nytt for å deaktivere)", + "cft_fg": "farge", + "cft_bg": "bakgrunnsfarge", + + "cdt_lim": "maks mengd filer å vise per mappe", + "cdt_ask": "vis knappar for å laste fleire filer nederst på sida i staden for å gradvis laste meir av mappea når man scroller ned", + "cdt_hsort": "antall sorteringsreglar (<code>,sorthref</code>) som skal inkluderast når media-URL'ar genererast. Dersom denne er 0 så vil sorteringsreglar i URL'ar korkje bli generert eller lest", + + "tt_entree": "bytt åt mappehierarki$NSnarvei: B", + "tt_detree": "bytt åt tradisjonell stivising$NSnarvei: B", + "tt_visdir": "bla ned åt den åpne mappa", + "tt_ftree": "bytt mellom filstruktur og tekstfiler$NSnarvei: V", + "tt_pdock": "vis dei overordna mappane i eit panel", + "tt_dynt": "øk bredda på panelet ettersom treet utvider seg", + "tt_wrap": "linjebryting", + "tt_hover": "vis heile mappenamnet når musepeikaren treff mappa$N( gjer diverre at scrollhjulet fusker dersom musepeikaren ikkje finn seg i grøfta )", + + "ml_pmode": "ved enden av mappa", + "ml_btns": "knapper", + "ml_tcode": "konvertering", + "ml_tcode2": "konvertér til", + "ml_tint": "tint", + "ml_eq": "audio equalizer (tonejustering)", + "ml_drc": "compressor (volumutjevning)", + + "mt_loop": "spel den same songen om og om igjen\">🔁", + "mt_one": "spel kun éin song\">1️⃣", + "mt_shuf": "songane i kvar mappe$Nspelast i tilfeldig rekkefølge\">🔀", + "mt_aplay": "prøv å starte avspeling viss linken du trykte på for å åpne nettsida inneheld ein song-ID$N$Nviss denne deaktiverast så vil heller ikkje nettside-URL'en bli oppdatert med song-ID'er når musikk spelast, i tilfelle innstillingane skulle gå tapt og nettsida lastast på ny\">a▶", + "mt_preload": "hent ned litt av neste song i forkant,$Nslik at pausa i overgangen blir mindre\">forsyn", + "mt_prescan": "ved behov, bla åt neste mappe$Nslik at nettlesaren lar oss$Nfortsetja å spele musikk\">bla", + "mt_fullpre": "hent ned heile neste song, ikkje berre litt:$N✅ skru på viss nettet ditt er <b>ustabilt</b>,$N❌ skru av viss nettet ditt er <b>treigt</b>\">full", + "mt_fau": "for telefoner: forhindre at avspeling stoppar viss nettet er for treigt åt å laste neste song i tide. Viss påskrudd kan det forårsake at songinfo ikkje visast korrekt i OS'et\">☕️", + "mt_waves": "waveform seekbar:$Nvis volumkurve i avspelingsfeltet\">~s", + "mt_npclip": "vis knappar for å kopiere info om songen du høyrer på\">/np", + "mt_m3u_c": "vis knapper for å kopiere dei valde$Nsongene som innslag i ei m3u8-speleliste\">📻", + "mt_octl": "integrering med operativsystemet (fjernkontroll, infoskjerm)\">os-ctl", + "mt_oseek": "gje løyve åt spoling med fjernkontroll$N$Nmerk: på nokon eininger (iPhones) så vil$Ndette erstatte knappen for neste song\">spoling", + "mt_oscv": "vis albumcover på infoskjermen\">bilde", + "mt_follow": "bla slik at songen som spelast alltid er synleg\">🎯", + "mt_compact": "tettpakka spelarpanel\">⟎", + "mt_uncache": "prøv denne viss ein song ikkje spelar riktig\">oppfrisk", + "mt_mloop": "repetér heile mappa\">🔁 gjenta", + "mt_mnext": "hopp åt neste mappe og fortsett\">📂 neste", + "mt_mstop": "stopp avspeling\">⏸ stopp", + "mt_cflac": "konvertér flac / wav-filer åt opus\">flac", + "mt_caac": "konvertér aac / m4a-filer åt to opus\">aac", + "mt_coth": "konvertér alt anna (men ikkje mp3) åt opus\">andre", + "mt_c2opus": "det beste valget for alle PCar og Android\">opus", + "mt_c2owa": "opus-weba, for iOS 17.5 og nyare\">owa", + "mt_c2caf": "opus-caf, for iOS 11 åt og med 17\">caf", + "mt_c2mp3": "bra valg for steinalder-utstyr (slår aldri feil)\">mp3", + "mt_c2flac": "gir best lydkvalitet, men et nettet ditt\">flac", + "mt_c2wav": "heilt rå lydstrøm (bruker enda meir data enn flac)\">wav", + "mt_c2ok": "bra valg!", + "mt_c2nd": "ikkje det føretrekte valget for din einheit, men funker sikkert greit", + "mt_c2ng": "ser verkelig ikkje ut som enheiten din taklar dette formatet... men ok, vi prøver", + "mt_xowa": "iOS har fortsatt problem med avspeling av owa-musikk i bakgrunnen. Bruk caf eller mp3 i staden for", + "mt_tint": "nivå av bakgrunnsfarge på søkestripa (0-100),$Ngjer oppdateringer mindre distraherande", + "mt_eq": "aktivér tonekontroll og forsterker;$N$Nboost <code>0</code> = normal volumskala$N$Nwidth <code>1  </code> = normal stereo$Nwidth <code>0.5</code> = 50% blanding venstre-høgre$Nwidth <code>0  </code> = mono$N$Nboost <code>-0.8</code> & width <code>10</code> = instrumental :^)$N$Nreduserer óg daudtid mellom songfiler", + "mt_drc": "aktivér volum-utjevning (dynamic range compressor); vil óg aktivere tonejustering, så sett alle EQ-feltene bortsett frå 'width' åt 0 viss du ikkje vil ha nokon EQ$N$Nfilteret vil dempe volumet på alt som er høgare enn TRESH dB; for kvar RATIO dB over grensa er det 1dB som treff høgtalarane, så standardverdiane tresh -24 og ratio 12 skal bety at volumet ikkje gjeng høgare enn -22 dB, slik at ein trygt kan øke boost-verdien i equalizeren åt rundt 0.8, eller 1.8 kombinert med ATK 0 og RLS 90 (berre mulig i firefox; andre nettlesarar tek ikkje høgare RLS enn 1)$N$Nwikipedia forklarar dette mykje betre forresten", + + "mb_play": "lytt", + "mm_hashplay": "spel denne songen?", + "mm_m3u": "trykk <code>Enter/OK</code> for å spele\ntrykk <code>ESC/Avbryt</code> for å redigere", + "mp_breq": "krev firefox 82+, chrome 73+, eller iOS 15+", + "mm_bload": "lastar inn...", + "mm_bconv": "konverterer åt {0}, vent litt...", + "mm_opusen": "nettlesaren din skjønar ikkje aac / m4a;\nkonvertering åt opus er no aktivert", + "mm_playerr": "avspeling feilet: ", + "mm_eabrt": "Avspelingsforespørselen blei avbroten", + "mm_enet": "Nettet ditt er ustabilt", + "mm_edec": "Noko er galt med musikkfila", + "mm_esupp": "Nettleseren din skjønar ikkje filtypen", + "mm_eunk": "Ukjent feil", + "mm_e404": "Avspeling feilet: Fil ikkje funnet.", + "mm_e403": "Avspeling feilet: Høve nekta.\n\nKanskje du blei logget ut?\nPrøv å trykk F5 for å laste sida på nytt.", + "mm_e500": "Avspeling feilet: Rusk i maskineriet, sjekk serverloggen.", + "mm_e5xx": "Avspeling feilet: ", + "mm_nof": "finn ikkje flere songer i nærheita", + "mm_prescan": "Leitar etter neste song...", + "mm_scank": "Fann neste song:", + "mm_uncache": "alle songer vil lastast på nytt ved neste avspeling", + "mm_hnf": "songen finnast ikkje lenger", + + "im_hnf": "bildet finnast ikkje lenger", + + "f_empty": 'denne mappa er tom', + "f_chide": 'dette vil skjule kolonna «{0}»\n\nfana for "andre innstillinger" let deg vise kolonna igjen', + "f_bigtxt": "denne fila er heeile {0} MiB -- vis som tekst?", + "f_bigtxt2": "vil du sjå bunnen av filen i staden for? du vil da óg sjå nye linjer som blir lagd åt på slutten av filen i sanntid", + "fbd_more": '<div id="blazy">visar <code>{0}</code> av <code>{1}</code> filer; <a href="#" id="bd_more">vis {2}</a> eller <a href="#" id="bd_all">vis alle</a></div>', + "fbd_all": '<div id="blazy">visar <code>{0}</code> av <code>{1}</code> filer; <a href="#" id="bd_all">vis alle</a></div>', + "f_anota": "kun {0} av totalt {1} element blei markert;\nfor å velje alt må du bla åt bunnen av mappa først", + + "f_dls": 'lenkane i denne mappa er no\nomgjort åt nedlastingsknappar', + + "f_partial": "For å laste ned ei fil som enda ikkje er ferdig opplasta, klikk på filen som har same filnamn som denne, men uten <code>.PARTIAL</code> på slutten. Da vil serveren passe på at nedlastinga går bra. Derfor anbefalast det sterkt å trykkje AVBRYT eller Escape-tasten.\n\nViss du verkelig ønskjer å laste ned denne <code>.PARTIAL</code>-filen på ein ukontrollert måte, trykk OK / Enter for å ignorere denne advarselen. Slik vil du høgst sannsynleg motta korrupt data.", + + "ft_paste": "Lim inn {0} filer$NSnarvei: ctrl-V", + "fr_eperm": 'kan ikkje endre namn:\ndu har ikkje høve åt “move” i denne mappa', + "fd_eperm": 'kan ikkje slette:\ndu har ikkje høve åt “delete” i denne mappa', + "fc_eperm": 'kan ikkje klippe ut:\ndu har ikkje høve åt “move” i denne mappa', + "fp_eperm": 'kan ikkje lime inn:\ndu har ikkje høve åt “write” i denne mappa', + "fr_emore": "vel minst éi fil som skal få nytt namn", + "fd_emore": "vel minst éi fil som skal slettast", + "fc_emore": "vel minst éi fil som skal klippast ut", + "fcp_emore": "vel minst éi fil som skal kopierast åt utklippstavla", + + "fs_sc": "del mappa du er i no", + "fs_ss": "del dei valde filene", + "fs_just1d": "du kan ikkje markere flere mapper samtidig,\neller kombinere mapper og filer", + "fs_abrt": "❌ avbryt", + "fs_rand": "🎲 tilfeldig namn", + "fs_go": "✅ opprett deling", + "fs_name": "namn", + "fs_src": "kjelde", + "fs_pwd": "passord", + "fs_exp": "varigheit", + "fs_tmin": "min", + "fs_thrs": "timar", + "fs_tdays": "dagar", + "fs_never": "for evig", + "fs_pname": "valfri namn (blir litt tilfeldig ellers)", + "fs_tsrc": "fil/mappe som skal delast", + "fs_ppwd": "valfri passord", + "fs_w8": "opprettar deling...", + "fs_ok": "trykk <code>Enter/OK</code> for å kopiere lenka (for CTRL-V)\ntrykk <code>ESC/Avbryt</code> for å kun bekrefta", + + "frt_dec": "kan korrigere visse ødelagte filnamn\">url-decode", + "frt_rst": "nullstillar endringar (tilbake åt dei originale filnamna)\">↺ reset", + "frt_abrt": "avbryt og lukk dette vindauget\">❌ avbryt", + "frb_apply": "IVERKSETT", + "fr_adv": "automasjon basert på metadata<br>og / eller mønster (regulære uttrykk)\">avansert", + "fr_case": "versalfølsomme uttrykk\">Aa", + "fr_win": "bytt ut bokstavane <code><>:"\\|?*</code> med$Ntilsvarande som windows ikkje får panikk av\">win", + "fr_slash": "bytt ut bokstaven <code>/</code> slik at den ikkje forårsakar at nye mapper opprettes\">ikke /", + "fr_re": "regex-mønster som køyrast på kvart filnamn. Grupper kan leses ut i format-feltet nedanfor, f.eks. <code>(1)</code> og <code>(2)</code> osv.", + "fr_fmt": "inspirert av foobar2000:$N<code>(title)</code> byttast ut med songtittel,$N<code>[(artist) - ](title)</code> dropper [dette] viss artist er blank$N<code>$lpad((tn),2,0)</code> visar songnr. med 2 siffer", + "fr_pdel": "slett", + "fr_pnew": "lagre som", + "fr_pname": "gje innstillingane dine eit namn", + "fr_aborted": "avbrote", + "fr_lold": "gamalt namn", + "fr_lnew": "nytt namn", + "fr_tags": "metadata for dei valde filene (kun for referanse):", + "fr_busy": "endrar namn på {0} filer...\n\n{1}", + "fr_efail": "endring av namn feila:\n", + "fr_nchg": "{0} av namna blei justert pga. <code>win</code> og/eller <code>ikkje /</code>\n\nvil du fortsetja med dei nye namna som blei valde?", + + "fd_ok": "sletting OK", + "fd_err": "sletting feila:\n", + "fd_none": "ingenting blei sletta; kanskje avvist av serverkonfigurasjon (xbd)?", + "fd_busy": "slettar {0} filer...\n\n{1}", + "fd_warn1": "SLETT disse {0} filene?", + "fd_warn2": "<b>Siste sjanse!</b> Dette kan ikkje angrast. Slett?", + + "fc_ok": "klipte ut {0} filer", + "fc_warn": 'klipte ut {0} filer\n\nmen: kun <b>denne</b> nettlesarfana har muligheit åt å lime dei inn ein annan plass, siden antallet filer er helt hinsides', + + "fcc_ok": "kopierte {0} filer åt utklippstavla", + "fcc_warn": 'kopierte {0} filer åt utklippstavla\n\nmen: kun <b>denne</b> nettlesarfana har muligheit åt å lime dei inn ein annan plass, sidan antallet filer er heilt på hi sida', + + "fp_apply": "bekreft og lim inn no", + "fp_ecut": "du må klippe ut eller kopiere nokre filer / mapper først\n\nmerk: du kan gjerne jobbe på kryss av nettlesarfaner; klippe ut i éi fane, lime inn i ei anna", + "fp_ename": "{0} filer kan ikkje flyttast åt målmappa fordi det allereie finnast filer med same namn. Gi dei nye namn nedanfor, eller gje dei eit blankt namn for å hoppe over dei:", + "fcp_ename": "{0} filer kan ikkje kopierast åt målmappa fordi det allereie finnast filer med same namn. Gi dei nye namn nedanfor, eller gje dei eit blankt namn for å hoppe over dei:", + "fp_emore": "det er fortsatt fleire namn som må endrast", + "fp_ok": "flytting OK", + "fcp_ok": "kopiering OK", + "fp_busy": "flyttar {0} filer...\n\n{1}", + "fcp_busy": "kopierar {0} filer...\n\n{1}", + "fp_err": "flytting feila:\n", + "fcp_err": "kopiering feila:\n", + "fp_confirm": "flytt disse {0} filene hit?", + "fcp_confirm": "kopiér disse {0} filene hit?", + "fp_etab": 'kunne ikkje lese lista med filer frå den andre nettlesarfana', + "fp_name": "Lastar opp éi fil frå einheita di. Velg filnamn:", + "fp_both_m": '<h6>kva skal limast inn her?</h6><code>Enter</code> = Flytt {0} filer frå «{1}»\n<code>ESC</code> = Last opp {2} filer frå einheita din', + "fcp_both_m": '<h6>kva skal limes inn her?</h6><code>Enter</code> = Kopiér {0} filer frå «{1}»\n<code>ESC</code> = Last opp {2} filer frå einheita din', + "fp_both_b": '<a href="#" id="modal-ok">Flytt</a><a href="#" id="modal-ng">Last opp</a>', + "fcp_both_b": '<a href="#" id="modal-ok">Kopiér</a><a href="#" id="modal-ng">Last opp</a>', + + "mk_noname": "skriv inn eit namn i tekstboksa åt venstre først :p", + + "tv_load": "Lastar inn tekstfil:\n\n{0}\n\n{1}% ({2} av {3} MiB lasta ned)", + "tv_xe1": "kunne ikkje laste tekstfil:\n\nfeil ", + "tv_xe2": "404, Fil ikkje funne", + "tv_lst": "tekstfiler i mappa", + "tvt_close": "gå tilbake åt mappa$NSnarvei: M (eller Esc)\">❌ lukk", + "tvt_dl": "last ned denne fila$NSnarvei: Y\">💾 last ned", + "tvt_prev": "vis førre dokument$NSnarvei: i\">⬆ forr.", + "tvt_next": "vis neste dokument$NSnarvei: K\">⬇ neste", + "tvt_sel": "markér fila   ( for utklipp / sletting / ... )$NSnarvei: S\">merk", + "tvt_edit": "redigér fila$NSnarvei: E\">✏️ endre", + "tvt_tail": "overvak fila for endringar og vis nye linjer i sanntid\">📡 følg", + "tvt_wrap": "tekstbryting\">↵", + "tvt_atail": "hald dei nyaste linjene synlege (lås åt botnen av sida)\">⚓", + "tvt_ctail": "skjøn og vis terminalfargar (ansi-sekvensar)\">🌈", + "tvt_ntail": "maksgrense for antal bokstavar som skal visast i vindauget", + + "m3u_add1": "songen blei lagd åt i m3u-spelelista", + "m3u_addn": "{0} songer blei lagde åt i m3u-spelelista", + "m3u_clip": "m3u-spelelista blei kopiert åt utklippstavla\n\nneste steg er å oppretta eit tekstdokument med filnamn som sluttar på <code>.m3u</code> og lime inn spelelista der", + + "gt_vau": "ikkje vis videofiler, berre spel lyden\">🎧", + "gt_msel": "markér filer i staden for å åpne dei; ctrl-klikk filer for å overstyre$N$N<em>når aktiv: dobbelklikk ei fil / mappe for å åpne</em>$N$NSnarvei: S\">markering", + "gt_crop": "skjer ikona slik at dei passar betre\">✂", + "gt_3x": "høgare oppløysing på ikon\">3x", + "gt_zoom": "zoom", + "gt_chop": "trim", + "gt_sort": "sortér", + "gt_name": "namn", + "gt_sz": "størr.", + "gt_ts": "dato", + "gt_ext": "type", + "gt_c1": "redusér makslengde på filnamn", + "gt_c2": "auk makslengde på filnamn", + + "sm_w8": "søker...", + "sm_prev": "søkeresultata er frå eit tidlegare søk:\n ", + "sl_close": "lukk søkeresultat", + "sl_hits": "visar {0} treff", + "sl_moar": "hent fleire", + + "s_sz": "størr.", + "s_dt": "dato", + "s_rd": "sti", + "s_fn": "namn", + "s_ta": "meta", + "s_ua": "up@", + "s_ad": "avns.", + "s_s1": "større enn ↓ MiB", + "s_s2": "mindre enn ↓ MiB", + "s_d1": "nyare enn <dato>", + "s_d2": "eldre enn", + "s_u1": "lasta opp etter", + "s_u2": "og/eller før", + "s_r1": "mappaamn inneheld", + "s_f1": "filnamn inneheld", + "s_t1": "song-info inneheld", + "s_a1": "konkrete eigenskapar", + + "md_eshow": "visar forenkla ", + "md_off": "[📜<em>readme</em>] er skrudd av i [⚙️] -- dokument skjult", + + "badreply": "Ugyldig svar frå serveren", + + "xhr403": "403: Høve nekta\n\nkanskje du blei logga ut? prøv å trykk F5", + "xhr0": "ukjend (enten nettverksproblem eller serverkræsj)", + "cf_ok": "om orsak -- liten tilfeldig kontroll, alt OK\n\nting skal fortsetja om ca. 30 sekund\n\nviss ikkje noko skjer, trykk F5 for å laste sida på nytt", + "tl_xe1": "kunne ikkje hente undermapper:\n\nfeil ", + "tl_xe2": "404: Mappa finnast ikkje", + "fl_xe1": "kunne ikkje hente filer i mappa:\n\nfeil ", + "fl_xe2": "404: Mappa finnast ikkje", + "fd_xe1": "kan ikkje opprette ny mappe:\n\nfeil ", + "fd_xe2": "404: Den overordna mappa finnast ikkje", + "fsm_xe1": "kunne ikkje sende melding:\n\nfeil ", + "fsm_xe2": "404: Den overordna mappa finnast ikkje", + "fu_xe1": "kunne ikkje hente lista med nyleg opplastede filer frå serveren:\n\nfeil ", + "fu_xe2": "404: Fila finnast ikkje??", + + "fz_tar": "ukomprimert gnu-tar arkiv, for linux og mac", + "fz_pax": "ukomprimert pax-tar arkiv, litt treigare", + "fz_targz": "gnu-tar pakket med gzip (nivå 3)$N$NNB: denne er veldig treig;$Nukomprimert tar er betre", + "fz_tarxz": "gnu-tar pakket med xz (nivå 1)$N$NNB: denne er veldig treig;$Nukomprimert tar er betre", + "fz_zip8": "zip med filnamn i utf8 (noko problematisk på windows 7 og eldre)", + "fz_zipd": "zip med filnamn i cp437, for høggamle maskiner", + "fz_zipc": "cp437 med tidlig crc32,$Nfor MS-DOS PKZIP v2.04g (oktober 1993)$N(øker behandlingstid på server)", + + "un_m1": "nedanfor kan du angre / slette filer som du nyleg har lastet opp, eller avbryte ufullstendige opplastinger", + "un_upd": "oppdater", + "un_m4": "eller viss du vil dele nedlastings-lenkene:", + "un_ulist": "vis", + "un_ucopy": "kopiér", + "un_flt": "valgfritt filter:  filnamn / filsti må inneholde", + "un_fclr": "nullstill filter", + "un_derr": 'unpost-sletting feilet:\n', + "un_f5": 'noko gjekk galt, prøv å oppdatere lista eller trykk F5', + "un_uf5": "om orsak, men du må laste sida på nytt (f.eks. ved å trykkje F5 eller CTRL-R) før denne opplastinga kan avbrytast", + "un_nou": '<b>advarsel:</b> kan ikkje vise ufullstendige opplastingar akkurat no; klikk på oppdater-lenka om litt', + "un_noc": '<b>advarsel:</b> angring av fullførte opplastingar er deaktivert i serverkonfigurasjonen', + "un_max": "visar dei første 2000 filene (bruk filteret for å snevre inn)", + "un_avail": "{0} nyleg opplasta filer kan slettast<br />{1} ufullstendige opplastingar kan avbrytast", + "un_m2": "sortert etter opplastingstid; nyaste først:", + "un_no1": "men nei, her var det jaggu ikkje noko som slettast kan", + "un_no2": "men nei, her var det jaggu ingenting som passa overens med filteret", + "un_next": "slett dei neste {0} filene nedanfor", + "un_abrt": "avbryt", + "un_del": "slett", + "un_m3": "hentar lista med nyleg opplasta filer...", + "un_busy": "slettar {0} filer...", + "un_clip": "{0} lenkar kopiert åt utklippstavla", + + "u_https1": "du burde", + "u_https2": "bytte åt https", + "u_https3": "for høgare hastigheit", + "u_ancient": 'nettlesaren din er prehistorisk -- mulig du burde <a href="#" onclick="goto(\'bup\')">bruke bup i staden for</a>', + "u_nowork": "krev firefox 53+, chrome 57+, eller iOS 11+", + "tail_2old": "krev firefox 105+, chrome 71+, eller iOS 14.5+", + "u_nodrop": 'nettlesaren din er for gamal åt å laste opp filer ved å drage dei inn i vindauget', + "u_notdir": "mottok ikkje mappa!\n\nnettlesaren din er for gamal,\nprøv å drage mappa inn i vindauget i staden for", + "u_uri": "for å laste opp bilder frå andre nettlesarvindauge,\nslipp bildet rett på den store last-opp-knappen", + "u_enpot": 'bytt åt <a href="#">enkelt UI</a> (gir sannsynleg raskere opplasting)', + "u_depot": 'bytt åt <a href="#">snæsent UI</a> (gir sannsynleg treigare opplasting)', + "u_gotpot": 'bytta åt eit enklare UI for å laste opp raskere,\n\ndu kan gjerne bytte tilbake altså!', + "u_pott": "<p>filer:   <b>{0}</b> ferdig,   <b>{1}</b> feilet,   <b>{2}</b> behandlast,   <b>{3}</b> i kø</p>", + "u_ever": "dette er den primitive opplastaren; up2k krev minst:<br>chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'dette er den primitive opplastaren; <a href="#" id="u2yea">up2k</a> er betre', + "u_uput": 'litt raskare (uten sha512)', + "u_ewrite": 'du har ikkje høve til å skrive i denne mappa', + "u_eread": 'du har ikkje høve til å lese i denne mappa', + "u_enoi": 'filsøk er deaktivert i serverkonfigurasjonen', + "u_enoow": "kan ikkje overskrive filer her (Delete-rettigheiten er nødvendig)", + "u_badf": 'Disse {0} filene (av totalt {1}) kan ikkje leses, kanskje pga rettigheitsproblem i filsystemet på datamaskinen din:\n\n', + "u_blankf": 'Disse {0} filene (av totalt {1}) er blanke / uten innhald; ønskjer du å laste dei opp uansett?\n\n', + "u_applef": 'Disse {0} filene (av totalt {1}) er antakeleg uønska;\nTrykk <code>OK/Enter</code> for å HOPPE OVER disse filene,\nTrykk <code>Avbryt/ESC</code> for å LASTE OPP disse filene óg:\n\n', + "u_just1": '\nFunkar kanskje betre viss du berre tar éi fil om gangen', + "u_ff_many": 'Viss du bruker <b>Linux / MacOS / Android,</b> så kan dette antalet filer<br /><a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank"><em>kanskje</em> kræsje Firefox!</a> Viss det skjer, så prøv igjen (eller bruk Chrome).', + "u_up_life": "Filene slettast frå serveren {0}\netter at opplastingen er fullført", + "u_asku": 'Laste opp disse {0} filene åt <code>{1}</code>', + "u_unpt": "Du kan angre / slette opplastinga med 🧯 oppe åt venstre", + "u_bigtab": 'Vil no vise {0} filer...\n\nDette kan kræsje nettlesaren din. Fortsette?', + "u_scan": 'Les mappane...', + "u_dirstuck": 'Nettleseren din fekk ikkje høve åt å lese følgande {0} filer/mapper, så dei blir hoppa over:', + "u_etadone": 'Ferdig ({0}, {1} filer)', + "u_etaprep": '(forberedar opplasting)', + "u_hashdone": 'synfaring ferdig', + "u_hashing": 'les', + "u_hs": 'serveren tenkjer...', + "u_started": "filene blir no lasta opp 🚀", + "u_dupdefer": "duplikat; vil bli håndtert åt slutt", + "u_actx": "klikk her for å forhindre tap av<br />yting ved bytte åt andre vindauge/faner", + "u_fixed": "OK!  Løyste seg 👍", + "u_cuerr": "kunne ikkje laste opp del {0} av {1};\nsikkert greit, fortsetjar\n\nfil: {2}", + "u_cuerr2": "server nekta opplastinga (del {0} av {1});\nprøver igjen senere\n\nfil: {2}\n\nerror ", + "u_ehstmp": "prøver igjen; se mld nederst", + "u_ehsfin": "server nekta forespørselen om å ferdigstille filen; prøver igjen...", + "u_ehssrch": "server nekta forespørselen om å utføre søk; prøver igjen...", + "u_ehsinit": "server nekta forespørselen om å begynne ei ny opplasting; prøver igjen...", + "u_eneths": "eit problem med nettverket gjorde at avtale om opplasting ikkje kunne inngås; prøver igjen...", + "u_enethd": "eit problem med nettverket gjorde at filsjekk ikkje kunne utførast; prøver igjen...", + "u_cbusy": "ventar på klarering frå server etter eit lite nettverksglipp...", + "u_ehsdf": "serveren er full!\n\nprøver igjen regelmessig,\ni tilfelle nokon ryddar litt...", + "u_emtleak1": "uff, det er mulig at nettlesaren din har ei minnelekkasje...\nForeslår", + "u_emtleak2": ' helst at du <a href="{0}">byttar åt https</a>, eller ', + "u_emtleak3": ' at du ', + "u_emtleakc": 'prøver følgande:\n<ul><li>trykk F5 for å laste sida på nytt</li><li>så skru av  <code>mt</code>  brytaren under  <code>⚙️ innstillinger</code></li><li>og prøv den same opplastinga igjen</li></ul>Opplasting vil gå litt treigare, men det får så vere.\nBeklager bryderiet!\n\nPS: feilen <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">skal vere fikset</a> i chrome v107', + "u_emtleakf": 'prøver følgende:\n<ul><li>trykk F5 for å laste sida på nytt</li><li>så skru på <code>🥔</code> ("enkelt UI") i opplastaren</li><li>og prøv den same opplastingen igjen</li></ul>\nPS: Firefox <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">fiksar forhåpentligvis feilen</a> ein eller annen gong', + "u_s404": "ikkje funne på serveren", + "u_expl": "forklar", + "u_maxconn": "dei fleste nettlesarar tillet ikkje meir enn 6, men firefox lar deg øke grensen med <code>connections-per-server</code> i <code>about:config</code>", + "u_tu": '<p class="warn">ADVARSEL: turbo er på, <span> avbrotne opplastingar vil muligens ikkje oppdagast og gjenopptakast; hald musepeikaren over turbo-knappen for meir info</span></p>', + "u_ts": '<p class="warn">ADVARSEL: turbo er på, <span> søkeresultat kan vere feil; hold musepeikaren over turbo-knappen for meir info</span></p>', + "u_turbo_c": "turbo er deaktivert i serverkonfigurasjonen", + "u_turbo_g": 'turbo blei deaktivert fordi du ikkje har\nhøve åt å sjå mappeinnhold i dette volumet', + "u_life_cfg": 'slett opplasting etter <input id="lifem" p="60" /> min (eller <input id="lifeh" p="3600" /> timar)', + "u_life_est": 'opplastingen slettast <span id="lifew" tt="lokal tid">---</span>', + "u_life_max": 'denne mappa tillet ikkje å \noppbevare filer i meir enn {0}', + "u_unp_ok": 'opplasting kan angrast i {0}', + "u_unp_ng": 'opplasting kan IKKE angrast', + "ue_ro": 'du har ikkje høve åt skriving i denne mappa\n\n', + "ue_nl": 'du er ikkje logga inn', + "ue_la": 'du er logga inn som "{0}"', + "ue_sr": 'du er i filsøk-modus\n\nbytt åt opplasting ved å klikke på forstørringsglaset 🔎 (ved siden av den store FILSØK-knappen) og prøv igjen\n\nsorry', + "ue_ta": 'prøv å last opp igjen, det burde fungere no', + "ue_ab": "den same filen er under opplasting åt ei anna mappe, og den må fullførast der før fila kan lastast opp andre plassar.\n\nDu kan avbryte og gløyme den påbegynte opplastinga ved hjelp av 🧯 oppe åt venstre", + "ur_1uo": "OK: Fila blei lastet opp", + "ur_auo": "OK: Alle {0} filene blei lastet opp", + "ur_1so": "OK: Fila blei funne på serveren", + "ur_aso": "OK: Alle {0} filene blei funne på serveren", + "ur_1un": "Opplasting feila!", + "ur_aun": "Alle {0} opplastingene gjekk feil!", + "ur_1sn": "Fila finnast IKKE på serveren", + "ur_asn": "Fann INGEN av dei {0} filene på serveren", + "ur_um": "Ferdig;\n{0} opplastingar gjekk bra,\n{1} opplastingar gjekk feil", + "ur_sm": "Ferdig;\n{0} filer blei funne,\n{1} filer finnast IKKJE på serveren", + + "lang_set": "passar det å laste sida på nytt?", + }, "rus": { "tt": "Русский", @@ -7560,7 +8190,7 @@ var Ls = { }, }; -var LANGS = ["eng", "nor", "chi", "cze", "deu", "fin", "grc", "ita", "nld", "rus", "spa", "ukr"]; +var LANGS = ["eng", "nor", "nno", "chi", "cze", "deu", "fin", "grc", "ita", "nld", "rus", "spa", "ukr"]; if (window.langmod) langmod(); diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index dc540fb4..6ffd95f6 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -343,6 +343,48 @@ var Ls = { "af1": "Recent geüploade bestanden weergeven", "ag1": "Bekende IdP-gebruikers weergeven", }, + "nno": { + "a1": "oppdatér", + "b1": "heisann   <small>(du er ikkje logga inn)</small>", + "c1": "logg ut", + "d1": "tilstand", + "d2": "vis tilstanden åt alle trådar", + "e1": "last innst.", + "e2": "les inn konfigurasjonsfiler på nytt$N(kontoer, volum, volumbrytarar)$Nog kartlegg alle e2ds-volum$N$Nmerk: endringer i globale parametrar$Nkrev ein full restart for å gjelde", + "f1": "du kan sjå på:", + "g1": "du kan laste opp åt:", + "cc1": "brytarar og slikt:", + "h1": "skru av k304", + "i1": "skru på k304", + "j1": "k304 bryt tilkoplinga for kvar HTTP 304. Dette hjelp mot visse mellomtjenarar som kan sette seg fast / plutselig sluttar å laste sider, men det sett óg ytinga ned betydelig", + "k1": "nullstill innstillinger", + "l1": "logg inn:", + "m1": "velkomen attende,", + "n1": "404: filen finnast ikkje  ┐( ´ -`)┌", + "o1": 'eller kanskje du ikkje har høve? prøv eit passord eller <a href="' + SR + '/?h">gå heim</a>', + "p1": "403: tilgang nektet  ~┻━┻", + "q1": 'prøv eit passord eller <a href="' + SR + '/?h">gå heim</a>', + "r1": "gå heim", + ".s1": "kartlegg", + "t1": "handling", + "u2": "tid sidan nokon sist skreiv åt serveren$N( opplastning / namnendring / ... )$N$N17d = 17 dagar$N1h23 = 1 time 23 minutt$N4m56 = 4 minutt 56 sekund", + "v1": "kople åt", + "v2": "bruk denne serveren som ein lokal harddisk", + "w1": "bytt åt https", + "x1": "bytt passord", + "y1": "dine delinger", + "z1": "lås opp område:", + "ta1": "du må skrive eit nytt passord først", + "ta2": "gjenta for å stadfeste nytt passord:", + "ta3": "fant ein skrivefeil; vennligst prøv igjen", + "aa1": "innkommande:", + "ab1": "skru av no304", + "ac1": "skru på no304", + "ad1": "no304 stoppar all bruk av cache. Hvis ikkje k304 var nok, prøv denne. Vil mangedoble dataforbruk!", + "ae1": "utgående:", + "af1": "vis nylig opplasta filer", + "ag1": "vis kjente IdP-brukarar", + }, "spa": { "a1": "actualizar", "b1": "hola   <small>(no has iniciado sesión)</small>", From 392a4db55bc4d27300bff5bc82638513e6c900d3 Mon Sep 17 00:00:00 2001 From: Artur Borecki <me@pufereq.pl> Date: Sat, 9 Aug 2025 00:55:57 +0200 Subject: [PATCH 127/174] Add Polish translation (#463) Add Polish translation --------- Signed-off-by: Artur Borecki <me@pufereq.pl> Signed-off-by: ed <s@ocv.me> Co-authored-by: dai <contact@daimond113.com> Co-authored-by: ed <s@ocv.me> --- copyparty/web/browser.js | 626 +++++++++++++++++++++++++++++++++++++++ copyparty/web/splash.js | 42 +++ 2 files changed, 668 insertions(+) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 70ee0acf..b3d174f8 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -6302,6 +6302,632 @@ var Ls = { "lang_set": "passar det å laste sida på nytt?", }, + "pol": { + "tt": "Polski", + "cols": { + "c": "przyciski akcji", + "dur": "czas trwania", + "q": "jakość / bitrate", + "Ac": "kodek audio", + "Vc": "kodek wideo", + "Fmt": "format / kontener", + "Ahash": "suma kontrolna audio", + "Vhash": "suma kontrolna wideo", + "Res": "rozdzielczość", + "T": "rodzaj pliku", + "aq": "jakość / bitrate audio", + "vq": "jakość / bitrate wideo", + "pixfmt": "podpróbkowanie / struktura pikseli", + "resw": "rozdzielczość pozioma", + "resh": "rozdzielczość pionowa", + "chs": "kanały audio", + "hz": "częstotliwość próbkowania" + }, + + "hks": [ + [ + "misc", + ["ESC", "zamknij różne rzeczy"], + + "file-manager", + ["G", "przełącz widok lista / siatka"], + ["T", "przełącz miniaturki / ikony"], + ["⇧ A/D", "wielkość miniaturki"], + ["ctrl-K", "usuń zaznaczone"], + ["ctrl-X", "wytnij zaznaczone do schowka"], + ["ctrl-C", "skopiuj zaznaczone do schowka"], + ["ctrl-V", "wklej (przenieś/skopiuj) tutaj"], + ["Y", "pobierz zaznaczone"], + ["F2", "zmień nazwę zaznaczonych"], + + "file-list-sel", + ["spacja", "przełącz zaznaczanie plików"], + ["↑/↓", "przenieś kursor zaznaczenia"], + ["ctrl ↑/↓", "przenieś kursor i widok"], + ["⇧ ↑/↓", "wybierz poprzedni/następny plik"], + ["ctrl-A", "wybierz wszystkie pliki/foldery"], + ], [ + "navigation", + ["B", "przełącz ścieżkę nawigacyjną / panel nawigacyjny"], + ["I/K", "poprzedni/następny folder"], + ["M", "folder nadrzędny (lub zwiń aktualny)"], + ["V", "przełącz foldery / pliki tekstowe w panelu nawigacyjnym"], + ["A/D", "rozmiar panelu nawigacyjnego"], + ], [ + "audio-player", + ["J/L", "poprzedni/następny utwór"], + ["U/O", "przejdź 10 sek. do tyłu/przodu"], + ["0..9", "przeskocz do 0%..90%"], + ["P", "odtwórz/pauza (również rozpoczyna)"], + ["S", "wybierz odtwarzany utwór"], + ["Y", "pobierz utwór"], + ], [ + "image-viewer", + ["J/L, ←/→", "poprzednie/następne zdjęcie"], + ["Home/End", "pierwsze/ostatnie zdjęcie"], + ["F", "pełny ekran"], + ["R", "obróć zgodnie ze wskaz. zegara"], + ["⇧ R", "obróć przeciwnie do ruchu wskaz. zegara"], + ["S", "wybierz zdjęcie"], + ["Y", "pobierz zdjęcie"], + ], [ + "video-player", + ["U/O", "przejdź 10 sek. do tyłu/przodu"], + ["P/K/Spacja", "odtwórz/pauza"], + ["C", "odtwarzaj następne po zakończeniu"], + ["V", "odtwarzaj w pętli"], + ["M", "wycisz"], + ["[ i ]", "ustaw opóźnienie pętli"], + ], [ + "textfile-viewer", + ["I/K", "poprzedni/następny plik"], + ["M", "zamknij plik"], + ["E", "edytuj plik"], + ["S", "wybierz plik (do wycięcia/skopiowania/zmiany nazwy)"], + ] + ], + + "m_ok": "OK", + "m_ng": "Anuluj", + + "enable": "Włącz", + "danger": "NIEBEZPIECZEŃSTWO", + "clipped": "skopiowano do schowka", + + "ht_s1": "sekunda", + "ht_s2": "sekund(y)", + "ht_m1": "minuta", + "ht_m2": "minuty", + "ht_h1": "godzina", + "ht_h2": "godziny", + "ht_d1": "dzień", + "ht_d2": "dni", + "ht_and": " i ", + + "goh": "panel sterowania", + "gop": 'poprzedni plik/folder">poprzedni', + "gou": 'nadrzędny folder">w górę', + "gon": 'następny folder">następny', + "logout": "Wyloguj ", + "access": " dostęp", + "ot_close": "zamknij pod-menu", + "ot_search": "szukaj plików po atrybutach, ścieżce / nazwie, tagach muzyki, bądź dowolnej ich kombinacji$N$N<code>foo bar</code> = musi zawierać «foo» oraz «bar»,$N<code>foo -bar</code> = musi zawierać «foo», lecz nie «bar»,$N<code>^yana .opus$</code> = musi zaczynać się od «yana» i być plikiem «opus»$N<code>"try unite"</code> = zawierać dokładnie «try unite»$N$Nformatem daty jest iso-8601, czyli$N<code>2009-12-31</code> lub <code>2020-09-12 23:30:00</code>", + "ot_unpost": "unpost: usuń ostatnio przesłane pliki lub przerwij przesyłanie", + "ot_bup": "bup: podstawowe przesyłanie danych, wspiera nawet netscape 4.0", + "ot_mkdir": "mkdir: tworzy nowy folder", + "ot_md": "new-md: tworzy nowy dokument markdown", + "ot_msg": "msg: wysyła wiadomość do loga serwera", + "ot_mp": "opcje odtwarzacza multimediów", + "ot_cfg": "opcje konfiguracji", + "ot_u2i": 'up2k: przesyła pliki (jeżeli masz dostęp do zapisu) lub uruchomia tryb wyszukiwania, aby sprawdzić czy już istnieją na serwerze$N$Nprzesyłanie można wznowić, jest wielowątkowe i znaczniki czasu są zachowywane, lecz zużywa więcej procesora niż [🎈]  (podstawowe przesyłanie)<br /><br />podczas przesyłania ta ikona zamienia się w wskaźnik postępu!', + "ot_u2w": 'up2k: przesyła pliki z możliwością wznowienia (można zamknąć przeglądarkę i dokończyć przesyłanie plików później)$N$Njest wielowątkowy i zachowuje znaczniki czasu plików, lecz zużywa więcej procesora od [🎈]  (podstawowego przesyłania)<br /><br />podczas przesyłania ta ikona zamienia się w wskaźnik postępu!', + "ot_noie": 'Użyj przeglądarki Chrome / Firefox / Edge', + + "ab_mkdir": "stwórz folder", + "ab_mkdoc": "stwórz dok. markdown", + "ab_msg": "wyślij wiad. do logów serwera", + + "ay_path": "przejdź do folderów", + "ay_files": "przejdź do plików", + + "wt_ren": "zmień nazwę zaznaczonych elementów$NSkrót: F2", + "wt_del": "usuń zaznaczone elementy$NSkrót: ctrl-K", + "wt_cut": "wytnij zaznaczone elementy <small>(aby wkleić gdzie indziej)</small>$NSkrót: ctrl-X", + "wt_cpy": "skopiuj zaznaczone pliki do schowka$N(aby wkleić gdzie indziej)$NSkrót: ctrl-C", + "wt_pst": "wklej wcześniej wycięte/skopiowane zaznaczenie$NSkrót: ctrl-V", + "wt_selall": "zaznacz wszystko$NHotkey: ctrl-A (when file focused)", + "wt_selinv": "odwróć zaznaczenie", + "wt_zip1": "pobierz folder jako archiwum", + "wt_selzip": "pobierz zaznaczone jako archiwum", + "wt_seldl": "pobierz zaznaczenie jako oddzielne pliki$NSkrót: Y", + "wt_npirc": "skopiuj informacje o utworze w formacie irc", + "wt_nptxt": "skopiuj informacje o utworze jako zwykły tekst", + "wt_m3ua": "dodaj to playlisty m3u (kliknij <code>📻copy</code> kliknij)", + "wt_m3uc": "skopiuj playlistę m3u do schowka", + "wt_grid": "przełącz widok siatki / listy$NSkrót: G", + "wt_prev": "poprzeni utwór$NSkrót: J", + "wt_play": "odtwórz / pauza$NSkrót: P", + "wt_next": "następny utwór$NSkrót: L", + + "ul_par": "przesyłane równolegle:", + "ut_rand": "losuj nazwy plików", + "ut_u2ts": "kopiuj znacznik ostatniej modyfikacji$Nz twojego systemu plików na serwer\">📅", + "ut_ow": "nadpisywać istniejące pliki na serwerzę?$N🛡️: nigdy (wygeneruje nową nazwę)$N🕒: nadpisz jeśli pliki na serwerze są starsze niż przesyłane$N♻️: zawsze nadpisuj jeśli zawartość plików się różni", + "ut_mt": "hashuj inne pliki podczas przesyłania$N$Nmożna wyłączyć w przypadku wystąpienia wąskiego gardła na CPU lub HDD", + "ut_ask": 'pytaj o potwierdzenie rozpoczęcia przesyłania">💭', + "ut_pot": "przyspiesz przesyłanie na słabszych urządzeniach,$Nupraszczając interfejs", + "ut_srch": "nie przesyłaj plików, jedynie sprawdź czy istnieją$Njuż na serwerze (przeskanuje wszystkie foldery dostępne do odczytu)", + "ut_par": "zatrzymuje przesyłanie jeśli wynosi 0$N$Nzwiększ w przypadku jeśli twoja sieć jest wolna / ma duże opóźnienia$N$Nustaw wartość 1 w sieci lokalnej lub w przypadku wolnego dysku serwerowego", + "ul_btn": "upuść pliki / foldery<br>tutaj (lub kliknij mnie)", + "ul_btnu": "P R Z E Ś L I J", + "ul_btns": "S Z U K A J", + + "ul_hash": "hashowanie", + "ul_send": "przesyłanie", + "ul_done": "gotowe", + "ul_idle1": "nic się jeszcze nie przesyła", + "ut_etah": "średnia prędkość <em>hashowania</em> i przewidywany czas do końca", + "ut_etau": "średnia prędkość <em>przesyłania</em> i przewidywany czas do końca", + "ut_etat": "średnia prędkość <em>ogólna</em> i przewidywany czas do końca", + + "uct_ok": "zakończone pomyślnie", + "uct_ng": "zakończono niepowodzeniem (odrzucono, nie znaleziono, itp.)", + "uct_done": "zakończono z błędami", + "uct_bz": "w trakcie (oblicznie sumy kontrolnej, przesyłanie)", + "uct_q": "oczekujące", + + "utl_name": "nazwa pliku", + "utl_ulist": "lista", + "utl_ucopy": "kopia", + "utl_links": "linki", + "utl_stat": "status", + "utl_prog": "postęp", + + // keep short: + "utl_404": "404", + "utl_err": "BŁĄD", + "utl_oserr": "błąd OS", + "utl_found": "znaleziono", + "utl_defer": "opóźnij", + "utl_yolo": "YOLO", + "utl_done": "gotowe", + + "ul_flagblk": "pliki zostały zakolejkowane,</b><br>lecz przesyłanie up2k już trwa (w innej zakładce),<br>oczekuję na zakończenie", + "ul_btnlk": "przełącznik zablokowany przez konfigurację serwera", + + "udt_up": "Prześlij", + "udt_srch": "Szukaj", + "udt_drop": "upuść tutaj", + + "u_nav_m": '<h6>co my tu mamy?</h6><code>Enter</code> = Pliki (jeden lub wiecej)\n<code>ESC</code> = Jeden folder (włącznie z podfolderami)', + "u_nav_b": '<a href="#" id="modal-ok">Pliki</a><a href="#" id="modal-ng">Jeden folder</a>', + + "cl_opts": "przełączniki", + "cl_themes": "motyw", + "cl_langs": "język", + "cl_ziptype": "pobieranie folderów", + "cl_uopts": "przełączniki przesyłania (up2k)", + "cl_favico": "favicon (ikona w przeglądarce)", + "cl_bigdir": "duże foldery", + "cl_hsort": "#sortowanie", + "cl_keytype": "notacja klucza", // not sure + "cl_hiddenc": "ukryte kolumny", + "cl_hidec": "ukryj", + "cl_reset": "zresetuj", + "cl_hpick": "kliknij nagłówki kolumn, aby ukryć je w tabeli niżej", + "cl_hcancel": "ukrywanie kolumn przerwane", + + "ct_grid": '田 siatka', + "ct_ttips": '◔ ◡ ◔">ℹ️ podpowiedzi', + "ct_thumb": 'w widoku siatki, przełącz ikony i miniaturki$NSkrót: T">🖼️ miniaturki', + "ct_csel": 'użyj CTRL i SHIFT do wybierania plików w widoku siatki">wybierz', + "ct_ihop": 'przejdź do ostatniego pliku po zamknięciu przeglądarki obrazów">g⮯', + "ct_dots": 'pokaż ukryte pliki (jeśli pozwala serwer)">ukryte', + "ct_qdel": 'pytaj o potwierdzenie przy usuwaniu tylko raz">pyt. us.', + "ct_dir1st": 'pokazuj foldery na początku">📁 najpierw', + "ct_nsort": 'naturalne sortowanie (dla numerowanych plików)">nsort', + "ct_utc": 'pokaż wszystkie daty/czas w UTC">UTC', + "ct_readme": 'pokazuj README.md w folderach">📜 readme', + "ct_idxh": 'pokazuj plik index.html zamiast zawartości folderu">htm', + "ct_sbars": 'pokazuj paski przewijania">⟊', + + "cut_umod": "uaktualnij znacznik ostatniej modyfikacji pliku, tak aby pasował do pliku lokalnego jeżeli plik już istnieje na serwerze (wymaga dostępu zapisu i usuwania)\">📅 ponownie", + + "cut_turbo": "przycisk „raz się żyje”, raczej NIE POWINIENEŚ tego włączać:$N$Nużywaj jeśli przesyłano ogromną liczbę plików i z jakiegoś powodu musisz przesłać pliki ponownie, kontynuując jak najszybciej$N$Nzamienia sprawdzanie sumy kontrolnej plików prostym <em>"czy ten plik jest tego samego rozmiaru jak ten na serwerze?"</em> więc jeśli pliki różnią się zawartością, ale są tego samego rozmiaru, NIE ZOSTANĄ przesłane ponownie$N$Nta opcja powinna zostać wyłączona po zakończeniu przesyłania, i potem "przesłać" te same pliki ponownie w celu weryfikacji\">turbo", + + "cut_datechk": "przy wyłączonym przycisku turbo nic nie robi$N$Nleciutko zmniejsza czynnik „raz się żyje”; dodatkowo sprawdza czy znaczniki modyfikacji pliku przesyłanego zgadzają się z serwerem$N$N<em>teorytycznie</em> powinno złapać to większość niedokończonych / uszkodzonych plików, lecz nie jest zamiennikiem wykonania ponownego sprawdzenia bez włączonego trybu turbo\">spr-daty", + + "cut_u2sz": "rozmiar (w MiB) każdego kawałka do przesłania; większe wartości szybciej latają po Atlantyku. Mniejsze wartości działają lepiej na bardzo niestabilnych połączeniach (neostrada?)", + + "cut_flag": "zapewnia, że tylko jedna karta przesyła dane w danym momencie$N -- opcja musi być włączona na innych kartach $N - dotyczy tylko kart w tej samej domenie", + + "cut_az": "przesyła pliki w kolejności alfabetycznej, zamiast rozpocząć od najmniejszego pliku$N$Nkolejność alfabetyczna może ułatwić oszacowanie, co mogło pójść nie tak na serwerze, lecz lekko spowalnia przesyłanie po światłowodzie lub w sieci lokalnej", + + "cut_nag": "powiadomienie systemowe po zakończeniu przesyłania$N(tylko jeśli przeglądarka lub karta nie jest aktywna)", + "cut_sfx": "sygnał dźwiękowy po zakończeniu przesyłania$N(tylko jeśli przeklądarka lub karta nie jest aktywna)", + + "cut_mt": "używaj wielowątkowości, aby przyspieszyć obliczanie sumy kontrolnej plików$N$Nużywa web workerów i wymaga$Nwięcej pamięci RAM (do 512 MiB)$N$Nprzyspiesza https o 30% i http 4,5-krotnie\">ww", + + "cut_wasm": "używaj WASM zamiast wbudowanego hashera przeglądarki; zwiększa prędkość na Chrome'o-pochodnych przeglądarkach, zwiększając zużycie procesora, ponadto wiele starszych wersji Chrome'a zawiera błędy powodujące zeżarcie całej pamięci RAM komputera i przymusowe zamknięcie przeglądarki jeżeli ta opcja jest włączona\">wasm", + + "cft_text": "tekst favicon (aby wyłączyć, usuń zawartość i przeładuj stronę)", + "cft_fg": "kolor tekstu", + "cft_bg": "kolor tła", + + "cdt_lim": "maksymalna liczba plików do pokazania na raz w folderze", + "cdt_ask": "przy przewijaniu w dół,$Nzapytaj co robić,$Nzamiast wczytywać kolejne pliki", + "cdt_hsort": "ile zasad sortowania (<code>,sorthref</code>) zawierać w generowanych linkach multimediów. Wartość 0 sprawi, że zasady sortowania zawarte w linkach multimediów przy otwarciu również będą ignorowane", + + "tt_entree": "pokaż panel nawigacyjny (panel boczny z drzewem folderów)$NSkrót: B", + "tt_detree": "pokaż ślad nawigacyjny$NSkrót: B", + "tt_visdir": "przewiń do wybranego folderu", + "tt_ftree": "przełącz drzewo folderów / pliki tekstowe$NSkrót: V", + "tt_pdock": "pokaż foldery nadrzędne w przypiętym u góry panelu", + "tt_dynt": "rozszerzaj panel wraz z drzewem", + "tt_wrap": "zawijaj tekst", + "tt_hover": "pokazuj za długie linie po najechaniu kursorem$N( psuje przewijanie gdy $N  kursor nie jest w lewym marginesie )", + + "ml_pmode": "na końcu folderu...", + "ml_btns": "komendy", + "ml_tcode": "transkoduj", + "ml_tcode2": "transkoduj do", + "ml_tint": "odcień", + "ml_eq": "korektor dźwięku (equalizer)", + "ml_drc": "kompresor zasięgu dynamiki", + + "mt_loop": "pętla/powtarzaj jeden utwór\">🔁", + "mt_one": "zatrzymaj po jednym utworze\">1️⃣", + "mt_shuf": "odtwarzaj losowo w każdym folderze\">🔀", + "mt_aplay": "autoodtwarzanie po kliknięciu linku do tego serwera, zawierającego identyfikator utworu$N$Nwyłączenie tej opcji zapobiegnie aktualizowaniu adresu strony podczas odtwarzania muzyki, aby zapobiec autoodtwarzaniu przy utracie ustawień\">a▶", + "mt_preload": "rozpocznij ładowanie kolejnego utworu blisko końca aktualnego w celu uzyskania odtwarzania bez przerw\">preload", + "mt_prescan": "przechodzi do następnego folderu przed zakończeniem ostatniego utworu,$Naby udobruchać przeglądarkę,$Nżeby nie zatrzymała odtwarzania\">naw", + "mt_fullpre": "próbuj zbuforować cały utwór;$N✅ włącz na <b>niestabilnych</b> połączeniach,$N❌ <b>wyłącz</b> na wolnych połączeniach\">pełnebuf", + "mt_fau": "nie zatrzymuj muzyki jeśli następna piosenka będzie się zbyt wolno buforować na telefonach (może sprawić, że tagi będą się niepoprawnie wyświetlać)\">☕️", + "mt_waves": "falisty pasek:$Npokazuj amplitudę dźwięku w pasku utworu\">~s", + "mt_npclip": "pokaż przyciski kopiowania aktualnie odtwarzanego utworu\">/np", + "mt_m3u_c": "pokaż przyciski kopiowania$Nwybranych piosenek jako playlista m3u8\">📻", + "mt_octl": "integracja z systemem operacyjnym (przyciski multimedialne / informacje o utworze)\">os-int", + "mt_oseek": "zezwól na przewijanie utworu poprzez integrację z systemem$N$Nuwaga: na niektórych urządzeniach (iPhone'y),$Nzamienia przycisk następnej piosenki\">seek", + "mt_oscv": "pokaż okładkę albumu w widoku systemu\">okładka", + "mt_follow": "podążaj za odtwarzanym utworem przewijając widok\">🎯", + "mt_compact": "kompaktowe sterowanie\">⟎", + "mt_uncache": "wyczyść pamięć podręczną  (spróbuj jeśli przeglądarka$Nzachowała zepsutą kopię utworu, przez co nie odtwarza się ona)\">uncache", + "mt_mloop": "odtwarzaj utwory w folderze w pętli\">🔁 loop", + "mt_mnext": "wczytaj następny folder i kontynuuj\">📂 next", + "mt_mstop": "zatrzymaj odtwarzanie\">⏸ stop", + "mt_cflac": "przekonwertuj format flac / wav na opus\">flac", + "mt_caac": "przekonwertuj format aac / m4a na opus\">aac", + "mt_coth": "przekonwertuj wszystkie inne formaty (nie będące mp3) na opus\">oth", + "mt_c2opus": "najlepszy wybór dla komputerów, laptopów i urządzeń z androidem\">opus", + "mt_c2owa": "opus-weba, dla iOS 17.5 i nowszych\">owa", + "mt_c2caf": "opus-caf, dla iOS 11 do 17\">caf", + "mt_c2mp3": "używaj na bardzo starych urządzeniach\">mp3", + "mt_c2ok": "cudownie, dobry wybór", + "mt_c2nd": "ten format nie jest rekomendowany dla twojego urządzenia, ale nadal jest w porządku", + "mt_c2ng": "wygląda na to, że to urządzenie nie wspiera tego formatu, lecz spróbujmy i tak", + "mt_xowa": "iOS zawiera błędy uniemożliwiające odtwarzanie w tle używając tego formatu; wybierz caf lub mp3", + "mt_tint": "jasność tła (0-100) paska,$Naby zmniejszyć widoczność buforowania", + "mt_eq": "włącza korektor dźwięku (equalizer) i kontrolę wzmocnienia dźwięku;$N$Nboost <code>0</code> = standardowa głośność 100% (niezmodyfikowana)$N$Nwidth <code>1  </code> = standardowe stereo (niezmodyfikowane)$Nwidth <code>0.5</code> = 50% crossfeed lewo-prawo$Nwidth <code>0  </code> = mono$N$Nboost <code>-0.8</code> & width <code>10</code> = usuwanie wokalu :^)$N$Nwłączenie korektora sprawia, że albumy bezprzerwowe są w pełni bez przerw, więc jeśli jest to dla ciebie ważne, zostaw wszystko na 0 (poza width = 1)", + "mt_drc": "włącza kompresor zakresu dynamiki (normalizacja głośności); włącza również korektor w celu zbalansowania tego spaghetti, więc ustaw wszystkie opcje korektora, oprócz 'width',na 0, jeśli go nie chcesz$N$Nobniża głośność audio nad THRESHOLD (próg) dB; dla każdego RATIO (współczynnika) dB, będącego ponad THRESHOLDem jest 1 dB wyjścia, więc domyślne wartości progu -24 i współczynnika 12 znaczą, że nigdy nie powinno być głośniej niż -22 dB i bezpieczne jest zwiększenie wzmocnienia korektora do 0.8, lub nawet 1.8 z ATK 0 i ogromnym RLS, jak 90 (działa tylko na firefoxie, inne przeglądarki mają limit RLS 1)$N$N(na wikipedii tłumaczą to dużo lepiej)", + + "mb_play": "odtwórz", + "mm_hashplay": "odtworzyć ten plik audio?", + "mm_m3u": "naciśnij <code>Enter/OK</code>, aby odtworzyć\nnaciśnij <code>ESC/Cancel</code>, aby edytować", + "mp_breq": "wymagany jest Firefox 82+, Chrome 73+ lub iOS 15+", + "mm_bload": "wczytywanie...", + "mm_bconv": "konwertowanie do {0}, proszę czekać...", + "mm_opusen": "ta przeglądarka nie może odtwarzać plików aac / m4a;\ntranskodowanie do formatu opus włączone", + "mm_playerr": "odtwarzanie nie powiodło się: ", + "mm_eabrt": "Odtwarzanie zostało przerwane", + "mm_enet": "Połączenie z internetem jest słabe", + "mm_edec": "Ten plik wydaje się uszkodzony??", + "mm_esupp": "Twoja przeglądarka nie rozumie tego formatu audio", + "mm_eunk": "Nieznany błąd", + "mm_e404": "Nie można odtworzyć; błąd 404: Nie znaleziono pliku.", + "mm_e403": "Nie można odtworzyć; błąd 403: Odmowa dostępu.\n\nSpróbuj przeładować stronę (F5), może cię wylogowało", + "mm_e500": "Nie można odtworzyć; błąd 500: Sprawdź logi serwera.", + "mm_e5xx": "Nie można odtworzyć; błąd serwera", + "mm_nof": "nie znaleziono więcej plików audio", + "mm_prescan": "Szukanie kolejnego utworu...", + "mm_scank": "Znaleziono następną piosenkę:", + "mm_uncache": "wyczyszczono pamięć podręczną; wszystkie utwory zostaną pobrane ponownie przy następnym odtworzeniu", + "mm_hnf": "ten utwór już nie istnieje", + + "im_hnf": "ten obraz już nie istnieje", + + "f_empty": 'ten folder jest pusty', + "f_chide": 'schowa kolumnę «{0}»\n\nkolumny można ponownie pokazać w zakładce ustwaień', + "f_bigtxt": "ten plik waży {0} MiB -- na pewno pokazać jako tekst?", + "f_bigtxt2": "odczytać jedynie koniec pliku? włączy również śledzenie, pokazując nowo-dodane linie tekstu w czasie rzeczywistym", + "fbd_more": '<div id="blazy">pokazuję <code>{0}</code> z <code>{1}</code> plików; <a href="#" id="bd_more">pokaż {2}</a> lub <a href="#" id="bd_all">pokaż wszystko</a></div>', + "fbd_all": '<div id="blazy">pokazuję <code>{0}</code> z <code>{1}</code> files; <a href="#" id="bd_all">pokaż wszystko</a></div>', + "f_anota": "{0} z {1} elementów zostało wybranych;\naby pokazać cały folder, zjedź na dół", + + "f_dls": 'linki do plików w aktualnym folderze\nzostały zmienione w linki pobierania', + + "f_partial": "Aby bezpiecznie pobrać plik, który aktualnie jest przesyłany, wybierz plik o tej samej nazwie, lecz bez rozszerzenia <code>.PARTIAL</code>. Żeby to zrobić, naciśnij ANULUJ lub klawisz ESC.\n\nWciśnięcie OK / Enter zignoruje to ostrzeżenie i pobierze plik tymczasowy <code>.PARTIAL</code>, który prawie z pewnością będzie zepsuty", + + "ft_paste": "wklej {0} elementów$NSkrót: ctrl-V", + "fr_eperm": 'nie można zmienić nazwy:\nnie posiadasz uprawnienia „move” w tym folderze', + "fd_eperm": 'nie można usunąć:\nnie posiadasz uprawnienia „delete” w tym folderze', + "fc_eperm": 'nie można wyciąć:\nnie posiadasz uprawnienia „move” w tym folderze', + "fp_eperm": 'nie można wkleić:\nnie posiadasz uprawnienia „write” w tym folderze', + "fr_emore": "wybierz przynajmniej jeden element do zmiany nazwy", + "fd_emore": "wybierz przynajmniej jeden element do usunięcia", + "fc_emore": "wybierz przynajmniej jeden element do wycięcia", + "fcp_emore": "wybierz przynajmniej jeden element do skopiowania", + + "fs_sc": "udostępnij ten folder", + "fs_ss": "udostępnij zaznaczone pliki", + "fs_just1d": "nie można wybrać więcej niż jednego folderu,\nani mieszać plików i folderów w jednym zaznaczeniu", + "fs_abrt": "❌ przerwij", + "fs_rand": "🎲 losuj nazwę", + "fs_go": "✅ stwórz udostępnienie", + "fs_name": "nazwa", + "fs_src": "źródło", + "fs_pwd": "hasło", + "fs_exp": "wygaśnięcie", + "fs_tmin": "min", + "fs_thrs": "godz.", + "fs_tdays": "dni", + "fs_never": "na zawsze", + "fs_pname": "opcjonalna nazwa linku; zostanie wylosowana jeśli pusta", + "fs_tsrc": "plik lub folder do udostępnienia", + "fs_ppwd": "hasło (opcjonalnie)", + "fs_w8": "udostępnianie...", + "fs_ok": "naciśnij <code>Enter/OK</code>, aby skopiować do schowka\nnaciśnij <code>ESC/Anuluj</code>, aby zamknąć", + + "frt_dec": "może naprawić niektóre zepsute nazwy plików\">dekoduj-url", + "frt_rst": "zresetuj zmodyfikowane nazwy plików do oryginalnych\">↺ zresetuj", + "frt_abrt": "przerwij i zamknij to okno\">❌ anuluj", + "frb_apply": "ZASTOSUJ ZMIANĘ NAZWY", + "fr_adv": "zmiana nazwy hurtowa / metadanych / wzorcem\">zaawansowane", + "fr_case": "rozróżnianie wielkości liter w regex\">wlit", + "fr_win": "nazwy bezpieczne dla systemu Windows; zamienia symbole <code><>:"\\|?*</code> na japońskie odpowiedniki\">win", + "fr_slash": "zamienia <code>/</code> symbolem, który nie tworzy nowych folderów\">brak /", + "fr_re": "wzorzec wyszukiwania regex stosowany do oryginalnych nazw plików; do grup przechwytywania można się odwołać w polu formatu poniżej, np. <code>(1)</code> i <code>(2)</code> itd.", + "fr_fmt": "inspirowane programem foobar2000:$N<code>(title)</code> zostaje zamienione na tytuł utworu,$N<code>[(artist) - ](title)</code> pomija [tą] część jeśli pole artysty jest puste$N<code>$lpad((tn),2,0)</code> wyrównuje numer utworu do 2 cyfr (np. 01, 06, 09, 16)", + "fr_pdel": "usuń", + "fr_pnew": "zapisz jako", + "fr_pname": "podaj nazwę nowego szablonu", + "fr_aborted": "anulowano", + "fr_lold": "poprzednia nazwa", + "fr_lnew": "nowa nazwa", + "fr_tags": "znaczniki dla wybranych plików (tylko do odczytu, w celach informacyjnych):", + "fr_busy": "zmienianie nazwy {0} plików...\n\n{1}", + "fr_efail": "zmiana nazwy zakończona niepowodzeniem:\n", + "fr_nchg": "{0} nowych nazw zostało zmienionych przez opcje <code>win</code> i/lub <code>brak /</code>\n\nKontynuować ze zmienionymi nazwami?", + + "fd_ok": "usunięto", + "fd_err": "usuwanie zakończone niepowodzeniem:\n", + "fd_none": "nie usunięto nic; usunięcie mogło zostać zablokowane przez konfigurację serwera (xbd)?", + "fd_busy": "usuwanie {0} elementów...\n\n{1}", + "fd_warn1": "USUNĄĆ {0} elementów?", + "fd_warn2": "<b>OSTATNIA SZANSA!</b> Tej operacji nie da się cofnąć. Usunąć?", + + "fc_ok": "wycięto {0} elementów", + "fc_warn": 'wycięto {0} elementów,\n\nlecz można je wkleić tylko w <b>tej</b> karcie\n(ze względu na ogromną ilość wybranych elementów)', + + "fcc_ok": "skopiowano {0} elementów do schowka", + "fcc_warn": 'skopiowano {0} elementów,\n\nlecz można je wkleić tylko w <b>tej</b> karcie\n(ze względu na ogromną ilość wybranych elementów)', + + "fp_apply": "zastosuj te nazwy", + "fp_ecut": "najpierw wytnij lub skopiuj pliki / foldery, aby je wkleić / przenieść\n\nuwaga: można wycinać / wklejać pomiędzy różnymi kartami przeglądarki", + "fp_ename": "Nie udało się przenieść {0} elementów, gdyż ich nazwy już istnieją w tym folderze. Nadaj im nowe nazwy poniżej, bądź zostaw pole nazwy puste, aby je pominąć:", + "fcp_ename": "Nie udało się przekopiować {0} elementów, gdyż ich nazwy już istnieją w tym folderze. Nadaj im nowe nazwy poniżej, bądź zostaw pole nazwy puste, aby je pominąć:", + "fp_emore": "pozostało jeszcze kilka kolizji nazw plików do poprawy", + "fp_ok": "przeniesiono", + "fcp_ok": "przekopiowano", + "fp_busy": "przenoszenie {0} elementów...\n\n{1}", + "fcp_busy": "kopiowanie {0} elementów...\n\n{1}", + "fp_err": "nie udało się przenieść:\n", + "fcp_err": "nie udało się skopiować:\n", + "fp_confirm": "przenieść tutaj {0} elementy(ów)?", + "fcp_confirm": "skopiować tutaj {0} elementy(ów)?", + "fp_etab": 'nie udało się odczytać schowka z innej karty przeglądarki', + "fp_name": "przesyłanie pliku z twojego urządzenia. Nadaj nazwę:", + "fp_both_m": '<h6>wybierz metodę wklejenia</h6><code>Enter</code> = Przenieś {0} pliki(ów) z «{1}»\n<code>ESC</code> = Prześlij {2} pliki(ów) z twojego urządzenia', + "fcp_both_m": '<h6>wybierz metodę wklejenia</h6><code>Enter</code> = Skopiuj {0} pliki(ów) z «{1}»\n<code>ESC</code> = Prześlij {2} pliki(ów) z twojego urządzenia', + "fp_both_b": '<a href="#" id="modal-ok">Przenieś</a><a href="#" id="modal-ng">Prześlij</a>', + "fcp_both_b": '<a href="#" id="modal-ok">Kopiuj</a><a href="#" id="modal-ng">Prześlij</a>', + + "mk_noname": "wpisz nazwę do pola po lewej zanim to zrobisz :p", + + "tv_load": "Wczytywanie pliku tekstowego:\n\n{0}\n\n{1}% (wczytano {2} z {3} MiB)", + "tv_xe1": "nie udało się wczytać pliku:\n\nbłąd ", + "tv_xe2": "404, nie znaleziono pliku", + "tv_lst": "lista plików tekstowych w", + "tvt_close": "powróć do widoku folderów$NSkrót: M (lub Esc)\">❌ zamknij", + "tvt_dl": "pobierz ten plik$NHotkey: Y\">💾 pobierz", + "tvt_prev": "pokaż poprzedni dokument$NSkrót: i\">⬆ poprzedni", + "tvt_next": "pokaż następny dokument$NSkrót: K\">⬇ następny", + "tvt_sel": "wybierz plik   ( do wycięcia / skopiowania / usunięcia / itp. )$NSkrót: S\">wyb", + "tvt_edit": "otwórz plik w edytorze tekstu$NSkrót: E\">✏️ edytuj", + "tvt_tail": "śledź zmiany w pliku; pokazuj nowe linie w czasie rzeczywistym\">📡 śledź", + "tvt_wrap": "zawijaj tekst\">↵", + "tvt_atail": "utrzymuj widok na dole strony\">⚓", + "tvt_ctail": "dekoduj kolory terminala (sekwencje sterujące ANSI)\">🌈", + "tvt_ntail": "limit przewijania (ile bajtów tekstu przechowywać w pamięci)", + + "m3u_add1": "dodano utwór do playlisty m3u", + "m3u_addn": "dodano {0} utwory(ów) do playlisty m3u", + "m3u_clip": "skopiowano playlistę m3u do schowka\n\nutwórz", + + "gt_vau": "nie pokazuj obrazu, odtwarzaj tylko dźwięk\">🎧", + "gt_msel": "wybierz pliki; kliknij plik z wciśniętym klawiszem CTRL, aby zastąpić$N$N<em>gdy tryb jest aktywny, kliknij dwukrotnie na plik / folder, żeby go otworzyć</em>$N$NSkrót: S\">wybierz wiele", + "gt_crop": "kadruj miniaturki do środka\">kadruj", + "gt_3x": "miniaturki w wysokiej rozdzielczości\">3x", + "gt_zoom": "przybliż", + "gt_chop": "przytnij", + "gt_sort": "sortuj według", + "gt_name": "nazwa", + "gt_sz": "rozmiar", + "gt_ts": "data", + "gt_ext": "typ", + "gt_c1": "przycinaj większą część nazw plików (pokazuj mniej)", + "gt_c2": "przycinaj mniejszą część nazw plików (pokazuj więcej)", + + "sm_w8": "wyszukiwanie...", + "sm_prev": "wyniki wyszukiwania poniżej pochodzą z poprzedniego zapytania:\n ", + "sl_close": "zamknij wyniki wyszukiwania", + "sl_hits": "pokazuję {0} wyniki(ów)", + "sl_moar": "pokaż więcej", + + "s_sz": "rozmiar", + "s_dt": "data", + "s_rd": "ścieżka", + "s_fn": "nazwa", + "s_ta": "znaczniki", + "s_ua": "data przesłania", + "s_ad": "zaawansowane", + "s_s1": "min. rozmiar (MiB)", + "s_s2": "maks. rozmiar (MiB)", + "s_d1": "min. data iso8601", + "s_d2": "maks. data iso8601", + "s_u1": "przesłane po", + "s_u2": "i/lub przed", + "s_r1": "ścieżka zawiera   (oddzielone spacją)", + "s_f1": "nazwa zawiera   (odwróć za pomocą -nope)", + "s_t1": "znaczniki zawierają   (^=start, koniec=$)", + "s_a1": "dokładne właściwości metadanych", + + "md_eshow": "nie można wyświetlić ", + "md_off": "[📜<em>readme</em>] wyłączone w [⚙️] -- dokument ukryty", + + "badreply": "Nie udało się przeanalizować odpowiedzi serwera", + + "xhr403": "403: Odmowa dostępu.\n\nSpróbuj przeładować stronę (F5), możliwe, że cię wylogowano", + "xhr0": "nieznany (być może utracono połączenie z serwerem, lub jest on nieaktywny)", + "cf_ok": "przepraszamy, włączyła się ochrona przed DD" + wah + "oS\n\nwszystko powinno wrócić do normy za około 30 sekund\n\njeśli nic się nie zmieni, naciśnij klawisz F5, aby przeładować stronę", + "tl_xe1": "nie można wyświetlić podfolderów:\n\nbłąd ", + "tl_xe2": "404: Nie znaleziono folderu", + "fl_xe1": "nie można wyświetlić plików w folderze:\n\nbłąd ", + "fl_xe2": "404: Nie znaleziono folderu", + "fd_xe1": "nie można stworzyć podfolderu:\n\nbłąd ", + "fd_xe2": "404: Nie znaleziono folderu nadrzędnego", + "fsm_xe1": "nie można wysłać wiadomości:\n\nbłąd ", + "fsm_xe2": "404: Nie znaleziono folderu nadrzędnego", + "fu_xe1": "nie udało się wczytać listy unpost z serwera:\n\nbłąd ", + "fu_xe2": "404: Nie znaleziono pliku??", + + "fz_tar": "nieskompresowane archiwum gnu-tar (linux / mac)", + "fz_pax": "nieskompresowane archiwum tar w formacie pax (wolniejsze)", + "fz_targz": "gnu-tar z kompresją gzip poziomu 3.,$N$Nzazwyczaj bardzo wolne, używaj nieskompresowanego tar", + "fz_tarxz": "gnu-tar z kompresją xz poziomu 3.$N$Nzazwyczaj bardzo wolne, używaj nieksompresowanego tar", + "fz_zip8": "zip z nazwami plików UTF-8 (może działać nieprawidłowo na systemie Windows 7 i starszych)", + "fz_zipd": "zip z nazwami plików cp437, dobre dla bardzo starego oprogramowania", + "fz_zipc": "cp437 z CRC32 obliczonym wcześniej,$Ndla MS-DOS PKZIP v2.04g (październik 1993)$N(przetwarzanie do pobrania trwa dłużej)", + + "un_m1": "można usunąć ostatnio przesłane pliki (lub przerwać trwające) poniżej", + "un_upd": "odśwież", + "un_m4": "lub udostępnij pliki widoczne poniżej:", + "un_ulist": "pokaż", + "un_ucopy": "kopiuj", + "un_flt": "filtruj (opcjonalnie):  URL musi zawierać", + "un_fclr": "wyczyść kryteria filtrowania", + "un_derr": 'nie udało się usunąć unpost:\n', + "un_f5": 'coś poszło nie tak, spróbuj odświeżyć lub wciśnij F5', + "un_uf5": "przed przerwaniem przesyłania trzeba odświeżyć stronę (za pomocą CTRL-R lub F5)", + "un_nou": '<b>ostrzeżenie:</b> serwer jest aktualnie zbyt obciążony, żeby pokazać niedokończone przesłania; kliknij link "odśwież" za chwilę', + "un_noc": '<b>ostrzeżenie:</b> unpost w pełni przesłanych plików jest wyłączone/zabronione w konfiguracji serwera', + "un_max": "pokazuję pierwsze 2000 plików (użyj filtrowania)", + "un_avail": "{0} ostatnio przesłanych elementów może zostać usunięte<br />{1} niedokończonych można przerwać", + "un_m2": "przesortowano po czasie przesłania; najnowsze elementy pierwsze: ", + "un_no1": "cholibka! żaden przesłany element nie jest wystarczająco niedawny", + "un_no2": "cholibka! żaden przesłany element pasujący do filtra nie jest wystarczająco niedawny", + "un_next": "usuń następne {0} pliki(ów) poniżej", + "un_abrt": "przerwij", + "un_del": "usuń", + "un_m3": "wczytywanie ostatnio przesłanych elementów...", + "un_busy": "usuwanie {0} plików...", + "un_clip": "skopiowano {0} linków do schowka", + + "u_https1": "powinieneś przejść", + "u_https2": "na HTTPS w celu", + "u_https3": "uzyskania lepszej wydajności", + "u_ancient": 'twoja przeglądarka jest niezwykle zabytkowa -- powinieneś zamiast tego <a href="#" onclick="goto(\'bup\')">użyć bup</a>', + "u_nowork": "wymaga Firefox 53+, Chrome 57+ lub iOS 11+", + "tail_2old": "wymaga Firefox 105+, Chrome 71+ lub iOS 14.5+", + "u_nodrop": 'ta przeglądarka jest za stara, nie wspiera przesyłania "przeciągnij i upuść"', + "u_notdir": "to nie jest folder!\n\nta przeglądarka jest za stara\nspróbuj przeciągnąć i upuścić", + "u_uri": "aby przeciągnąć i upuścić obrazy z innych okien przeglądarki,\nupuść je na duży przycisk przesyłania", + "u_enpot": 'przełącz na <a href="#">lekki interfejs</a> (może zwiększyć prędkość przesyłania)', + "u_depot": 'przełącz na <a href="#">ładny interfejs</a> (może zmniejszyć prędkośc przesyłania)', + "u_gotpot": 'przełączanie na lekki interfejs w celu poprawy prędkości przesyłania,\n\nzawsze można przełączyć się na ładny interfejs!', + "u_pott": "<p>pliki:   <b>{0}</b> ukończonych,   <b>{1}</b> nie powiodło się,   <b>{2}</b> w trakcie,   <b>{3}</b> oczekujących</p>", + "u_ever": "podstawowe przesyłanie; up2k wymaga minimalnie przeglądarek:<br>Chrome 21 // Firefox 13 // Edge 12 // Opera 12 // Safari 5.1", + "u_su2k": 'podstawowe przesyłanie; <a href="#" id="u2yea">up2k</a> jest lepszy', + "u_uput": 'optymalizuj dla prędkości (pomijając spr. sum kontrolnych)', + "u_ewrite": 'nie masz dostępu do zapisu (write) w tym folderze', + "u_eread": 'nie masz dostępu do odczytu (read) tego folderu', + "u_enoi": 'wyszukiwanie plików jest wyłączone w konfiguracji serwera', + "u_enoow": "nadpisanie nie zadziała, wymagany dostęp do usuwania (delete)", + "u_badf": '{0} (z {1}) plików zostało pominiętych, prawdopodobnie przez opcje dostępu systemu plików:\n\n', + "u_blankf": '{0} (z {1}) plików jest pustych; przesłać mimo to?\n\n', + "u_applef": '{0} (z {1}) plików może być niepożądane;\nNaciśnij <code>OK/Enter</code>, aby pominąć je (wypisane poniżej);\nNaciśnij <code>Anuluj/ESC</code>, by je przesłać mimo to:\n\n', + "u_just1": '\nTa funkcja może działać lepiej z wybranym jednym plikiem', + "u_ff_many": "na systemach <b>Linux / MacOS / Android,</b> ta ilośc plików <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>może</em> spowodować przymusowe zamknięcie przeglądarki Firefox</a>\nw takim przypadku, spróbuj ponownie (lub użyj Chrome'a).", + "u_up_life": "Ten przesyłany plik zostanie usunięty z serwera\n{0} po zakończeniu przesyłania", + "u_asku": 'prześlij {0} pliki(ów) do <code>{1}</code>', + "u_unpt": "można cofnąć / usunąć ten przesłany plik za pomocą 🧯 w lewym górnym rogu", + "u_bigtab": 'zaraz pokażę {0} plików\n\nta operacja może zawiesić twoją przeglądarkę, na pewno kontynuować?', + "u_scan": 'Skanowanie plików...', + "u_dirstuck": 'iterator katalogów utknął podczas próby dostępu poniższych {0} elementów, pominięto:', + "u_etadone": 'Ukończono ({0}, {1} plików)', + "u_etaprep": '(przygotowywanie do przesłania)', + "u_hashdone": 'obliczono sumę kontrolną', + "u_hashing": 'obliczanie sumy kontrolnej', + "u_hs": 'nawiązywanie połączenia...', + "u_started": "rozpoczęto przesyłanie; zobacz w [🚀]", + "u_dupdefer": "duplikat; zostanie przetworzony na końcu", + "u_actx": "kliknij ten napis, aby zapobiec spadkowi <br />wydajności po zmianie aktywnego okna/karty przeglądarki", + "u_fixed": "OK!  Naprawiono 👍", + "u_cuerr": "nie udało się przesłać fragmentu {0} z {1};\nprawdopodobnie niegroźne, kontynuowanie\n\nplik: {2}", + "u_cuerr2": "serwer odrzucił przesyłanie (kawałek {0} z {1});\nspróbuję ponownie później\n\nplik: {2}\n\nbłąd ", + "u_ehstmp": "spróbuję ponownie; więcej informacji w prawym dolnym rogu", + "u_ehsfin": "serwer odrzucił prośbę o zakończenie przesyłania; próbuję ponownie...", + "u_ehssrch": "serwer odrzucił prośbę o wykonanie wyszukania; próbuję ponownie...", + "u_ehsinit": "serwer odrzucił prośbę o rozpoczęcie przesyłania; próbuję ponownie...", + "u_eneths": "błąd sieci podczas negocjacji warunków przesyłania; próbuję ponownie...", + "u_enethd": "błąd sieci podczas sprawdzania istnienia celu; próbuję ponownie...", + "u_cbusy": "oczekiwanie na ponowne zaufanie serwera po błędzie sieci...", + "u_ehsdf": "brak miejsca na dysku serwera!\n\npróby będą ponawiane na wypadek\nzwolnienia wystarczająco dużo miejsca aby kontynuować", + "u_emtleak1": "wygląda na to, że twoja przeglądarka może mieć wyciek pamięci;\n", + "u_emtleak2": ' <a href="{0}">przejdź na HTTPS (zalecane)</a> lub ', + "u_emtleak3": ' ', + "u_emtleakc": 'spróbuj:\n<ul><li>wciśnij <code>F5</code>, aby odświeżyć stronę</li><li>wyłącz przycisk  <code>ww</code>  w menu <code>⚙️ ustawienia</code></li><li>i spróbuj przesłać ponownie</li></ul>Prędkość przesyłania będzie niższa, ale cóż zrobisz.\nPrzepraszamy za problemu!\n\nPS: Chrome v107 <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">ma poprawkę tego błędu</a>.', + "u_emtleakf": 'spróbuj:\n<ul><li>wciśnij <code>F5</code>, aby odświeżyć stronę</li><li>włącz tryb <code>🥔</code> (lekkiego interfejsu) w interfejsie przesyłania<li>i spróbuj przesłać ponownie</li></ul>\nPS: Firefox może kiedyś mieć <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">poprawkę tego błędu</a>', + "u_s404": "nie znaleziono na serwerze", + "u_expl": "wytłumacz", + "u_maxconn": "większość przeglądarek ogranicza to do 6, ale Firefox pozwala zwiększyć tą wartość, ustawiając <code>connections-per-server</code> w <code>about:config</code>", + "u_tu": '<p class="warn">UWAGA: tryb turbo włączony, <span> klient może nie wykryć i nie kontynuować niedokończonych przesłań; patrz wskazówka przycisku turbo</span></p>', + "u_ts": '<p class="warn">UWAGA: tryb turbo włączony, <span> wyniki wyszukiwania mogą być niepoprawne; patrz wskazówka przycisku turbo</span></p>', + "u_turbo_c": "tryb turbo jest wyłączony w konfiguracji serwera", + "u_turbo_g": "wyłączanie trybu turbo, nie posiadasz dostępu\ndo listy katalogu w tym wolumenie", + "u_life_cfg": 'autousuwanie po <input id="lifem" p="60" /> min (lub <input id="lifeh" p="3600" /> godz.)', + "u_life_est": 'przesłany plik zostanie usunięty <span id="lifew" tt="local time">---</span>', + "u_life_max": 'ten folder wymaga\nmaks. czasu do usunięcia równego {0}', + "u_unp_ok": 'unpost jest dozwolony przez {0}', + "u_unp_ng": 'unpost NIE jest dozwolony', + "ue_ro": 'dostęp tylko-do-odczytu\n\n', + "ue_nl": 'nie jesteś zalogowany', + "ue_la": 'zalogowano jako "{0}"', + "ue_sr": 'jesteś w trybie wyszukiwania\n\nprzełącz się na tryb przesyłania, klikając lupę 🔎 (obok przycisku Szukaj), i spróbuj ponownie', + "ue_ta": 'spróbuj przesłać ponownie, wszystko powinno być w porządku', + "ue_ab": "ten plik już jest przesyłany do innego folderu, przesyłanie musi się zakończyć, zanim będzie mógł być on przesłany gdzie indziej.\n\nMożna przerwać pierwsze przesyłanie za pomocą 🧯 w lewym górnym rogu", + "ur_1uo": "OK: Plik przesłany pomyślnie", + "ur_auo": "OK: Wszystkie ({0}) pliki zostały przesłane pomyślnie", + "ur_1so": "OK: Znaleziono plik na serwerze", + "ur_aso": "OK: Znaleziono wszystkie ({0}) pliki na serwerze", + "ur_1un": "Przesyłanie nie powiodło się", + "ur_aun": "Wszystkie ({0}) przesłania nie powiodły się", + "ur_1sn": "NIE znaleziono pliku na serwerze", + "ur_asn": "NIE znaleziono {0} plików na serwerze", + "ur_um": "Zakończono;\n{0} przesłań OK,\n{1} przesłań nie powiodło się", + "ur_sm": "Zakończono;\nznaleziono {0} pliki(ów),\nnie znaleziono {1} pliki(ów) na serwerze", + + "lang_set": "odśwież stronę (F5), aby zastosować zmianę.", + }, "rus": { "tt": "Русский", diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 6ffd95f6..2475cb97 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -385,6 +385,48 @@ var Ls = { "af1": "vis nylig opplasta filer", "ag1": "vis kjente IdP-brukarar", }, + "pol": { + "a1": "odśwież", + "b1": "witaj, nieznajomy   <small>(nie jesteś zalogowany)</small>", + "c1": "wyloguj się", + "d1": "zrzut stosu", + "d2": "pokazuje status wszystkich aktywnych wątków", + "e1": "przeładuj konfigurację", + "e2": "przeładuj pliki konfiguracyjne (konta/wolumeny/flagi wolumenów),$Ni przeskanuje wszystkie wolumeny e2ds$N$Nnotka: zmiany konfiguracji globalnej$Nwymagają pełnego uruchomienia ponownie serwera, aby zaczęły obowiązywać", + "f1": "możesz przeglądać:", + "g1": "możesz przesyłać do:", + "cc1": "inne:", + "h1": "wyłącz k304", + "i1": "włącz k304", + "j1": "włączenie k304 będzie odłączało klienta przy każdorazowym otrzymaniu kodu HTTP 304, co może zapobiec wieszaniu się wadliwych proxy, <em>ale</em> spowolni ogólne działanie", + "k1": "zresetuj ustawienia klienta", + "l1": "zaloguj się po więcej:", + "m1": "Witaj,", + "n1": "404 nie znaleziono  ┐( ´ -`)┌", + "o1": 'lub możesz nie mieć dostępu -- spróbuj wprowadzić hasło lub <a href="' + SR + '/?h">przejdź do strony głównej</a>', + "p1": "403 odmowa dostępu  ~┻━┻", + "q1": 'użyj hasła lub <a href="' + SR + '/?h">przejdź do strony głównej</a>', + "r1": "idź do strony głównej", + ".s1": "przeskanuj ponownie", + "t1": "akcje", + "u2": "czas od ostatniej interakcji z serwerem$N( przesyłania / zmiany nazwy / ... )$N$N17d = 17 dni$N1h23 = 1 godzina 23 minuty$N4m56 = 4 minuty 56 sekund", + "v1": "połącz", + "v2": "używaj tego serwera jako dysku lokalnego", + "w1": "przejdź na HTTPS", + "x1": "zmień hasło", + "y1": "edytuj udostępnione", + "z1": "odblokuj udostępnienie:", + "ta1": "najpierw wprowadź nowe hasło", + "ta2": "powtórz hasło dla potwierdzenia:", + "ta3": "znaleziono literówkę, spróbuj ponownie", + "aa1": "pliki przychodzące:", + "ab1": "wyłącz no304", + "ac1": "włącz no304", + "ad1": "włączenie no304 wyłączy przechowywanie jakiejkolwiek pamięci podręcznej. Zmarnuje to olbrzymią ilość ruchu sieciowego!", + "ae1": "trwające pobierania:", + "af1": "pokaż ostatnio przesłane pliki", + "ag1": "pokaż znanych użytkowników IdP", + }, "spa": { "a1": "actualizar", "b1": "hola   <small>(no has iniciado sesión)</small>", From 91ce7a29aa3324a8e321d002e62fbf4e5b1f13ec Mon Sep 17 00:00:00 2001 From: Chloe Surett <chloe@surett.me> Date: Sat, 9 Aug 2025 16:17:20 -0400 Subject: [PATCH 128/174] Add .idea to .gitignore (#547) --- .gitignore | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.gitignore b/.gitignore index b7b533b4..b70c56b7 100644 --- a/.gitignore +++ b/.gitignore @@ -43,3 +43,6 @@ scripts/docker/*.err # nix build output link result + +# IDEA config +.idea/ From e9ddfccfb60f43a66e04f463a2084c666d4cb557 Mon Sep 17 00:00:00 2001 From: Tr3yWay996 <112483771+Tr3yWay996@users.noreply.github.com> Date: Sat, 9 Aug 2025 22:26:52 +0200 Subject: [PATCH 129/174] Add French translation (#486) Add French translation (#486) --------- Signed-off-by: ed <s@ocv.me> Co-authored-by: Packingdustry <alois.mc@hotmail.com> Co-authored-by: Andrew Lee <andrew@alee14.me> Co-authored-by: A. Jakubiak <contact@jakubiak.fr> --- copyparty/web/browser.js | 631 ++++++++++++++++++++++++++++++++++++++- 1 file changed, 630 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index b3d174f8..f6b1e3eb 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3785,6 +3785,635 @@ var Ls = { "lang_set": "ladataanko sivu uudestaan kielen vaihtamiseksi?", }, + "fra": { + "tt": "français", + + "cols": { + "c": "bouton d'action", + "dur": "durée", + "q": "qualité / débit binaire", + "Ac": "codec audio", + "Vc": "codec vidéo", + "Fmt": "format / conteneur", + "Ahash": "somme de contrôle audio", + "Vhash": "somme de contrôle vidéo", + "Res": "résolution", + "T": "type de fichier", + "aq": "qualité audio / débit binaire", + "vq": "qualité vidéo / débit binaire", + "pixfmt": "sous-échantillonnage / structure de pixel", + "resw": "résolution horizontale", + "resh": "résolution verticale", + "chs": "canaux audio", + "hz": "fréquence" + }, + + "hks": [ + [ + "misc", + ["Échap", "ferme divers menus"], + + "gestionaire de fichiers", + ["G", "activer vue en liste / vue en grille"], + ["T", "activer les miniatures / icônes"], + ["⇧ A/D", "taille des miniatures"], + ["ctrl-K", "suprimer la sélection"], + ["ctrl-X", "couper la sélection au presse-papier"], + ["ctrl-C", "copier la sélection au presse-papier"], + ["ctrl-V", "coller (déplacer/copier) ici"], + ["Y", "télécharger la sélection"], + ["F2", "renomer la sélection"], + + "file-list-sel", + ["Espace", "activer la sélection de fichiers"], + ["↑/↓", "déplacer le selecteur"], + ["ctrl ↑/↓", "déplacer le curseur et la zone d'affichage"], + ["⇧ ↑/↓", "sélectioner le fichier précédent/suivant"], + ["ctrl-A", "sélectionner tout les fichiers / dossiers"], + ], [ + "navigation", + ["B", "basculer la vue en fil d'Ariane / panneau de navigation"], + ["I/K", "dossier précédent/suivant"], + ["M", "dossier parent (ou réduire le dossier actuel)"], + ["V", "activer les dossiers / fichiers texte dans le volet de navigation"], + ["A/D", "taille du volet de navigation"], + ], [ + "lecteur-audio", + ["J/L", "chanson précédente/suivante"], + ["U/O", "sauter 10s en arrière/avant"], + ["0..9", "sauter à 0%..90%"], + ["P", "lecture/pause (démarre également la lecture)"], + ["S", "sélectionner la chanson en cours"], + ["Y", "télécharger le morceau"], + ], [ + "visionneuse d'image", + ["J/L, ←/→", "image précédente/suivante"], + ["Début/Fin, ⭦/Fin", "première/dernière image"], + ["F", "plein écran"], + ["R", "rotation horaire"], + ["⇧ R", "rotation antihoraire"], + ["S", "sélectionner l'image"], + ["Y", "télécharger l'image"], + ], [ + "lecteur vidéo", + ["U/O", "sauter 10s en arrière/avant"], + ["P/K/Espace", "lecture/pause"], + ["C", "continuer de lire la suivante"], + ["V", "lire en boucle"], + ["M", "couper le son"], + ["[ and ]", "définir l'intervalle de boucle"], + ], [ + "visionneuse de texte", + ["I/K", "fichier précédent/suivant"], + ["M", "fermer le fichier texte"], + ["E", "modifier le fichier texte"], + ["S", "sélectioner le fichier (pour le couper/copier/renommer)"], + ] + ], + + "m_ok": "OK", + "m_ng": "Annuler", + + "enable": "Activer", + "danger": "DANGER", + "clipped": "copié dans le presse-papier", + + "ht_s1": "seconde", + "ht_s2": "secondes", + "ht_m1": "minute", + "ht_m2": "minutes", + "ht_h1": "heure", + "ht_h2": "heures", + "ht_d1": "jour", + "ht_d2": "jours", + "ht_and": " et ", + + "goh": "panneau-de-commande", + "gop": 'élément "frère" précédent">précédent', + "gou": 'dossier parent">haut', + "gon": 'dossier suivant">suivant', + "logout": "Déconnexion ", + "access": "accès", + "ot_close": "fermer le sous-menu", + "ot_search": "chercher des fichiers par leurs attributs, chemin / nom, tag musicaux, ou nimporte quelle combinaison de ces options$N$N<code>foo bar</code> = doit contenir à la fois «foo» et «bar»,$N<code>foo -bar</code> = doit contenir «foo» mais pas «bar»,$N<code>^yana .opus$</code> = commence par «yana» et est un fichier «opus»$N<code>"try unite"</code> = contient exactement «try unite»$N$Nle format de date est iso-8601, comme$N<code>2009-12-31</code> ou <code>2020-09-12 23:30:00</code>", + "ot_unpost": "unpost: supprimer vos téléchargements récents, ou annuler ceux en cours", + "ot_bup": "bup: téléverseur de base, prend même en charge netscape 4.0", + "ot_mkdir": "mkdir: créer un nouveau répertoire", + "ot_md": "new-md: créer un nouveau document markdown", + "ot_msg": "msg: envoyer un message au journal du serveur", + "ot_mp": "options du lecteur multimedia", + "ot_cfg": "options de configuration", + "ot_u2i": 'up2k : téléverser des fichiers (si vous avez un accès en écriture) ou basculer en mode recherche pour voir s\'ils existent quelque part sur le serveur$N$Nles téléversements peuvent être repris, ils sont multithreadé, et les horodatages des fichiers sont préservés, mais cela utilise plus de CPU que [🎈]  (le téléverseur de base)<br /><br />pendant les téléversements, cette icône devient un indicateur de progression!', + "ot_u2w": 'up2k : téléverser des fichiers avec prise en charge de la reprise (fermez votre navigateur et déposez les mêmes fichiers plus tard)$N$multithreadé, et les horodatages des fichiers sont préservés, mais cela utilise plus de CPU que [🎈]  (le téléverseur de base)<br /><br />pendant les téléversements, cette icône devient un indicateur de progression!', + "ot_noie": 'Utilisez Chrome / Firefox / Edge', + + "ab_mkdir": "créer un nouveau répertoire", + "ab_mkdoc": "faire un nouveau document markdown", + "ab_msg": "envoyer un message au journal du serveur", + + "ay_path": "passer aux dossiers", + "ay_files": "passer aux fichiers", + + "wt_ren": "renommer les éléments sélectionnés$NHotkey: F2", + "wt_del": "supprimer les éléments sélectionnés$NHotkey: ctrl-K", + "wt_cut": "couper les éléments sélectionnés <small>(puis coller ailleurs)</small>$NHotkey: ctrl-X", + "wt_cpy": "copier les éléments sélectionnés dans le presse-papiers$N(pour les coller ailleurs)$NHotkey: ctrl-C", + "wt_pst": "coller une sélection précédemment coupée / copiée$NHotkey: ctrl-V", + "wt_selall": "sélectionner tous les fichiers$NHotkey: ctrl-A (lorsque le fichier est sélectionné)", + "wt_selinv": "inverser la sélection", + "wt_zip1": "télécharger ce dossier en tant qu'archive", + "wt_selzip": "télécharger la sélection en tant qu'archive", + "wt_seldl": "télécharger la sélection en tant que fichiers séparés$NHotkey: Y", + "wt_npirc": "copier les informations de la musique au format irc", + "wt_nptxt": "copier les informations de la musique en texte brut", + "wt_m3ua": "ajouter à la playlist m3u (cliquez sur <code>📻copier</code> plus tard)", + "wt_m3uc": "copier la playlist m3u dans le presse-papiers", + "wt_grid": "basculer entre la vue en grille / liste$NHotkey: G", + "wt_prev": "musique précédente$NHotkey: J", + "wt_play": "lecture / pause$NHotkey: P", + "wt_next": "musique suivante$NHotkey: L", + + "ul_par": "téléversements parallèles:", + "ut_rand": "attribution de noms de fichiers aléatoires", + "ut_u2ts": "copier l'horodatage de dernière modification$Nde votre système de fichiers vers le serveur\">📅", + "ut_ow": "écraser les fichiers existants sur le serveur?$N🛡️: jamais (générera un nouveau nom de fichier à la place)$N🕒: écraser si le fichier sur le serveur est plus ancien que le vôtre$N♻️: toujours écraser si les fichiers sont différents", + "ut_mt": "continuer à calculer la somme de contrôle d'autres fichiers pendant le téléversement$N$Npeut-être désactiver si votre CPU ou HDD est la cause de perte de performances", + "ut_ask": 'demander confirmation avant le début du téléversement">💭', + "ut_pot": "améliorer la vitesse de téléversement sur les appareils lents$Nen simplifiant l'interface utilisateur", + "ut_srch": "ne pas réellement téléverser, mais vérifier si les fichiers existent déjà$N sur le serveur (scannera tous les dossiers que vous pouvez lire)", + "ut_par": "mettre en pause les téléversements en le réglant sur 0$N$Naugmenter si votre connexion est lente / à forte latence$N$Nle garder à 1 sur le LAN ou si le HDD du serveur est un goulot d'étranglement", + "ul_btn": "déposer des fichiers / dossiers<br>ici (ou cliquez sur moi)", + "ul_btnu": "T É L É V E R S E R", + "ul_btns": "C H E R C H E R", + + "ul_hash": "somme de contrôle", + "ul_send": "envoyer", + "ul_done": "terminé", + "ul_idle1": "aucun téléversement n'est encore dans la file d'attente", + "ut_etah": "moyenne <em>hashing</em> vitesse, et temps estimé jusqu'à la fin", + "ut_etau": "moyenne <em>upload</em> vitesse et temps estimé jusqu'à la fin", + "ut_etat": "moyenne <em>total</em> vitesse et temps estimé jusqu'à la fin", + + "uct_ok": "terminé avec succès", + "uct_ng": "non réussi : échoué / rejeté / non trouvé", + "uct_done": "terminés et échoué combinés", + "uct_bz": "hachage ou téléversement", + "uct_q": "inactif, en attente", + + "utl_name": "nom de fichier", + "utl_ulist": "liste", + "utl_ucopy": "copie", + "utl_links": "liens", + "utl_stat": "état", + "utl_prog": "progrès", + + // keep short: + "utl_404": "404", + "utl_err": "ERREUR", + "utl_oserr": "OS-ERREUR", + "utl_found": "trouvé", + "utl_defer": "état", + "utl_yolo": "YOLO", + "utl_done": "terminé", + + "ul_flagblk": "les fichiers ont été ajoutés à la file d'attente</b><br>cependant, il y a un processus up2k actif dans un autre onglet du navigateur,<br>en attente qu'il finisse d'abord", + "ul_btnlk": "la configuration du serveur a verrouillé cette options dans cet état", + + "udt_up": "Téléverser", + "udt_srch": "Chercher", + "udt_drop": "déposer ici", + + "u_nav_m": '<h6>aight, ques-que tu à ?</h6><code>Enter</code> = Fichiers (un ou plus)\n<code>ESC</code> = Un dossier (sous-dossiers inclus)', + "u_nav_b": '<a href="#" id="modal-ok">Fichiers</a><a href="#" id="modal-ng">Un dossier</a>', + + "cl_opts": "options", + "cl_themes": "thème", + "cl_langs": "langue", + "cl_ziptype": "téléchargement de dossier", + "cl_uopts": "up2k", + "cl_favico": "favicon", + "cl_bigdir": "gros dossiers", + "cl_hsort": "#sort", + "cl_keytype": "notation des touches", + "cl_hiddenc": "colonnes masquées", + "cl_hidec": "masquer", + "cl_reset": "réinitialiser", + "cl_hpick": "cliquez sur les en-têtes de colonnes pour les masquer dans le tableau ci-dessous", + "cl_hcancel": "masquage des colonnes annulé", + + "ct_grid": '田 grille', + "ct_ttips": '◔ ◡ ◔">ℹ️ infobulles', + "ct_thumb": 'vue en grille, activer les icônes ou les miniatures$NHotkey: T">🖼️ minia', + "ct_csel": 'utiliser CTRL et MAJ pour selectioner des fichiers en vue en grille">sel', + "ct_ihop": 'quand le visionneuse d\'image est fermé, faire defiller vers le bas jusqu\'au dernier fichier">g⮯', + "ct_dots": 'voir les fichiers caché (si le serveur le permet)">dotfiles', + "ct_qdel": 'ne demander qu\'une confirmation lors de la suppression de fichiers>qdel', + "ct_dir1st": 'trier les dossiers avant les fichiers">📁 first', + "ct_nsort": 'triage par numérotation (pour les nom de fichiers qui sont numérotés)">nsort', + "ct_utc": 'voir tout les horodatage en format UTC">UTC', + "ct_readme": 'voir le fichier README.md dans le listage des dossiers">📜 readme', + "ct_idxh": 'voir une version html (index.html) au-lieu du listage des dossiers normal">htm', + "ct_sbars": 'montrer la barre de defilement">⟊', + + "cut_umod": "si un fichier existe déjà sur le server, mettre à jour l'horodatage de dernière modification du serveur pour qu'il corresponde à votre fichier local (nécessite des autorisations d'écriture et de suppression)\">re📅", + + "cut_turbo": "le bouton yolo, vous ne voulez probablement PAS activer ceci:$N$Nutilisez ceci si vous téléchargez une grande quantité de fichiers et que vous devez redémarrer pour une raison quelconque, et que vous souhaitez continuer le téléchargement dès que possible$N$Ncela remplace la vérification de hachage par une simple <em>"est-ce que cela a la même taille de fichier sur le serveur?"</em> donc si le contenu du fichier est différent, il ne sera PAS téléchargé$N$Nvous devriez désactiver cela lorsque le téléchargement est terminé, puis "télécharger" les mêmes fichiers à nouveau pour laisser le client les vérifier\">turbo", + + "cut_datechk": "n'a aucun effet à moins que le bouton turbo ne soit activé$N$Nréduit le facteur yolo d'un tout petit peu ; vérifie si les horodatages des fichiers sur le serveur correspondent aux vôtres$N$Ndevrait <em>théoriquement</em> attraper la plupart des téléchargements inachevés / corrompus, mais n'est pas un substitut à un passage de vérification avec turbo désactivé par la suite\">date-chk", + + "cut_u2sz": "taille (en MiB) de chaque morceau de téléversement; des grosse valeurs vont mieux passer si la distance entre le serveur et vous est trés grande. Si vous avez une connection trés instable, essayer de plus petites valeurs", + + "cut_flag": "s'assurer qu'un seul onglet est entrain de mettre un fichier en ligne a la fois $N -- les autres onglets doivent avoir cette option activé aussi $N -- affecte seulement les onglets qui sont sur le même domaine", + + "cut_az": "mettre en ligne les fichiers dans l'ordre alphabétique, plutôt que le plus petit fichier en premier$N$Nl'ordre alphabétique peut rendre la lecture plus douce sur pour les yeux si quelque chose s'est mal passé sur le serveur, mais cela rend le téléversement légèrement plus lent sur fibre / LAN", + + "cut_nag": "recevoir une notification via l'OS quand un téléversement finit$N(seulement si le navigateur ou l'onglet n'est pas actif)", + "cut_sfx": "alerte audible quand le téléversement finit$N(seulement si le navigateur ou l'onglet n'est pas actif)", + + "cut_mt": "utiliser le calcul de somme de contrôle multithreadé pour accelerer le processus$N$Ncela utilise des web-workers et nécessite$Nplus de RAM (jusqu'à 512 MiB supplémentaires)$N$NCela rend https 30% plus rapide, http 4.5x plus rapide\">mt", + + "cut_wasm": "utiliser wasm au lieu du hachage intégré du navigateur; améliore la vitesse sur les navigateurs basés sur chrome mais augmente la charge CPU, et de nombreuses anciennes versions de chrome ont des bugs qui font que le navigateur consomme toute la RAM et plante si cela est activé\">wasm", + + "cft_text": "text favicon (laisser vide et rafraîchir pour désactiver)", + "cft_fg": "couleur de premier plan", + "cft_bg": "couleur d'arrière-plan", + + "cdt_lim": "nombre maximum de fichiers à afficher dans un dossier", + "cdt_ask": "lorsque vous faites défiler vers le bas,$Nau lieu de charger plus de fichiers,$Ndemander quoi faire", + "cdt_hsort": "combien de règles de tri (<code>,sorthref</code>) à inclure dans les media-URLs. Définir cette valeur à 0 ignorera également les règles de tri incluses dans les liens média lorsque vous cliquez dessus.", + + "tt_entree": "afficher le panneau de navigation (arborescence des dossiers)$NHotkey: B", + "tt_detree": "afficher le fil d’Ariane$NHotkey: B", + "tt_visdir": "faire défiler jusqu'au dossier sélectionné", + "tt_ftree": "basculer l'arborescence des dossiers / fichiers texte$NHotkey: V", + "tt_pdock": "afficher les dossiers parents dans un panneau ancré en haut", + "tt_dynt": "croissance automatique à mesure que l'arborescence s'étend", + "tt_wrap": "retour à la ligne", + "tt_hover": "révéler les lignes débordantes au survol$N( interrompt le défilement à moins que le curseur de la souris ne soit dans la gouttière gauche )", + + "ml_pmode": "à la fin du dossier…", + "ml_btns": "cmds", + "ml_tcode": "transcoder", + "ml_tcode2": "transcoder vers", + "ml_tint": "teinte", + "ml_eq": "égaliseur audio", + "ml_drc": "compresseur de plage dynamique", + + "mt_loop": "répéter en boucle une musique\">🔁", + "mt_one": "stopper après une musique\">1️⃣", + "mt_shuf": "mélanger les musiques dans chaque dossiers\">🔀", + "mt_aplay": "jouer automatiquement si le lien utilisé pour accéder au serveur a un song-ID $N$N, désactiver cela arrêtera également la mise à jour de l'URL de la page avec les song-IDs lors de la lecture de la musique, pour éviter la lecture automatique si ces paramètres sont perdus mais que l'URL reste\">a▶", + "mt_preload": "commencer à charger la prochaine chanson près de la fin pour une lecture sans interruption\">preload", + "mt_prescan": "explorer le dossier suivant avant la dernière musique$Nne finisse, pour garder le navigateur content$Npour qu'il n'arrête pas la lecture\">nav", + "mt_fullpre": "essayer de pré-charger la musique entière;$N✅ activer en cas de connection instable,$N❌ désactiver en revanche sur une connection lente va probablement être mieux\">full", + "mt_fau": "sur téléphone, empêche la musique de s'arrêter de jouer si la prochaine n'est pas pré-chargée assez rapidement (peut rendre l'affichage des tags buggé)\">☕️", + "mt_waves": "barre de progression en spectrograme:$Nmontrer l'amplitude audio dans la miniature\">~s", + "mt_npclip": "montrer les boutons pour copier le morceau en cours de lecture\">/np", + "mt_m3u_c": "montrer les boutons pour copier les$morceaux sélectionnées en tant qu'entrées de playlist m3u8\">📻", + "mt_octl": "intégration os (touches de raccourci multimédia / osd)\">os-ctl", + "mt_oseek": "permettre la recherche via l'intégration os$N$Nremarque : sur certains appareils (iPhones),$Ncela remplace le bouton de la chanson suivante\">seek", + "mt_oscv": "montrer la couverture de l'album dans l'osd\">art", + "mt_follow": "garder la piste en cours défilée dans la vue\">🎯", + "mt_compact": "contrôles compacts\">⟎", + "mt_uncache": "effacer le cache  (essayez ceci si votre navigateur a mis en cache$Nun copie défectueuse d'une chanson, ce qui empêche sa lecture)\">uncache", + "mt_mloop": "lire en boucle le dossier ouvert\">🔁 loop", + "mt_mnext": "charger le dossier suivant et continuer\">📂 next", + "mt_mstop": "arrêter la lecture\">⏸ stop", + "mt_cflac": "convertir flac / wav en opus\">flac", + "mt_caac": "convertir aac / m4a en opus\">aac", + "mt_coth": "convertir tout les autres (pas mp3) en opus\">oth", + "mt_c2opus": "meilleur choix pour PC fixe, PC portable, android\">opus", + "mt_c2owa": "opus-weba, pour iOS 17.5 et supérieur\">owa", + "mt_c2caf": "opus-caf, pour iOS 11 à 17\">caf", + "mt_c2mp3": "utilisez ceci sur des appareils très anciens\">mp3", + "mt_c2flac": "meilleure qualité sonore, mais téléchargements énormes\">flac", + "mt_c2wav": "lecture non compressée (encore plus gros)\">wav", + "mt_c2ok": "bien, bon choix", + "mt_c2nd": "ce n'est pas le format de sortie recommandé pour votre appareil, mais ça devrait aller", + "mt_c2ng": "votre appareil ne semble pas prendre en charge ce format de sortie, mais essayons quand même", + "mt_xowa": "il y a des bugs dans iOS qui empeche d'avoir une lecture en ariere plan en utilisant ce format; utilisez caf ou mp3 à la place", + "mt_tint": "niveau d’arrière-plan (0–100) de la barre de progression$Npour rendre la mise en mémoire tampon moins gênante", + "mt_eq": "active l'égaliseur et le contrôle de gain;$N$Nboost <code>0</code> = volume standard 100% (non modifié)$N$Nwidth <code>1  </code> = stéréo standard (non modifié)$Nwidth <code>0.5</code> = 50% de crossfeed gauche-droite$Nwidth <code>0  </code> = mono$N$Nboost <code>-0.8</code> & width <code>10</code> = suppression vocale :^)$N$Nl'activation de l'égaliseur rend les albums gapless entièrement gapless, alors laissez-le activé avec toutes les valeurs à zéro (sauf largeur = 1) si vous vous en souciez", + "mt_drc": "active le compresseur de plage dynamique (aplanisseur de volume / brickwaller); activera également l'EQ pour équilibrer les choses, donc définissez tous les champs EQ sauf 'width' sur 0 si vous ne le voulez pas$N$Ndiminue le volume de l'audio au-dessus de THRESHOLD dB; pour chaque RATIO dB au-delà de THRESHOLD, il y a 1 dB de sortie, donc des valeurs par défaut de tresh -24 et ratio 12 signifient qu'il ne devrait jamais être plus fort que -22 dB et qu'il est sûr d'augmenter le boost de l'égaliseur à 0.8, ou même 1.8 avec ATK 0 et un énorme RLS comme 90 (ne fonctionne que dans firefox; RLS est max 1 dans les autres navigateurs)$N$N(voir wikipedia, ils expliquent cela beaucoup mieux)", + + "mb_play": "lecture", + "mm_hashplay": "lire ce fichier audio ?", + "mm_m3u": "appuyez sur <code>Entrée/OK</code> pour lire\nappuyez sur <code>Échap/Annuler</code> pour modifier", + "mp_breq": "nécessite firefox 82+ ou chrome 73+ ou iOS 15+", + "mm_bload": "chargement en cours…", + "mm_bconv": "conversion en {0}, veuillez patienter…", + "mm_opusen": "votre navigateur ne peut pas lire les fichiers aac / m4a ;\nle transcodage en opus est maintenant activé", + "mm_playerr": "échec de la lecture : ", + "mm_eabrt": "La tentative de lecture a été annulée", + "mm_enet": "Votre connexion internet est instable ou inexistante", + "mm_edec": "Ce fichier est supposément corrompu??", + "mm_esupp": "Votre navigateur ne comprend pas ce format audio", + "mm_eunk": "Erreur inconnue", + "mm_e404": "Impossible de lire l'audio ; erreur 404 : fichier introuvable.", + "mm_e403": "Impossible de lire l'audio ; erreur 403 : accès refusé.\n\nEssayez d'appuyer sur F5 pour recharger, peut-être que vous avez été déconnecté", + "mm_e500": "Impossible de lire l'audio ; erreur 500 : vérifiez les journaux du serveur.", + "mm_e5xx": "Impossible de lire l'audio ; erreur serveur ", + "mm_nof": "Pas d'autres fichiers audio trouvés par ici", + "mm_prescan": "En recherche d'une autre musique à lire…", + "mm_scank": "Prochaine musique trouvée :", + "mm_uncache": "cache vidé ; toutes les chansons seront retéléchargées lors de la prochaine lecture", + "mm_hnf": "cette chanson n'existe plus", + + "im_hnf": "cette image n'existe plus", + + "f_empty": 'ce dossier est vide', + "f_chide": 'ceci va cacher les colonnes «{0}»\n\ntu peut les réafficher dans les options', + "f_bigtxt": "ce fichier fait {0} MiB -- tu veut vraiment le voir en tant que texte ?", + "f_bigtxt2": "voir seulement la fin du fichier à la place ? ceci activera aussi le suivi en temps réel, affichant les nouvelles lignes de texte au fur et à mesure", + "fbd_more": '<div id="blazy">showing <code>{0}</code> of <code>{1}</code> files; <a href="#" id="bd_more">show {2}</a> or <a href="#" id="bd_all">show all</a></div>', + "fbd_all": '<div id="blazy">showing <code>{0}</code> of <code>{1}</code> files; <a href="#" id="bd_all">show all</a></div>', + "f_anota": "seulement {0} des {1} elements sont selectioné;\npour selectioner le dossier entier, fait défiler jusqu'au fond", + + "f_dls": 'le lien de fichier dans le répertoire actuel\nà été changé en lien de téléchargement', + + "f_partial": "Pour télécharger de façon sécurisée un fichier qui est entrain de se faire téléverser, cliquez sur le fichier qui a le même nom, mais sans l'extension de fichier <code>.PARTIAL</code>. Choisissez CANCEL ou appuiez sur la touche Échap pour faire cela.\n\nAppuyer sur OK / Entrée ignorera cet avertissement et continuera à télécharger le fichier temporaire <code>.PARTIAL</code> à la place, ce qui donnera presque certainement des données corrompues.", + + "ft_paste": "coller {0} éléments$NHotkey: ctrl-V", + "fr_eperm": 'impossible de renommer:\n vous n\'avez pas la permission “move” dans ce dossier', + "fd_eperm": 'impossible de supprimer:\nvous n\'avez pas la permission “delete” dans ce dossier', + "fc_eperm": 'impossible de couper:\nvous n\'avez pas la permission “move” dans ce dossier', + "fp_eperm": 'impossible de coller:\nvous n\'avez pas la permission “write” dans ce dossier', + "fr_emore": "sélectionnez au moins un élément à renommer", + "fd_emore": "sélectionnez au moins un élément à supprimer", + "fc_emore": "sélectionnez au moins un élément à couper", + "fcp_emore": "sélectionnez au moins un élément à copier dans le presse-papiers", + + "fs_sc": "partager le dossier dans lequel vous vous trouvez", + "fs_ss": "partager les fichiers sélectionnés", + "fs_just1d": "vous ne pouvez pas sélectionner plus d'un dossier,\nou mélanger des fichiers et des dossiers dans une seule sélection", + "fs_abrt": "❌ abandonner", + "fs_rand": "🎲 nom.aleatoire", + "fs_go": "✅ créer partage", + "fs_name": "nom", + "fs_src": "source", + "fs_pwd": "mdp", + "fs_exp": "expiration", + "fs_tmin": "min", + "fs_thrs": "heures", + "fs_tdays": "jours", + "fs_never": "éternel", + "fs_pname": "nom de lien optionnel ; sera aléatoire si vide", + "fs_tsrc": "le fichier ou le dossier à partager", + "fs_ppwd": "mot de passe optionnel", + "fs_w8": "création du partage…", + "fs_ok": "appuyez sur <code>Entrée/OK</code> pour le Presse-papiers\nappuyez sur <code>Échap/Annuler</code> pour fermer", + + "frt_dec": "peut potentiellement réparer certaines instances de noms de fichiers cassés\">url-decode", + "frt_rst": "réinitialiser les noms de fichiers modifiés à leurs originaux\">↺ reset", + "frt_abrt": "abandonner et fermer cette fenêtre\">❌ cancel", + "frb_apply": "APPLIQUER RENOMMER", + "fr_adv": "renommage par lot / métadonnées / motif\">advanced", + "fr_case": "regex sensible à la casse\">case", + "fr_win": "noms windows-safe; remplacer <code><>:"\\|?*</code> par des caractères japonais en pleine largeur\">win", + "fr_slash": "remplacer <code>/</code> par un caractère qui ne provoque pas la création de nouveaux dossiers\">no /", + "fr_re": "modèle de recherche regex à appliquer aux noms de fichiers originaux ; les groupes capturés peuvent être référencés dans le champ de format ci-dessous comme <code>(1)</code> et <code>(2)</code> et ainsi de suite", + "fr_fmt": "inspiré par foobar2000 : $N<code>(title)</code> est remplacé par le titre de la chanson, $N<code>[(artist) - ](title)</code> saute [cette] partie si l'artiste est vide, $N<code>$lpad((tn),2,0)</code> remplit le numéro de piste à 2 chiffres", + "fr_pdel": "supprimer", + "fr_pnew": "enregistrer sous", + "fr_pname": "donnez un nom pour le nouveau preset", + "fr_aborted": "abandonné", + "fr_lold": "ancien nom", + "fr_lnew": "nouveau nom", + "fr_tags": "tags pour les fichier selectioné (lecture-seule, juste pour référence):", + "fr_busy": "renomage de {0} items…\n\n{1}", + "fr_efail": "renomage a échoué:\n", + "fr_nchg": "{0} des nouveaux noms ont été modifiés en raison de <code>win</code> et/ou <code>no /</code>\n\nOK pour continuer avec ces nouveaux noms modifiés ?", + + "fd_ok": "suppression réussie", + "fd_err": "impossible de supprimer:\n", + "fd_none": "rien n'a été supprimé ; peut-être bloqué par la configuration du serveur (xbd) ?", + "fd_busy": "suppression de {0} éléments…\n\n{1}", + "fd_warn1": "SUPPRIMER ces {0} éléments ?", + "fd_warn2": "<b>Dernière chance !</b> Impossible de revenir en arrière. Supprimer ?", + + "fc_ok": "couper {0} éléments", + "fc_warn": 'couper {0} éléments\n\nmais : seul <b>cet</b> onglets peut les coller\n(puisque la sélection est si absolument massive)', + + "fcc_ok": "copié {0} éléments dans le presse-papiers", + "fcc_warn": 'copié {0} éléments dans le presse-papiers\n\nmais : seul <b>cet</b> onglet peut les coller\n(puisque la sélection est si absolument massive)', + + "fp_apply": "utiliser ces noms", + "fp_ecut": "en premier, coupez ou copiez quelques fichiers / dossiers à coller / déplacer\n\nnote: vous pouvez couper / coller a travers different onglets", + "fp_ename": "{0} éléments ne peuvent pas être déplacés ici parceque leurs noms sont déjà pris. Donnez leurs un nouveau nom ci-dessous pour continuer, ou laissez les vides pour les sauter:", + "fcp_ename": "{0} éléments ne peuvent pas être copiés ici parce que les noms sont déjà pris. Donnez-leur un nouveau nom ci-dessous pour continuer, ou laissez-les vides pour les sauter :", + "fp_emore": "il reste encore des collisions de noms de fichiers à corriger", + "fp_ok": "déplacement OK", + "fcp_ok": "copie OK", + "fp_busy": "déplacement de {0} éléments…\n\n{1}", + "fcp_busy": "copie de {0} éléments…\n\n{1}", + "fp_err": "deplacement échoué:\n", + "fcp_err": "copie échouée:\n", + "fp_confirm": "déplacer ces {0} éléments ici ?", + "fcp_confirm": "copier ces {0} éléments ici ?", + "fp_etab": 'lecture du presse-papier venant d\'un autre onglet échoué', + "fp_name": "téléversement d'un fichier de votre apareil. Donnez lui un nom:", + "fp_both_m": '<h6>choisisez ce qu\'il faut coller</h6><code>Entrer</code> = Déplacer {0} fichiers de «{1}»\n<code>ESC</code> = Téléverser {2} fichiers de votre appareil', + "fcp_both_m": '<h6>choisissez ce qu\'il faut coller</h6><code>Entrer</code> = Copier {0} fichiers de «{1}»\n<code>ESC</code> = Téléverser {2} fichiers de votre appareil', + "fp_both_b": '<a href="#" id="modal-ok">Déplacer</a><a href="#" id="modal-ng">Téléverser</a>', + "fcp_both_b": '<a href="#" id="modal-ok">Copier</a><a href="#" id="modal-ng">Téléverser</a>', + + "mk_noname": "entrez un nom dans le champ de texte à gauche avant de faire ça :p", + + "tv_load": "Chargement du document texte:\n\n{0}\n\n{1}% ({2} de {3} MiB chargés)", + "tv_xe1": "impossible de charger le fichier texte:\n\nerreur", + "tv_xe2": "404, fichier introuvable", + "tv_lst": "liste des fichiers texte dans", + "tvt_close": "retour a la vue de dossier$NHotkey: M (ou Échap)\">❌ fermer", + "tvt_dl": "télécharger ce fichier$NHotkey: Y\">💾 télécharger", + "tvt_prev": "montrer le document précédent$NHotkey: i\">⬆ précédent", + "tvt_next": "montrer le document suivant$NHotkey: K\">⬇ suivant", + "tvt_sel": "sélectionner le fichier   ( pour couper / copier / supprimer / … )$NHotkey: S\">sel", + "tvt_edit": "ouvrir le fichier dans l'éditeur de texte$NHotkey: E\">✏️ modifier", + "tvt_tail": "surveiller le fichier pour les changements; montrer les nouvelles lignes en temps réel\">📡 suivre", + "tvt_wrap": "retour à la ligne\">↵", + "tvt_atail": "ancrer le défilement au fond de la page\">⚓", + "tvt_ctail": "décoder les couleurs du terminal (ansi escape codes)\">🌈", + "tvt_ntail": "limite de défilement en arrière (combien d'octets de texte à garder chargé)", + + "m3u_add1": "musique ajoutée à la playlist m3u", + "m3u_addn": "{0} musiques ajoutées à la playlist m3u", + "m3u_clip": "la playlist m3u est maintenant copiée dans le presse-papier\n\nvous devriez créer un nouveau fichier texte nommé par exemple playlist.m3u et coller la playlist dans ce fichier ; cela la rendra lisible en tant que playlist", + + "gt_vau": "ne pas voir les vidéos, juste jouer l'audio\">🎧", + "gt_msel": "activer la séléction de fichiers ; ctrl-clic sur un fichier pour override écraser$N$N<em>quand actif : double-cliquer sur un fichier / dossier pour l'ouvrir</em>$N$NHotkey: S\">multiséléction", + "gt_crop": "rogner les miniatures au centre\">rogner", + "gt_3x": "miniatures haute résolution\">3x", + "gt_zoom": "zoomer", + "gt_chop": "rogner", + "gt_sort": "trier par", + "gt_name": "nom", + "gt_sz": "taille", + "gt_ts": "date", + "gt_ext": "type", + "gt_c1": "tronquer les noms de fichiers (montrer moins)", + "gt_c2": "tronquer les noms de fichiers (montrer plus)", + + "sm_w8": "recherche…", + "sm_prev": "les résultats de recherche ci-dessous proviennent d'une requête précédente:\n ", + "sl_close": "fermer les résultats de recherche", + "sl_hits": "affichage de {0} résultats", + "sl_moar": "chercher plus", + + "s_sz": "taille", + "s_dt": "date", + "s_rd": "chemin", + "s_fn": "nom", + "s_ta": "tags", + "s_ua": "up@", + "s_ad": "adv.", + "s_s1": "minimum MiB", + "s_s2": "maximum MiB", + "s_d1": "min. iso8601", + "s_d2": "max. iso8601", + "s_u1": "téléverser après", + "s_u2": "et/ou avant", + "s_r1": "le chemin contient   (séparé par des espaces)", + "s_f1": "le nom contient   (négation avec -nope)", + "s_t1": "les tags contiennent   (^=début, fin=$)", + "s_a1": "propriétés de métadonnées spécifiques", + + "md_eshow": "impossible d'afficher le rendu ", + "md_off": "[📜<em>readme</em>] disabled in [⚙️] -- document caché", + + "badreply": "Échec de l'analyse de la réponse du serveur", + + "xhr403": "403: Accès refusé\n\nessayez d'appuyer sur F5, peut-être que vous avez été déconnecté", + "xhr0": "inconnu (vous avez probablement perdu la connexion au serveur, ou le serveur est hors ligne)", + "cf_ok": "désolé pour cela -- la protection DD" + wah + "oS a été déclenché\n\nles choses devraient reprendre dans environ 30 secondes\n\nsi rien ne se passe, appuyez sur F5 pour recharger la page", + "tl_xe1": "impossible de lister les sous-dossiers:\n\nerreur ", + "tl_xe2": "404: Dossier introuvable", + "fl_xe1": "impossible de lister les fichiers dans le dossier:\n\nerreur ", + "fl_xe2": "404: Dossier introuvable", + "fd_xe1": "impossible de créer le sous-dossier:\n\nerreur ", + "fd_xe2": "404: Dossier parent introuvable", + "fsm_xe1": "impossible d'envoyer le message:\n\nerreur ", + "fsm_xe2": "404: Dossier parent introuvable", + "fu_xe1": "échec du chargement de la liste des unpost du serveur:\n\nerreur ", + "fu_xe2": "404: Fichier introuvable??", + + "fz_tar": "fichier gnu-tar non compressé (linux / mac)", + "fz_pax": "tar au format pax non compressé (plus lent)", + "fz_targz": "gnu-tar avec compression gzip niveau 3$N$Ncela est généralement très lent, donc$Nutilisez plutôt tar non compressé", + "fz_tarxz": "gnu-tar avec compression xz niveau 1$N$Ncela est généralement très lent, donc$Nutilisez plutôt tar non compressé", + "fz_zip8": "zip avec noms de fichiers utf8 (peut être instable sur windows 7 et versions antérieures)", + "fz_zipd": "zip avec noms de fichiers cp437 traditionnels, pour les très anciens logiciels", + "fz_zipc": "cp437 avec crc32 calculé tôt,$Nfor MS-DOS PKZIP v2.04g (octobre 1993)$N(prend plus de temps à charger avant que le téléchargement ne commence)", + + "un_m1": "vous pouvez supprimer vos téléchargements récents (ou annuler ceux en cours) ci-dessous", + "un_upd": "rafraîchir", + "un_m4": "ou partager les fichiers visibles ci-dessous:", + "un_ulist": "montrer", + "un_ucopy": "copier", + "un_flt": "filtre optionnel:  l'URL doit contenir", + "un_fclr": "effacer le filtre", + "un_derr": 'échec de l\'unpost-delete:\n', + "un_f5": 'quelque chose a cassé, veuillez essayer de rafraîchir ou d\'appuyer sur F5', + "un_uf5": "désolé mais vous devez rafraîchir la page (par exemple en appuyant sur F5 ou CTRL-R) avant que ce téléchargement puisse être annulé", + "un_nou": '<b>warning:</b> serveur trop occupé pour afficher les téléversements non finis; cliquez sur le lien "rafraîchir" dans un instant', + "un_noc": '<b>warning:</b> unpost des fichiers entièrement téléchargés n\'est pas activé/permis dans la configuration du serveur', + "un_max": "affichage des 2000 premiers fichiers (utilisez le filtre)", + "un_avail": "{0} téléchargements récents peuvent être supprimés<br />{1} ceux en cours peuvent être annulés", + "un_m2": "triés par date de téléchargement; les plus récents en premier:", + "un_no1": "sike! aucun téléchargement n'est suffisamment récent", + "un_no2": "sike! aucun téléchargement correspondant à ce filtre n'est suffisamment récent", + "un_next": "supprimer les {0} fichiers suivants ci-dessous", + "un_abrt": "abandonner", + "un_del": "supprimer", + "un_m3": "chargement de vos téléchargements récents…", + "un_busy": "suppression de {0} fichiers…", + "un_clip": "{0} liens copiés dans le presse-papiers", + + "u_https1": "vous devriez", + "u_https2": "passer à https", + "u_https3": "pour de meilleure performances", + "u_ancient": 'votre navigateur est impressionnamment ancien -- vous devriez peut-être <a href="#" onclick="goto(\'bup\')">utiliser bup à la place</a>', + "u_nowork": "nécessite firefox 53+ ou chrome 57+ ou iOS 11+", + "tail_2old": "nécessite firefox 105+ ou chrome 71+ ou iOS 14.5+", + "u_nodrop": 'votre navigateur est trop ancien pour le téléversement par glisser-déposer', + "u_notdir": "ce n'est pas un dossier!\n\nvotre navigateur est trop ancien,\nveuillez essayer le glisser-déposer à la place", + "u_uri": "pour glisser-déposer des images depuis d'autres fenêtres de navigateur,\nveuillez les déposer sur le gros bouton de téléversement", + "u_enpot": 'passer à <a href="#">l\'interface utilisateur potato</a> (peut améliorer la vitesse de téléversement)', + "u_depot": 'passer à <a href="#">l\'interface utilisateur fancy</a> (peut réduire la vitesse de téléversement)', + "u_gotpot": 'passage à l\'interface utilisateur potato pour une vitesse de téléversement améliorée,\n\nn\'hésitez pas à revenir en arrière si ça ne vous plaît pas !', + "u_pott": "<p>fichiers:   <b>{0}</b> fini,   <b>{1}</b> échoué,   <b>{2}</b> en cours,   <b>{3}</b> en attente</p>", + "u_ever": "ceci est le téléverseur de base ; up2k nécessite au moins chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'ceci est le téléverseur de base; <a href="#" id="u2yea">up2k</a> est meilleur', + "u_uput": 'optimiser pour la vitesse (ignorer la somme de contrôle)', + "u_ewrite": 'vous n\'avez pas accès en écriture à ce dossier', + "u_eread": 'vous n\'avez pas accès en lecture à ce dossier', + "u_enoi": 'la recherche de fichiers n\'est pas activée dans la configuration du serveur', + "u_enoow": "l'écrasage ne fonctionnera pas ici; besoin de permissions de suppression", + "u_badf": 'Ces {0} fichiers (sur {1} au total) ont été ignorés, probablement en raison de permissions système de fichiers:\n\n', + "u_blankf": 'Ces {0} fichiers (sur {1} au total) sont vides; les téléverser quand même ?\n\n', + "u_applef": 'Ces {0} fichiers (sur {1} au total) sont probablement indésirables;\nAppuyez sur <code>OK/Enter</code> pour IGNORER les fichiers suivants,\nAppuyez sur <code>Cancel/Échap</code> pour NE PAS exclure, et TÉLÉVERSER ceux-ci également:\n\n', + "u_just1": '\nPeut-être que cela fonctionne mieux si vous sélectionnez juste un fichier', + "u_ff_many": "si vous utilisez <b>Linux / MacOS / Android,</b> alors ce nombre de fichiers <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>peut</em> faire planter Firefox!</a>\nSi cela se produit, veuillez réessayer (ou utiliser Chrome).", + "u_up_life": "Ce téléversement va être supprimé du serveur\n{0} après son achèvement", + "u_asku": 'téléverser ces {0} fichiers vers <code>{1}</code>', + "u_unpt": "vous pouvez défaire / supprimer ce téléversement en utilisant le 🧯 en haut à gauche", + "u_bigtab": 'sur le point d\'afficher {0} fichiers\n\ncela peut faire planter votre navigateur, êtes-vous sûr ?', + "u_scan": 'Analyse des fichiers…', + "u_dirstuck": 'l\'itérateur de répertoire est bloqué en essayant d\'accéder aux {0} éléments suivants ; il sera ignoré :', + "u_etadone": 'Terminé ({0}, {1} fichiers)', + "u_etaprep": '(préparation au téléversement)', + "u_hashdone": 'calcul de la somme de contrôle terminé', + "u_hashing": 'calcul de la somme de contrôle', + "u_hs": 'établissement d\'une liaison…', + "u_started": "les fichiers sont maintenant en cours de téléversement ; voir [🚀]", + "u_dupdefer": "dupliqué ; sera traité après tous les autres fichiers", + "u_actx": "cliquez sur ce texte pour éviter la perte de<br />performance lors du passage à d'autres fenêtres/onglets", + "u_fixed": "OK!  Résolu 👍", + "u_cuerr": "echec du téléversement du morceau {0} de {1};\nprobablement inoffensif, poursuite\n\nfichier : {2}", + "u_cuerr2": "le serveur a rejeté le téléversement (morceau {0} de {1});\nréessaiera plus tard\n\nfichier : {2}\n\nerreur ", + "u_ehstmp": "réessaiera ; voir en bas à droite", + "u_ehsfin": "le serveur a rejeté la demande de finalisation du téléversement ; nouvelle tentative…", + "u_ehssrch": "le serveur a rejeté la demande d'effectuer une recherche ; nouvelle tentative…", + "u_ehsinit": "le serveur a rejeté la demande d'initier le téléversement ; nouvelle tentative…", + "u_eneths": "erreur réseau lors de l'exécution de l'initialisation du téléversement ; nouvelle tentative…", + "u_enethd": "erreur réseau lors du test de l'existence de la cible ; nouvelle tentative…", + "u_cbusy": "attente que le serveur nous fasse à nouveau confiance après un problème réseau…", + "u_ehsdf": "le serveur est à court d'espace disque !\n\nil va continuer de réessayer, au cas où quelqu'un\nlibérerait suffisamment d'espace pour continuer", + "u_emtleak1": "il semble que votre navigateur web ait une fuite de mémoire ;\nveuillez", + "u_emtleak2": ' <a href="{0}">passer à https (recommandé)</a> ou ', + "u_emtleak3": ' ', + "u_emtleakc": 'essayez la solution suivante:\n<ul><li>appuyez sur <code>F5</code> pour rafraîchir la page</li><li>ensuite désactivez le bouton  <code>mt</code>  dans les  <code>⚙️ paramètres</code></li><li>et réessayez ce téléversement</li></ul>Les téléversements seront un peu plus lents, mais tant pis.\nDésolé pour le dérangement !\n\nPS : chrome v107 <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">a un correctif</a> pour cela', + "u_emtleakf": 'essayez la solution suivante:\n<ul><li>appuyez sur <code>F5</code> pour rafraîchir la page</li><li>ensuite activez <code>🥔</code> (pomme de terre) dans l\'interface de téléversement</li><li>et réessayez ce téléversement</li></ul>\nPS : firefox <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">aura probablement un correctif</a> à un moment donné', + "u_s404": "pas trouvé sur le serveur", + "u_expl": "expliquer", + "u_maxconn": "la plupart des navigateur limite ceci à 6, mais firefox vous permet de l'augmenter avec <code>connections-per-server</code> dans <code>about:config</code>", + "u_tu": '<p class="warn">WARNING: turbo enclenché, <span> le client peut ne pas détecter et reprendre les téléversements incomplets ; voir l\'info-bulle du bouton turbo</span></p>', + "u_ts": '<p class="warn">WARNING: turbo enclenché, <span> les résultats de recherche peuvent être incorrects ; voir l\'info-bulle du bouton turbo</span></p>', + "u_turbo_c": "turbo est désactivé dans la configuration du serveur", + "u_turbo_g": "désactivation de turbo car vous n'avez pas de\nprivilèges de listing de répertoires dans ce volume", + "u_life_cfg": 'suppression automatique après <input id="lifem" p="60" /> min (ou <input id="lifeh" p="3600" /> heures)', + "u_life_est": 'le téléversement sera supprimé <span id="lifew" tt="local time">---</span>', + "u_life_max": 'ce dossier impose une\ndurée de vie maximale de {0}', + "u_unp_ok": 'unpost est autorisé pour {0}', + "u_unp_ng": 'unpost ne sera PAS autorisé', + "ue_ro": 'votre accès à ce dossier est en lecture seule\n\n', + "ue_nl": 'vous n\'êtes actuellement pas connecté', + "ue_la": 'vous êtes actuellement connecté en tant que "{0}"', + "ue_sr": 'vous êtes actuellement en mode recherche de fichiers\n\nchangez en mode téléversement en cliquant sur la loupe 🔎 (à côté du grand bouton RECHERCHER), et essayez de téléverser à nouveau\n\ndésolé', + "ue_ta": 'essayez de téléverser à nouveau, cela devrait fonctionner maintenant', + "ue_ab": "ce fichier a déjà été téléversé dans un autre dossier, et ce téléversement doit être terminé avant que le fichier puisse être téléversé ailleurs.\n\nVous pouvez annuler et oublier le téléversement initial en utilisant le bouton 🧯 en haut à gauche.", + "ur_1uo": "OK: Fichier téléversé avec succès", + "ur_auo": "OK: Tous les {0} fichiers téléversés avec succès", + "ur_1so": "OK: Fichier trouvé sur le serveur", + "ur_aso": "OK: Tous les {0} fichiers trouvés sur le serveur", + "ur_1un": "Échec du téléversement, désolé", + "ur_aun": "Tous les {0} téléversements ont échoué, désolé", + "ur_1sn": "Fichier NON trouvé sur le serveur", + "ur_asn": "Les {0} fichiers n'ont PAS ÉTÉ trouvés sur le serveur", + "ur_um": "Terminé;\n{0} téléversements OK,\n{1} téléversements échoués, désolé", + "ur_sm": "Terminé;\n{0} fichiers trouvés sur le serveur,\n{1} fichiers NON trouvés sur le serveur", + + "lang_set": "rafraîchir pour que les changements prennent effet ?", + }, "grc": { "tt": "Ελληνικά", @@ -8816,7 +9445,7 @@ var Ls = { }, }; -var LANGS = ["eng", "nor", "nno", "chi", "cze", "deu", "fin", "grc", "ita", "nld", "rus", "spa", "ukr"]; +var LANGS = ["eng", "nor", "nno", "chi", "cze", "deu", "fin", "fra", "grc", "ita", "nld", "rus", "spa", "ukr"]; if (window.langmod) langmod(); From 074e106e24afe5cf2cac100d82a78c872446de9a Mon Sep 17 00:00:00 2001 From: icxes <dev.null@need.moe> Date: Sun, 10 Aug 2025 12:17:00 +0300 Subject: [PATCH 130/174] fix PRTY_CONFIG not reading global flags from the config --- copyparty/__main__.py | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index e87fe487..8c630115 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1767,6 +1767,10 @@ def main(argv: Optional[list[str]] = None) -> None: ensure_webdeps() + if CFG_DEF: + supp = args_from_cfg(CFG_DEF[0]) + argv.extend(supp) + for k, v in zip(argv[1:], argv[2:]): if k == "-c" and os.path.isfile(v): supp = args_from_cfg(v) From 7aa21483c5789681a4796def4e9ba7f461035358 Mon Sep 17 00:00:00 2001 From: Andrew Lee <andrew@alee14.me> Date: Sun, 10 Aug 2025 06:06:23 -0400 Subject: [PATCH 131/174] French translation part 2: splash.js #553 --- copyparty/web/browser.js | 2 +- copyparty/web/splash.js | 53 +++++++++++++++++++++++++++++++++++----- 2 files changed, 48 insertions(+), 7 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index f6b1e3eb..9739e55a 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -3893,7 +3893,7 @@ var Ls = { "gou": 'dossier parent">haut', "gon": 'dossier suivant">suivant', "logout": "Déconnexion ", - "access": "accès", + "access": " accès", "ot_close": "fermer le sous-menu", "ot_search": "chercher des fichiers par leurs attributs, chemin / nom, tag musicaux, ou nimporte quelle combinaison de ces options$N$N<code>foo bar</code> = doit contenir à la fois «foo» et «bar»,$N<code>foo -bar</code> = doit contenir «foo» mais pas «bar»,$N<code>^yana .opus$</code> = commence par «yana» et est un fichier «opus»$N<code>"try unite"</code> = contient exactement «try unite»$N$Nle format de date est iso-8601, comme$N<code>2009-12-31</code> ou <code>2020-09-12 23:30:00</code>", "ot_unpost": "unpost: supprimer vos téléchargements récents, ou annuler ceux en cours", diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 2475cb97..d6c3a97f 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -98,33 +98,33 @@ var Ls = { "a1": "obnovit", "b1": "ahoj cizinče   <small>(nejsi přihlášen)</small>", "c1": "odhlásit se", - "d1": "vypsat zásobníku", // TLNote: "d2" is the tooltip for this button + "d1": "vypsat zásobníku", "d2": "zobrazit stav všech aktivních vláken", "e1": "znovu načíst konfiguraci", "e2": "znovu načíst konfigurační soubory (accounts/volumes/volflags),$Na prohledat všechny e2ds úložiště$N$Npoznámka: všechny změny globálních nastavení$Nvyžadují úplné restartování, aby se projevily", "f1": "můžeš procházet:", "g1": "můžeš nahrávat do:", "cc1": "další věci:", - "h1": "zakázat k304", // TLNote: "j1" explains what k304 is + "h1": "zakázat k304", "i1": "povolit k304", "j1": "povolení k304 odpojí vašeho klienta při každém HTTP 304, což může zabránit některým chybovým proxy serverům, aby se zasekly (náhle nenačítaly stránky), <em>ale</em> také to obecně zpomalí věci", "k1": "resetovat nastavení klienta", "l1": "přihlaste se pro více:", - "m1": "vítej zpět,", // TLNote: "welcome back, USERNAME" + "m1": "vítej zpět,", "n1": "404 nenalezeno  ┐( ´ -`)┌", "o1": 'nebo možná nemáš přístup -- zkus heslo nebo <a href="' + SR + '/?h">jdi domů</a>', "p1": "403 zakázáno  ~┻━┻", "q1": 'použij heslo nebo <a href="' + SR + '/?h">jdi domů</a>', "r1": "jdi domů", ".s1": "znovu prohledat", - "t1": "akce", // TLNote: this is the header above the "rescan" buttons + "t1": "akce", "u2": "čas od posledního zápisu na server$N( upload / rename / ... )$N$N17d = 17 dní$N1h23 = 1 hodina 23 minut$N4m56 = 4 minuty 56 sekund", "v1": "připojit", "v2": "použít tento server jako místní HDD", "w1": "přepnout na https", "x1": "změnit heslo", - "y1": "upravit sdílení", // TLNote: shows the list of folders that the user has decided to share - "z1": "odblokovat toto sdílení:", // TLNote: the password prompt to see a hidden share + "y1": "upravit sdílení", + "z1": "odblokovat toto sdílení:", "ta1": "nejprve vyplňte své nové heslo", "ta2": "zopakujte pro potvrzení nového hesla:", "ta3": "nalezen překlep; zkuste to prosím znovu", @@ -218,6 +218,47 @@ var Ls = { "af1": "näytä viimeaikaiset lataukset", "ag1": "näytä tunnetut IdP-käyttäjät", }, + "fra": { + "a1": "rafraîchir", + "b1": "salut étranger   <small>(vous n'êtes pas connecté.)</small>", + "c1": "déconnexion", + "d1": "vidange de la pile", + "d2": "affiche l'état de tous les threads actifs", + "e1": "recharger la configuration", + "e2": "recharger le fichier configuration (comptes/volumes/indicateurs de volume),$Net rescanner tous les e2ds volumes$N$Nnote: n'importe quel changement aux paramètres globaux$Nnécessite un redémarrage complet pour prendre effet", + "f1": "vous pouvez naviguer:", + "g1": "vous pouvez télécharger sur:", + "cc1": "autres choses:", + "h1": "désactiver k304", + "i1": "activer k304", + "j1": "activer k304 vouloir déconnecter votre client sur chaque HTTP 304, ce qui peut éviter certains proxys défectueux d'avoir bloqué (les pages ne se chargent soudainement plus), <em>mais</em> cela ralentira également les choses en général", + "k1": "réinitialiser client paramètres", + "l1": "connectez-vous pour en savoir plus:", + "m1": "heureux vous revoir,", + "n1": "404 introuvable  ┐( ´ -`)┌", + "o1": 'ou peut-être que vous n\'y avez pas accès -- essayer un mot de passe ou <a href="' + SR + '/?h">aller à la page d\'accueil</a>', + "p1": "403 interdita  ~┻━┻", + "q1": 'utiliser un mot de passe or <a href="' + SR + '/?h">aller à la page d\'accueil</a>', + "r1": "aller à la page d\'accueil", + ".s1": "rescan", + "t1": "action", + "u2": "temps écoulé depuis la dernière écriture sur le serveur$N( télécharger / renommer / ... )$N$N17j = 17 jours$N1h23 = 1 heur 23 minutes$N4m56 = 4 minutes 56 secondes", + "v1": "connecter", + "v2": "utilisez ce serveur en tant que disque dur local", + "w1": "passer à https", + "x1": "changer mot de passe", + "y1": "modifier les partages", + "z1": "déverrouiller ce partage:", + "ta1": "entrez d'abord votre nouveau mot de passe", + "ta2": "répétez pour confirmer le nouveau mot de passe:", + "ta3": "une faute de frappe a été détectée ; veuillez réessayer.", + "aa1": "fichiers entrants:", + "ab1": "désactiver no304", + "ac1": "activer no304", + "ad1": "l'activation de no304 désactivera toute mise en cache; essayez ceci si k304 n'était pas suffisant. Cela va générer un trafic réseau considérable!", + "ae1": "téléchargements actifs:", + "af1": "afficher les derniers téléchargements", + }, "grc": { "a1": "ανανέωση", "b1": "γεια σου ξένε!   <small>(δεν είσαι συνδεδεμένος)</small>", From 3c78c6a880807d78efa754aeb7a559d99d3b6b57 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Sun, 10 Aug 2025 10:03:42 +0000 Subject: [PATCH 132/174] custom mdns domain, closes #549 --- copyparty/__main__.py | 1 + copyparty/mdns.py | 3 ++- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 8c630115..5c2771fe 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1201,6 +1201,7 @@ def add_zc_mdns(ap): ap2.add_argument("--zm-lh", metavar="PATH", type=u, default="", help="link a specific folder for http shares") ap2.add_argument("--zm-lf", metavar="PATH", type=u, default="", help="link a specific folder for ftp shares") ap2.add_argument("--zm-ls", metavar="PATH", type=u, default="", help="link a specific folder for smb shares") + ap2.add_argument("--zm-fqdn", metavar="FQDN", type=u, default="--name.local", help="the domain to announce; NOTE: using anything other than .local is nonstandard and could cause problems") ap2.add_argument("--zm-mnic", action="store_true", help="merge NICs which share subnets; assume that same subnet means same network") ap2.add_argument("--zm-msub", action="store_true", help="merge subnets on each NIC -- always enabled for ipv6 -- reduces network load, but gnome-gvfs clients may stop working, and clients cannot be in subnets that the server is not") ap2.add_argument("--zm-noneg", action="store_true", help="disable NSEC replies -- try this if some clients don't see copyparty") diff --git a/copyparty/mdns.py b/copyparty/mdns.py index a9d36fe7..0dde707d 100644 --- a/copyparty/mdns.py +++ b/copyparty/mdns.py @@ -76,7 +76,8 @@ class MDNS(MCast): if not self.args.zm_nwa_1: set_avahi_379() - zs = self.args.name + ".local." + zs = self.args.zm_fqdn or (self.args.name + ".local") + zs = zs.replace("--name", self.args.name).rstrip(".") + "." zs = zs.encode("ascii", "replace").decode("ascii", "replace") self.hn = "-".join(x for x in zs.split("?") if x) or ( "vault-{}".format(random.randint(1, 255)) From 8ba98877ee15af62bac082e28248ef0feb8c6ee6 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Sun, 10 Aug 2025 11:23:22 +0000 Subject: [PATCH 133/174] patch pyftpdlib, fixes #539 upgrading pyftpdlib brings only pain and no benefits so grafting a patch for this instead --- scripts/make-sfx.sh | 1 + scripts/patches/pyftpdlib-win313.patch | 41 ++++++++++++++++++++++++++ 2 files changed, 42 insertions(+) create mode 100644 scripts/patches/pyftpdlib-win313.patch diff --git a/scripts/make-sfx.sh b/scripts/make-sfx.sh index a1ee1a06..f752f8a8 100755 --- a/scripts/make-sfx.sh +++ b/scripts/make-sfx.sh @@ -217,6 +217,7 @@ necho() { tar -zxf $f mv pyftpdlib-*/pyftpdlib . rm -rf pyftpdlib-* pyftpdlib/test + patch -p1 <../scripts/patches/pyftpdlib-win313.patch for f in pyftpdlib/_async{hat,ore}.py; do [ -e "$f" ] || continue; iawk 'NR<4||NR>27||!/^#/;NR==4{print"# license: https://opensource.org/licenses/ISC\n"}' $f diff --git a/scripts/patches/pyftpdlib-win313.patch b/scripts/patches/pyftpdlib-win313.patch new file mode 100644 index 00000000..62601108 --- /dev/null +++ b/scripts/patches/pyftpdlib-win313.patch @@ -0,0 +1,41 @@ +Date: Tue, 22 Oct 2024 12:47:30 +0200 +Subject: Workaround for isabs() on Windows + Python 3.13 (#652) + +Starting from Python 3.13, `os.path.isabs("/foo")` on Windows return `False` + +diff --git a/pyftpdlib/filesystems.py b/pyftpdlib/filesystems.py +index 9b9326bf..320ffe40 100644 +--- a/pyftpdlib/filesystems.py ++++ b/pyftpdlib/filesystems.py +@@ -132,6 +132,16 @@ def cwd(self, path): + + # --- Pathname / conversion utilities + ++ @staticmethod ++ def _isabs(path, _windows=os.name == "nt"): ++ # Windows + Python 3.13: isabs() changed so that a path ++ # starting with "/" is no longer considered absolute. ++ # https://github.com/python/cpython/issues/44626 ++ # https://github.com/python/cpython/pull/113829/ ++ if _windows and path.startswith("/"): ++ return True ++ return os.path.isabs(path) ++ + def ftpnorm(self, ftppath): + """Normalize a "virtual" ftp pathname (typically the raw string + coming from client) depending on the current working directory. +@@ -146,3 +156,3 @@ + assert isinstance(ftppath, unicode), ftppath +- if os.path.isabs(ftppath): ++ if self._isabs(ftppath): + p = os.path.normpath(ftppath) +@@ -162,3 +172,3 @@ + # This is for extra protection, maybe not really necessary. +- if not os.path.isabs(p): ++ if not self._isabs(p): + p = u("/") +@@ -201,3 +211,3 @@ + assert isinstance(fspath, unicode), fspath +- if os.path.isabs(fspath): ++ if self._isabs(fspath): + p = os.path.normpath(fspath) From 347cf6a5461d281ef90935d1e9023d95b909230c Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Sun, 10 Aug 2025 11:28:22 +0000 Subject: [PATCH 134/174] fix dropdown color --- copyparty/web/browser.css | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/web/browser.css b/copyparty/web/browser.css index b5506f39..298de0ea 100644 --- a/copyparty/web/browser.css +++ b/copyparty/web/browser.css @@ -1377,7 +1377,7 @@ html.y #ops svg circle { .opview select, .opview input[type=text] { color: var(--fg); - background: var(--txt-bg); + background: var(--bg-u5); border: none; box-shadow: 0 0 2px var(--txt-sh); border-bottom: 1px solid #999; From e5e822951d66f5a9b60e3f4a22ffa6b1a46f6b43 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Sun, 10 Aug 2025 11:31:12 +0000 Subject: [PATCH 135/174] fix filter case-sensitivity --- copyparty/httpcli.py | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 402d623f..b981172d 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -5481,13 +5481,13 @@ class HttpCli(object): q = "select sz, rd, fn, at from up where ip=? and at>? order by at desc" for sz, rd, fn, at in cur.execute(q, (self.ip, lim)): vp = "/" + "/".join(x for x in [vol.vpath, rd, fn] if x) - if nfi == 0 or (nfi == 1 and vfi in vp): + if nfi == 0 or (nfi == 1 and vfi in vp.lower()): pass elif nfi == 2: - if not vp.startswith(vfi): + if not vp.lower().startswith(vfi): continue elif nfi == 3: - if not vp.endswith(vfi): + if not vp.lower().endswith(vfi): continue else: continue @@ -5607,13 +5607,13 @@ class HttpCli(object): q = "select sz, rd, fn, ip, at from up where at>0 order by at desc" for sz, rd, fn, ip, at in cur.execute(q): vp = "/" + "/".join(x for x in [vol.vpath, rd, fn] if x) - if nfi == 0 or (nfi == 1 and vfi in vp): + if nfi == 0 or (nfi == 1 and vfi in vp.lower()): pass elif nfi == 2: - if not vp.startswith(vfi): + if not vp.lower().startswith(vfi): continue elif nfi == 3: - if not vp.endswith(vfi): + if not vp.lower().endswith(vfi): continue else: continue From 03acd65e9666c456703a3c73b809556b93577291 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Sun, 10 Aug 2025 11:45:20 +0000 Subject: [PATCH 136/174] avoid ios bug (keystore spam) --- copyparty/web/splash.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index d6c3a97f..08551999 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -625,7 +625,7 @@ catch (ex) { } tt.init(); var o = QS('input[name="uname"]') || QS('input[name="cppwd"]'); -if (!ebi('c') && o.offsetTop + o.offsetHeight < window.innerHeight) +if (!MOBILE && !ebi('c') && o.offsetTop + o.offsetHeight < window.innerHeight) o.focus(); o = ebi('u'); From 3560eeb10eed1ae6641f6855f8cd1f5e77f8746b Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Sun, 10 Aug 2025 13:55:29 +0200 Subject: [PATCH 137/174] better with sessions --- copyparty/httpcli.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index b981172d..29e4c610 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -4971,7 +4971,7 @@ class HttpCli(object): rip = host vp = (self.uparam["hc"] or "").lstrip("/") - pw = self.pw or "hunter2" + pw = self.ouparam.get("pw") or "hunter2" if pw in self.asrv.sesa: pw = "hunter2" From a01870b744dc12292808f2002ad86b1e7f1268f8 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Sun, 10 Aug 2025 13:55:52 +0200 Subject: [PATCH 138/174] avoid macos bug (finder hangs on connect) --- copyparty/httpcli.py | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 29e4c610..72bff45e 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -1599,6 +1599,10 @@ class HttpCli(object): "quota-available-bytes": str(bfree), "quota-used-bytes": str(btot - bfree), } + if "quotaused" in props: # macos finder crazytalk + df["quotaused"] = df["quota-used-bytes"] + if "quota" in props: + df["quota"] = df["quota-available-bytes"] # idk, makes it happy else: df = {} else: From b87f8f1b01dc9e53005bbe9ec004c66fd25b34c8 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Sun, 10 Aug 2025 12:10:54 +0000 Subject: [PATCH 139/174] french improvements by @Equinoxs #553 --- copyparty/web/splash.js | 34 +++++++++++++++++----------------- 1 file changed, 17 insertions(+), 17 deletions(-) diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 08551999..8cbca1e6 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -225,38 +225,38 @@ var Ls = { "d1": "vidange de la pile", "d2": "affiche l'état de tous les threads actifs", "e1": "recharger la configuration", - "e2": "recharger le fichier configuration (comptes/volumes/indicateurs de volume),$Net rescanner tous les e2ds volumes$N$Nnote: n'importe quel changement aux paramètres globaux$Nnécessite un redémarrage complet pour prendre effet", - "f1": "vous pouvez naviguer:", - "g1": "vous pouvez télécharger sur:", - "cc1": "autres choses:", + "e2": "recharger le fichier de configuration (comptes/volumes/indicateurs de volume),$Net rescanner tous les volumes e2ds$N$Nnote : n'importe quel changement aux paramètres globaux$Nnécessite un redémarrage complet pour prendre effet", + "f1": "vous pouvez naviguer :", + "g1": "vous pouvez télécharger sur :", + "cc1": "autres choses :", "h1": "désactiver k304", "i1": "activer k304", - "j1": "activer k304 vouloir déconnecter votre client sur chaque HTTP 304, ce qui peut éviter certains proxys défectueux d'avoir bloqué (les pages ne se chargent soudainement plus), <em>mais</em> cela ralentira également les choses en général", - "k1": "réinitialiser client paramètres", - "l1": "connectez-vous pour en savoir plus:", - "m1": "heureux vous revoir,", + "j1": "activer k304 va déconnecter votre client sur chaque HTTP 304, ce qui peut éviter à certains proxies défectueux de rester bloqués (les pages ne se chargent soudainement plus), <em>mais</em> cela ralentira également les choses en général", + "k1": "réinitialiser les paramètres du client", + "l1": "connectez-vous pour en savoir plus :", + "m1": "heureux de vous revoir,", "n1": "404 introuvable  ┐( ´ -`)┌", "o1": 'ou peut-être que vous n\'y avez pas accès -- essayer un mot de passe ou <a href="' + SR + '/?h">aller à la page d\'accueil</a>', - "p1": "403 interdita  ~┻━┻", - "q1": 'utiliser un mot de passe or <a href="' + SR + '/?h">aller à la page d\'accueil</a>', + "p1": "403 interdit  ~┻━┻", + "q1": 'utiliser un mot de passe ou <a href="' + SR + '/?h">aller à la page d\'accueil</a>', "r1": "aller à la page d\'accueil", - ".s1": "rescan", + ".s1": "rescanner", "t1": "action", - "u2": "temps écoulé depuis la dernière écriture sur le serveur$N( télécharger / renommer / ... )$N$N17j = 17 jours$N1h23 = 1 heur 23 minutes$N4m56 = 4 minutes 56 secondes", + "u2": "temps écoulé depuis la dernière écriture sur le serveur$N(téléchargement/renommage/...)$N$N17j = 17 jours$N1h23 = 1 heure 23 minutes$N4m56 = 4 minutes 56 secondes", "v1": "connecter", "v2": "utilisez ce serveur en tant que disque dur local", "w1": "passer à https", "x1": "changer mot de passe", "y1": "modifier les partages", - "z1": "déverrouiller ce partage:", + "z1": "déverrouiller ce partage :", "ta1": "entrez d'abord votre nouveau mot de passe", - "ta2": "répétez pour confirmer le nouveau mot de passe:", + "ta2": "répétez pour confirmer le nouveau mot de passe :", "ta3": "une faute de frappe a été détectée ; veuillez réessayer.", - "aa1": "fichiers entrants:", + "aa1": "fichiers entrants :", "ab1": "désactiver no304", "ac1": "activer no304", - "ad1": "l'activation de no304 désactivera toute mise en cache; essayez ceci si k304 n'était pas suffisant. Cela va générer un trafic réseau considérable!", - "ae1": "téléchargements actifs:", + "ad1": "l'activation de no304 désactivera toute mise en cache ; essayez ceci si k304 n'était pas suffisant. Cela va générer un trafic réseau considérable !", + "ae1": "téléchargements actifs :", "af1": "afficher les derniers téléchargements", }, "grc": { From c2cee222bdc3ae836ca1adb52f2af6381e8a5a1f Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Sun, 10 Aug 2025 12:26:12 +0000 Subject: [PATCH 140/174] v1.19.1 --- copyparty/__main__.py | 6 +++--- copyparty/__version__.py | 4 ++-- docs/changelog.md | 46 ++++++++++++++++++++++++++++++++++++++++ scripts/rls.sh | 2 ++ scripts/tl.js | 1 + 5 files changed, 54 insertions(+), 5 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 5c2771fe..2468afe6 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1391,7 +1391,7 @@ def add_shutdown(ap): def add_logging(ap): ap2 = ap.add_argument_group("logging options") ap2.add_argument("-q", action="store_true", help="quiet; disable most STDOUT messages") - ap2.add_argument("-lo", metavar="PATH", type=u, default="", help="logfile, example: \033[32mcpp-%%Y-%%m%%d-%%H%%M%%S.txt.xz\033[0m (NB: some errors may appear on STDOUT only)") + ap2.add_argument("-lo", metavar="PATH", type=u, default="", help="logfile; use .txt for plaintext or .xz for compressed. Example: \033[32mcpp-%%Y-%%m%%d-%%H%%M%%S.txt.xz\033[0m (NB: some errors may appear on STDOUT only)") ap2.add_argument("--no-ansi", action="store_true", default=not VT100, help="disable colors; same as environment-variable NO_COLOR") ap2.add_argument("--ansi", action="store_true", help="force colors; overrides environment-variable NO_COLOR") ap2.add_argument("--no-logflush", action="store_true", help="don't flush the logfile after each write; tiny bit faster") @@ -1494,8 +1494,8 @@ def add_db_general(ap, hcores): ap2.add_argument("-e2vp", action="store_true", help="on hash mismatch: panic and quit copyparty") ap2.add_argument("--hist", metavar="PATH", type=u, default="", help="where to store volume data (db, thumbs); default is a folder named \".hist\" inside each volume (volflag=hist)") ap2.add_argument("--dbpath", metavar="PATH", type=u, default="", help="override where the volume databases are to be placed; default is the same as \033[33m--hist\033[0m (volflag=dbpath)") - ap2.add_argument("--no-hash", metavar="PTN", type=u, default="", help="regex: disable hashing of matching absolute-filesystem-paths during e2ds folder scans (volflag=nohash)") - ap2.add_argument("--no-idx", metavar="PTN", type=u, default=noidx, help="regex: disable indexing of matching absolute-filesystem-paths during e2ds folder scans (volflag=noidx)") + ap2.add_argument("--no-hash", metavar="PTN", type=u, default="", help="regex: disable hashing of matching absolute-filesystem-paths during e2ds folder scans (must be specified as one big regex, not multiple times) (volflag=nohash)") + ap2.add_argument("--no-idx", metavar="PTN", type=u, default=noidx, help="regex: disable indexing of matching absolute-filesystem-paths during e2ds folder scan (must be specified as one big regex, not multiple times) (volflag=noidx)") ap2.add_argument("--no-dirsz", action="store_true", help="do not show total recursive size of folders in listings, show inode size instead; slightly faster (volflag=nodirsz)") ap2.add_argument("--re-dirsz", action="store_true", help="if the directory-sizes in the UI are bonkers, use this along with \033[33m-e2dsa\033[0m to rebuild the index from scratch") ap2.add_argument("--no-dhash", action="store_true", help="disable rescan acceleration; do full database integrity check -- makes the db ~5%% smaller and bootup/rescans 3~10x slower") diff --git a/copyparty/__version__.py b/copyparty/__version__.py index 3350f413..1cea26c5 100644 --- a/copyparty/__version__.py +++ b/copyparty/__version__.py @@ -1,8 +1,8 @@ # coding: utf-8 -VERSION = (1, 19, 0) +VERSION = (1, 19, 1) CODENAME = "usernames" -BUILD_DT = (2025, 8, 7) +BUILD_DT = (2025, 8, 10) S_VERSION = ".".join(map(str, VERSION)) S_BUILD_DT = "{0:04d}-{1:02d}-{2:02d}".format(*BUILD_DT) diff --git a/docs/changelog.md b/docs/changelog.md index a3c0aaae..91fd1b84 100644 --- a/docs/changelog.md +++ b/docs/changelog.md @@ -1,3 +1,49 @@ +▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀ +# 2025-0807-2213 `v1.19.0` usernames + +## 🧪 new features + +* #511 login with username and password (not just password) can now optionally be enabled with `--usernames` 346515cc + * if you have enabled password hashing (`ah-alg: argon2` or similar) then you will need to hash your passwords again after enabling usernames, hashing them as `username:password:` +* #468 add Greek translation (thx @chamdim!) 50f46187 392abd06 +* #471 add Czech translation (thx @kubakubakuba!) c9556583 +* #515 support systemd socket acivation (thx @mati1210!) 9b9d2a92 +* #523 add QR-code to the connectpage bcc3b156 +* #513 optional EOL-conversion for texteditor 8b31ed88 +* controlpanel refresh-button now toggles automatic refresh 7ae84dea + +## 🩹 bugfixes + +* fix stuck uploads when the up2k database (`e2d`) is not enabled 4a043568 + * if more than 60'000 files were uploaded and there were several dupes of some files, they could get stuck and never upload + * upload performance is improved remarkably by enabling `e2d` so such huge uploads non-e2d had not been tested in a long time +* #467 #470 fix ui-crash when exporting links of all uploaded files to clipboard (thx @geekalaa!) 0df1901f +* #487 fix ui-crash when the location url-part is `//` 0f55a1ae +* fix viewing `.MD` files (8a0746c6) + +## 🔧 other changes + +* when a reverse-proxy is detected, force explicit configuration of `--rproxy` to obtain correct client IP 3f8cb7e8 + * a bit inconvenient, but helps prevent potentially-dangerous misconfiguration + * the necessary configuration changes are explained in the serverlog (you can't miss it) + * thanks to @person4268 for pointing out that there was room for improvements! +* failed login attempts now only log a sha512 hash of the provided password + * to see login-attempts with incorrect passwords as plaintext like before, `log-badpwd: 1` +* #502 add systemd user services and templated services (thx @icxes!) 34d98e99 +* #475 improve helptext for multivalue global-options c2ac57a2 +* #475 add [chungus.conf](https://github.com/9001/copyparty/blob/hovudstraum/docs/chungus.conf), massive extensive nonsensical demo config b664ebb0 +* try to detect proxies with incorrect caching behavior 9e980bb5 +* recent-uploads now support ie9 a57f7cc2 +* languages and themes are now dropdowns a9ee4f24 +* copyparty.exe: upgrade python to 3.13.6 a98360f2 +* introduce [copyparty-en.py](https://github.com/9001/copyparty/releases/latest/download/copyparty-en.py), english-only edition of copyparty-sfx.py to save space 33497e6b + +## 🗿 known issues + +* the `copyparty.pyz` in this release is english-only, and does not include the translations -- they got lost in transit while adjusting the buildscripts to make `copyparty-en.py` + + + ▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀ # 2025-0804-0013 `v1.18.10` idp speedboost diff --git a/scripts/rls.sh b/scripts/rls.sh index 767ca2de..d18f12f2 100755 --- a/scripts/rls.sh +++ b/scripts/rls.sh @@ -43,6 +43,8 @@ while [ "$1" ]; do shift done +./make-pyz.sh + ./make-sfx.sh re lang eng "$@" mv ../dist/copyparty-{sfx,en}.py mv ../dist/copyparty-{int,sfx}.py diff --git a/scripts/tl.js b/scripts/tl.js index 97bf061d..4d4fcaba 100644 --- a/scripts/tl.js +++ b/scripts/tl.js @@ -81,6 +81,7 @@ var tl_cpanel = { "ad1": "enabling no304 will disable all caching; try this if k304 wasn't enough. This will waste a huge amount of network traffic!", "ae1": "active downloads:", "af1": "show recent uploads", + "ag1": "view idp cache", }, }; From db2a03409c7bdbcd0752e4b97dad2139db5c77ef Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Sun, 10 Aug 2025 12:44:51 +0000 Subject: [PATCH 141/174] update pkgs to 1.19.1 --- contrib/package/arch/PKGBUILD | 4 ++-- contrib/package/makedeb-mpr/PKGBUILD | 4 ++-- contrib/package/nix/copyparty/pin.json | 6 +++--- 3 files changed, 7 insertions(+), 7 deletions(-) diff --git a/contrib/package/arch/PKGBUILD b/contrib/package/arch/PKGBUILD index e5896fd7..3c6a4abb 100644 --- a/contrib/package/arch/PKGBUILD +++ b/contrib/package/arch/PKGBUILD @@ -3,7 +3,7 @@ # NOTE: You generally shouldn't use this PKGBUILD on Arch, as it is mainly for testing purposes. Install copyparty using pacman instead. pkgname=copyparty -pkgver="1.19.0" +pkgver="1.19.1" pkgrel=1 pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" arch=("any") @@ -23,7 +23,7 @@ optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tag ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") backup=("etc/${pkgname}/copyparty.conf" ) -sha256sums=("179b027d51e4fe7ebdab2b18c07475d52c57e2ce69256292b157a8efacd82118") +sha256sums=("bbc250db23eb80bc96c27b2efa456ce1e7f49c7dfaabadb91a571f70064b6f91") build() { cd "${srcdir}/${pkgname}-${pkgver}/copyparty/web" diff --git a/contrib/package/makedeb-mpr/PKGBUILD b/contrib/package/makedeb-mpr/PKGBUILD index 645623c0..6201430a 100644 --- a/contrib/package/makedeb-mpr/PKGBUILD +++ b/contrib/package/makedeb-mpr/PKGBUILD @@ -2,7 +2,7 @@ pkgname=copyparty -pkgver=1.19.0 +pkgver=1.19.1 pkgrel=1 pkgdesc="File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++" arch=("any") @@ -20,7 +20,7 @@ optdepends=("ffmpeg: thumbnails for videos, images (slower) and audio, music tag ) source=("https://github.com/9001/${pkgname}/releases/download/v${pkgver}/${pkgname}-${pkgver}.tar.gz") backup=("/etc/${pkgname}.d/init" ) -sha256sums=("179b027d51e4fe7ebdab2b18c07475d52c57e2ce69256292b157a8efacd82118") +sha256sums=("bbc250db23eb80bc96c27b2efa456ce1e7f49c7dfaabadb91a571f70064b6f91") build() { cd "${srcdir}/${pkgname}-${pkgver}/copyparty/web" diff --git a/contrib/package/nix/copyparty/pin.json b/contrib/package/nix/copyparty/pin.json index c0d8c68a..55f1c2e1 100644 --- a/contrib/package/nix/copyparty/pin.json +++ b/contrib/package/nix/copyparty/pin.json @@ -1,5 +1,5 @@ { - "url": "https://github.com/9001/copyparty/releases/download/v1.19.0/copyparty-sfx.py", - "version": "1.19.0", - "hash": "sha256-9A+zPtkVtUuGHB/JJV3fhVtJderLUGxHqvuJQz0/1+Q=" + "url": "https://github.com/9001/copyparty/releases/download/v1.19.1/copyparty-sfx.py", + "version": "1.19.1", + "hash": "sha256-Orn0N//DD5/5rIWK9yYRcvyOnt/bKCE9CeoxkfNO76s=" } \ No newline at end of file From 0da93659a4def6c3adbba71c4887d368a7e3a1cd Mon Sep 17 00:00:00 2001 From: Bevinsky <vin@r-a-d.io> Date: Sun, 10 Aug 2025 17:19:43 +0200 Subject: [PATCH 142/174] Add missing translated string in up2k. --- copyparty/web/up2k.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/web/up2k.js b/copyparty/web/up2k.js index 2743ea36..6ae7046f 100644 --- a/copyparty/web/up2k.js +++ b/copyparty/web/up2k.js @@ -2831,7 +2831,7 @@ function up2k_init(subtle) { if (!t.t_uploading) t.t_uploading = Date.now(); - pvis.seth(t.n, 1, "🚀 send"); + pvis.seth(t.n, 1, "🚀 " + L.ul_send); var chunksize = get_chunksize(t.size), car = pcar * chunksize, From 9435e6b2e2d3fb3a32fcba7abc3b755a840ec5a6 Mon Sep 17 00:00:00 2001 From: AppleTheGolden <scotsbox@protonmail.com> Date: Sun, 10 Aug 2025 22:30:37 +0200 Subject: [PATCH 143/174] EPUB Thumbnailing support (#561) * EPUB Thumbnailing support --------- Signed-off-by: ed <s@ocv.me> Co-authored-by: ed <s@ocv.me> --- copyparty/__main__.py | 6 ++-- copyparty/dxml.py | 3 ++ copyparty/mtag.py | 73 ++++++++++++++++++++++++++++++++++++++++++- 3 files changed, 78 insertions(+), 4 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 2468afe6..be7c9979 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1443,13 +1443,13 @@ def add_thumbnail(ap): # https://pillow.readthedocs.io/en/stable/handbook/image-file-formats.html # https://github.com/libvips/libvips # ffmpeg -hide_banner -demuxers | awk '/^ D /{print$2}' | while IFS= read -r x; do ffmpeg -hide_banner -h demuxer=$x; done | grep -E '^Demuxer |extensions:' - ap2.add_argument("--th-r-pil", metavar="T,T", type=u, default="avif,avifs,blp,bmp,cbz,dcx,dds,dib,emf,eps,fits,flc,fli,fpx,gif,heic,heics,heif,heifs,icns,ico,im,j2p,j2k,jp2,jpeg,jpg,jpx,pbm,pcx,pgm,png,pnm,ppm,psd,qoi,sgi,spi,tga,tif,tiff,webp,wmf,xbm,xpm", help="image formats to decode using pillow") + ap2.add_argument("--th-r-pil", metavar="T,T", type=u, default="avif,avifs,blp,bmp,cbz,dcx,dds,dib,emf,eps,epub,fits,flc,fli,fpx,gif,heic,heics,heif,heifs,icns,ico,im,j2p,j2k,jp2,jpeg,jpg,jpx,pbm,pcx,pgm,png,pnm,ppm,psd,qoi,sgi,spi,tga,tif,tiff,webp,wmf,xbm,xpm", help="image formats to decode using pillow") ap2.add_argument("--th-r-vips", metavar="T,T", type=u, default="avif,exr,fit,fits,fts,gif,hdr,heic,jp2,jpeg,jpg,jpx,jxl,nii,pfm,pgm,png,ppm,svg,tif,tiff,webp", help="image formats to decode using pyvips") - ap2.add_argument("--th-r-ffi", metavar="T,T", type=u, default="apng,avif,avifs,bmp,cbz,dds,dib,fit,fits,fts,gif,hdr,heic,heics,heif,heifs,icns,ico,jp2,jpeg,jpg,jpx,jxl,pbm,pcx,pfm,pgm,png,pnm,ppm,psd,qoi,sgi,tga,tif,tiff,webp,xbm,xpm", help="image formats to decode using ffmpeg") + ap2.add_argument("--th-r-ffi", metavar="T,T", type=u, default="apng,avif,avifs,bmp,cbz,dds,dib,epub,fit,fits,fts,gif,hdr,heic,heics,heif,heifs,icns,ico,jp2,jpeg,jpg,jpx,jxl,pbm,pcx,pfm,pgm,png,pnm,ppm,psd,qoi,sgi,tga,tif,tiff,webp,xbm,xpm", help="image formats to decode using ffmpeg") ap2.add_argument("--th-r-ffv", metavar="T,T", type=u, default="3gp,asf,av1,avc,avi,flv,h264,h265,hevc,m4v,mjpeg,mjpg,mkv,mov,mp4,mpeg,mpeg2,mpegts,mpg,mpg2,mts,nut,ogm,ogv,rm,ts,vob,webm,wmv", help="video formats to decode using ffmpeg") ap2.add_argument("--th-r-ffa", metavar="T,T", type=u, default="aac,ac3,aif,aiff,alac,alaw,amr,apac,ape,au,bonk,dfpwm,dts,flac,gsm,ilbc,it,itgz,itxz,itz,m4a,mdgz,mdxz,mdz,mo3,mod,mp2,mp3,mpc,mptm,mt2,mulaw,oga,ogg,okt,opus,ra,s3m,s3gz,s3xz,s3z,tak,tta,ulaw,wav,wma,wv,xm,xmgz,xmxz,xmz,xpk", help="audio formats to decode using ffmpeg") ap2.add_argument("--th-spec-cnv", metavar="T", type=u, default="it,itgz,itxz,itz,mdgz,mdxz,mdz,mo3,mod,s3m,s3gz,s3xz,s3z,xm,xmgz,xmxz,xmz,xpk", help="audio formats which provoke https://trac.ffmpeg.org/ticket/10797 (huge ram usage for s3xmodit spectrograms)") - ap2.add_argument("--au-unpk", metavar="E=F.C", type=u, default="mdz=mod.zip, mdgz=mod.gz, mdxz=mod.xz, s3z=s3m.zip, s3gz=s3m.gz, s3xz=s3m.xz, xmz=xm.zip, xmgz=xm.gz, xmxz=xm.xz, itz=it.zip, itgz=it.gz, itxz=it.xz, cbz=jpg.cbz", help="audio/image formats to decompress before passing to ffmpeg") + ap2.add_argument("--au-unpk", metavar="E=F.C", type=u, default="mdz=mod.zip, mdgz=mod.gz, mdxz=mod.xz, s3z=s3m.zip, s3gz=s3m.gz, s3xz=s3m.xz, xmz=xm.zip, xmgz=xm.gz, xmxz=xm.xz, itz=it.zip, itgz=it.gz, itxz=it.xz, cbz=jpg.cbz, epub=jpg.epub", help="audio/image formats to decompress before passing to ffmpeg") def add_transcoding(ap): diff --git a/copyparty/dxml.py b/copyparty/dxml.py index b49f060b..6271da8e 100644 --- a/copyparty/dxml.py +++ b/copyparty/dxml.py @@ -65,6 +65,9 @@ DXMLParser = _DXMLParser def parse_xml(txt: str) -> ET.Element: + """ + Parse XML into an xml.etree.ElementTree.Element while defusing some unsafe parts. + """ parser = DXMLParser() parser.feed(txt) return parser.close() # type: ignore diff --git a/copyparty/mtag.py b/copyparty/mtag.py index e9428b79..660f8180 100644 --- a/copyparty/mtag.py +++ b/copyparty/mtag.py @@ -29,7 +29,7 @@ from .util import ( ) if True: # pylint: disable=using-constant-test - from typing import Any, Optional, Union + from typing import IO, Any, Optional, Union from .util import NamedLogger, RootLogger @@ -176,6 +176,9 @@ def au_unpk( raise Exception("no images inside cbz") fi = zf.open(using) + elif pk == "epub": + fi = get_cover_from_epub(log, abspath) + else: raise Exception("unknown compression %s" % (pk,)) @@ -365,6 +368,74 @@ def parse_ffprobe(txt: str) -> tuple[dict[str, tuple[int, Any]], dict[str, list[ return zd, md +def get_cover_from_epub(log: "NamedLogger", abspath: str) -> IO[bytes] | None: + import zipfile + + from .dxml import parse_xml + + try: + from urlparse import urljoin # Python2 + except ImportError: + from urllib.parse import urljoin # Python3 + + with zipfile.ZipFile(abspath, "r") as z: + # First open the container file to find the package document (.opf file) + try: + container_root = parse_xml(z.read("META-INF/container.xml").decode()) + except KeyError: + log("epub: no container file found in %s" % (abspath,)) + return None + + # https://www.w3.org/TR/epub-33/#sec-container.xml-rootfile-elem + container_ns = {"": "urn:oasis:names:tc:opendocument:xmlns:container"} + # One file could contain multiple package documents, default to the first one + rootfile_path = container_root.find("./rootfiles/rootfile", container_ns).get( + "full-path" + ) + + # Then open the first package document to find the path of the cover image + try: + package_root = parse_xml(z.read(rootfile_path).decode()) + except KeyError: + log("epub: no package document found in %s" % (abspath,)) + return None + + # https://www.w3.org/TR/epub-33/#sec-package-doc + package_ns = {"": "http://www.idpf.org/2007/opf"} + # https://www.w3.org/TR/epub-33/#sec-cover-image + coverimage_path_node = package_root.find( + "./manifest/item[@properties='cover-image']", package_ns + ) + if coverimage_path_node is not None: + coverimage_path = coverimage_path_node.get("href") + else: + # This might be an EPUB2 file, try the legacy way of specifying covers + coverimage_path = _get_cover_from_epub2(log, package_root, package_ns) + + # This url is either absolute (in the .epub) or relative to the package document + adjusted_cover_path = urljoin(rootfile_path, coverimage_path) + + return z.open(adjusted_cover_path) + + +def _get_cover_from_epub2(log: "NamedLogger", package_root, package_ns) -> str | None: + # <meta name="cover" content="id-to-cover-image"> in <metadata>, then + # <item> in <manifest> + cover_id = package_root.find("./metadata/meta[@name='cover']", package_ns).get( + "content" + ) + + if not cover_id: + return None + + for node in package_root.iterfind("./manifest/item", package_ns): + if node.get("id") == cover_id: + cover_path = node.get("href") + return cover_path + + return None + + class MTag(object): def __init__(self, log_func: "RootLogger", args: argparse.Namespace) -> None: self.log_func = log_func From 0177a9b4025164b9d961f65fd71a49077817d8a2 Mon Sep 17 00:00:00 2001 From: "Adam R. Nelson" <adam@nels.onl> Date: Mon, 11 Aug 2025 13:28:01 -0400 Subject: [PATCH 144/174] Add RAW file thumbnailing support via rawpy (#567) * add RAW image file types to mimetype list * add RAW thumbnailer via rawpy --------- Signed-off-by: Adam R. Nelson <adam@nels.onl> Signed-off-by: ed <s@ocv.me> --- README.md | 3 + copyparty/__main__.py | 4 +- copyparty/svchub.py | 4 ++ copyparty/th_cli.py | 6 +- copyparty/th_srv.py | 127 +++++++++++++++++++++++++++++++++--------- copyparty/util.py | 3 + docs/chungus.conf | 5 +- 7 files changed, 122 insertions(+), 30 deletions(-) diff --git a/README.md b/README.md index 8f470a50..b3a2d134 100644 --- a/README.md +++ b/README.md @@ -266,6 +266,7 @@ also see [comparison to similar software](./docs/versus.md) * ☑ realtime streaming of growing files (logfiles and such) * ☑ [thumbnails](#thumbnails) * ☑ ...of images using Pillow, pyvips, or FFmpeg + * ☑ ...of RAW images using rawpy * ☑ ...of videos using FFmpeg * ☑ ...of audio (spectrograms) using FFmpeg * ☑ cache eviction (max-age; maybe max-size eventually) @@ -2795,6 +2796,7 @@ enable [thumbnails](#thumbnails) of... * **HEIF pictures:** `pyvips` or `ffmpeg` or `pyheif-pillow-opener` (requires Linux or a C compiler) * **AVIF pictures:** `pyvips` or `ffmpeg` or `pillow-avif-plugin` or pillow v11.3+ * **JPEG XL pictures:** `pyvips` or `ffmpeg` +* **RAW images:** `rawpy`, plus one of `pyvips` or `Pillow` (for some formats) enable sending [zeromq messages](#zeromq) from event-hooks: `pyzmq` @@ -2828,6 +2830,7 @@ set any of the following environment variables to disable its associated optiona | `PRTY_NO_PIL_HEIF` | disable 3rd-party Pillow plugin for [HEIF support](https://pypi.org/project/pyheif-pillow-opener/) | | `PRTY_NO_PIL_WEBP` | disable use of native webp support in Pillow | | `PRTY_NO_PSUTIL` | do not use [psutil](https://pypi.org/project/psutil/) for reaping stuck hooks and plugins on Windows | +| `PRTY_NO_RAW` | disable all [rawpy](https://pypi.org/project/rawpy/)-based thumbnail support for RAW images | | `PRTY_NO_VIPS` | disable all [libvips](https://pypi.org/project/pyvips/)-based thumbnail support; will fallback to Pillow or ffmpeg | example: `PRTY_NO_PIL=1 python3 copyparty-sfx.py` diff --git a/copyparty/__main__.py b/copyparty/__main__.py index be7c9979..946b3d0a 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1431,7 +1431,7 @@ def add_thumbnail(ap): ap2.add_argument("--th-ram-max", metavar="GB", type=float, default=th_ram, help="max memory usage (GiB) permitted by thumbnailer; not very accurate") ap2.add_argument("--th-crop", metavar="TXT", type=u, default="y", help="crop thumbnails to 4:3 or keep dynamic height; client can override in UI unless force. [\033[32my\033[0m]=crop, [\033[32mn\033[0m]=nocrop, [\033[32mfy\033[0m]=force-y, [\033[32mfn\033[0m]=force-n (volflag=crop)") ap2.add_argument("--th-x3", metavar="TXT", type=u, default="n", help="show thumbs at 3x resolution; client can override in UI unless force. [\033[32my\033[0m]=yes, [\033[32mn\033[0m]=no, [\033[32mfy\033[0m]=force-yes, [\033[32mfn\033[0m]=force-no (volflag=th3x)") - ap2.add_argument("--th-dec", metavar="LIBS", default="vips,pil,ff", help="image decoders, in order of preference") + ap2.add_argument("--th-dec", metavar="LIBS", default="vips,pil,raw,ff", help="image decoders, in order of preference") ap2.add_argument("--th-no-jpg", action="store_true", help="disable jpg output") ap2.add_argument("--th-no-webp", action="store_true", help="disable webp output") ap2.add_argument("--th-ff-jpg", action="store_true", help="force jpg output for video thumbs (avoids issues on some FFmpeg builds)") @@ -1442,9 +1442,11 @@ def add_thumbnail(ap): ap2.add_argument("--th-covers", metavar="N,N", type=u, default="folder.png,folder.jpg,cover.png,cover.jpg", help="folder thumbnails to stat/look for; enabling \033[33m-e2d\033[0m will make these case-insensitive, and try them as dotfiles (.folder.jpg), and also automatically select thumbnails for all folders that contain pics, even if none match this pattern") # https://pillow.readthedocs.io/en/stable/handbook/image-file-formats.html # https://github.com/libvips/libvips + # https://stackoverflow.com/a/47612661 # ffmpeg -hide_banner -demuxers | awk '/^ D /{print$2}' | while IFS= read -r x; do ffmpeg -hide_banner -h demuxer=$x; done | grep -E '^Demuxer |extensions:' ap2.add_argument("--th-r-pil", metavar="T,T", type=u, default="avif,avifs,blp,bmp,cbz,dcx,dds,dib,emf,eps,epub,fits,flc,fli,fpx,gif,heic,heics,heif,heifs,icns,ico,im,j2p,j2k,jp2,jpeg,jpg,jpx,pbm,pcx,pgm,png,pnm,ppm,psd,qoi,sgi,spi,tga,tif,tiff,webp,wmf,xbm,xpm", help="image formats to decode using pillow") ap2.add_argument("--th-r-vips", metavar="T,T", type=u, default="avif,exr,fit,fits,fts,gif,hdr,heic,jp2,jpeg,jpg,jpx,jxl,nii,pfm,pgm,png,ppm,svg,tif,tiff,webp", help="image formats to decode using pyvips") + ap2.add_argument("--th-r-raw", metavar="T,T", type=u, default="arw,cr2,crw,dcr,dng,erf,k25,kdc,mrw,nef,orf,pef,raf,raw,sr2,srf,x3f", help="image formats to decode using rawpy") ap2.add_argument("--th-r-ffi", metavar="T,T", type=u, default="apng,avif,avifs,bmp,cbz,dds,dib,epub,fit,fits,fts,gif,hdr,heic,heics,heif,heifs,icns,ico,jp2,jpeg,jpg,jpx,jxl,pbm,pcx,pfm,pgm,png,pnm,ppm,psd,qoi,sgi,tga,tif,tiff,webp,xbm,xpm", help="image formats to decode using ffmpeg") ap2.add_argument("--th-r-ffv", metavar="T,T", type=u, default="3gp,asf,av1,avc,avi,flv,h264,h265,hevc,m4v,mjpeg,mjpg,mkv,mov,mp4,mpeg,mpeg2,mpegts,mpg,mpg2,mts,nut,ogm,ogv,rm,ts,vob,webm,wmv", help="video formats to decode using ffmpeg") ap2.add_argument("--th-r-ffa", metavar="T,T", type=u, default="aac,ac3,aif,aiff,alac,alaw,amr,apac,ape,au,bonk,dfpwm,dts,flac,gsm,ilbc,it,itgz,itxz,itz,m4a,mdgz,mdxz,mdz,mo3,mod,mp2,mp3,mpc,mptm,mt2,mulaw,oga,ogg,okt,opus,ra,s3m,s3gz,s3xz,s3z,tak,tta,ulaw,wav,wma,wv,xm,xmgz,xmxz,xmz,xpk", help="audio formats to decode using ffmpeg") diff --git a/copyparty/svchub.py b/copyparty/svchub.py index 1af13276..5e0ab886 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -40,6 +40,7 @@ from .th_srv import ( HAVE_PIL, HAVE_VIPS, HAVE_WEBP, + HAVE_RAW, ThumbSrv, ) from .up2k import Up2k @@ -321,6 +322,8 @@ class SvcHub(object): decs.pop("vips", None) if not HAVE_PIL: decs.pop("pil", None) + if not HAVE_RAW: + decs.pop("raw", None) if not HAVE_FFMPEG or not HAVE_FFPROBE: decs.pop("ff", None) @@ -811,6 +814,7 @@ class SvcHub(object): (HAVE_ZMQ, "pyzmq", "send zeromq messages from event-hooks"), (HAVE_HEIF, "pillow-heif", "read .heif images with pillow (rarely useful)"), (HAVE_AVIF, "pillow-avif", "read .avif images with pillow (rarely useful)"), + (HAVE_RAW, "rawpy", "read RAW images"), ] if ANYWIN: to_check += [ diff --git a/copyparty/th_cli.py b/copyparty/th_cli.py index 6c2632b5..384316ee 100644 --- a/copyparty/th_cli.py +++ b/copyparty/th_cli.py @@ -36,11 +36,15 @@ class ThumbCli(object): if not c: raise Exception() except: - c = {k: set() for k in ["thumbable", "pil", "vips", "ffi", "ffv", "ffa"]} + c = { + k: set() + for k in ["thumbable", "pil", "vips", "raw", "ffi", "ffv", "ffa"] + } self.thumbable = c["thumbable"] self.fmt_pil = c["pil"] self.fmt_vips = c["vips"] + self.fmt_raw = c["raw"] self.fmt_ffi = c["ffi"] self.fmt_ffv = c["ffv"] self.fmt_ffa = c["ffa"] diff --git a/copyparty/th_srv.py b/copyparty/th_srv.py index 7b07376a..b37591c6 100644 --- a/copyparty/th_srv.py +++ b/copyparty/th_srv.py @@ -2,6 +2,7 @@ from __future__ import print_function, unicode_literals import hashlib +import io import logging import os import re @@ -121,6 +122,17 @@ try: except: HAVE_VIPS = False +try: + if os.environ.get("PRTY_NO_RAW"): + raise Exception() + + HAVE_RAW = True + import rawpy + + logging.getLogger("rawpy").setLevel(logging.WARNING) +except: + HAVE_RAW = False + th_dir_cache = {} @@ -205,11 +217,19 @@ class ThumbSrv(object): if self.args.th_clean: Daemon(self.cleaner, "thumb.cln") - self.fmt_pil, self.fmt_vips, self.fmt_ffi, self.fmt_ffv, self.fmt_ffa = [ + ( + self.fmt_pil, + self.fmt_vips, + self.fmt_raw, + self.fmt_ffi, + self.fmt_ffv, + self.fmt_ffa, + ) = [ set(y.split(",")) for y in [ self.args.th_r_pil, self.args.th_r_vips, + self.args.th_r_raw, self.args.th_r_ffi, self.args.th_r_ffv, self.args.th_r_ffa, @@ -232,6 +252,9 @@ class ThumbSrv(object): if "vips" in self.args.th_dec: self.thumbable |= self.fmt_vips + if "raw" in self.args.th_dec: + self.thumbable |= self.fmt_raw + if "ff" in self.args.th_dec: for zss in [self.fmt_ffi, self.fmt_ffv, self.fmt_ffa]: self.thumbable |= zss @@ -313,6 +336,7 @@ class ThumbSrv(object): "thumbable": self.thumbable, "pil": self.fmt_pil, "vips": self.fmt_vips, + "raw": self.fmt_raw, "ffi": self.fmt_ffi, "ffv": self.fmt_ffv, "ffa": self.fmt_ffa, @@ -368,6 +392,8 @@ class ThumbSrv(object): funs.append(self.conv_pil) elif lib == "vips" and ext in self.fmt_vips: funs.append(self.conv_vips) + elif lib == "raw" and ext in self.fmt_raw: + funs.append(self.conv_raw) elif can_au and (want_png or want_au): if want_opus: funs.append(self.conv_opus) @@ -480,35 +506,38 @@ class ThumbSrv(object): return im + def conv_image_pil(self, im: "Image.Image", tpath: str, fmt: str, vn: VFS) -> None: + try: + im = self.fancy_pillow(im, fmt, vn) + except Exception as ex: + self.log("fancy_pillow {}".format(ex), "90") + im.thumbnail(self.getres(vn, fmt)) + + fmts = ["RGB", "L"] + args = {"quality": 40} + + if tpath.endswith(".webp"): + # quality 80 = pillow-default + # quality 75 = ffmpeg-default + # method 0 = pillow-default, fast + # method 4 = ffmpeg-default + # method 6 = max, slow + fmts.extend(("RGBA", "LA")) + args["method"] = 6 + else: + # default q = 75 + args["progressive"] = True + + if im.mode not in fmts: + # print("conv {}".format(im.mode)) + im = im.convert("RGB") + + im.save(tpath, **args) + def conv_pil(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: self.wait4ram(0.2, tpath) with Image.open(fsenc(abspath)) as im: - try: - im = self.fancy_pillow(im, fmt, vn) - except Exception as ex: - self.log("fancy_pillow {}".format(ex), "90") - im.thumbnail(self.getres(vn, fmt)) - - fmts = ["RGB", "L"] - args = {"quality": 40} - - if tpath.endswith(".webp"): - # quality 80 = pillow-default - # quality 75 = ffmpeg-default - # method 0 = pillow-default, fast - # method 4 = ffmpeg-default - # method 6 = max, slow - fmts.extend(("RGBA", "LA")) - args["method"] = 6 - else: - # default q = 75 - args["progressive"] = True - - if im.mode not in fmts: - # print("conv {}".format(im.mode)) - im = im.convert("RGB") - - im.save(tpath, **args) + self.conv_image_pil(im, tpath, fmt, vn) def conv_vips(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: self.wait4ram(0.2, tpath) @@ -531,6 +560,50 @@ class ThumbSrv(object): assert img # type: ignore # !rm img.write_to_file(tpath, Q=40) + def conv_raw(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: + self.wait4ram(0.2, tpath) + with rawpy.imread(abspath) as raw: + thumb = raw.extract_thumb() + if thumb.format == rawpy.ThumbFormat.JPEG and tpath.endswith(".jpg"): + # if we have a jpg thumbnail and no webp output is available, + # just write the jpg directly (it'll be the wrong size, but it's fast) + with open(tpath, "wb") as f: + f.write(thumb.data) + if HAVE_VIPS: + crops = ["centre", "none"] + if "f" in fmt: + crops = ["none"] + w, h = self.getres(vn, fmt) + kw = {"height": h, "size": "down", "intent": "relative"} + + for c in crops: + try: + kw["crop"] = c + if thumb.format == rawpy.ThumbFormat.BITMAP: + img = pyvips.Image.new_from_array( + thumb.data, interpretation="rgb" + ) + img = img.thumbnail_image(w, **kw) + else: + img = pyvips.Image.thumbnail_buffer(thumb.data, w, **kw) + break + except: + if c == crops[-1]: + raise + + assert img # type: ignore # !rm + img.write_to_file(tpath, Q=40) + elif HAVE_PIL: + if thumb.format == rawpy.ThumbFormat.BITMAP: + im = Image.fromarray(thumb.data, "RGB") + else: + im = Image.open(io.BytesIO(thumb.data)) + self.conv_image_pil(im, tpath, fmt, vn) + else: + raise Exception( + "either pil or vips is needed to process embedded bitmap thumbnails in raw files" + ) + def conv_ffmpeg(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: self.wait4ram(0.2, tpath) ret, _ = ffprobe(abspath, int(vn.flags["convt"] / 2)) diff --git a/copyparty/util.py b/copyparty/util.py index b5a2c45e..1578d291 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -399,6 +399,9 @@ application swf=x-shockwave-flash m3u=vnd.apple.mpegurl db3=vnd.sqlite3 sqlite=v text ass=plain ssa=plain image jpg=jpeg xpm=x-xpixmap psd=vnd.adobe.photoshop jpf=jpx tif=tiff ico=x-icon djvu=vnd.djvu image heic=heic-sequence heif=heif-sequence hdr=vnd.radiance svg=svg+xml +image arw=x-sony-arw cr2=x-canon-cr2 crw=x-canon-crw dcr=x-kodak-dcr dng=x-adobe-dng erf=x-epson-erf +image k25=x-kodak-k25 kdc=x-kodak-kdc mrw=x-minolta-mrw nef=x-nikon-nef orf=x-olympus-orf +image pef=x-pentax-pef raf=x-fuji-raf raw=x-panasonic-raw sr2=x-sony-sr2 srf=x-sony-srf x3f=x-sigma-x3f audio caf=x-caf mp3=mpeg m4a=mp4 mid=midi mpc=musepack aif=aiff au=basic qcp=qcelp video mkv=x-matroska mov=quicktime avi=x-msvideo m4v=x-m4v ts=mp2t video asf=x-ms-asf flv=x-flv 3gp=3gpp 3g2=3gpp2 rmvb=vnd.rn-realmedia-vbr diff --git a/docs/chungus.conf b/docs/chungus.conf index 7ac7ba9a..a8e34516 100644 --- a/docs/chungus.conf +++ b/docs/chungus.conf @@ -1083,7 +1083,7 @@ th-x3: n # default # image decoders, in order of preference - th-dec: vips,pil,ff # default + th-dec: vips,pil,raw,ff # default # disable jpg output th-no-jpg @@ -1115,6 +1115,9 @@ # image formats to decode using pyvips th-r-vips: a,very,long,list,of,file,extensions # hint + # image formats to decode using rawpy + th-r-raw: a,very,long,list,of,file,extensions # hint + # image formats to decode using ffmpeg th-r-ffi: a,very,long,list,of,file,extensions # hint From 6ccc9224f308a8a171478c5645dc065273012100 Mon Sep 17 00:00:00 2001 From: Kamalei Zestri <38802353+KamaleiZestri@users.noreply.github.com> Date: Sun, 10 Aug 2025 15:21:45 -0500 Subject: [PATCH 145/174] make rpm --- contrib/package/rpm/copyparty.spec | 62 ++++++++++++++++++++++++++++++ scripts/make-rpm.sh | 61 +++++++++++++++++++++++++++++ 2 files changed, 123 insertions(+) create mode 100644 contrib/package/rpm/copyparty.spec create mode 100755 scripts/make-rpm.sh diff --git a/contrib/package/rpm/copyparty.spec b/contrib/package/rpm/copyparty.spec new file mode 100644 index 00000000..6d3ebb3f --- /dev/null +++ b/contrib/package/rpm/copyparty.spec @@ -0,0 +1,62 @@ +Name: copyparty +Version: $pkgver +Release: $pkgrel +License: MIT +Group: Utilities +URL: https://github.com/9001/copyparty +Source0: copyparty-$pkgver.tar.gz +Summary: File server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++ +BuildArch: noarch +BuildRequires: python3, python3-devel, pyproject-rpm-macros, python-setuptools, python-wheel, make +Requires: python3, (python3-jinja2 or python-jinja2), lsof +Recommends: ffmpeg, (golang-github-cloudflare-cfssl or cfssl), python-mutagen, python-pillow, python-pyvips +Recommends: qm-vamp-plugins, python-argon2-cffi, (python-pyopenssl or pyopenssl), python-impacket + +%description +Portable file server with accelerated resumable uploads, dedup, WebDAV, FTP, TFTP, zeroconf, media indexer, thumbnails++ all in one file, no deps + +See release at https://github.com/9001/copyparty/releases + +%global debug_package %{nil} + +%generate_buildrequires +%pyproject_buildrequires + +%prep +%setup -q + +%build +cd "copyparty/web" +make +cd - +%pyproject_wheel + +%install +mkdir -p %{buildroot}%{_bindir} +mkdir -p %{buildroot}%{_libdir}/systemd/{system,user} +mkdir -p %{buildroot}/etc/%{name} +mkdir -p %{buildroot}/var/lib/%{name}-jail +mkdir -p %{buildroot}%{_datadir}/licenses/%{name} + +%pyproject_install +%pyproject_save_files copyparty + +install -m 0755 bin/prisonparty.sh %{buildroot}%{_bindir}/prisonpary.sh +install -m 0644 contrib/systemd/%{name}.conf %{buildroot}/etc/%{name}/%{name}.conf +install -m 0644 contrib/systemd/%{name}@.service %{buildroot}%{_libdir}/systemd/system/%{name}@.service +install -m 0644 contrib/systemd/%{name}-user.service %{buildroot}%{_libdir}/systemd/user/%{name}.service +install -m 0644 contrib/systemd/prisonparty@.service %{buildroot}%{_libdir}/systemd/system/prisonparty@.service +install -m 0644 contrib/systemd/index.md %{buildroot}/var/lib/%{name}-jail/README.md +install -m 0644 LICENSE %{buildroot}%{_datadir}/licenses/%{name}/LICENSE + +%files -n copyparty -f %{pyproject_files} +%license LICENSE +%{_bindir}/copyparty +%{_bindir}/partyfuse +%{_bindir}/u2c +%{_bindir}/prisonpary.sh +/etc/%{name}/%{name}.conf +%{_libdir}/systemd/system/%{name}@.service +%{_libdir}/systemd/user/%{name}.service +%{_libdir}/systemd/system/prisonparty@.service +/var/lib/%{name}-jail/README.md diff --git a/scripts/make-rpm.sh b/scripts/make-rpm.sh new file mode 100755 index 00000000..441ade5a --- /dev/null +++ b/scripts/make-rpm.sh @@ -0,0 +1,61 @@ +#!/bin/bash + +#--localbuild to build webdeps and tar locally; otherwise just download prebuilt +#--pm change packagemanager; otherwise default to dnf + +while [ ! -z "$1" ]; do + case $1 in + local-build) local_build=1 ; ;; + pm) shift;packagemanager="$1"; ;; + esac + shift +done + +[ -e copyparty/__main__.py ] || cd .. +[ -e copyparty/__main__.py ] || +{ + echo "run me from within the project root folder" + echo + exit 1 +} + + +packagemanager=${packagemanager:-dnf} +ver=$(awk '/^VERSION/{gsub(/[^0-9]/," ");printf "%d.%d.%d\n",$1,$2,$3}' copyparty/__version__.py) +releasedir="dist/temp_copyparty_$ver" +sourcepkg="copyparty-$ver.tar.gz" + +#make temporary directory to build rpm in +mkdir -p $releasedir/{BUILD,BUILDROOT,RPMS,SOURCES,SPECS,SRPMS} +trap "rm -rf $releasedir/" EXIT + +# make/get tarball +if [ $local_build ]; then + if [ ! -f "copyparty/web/deps/mini-fa.woff" ]; then + sudo $packagemanager update + sudo $packagemanager install podman-docker docker + make -C deps-docker + fi + if [ ! -f "dist/$sourcepkg" ]; then + ./$cppdir/scripts/make-tgz-release.sh "$ver" + fi +else + if [ ! -f "dist/$sourcepkg" ]; then + curl -OL https://github.com/9001/copyparty/releases/download/v$ver/$sourcepkg --output-dir dist + fi +fi + +cp dist/$sourcepkg "$releasedir/SOURCES/$sourcepkg" + +cp "contrib/package/rpm/copyparty.spec" "$releasedir/SPECS/" +sed -i "s/\$pkgver/$ver/g" "$releasedir/SPECS/copyparty.spec" +sed -i "s/\$pkgrel/1/g" "$releasedir/SPECS/copyparty.spec" + +sudo $packagemanager update +sudo $packagemanager install rpmdevtools python-wheel python-setuptools pyproject-rpm-macros +cd "$releasedir/" +rpmbuild --define "_topdir `pwd`" -bb SPECS/copyparty.spec +cd - + +rpm="copyparty-$ver-1.noarch.rpm" +mv "$releasedir/RPMS/noarch/$rpm" dist/$rpm From 88243ac8d6e4b98e9e10f52d4733c3f7eca4f614 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Mon, 11 Aug 2025 18:35:49 +0000 Subject: [PATCH 146/174] make-rpm: small tweaks; * fail fast on error * ensure all deps Signed-off-by: ed <s@ocv.me> --- scripts/make-rpm.sh | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/scripts/make-rpm.sh b/scripts/make-rpm.sh index 441ade5a..5dd6573c 100755 --- a/scripts/make-rpm.sh +++ b/scripts/make-rpm.sh @@ -1,4 +1,5 @@ #!/bin/bash +set -e #--localbuild to build webdeps and tar locally; otherwise just download prebuilt #--pm change packagemanager; otherwise default to dnf @@ -27,7 +28,7 @@ sourcepkg="copyparty-$ver.tar.gz" #make temporary directory to build rpm in mkdir -p $releasedir/{BUILD,BUILDROOT,RPMS,SOURCES,SPECS,SRPMS} -trap "rm -rf $releasedir/" EXIT +trap "rm -rf $releasedir" EXIT # make/get tarball if [ $local_build ]; then @@ -37,6 +38,7 @@ if [ $local_build ]; then make -C deps-docker fi if [ ! -f "dist/$sourcepkg" ]; then + ./$cppdir/scripts/make-sfx.sh gz fast # pulls some build-deps + good smoketest ./$cppdir/scripts/make-tgz-release.sh "$ver" fi else @@ -52,7 +54,10 @@ sed -i "s/\$pkgver/$ver/g" "$releasedir/SPECS/copyparty.spec" sed -i "s/\$pkgrel/1/g" "$releasedir/SPECS/copyparty.spec" sudo $packagemanager update -sudo $packagemanager install rpmdevtools python-wheel python-setuptools pyproject-rpm-macros +sudo $packagemanager install \ + rpmdevtools python-devel pyproject-rpm-macros \ + python-wheel python-setuptools python-jinja2 \ + make pigz cd "$releasedir/" rpmbuild --define "_topdir `pwd`" -bb SPECS/copyparty.spec cd - From 1ebe06f51efaf3e77974a48b16f907992c2aaabf Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Mon, 11 Aug 2025 20:49:09 +0000 Subject: [PATCH 147/174] sticky qr-code; #533 --- copyparty/__main__.py | 7 ++++--- copyparty/svchub.py | 40 +++++++++++++++++++++++++++++++++++++++- copyparty/tcpsrv.py | 6 ++++++ tests/util.py | 2 +- 4 files changed, 50 insertions(+), 5 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 946b3d0a..e1fcce2f 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1019,14 +1019,15 @@ def add_general(ap, nc, srvname): def add_qr(ap, tty): ap2 = ap.add_argument_group("qr options") - ap2.add_argument("--qr", action="store_true", help="show http:// QR-code on startup") - ap2.add_argument("--qrs", action="store_true", help="show https:// QR-code on startup") + ap2.add_argument("--qr", action="store_true", help="show QR-code on startup") + ap2.add_argument("--qrs", action="store_true", help="change the QR-code URL to https://") ap2.add_argument("--qrl", metavar="PATH", type=u, default="", help="location to include in the url, for example [\033[32mpriv/?pw=hunter2\033[0m]") ap2.add_argument("--qri", metavar="PREFIX", type=u, default="", help="select IP which starts with \033[33mPREFIX\033[0m; [\033[32m.\033[0m] to force default IP when mDNS URL would have been used instead") - ap2.add_argument("--qr-fg", metavar="COLOR", type=int, default=0 if tty else 16, help="foreground; try [\033[32m0\033[0m] if the qr-code is unreadable") + ap2.add_argument("--qr-fg", metavar="COLOR", type=int, default=0 if tty else 16, help="foreground; try [\033[32m0\033[0m] or [\033[32m-1\033[0m] if the qr-code is unreadable") ap2.add_argument("--qr-bg", metavar="COLOR", type=int, default=229, help="background (white=255)") ap2.add_argument("--qrp", metavar="CELLS", type=int, default=4, help="padding (spec says 4 or more, but 1 is usually fine)") ap2.add_argument("--qrz", metavar="N", type=int, default=0, help="[\033[32m1\033[0m]=1x, [\033[32m2\033[0m]=2x, [\033[32m0\033[0m]=auto (try [\033[32m2\033[0m] on broken fonts)") + ap2.add_argument("--qr-pin", metavar="N", type=int, default=0, help="sticky/pin the qr-code to always stay on-screen; [\033[32m0\033[0m]=disabled, [\033[32m1\033[0m]=with-url, [\033[32m2\033[0m]=just-qr") def add_fs(ap): diff --git a/copyparty/svchub.py b/copyparty/svchub.py index 5e0ab886..a6f87a85 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -2,6 +2,7 @@ from __future__ import print_function, unicode_literals import argparse +import atexit import errno import logging import os @@ -73,6 +74,7 @@ from .util import ( pybin, start_log_thrs, start_stackmon, + termsize, ub64enc, ) @@ -775,6 +777,39 @@ class SvcHub(object): def sigterm(self) -> None: self.signal_handler(signal.SIGTERM, None) + def sticky_qr(self) -> None: + tw, th = termsize() + zs1, qr = self.tcpsrv.qr.split("\n", 1) + url, colr = zs1.split(" ", 1) + nl = len(qr.split("\n")) # numlines + lp = 3 if nl * 2 + 4 < tw else 0 # leftpad + lp0 = lp + if self.args.qr_pin == 2: + url = "" + else: + while lp and (nl + lp) * 2 + len(url) + 1 > tw: + lp -= 1 + if (nl + lp) * 2 + len(url) + 1 > tw: + qr = url + "\n" + qr + url = "" + nl += 1 + lp = lp0 + sh = 1 + th - nl + if lp: + zs = " " * lp + qr = zs + qr.replace("\n", "\n" + zs) + if url: + url = "%s\033[%d;%dH%s\033[0m" % (colr, sh + 1, (nl + lp) * 2, url) + qr = colr + qr + + def unlock(): + print("\033[s\033[r\033[u", file=sys.stderr) + + atexit.register(unlock) + t = "%s\033[%dA" % ("\n" * nl, nl) + t = "%s\033[s\033[1;%dr\033[%dH%s%s\033[u" % (t, sh - 1, sh, qr, url) + self.pr(t, file=sys.stderr) + def cb_httpsrv_up(self) -> None: self.httpsrv_up += 1 if self.httpsrv_up != self.broker.num_workers: @@ -787,7 +822,10 @@ class SvcHub(object): break if self.tcpsrv.qr: - self.log("qr-code", self.tcpsrv.qr) + if self.args.qr_pin: + self.sticky_qr() + else: + self.log("qr-code", self.tcpsrv.qr) else: self.log("root", "workers OK\n") diff --git a/copyparty/tcpsrv.py b/copyparty/tcpsrv.py index 6c61fe61..55ea3a56 100644 --- a/copyparty/tcpsrv.py +++ b/copyparty/tcpsrv.py @@ -614,6 +614,10 @@ class TcpSrv(object): fg = self.args.qr_fg bg = self.args.qr_bg + nocolor = fg == -1 + if nocolor: + fg = 0 + pad = self.args.qrp zoom = self.args.qrz qrc = QrCode.encode_binary(btxt) @@ -641,6 +645,8 @@ class TcpSrv(object): qr = qr.replace("\n", "\033[K\n") + "\033[K" # win10do cc = " \033[0;38;5;{0};47;48;5;{1}m" if fg else " \033[0;30;47m" + if nocolor: + cc = " \033[0m" t = cc + "\n{2}\033[999G\033[0m\033[J" t = t.format(fg, bg, qr) if ANYWIN: diff --git a/tests/util.py b/tests/util.py index d3884ca9..f02f81ba 100644 --- a/tests/util.py +++ b/tests/util.py @@ -161,7 +161,7 @@ class Cfg(Namespace): ex = "au_vol dl_list mtab_age reg_cap s_thead s_tbody tail_tmax tail_who th_convt ups_who zip_who" ka.update(**{k: 9 for k in ex.split()}) - ex = "ctl_re db_act forget_ip idp_cookie idp_store k304 loris no304 nosubtle re_maxage rproxy rsp_jtr rsp_slp s_wr_slp snap_wri theme themes turbo u2ow zipmaxn zipmaxs" + ex = "ctl_re db_act forget_ip idp_cookie idp_store k304 loris no304 nosubtle qr_pin re_maxage rproxy rsp_jtr rsp_slp s_wr_slp snap_wri theme themes turbo u2ow zipmaxn zipmaxs" ka.update(**{k: 0 for k in ex.split()}) ex = "ah_alg bname chmod_f chpw_db doctitle df exit favico idp_h_usr ipa html_head lg_sba lg_sbf log_fk md_sba md_sbf name og_desc og_site og_th og_title og_title_a og_title_v og_title_i shr tcolor textfiles txt_eol unlist vname xff_src zipmaxt R RS SR" From a407eb92692cf09f27892cc0921a14db78abc403 Mon Sep 17 00:00:00 2001 From: mat <matheuz1210@gmail.com> Date: Tue, 12 Aug 2025 12:31:23 -0300 Subject: [PATCH 148/174] warn if failed to import pyvips pyvips uses `ffi.callback()`, which fails on on some systems --- copyparty/th_srv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/copyparty/th_srv.py b/copyparty/th_srv.py index b37591c6..945c6d56 100644 --- a/copyparty/th_srv.py +++ b/copyparty/th_srv.py @@ -113,14 +113,17 @@ except: try: if os.environ.get("PRTY_NO_VIPS"): - raise Exception() + raise ImportError() HAVE_VIPS = True import pyvips logging.getLogger("pyvips").setLevel(logging.WARNING) -except: +except Exception as e: HAVE_VIPS = False + if not isinstance(e, ImportError): + logging.warning("libvips found, but failed to load: " + str(e)) + try: if os.environ.get("PRTY_NO_RAW"): From d8662aeb0ecb0409d1807e466fc5818303437e53 Mon Sep 17 00:00:00 2001 From: varphi <160203809+varphi-online@users.noreply.github.com> Date: Tue, 12 Aug 2025 12:01:41 -0700 Subject: [PATCH 149/174] add win95 light and dark themes (#581) --- copyparty/__main__.py | 2 +- copyparty/web/browser.css | 787 ++++++++++++++++++++++++++++++++++++++ copyparty/web/browser.js | 2 +- 3 files changed, 789 insertions(+), 2 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index e1fcce2f..97392c87 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1570,7 +1570,7 @@ def add_ui(ap, retry): ap2.add_argument("--localtime", action="store_true", help="default to local timezone instead of UTC") ap2.add_argument("--lang", metavar="LANG", type=u, default="eng", help="language; one of the following: \033[32meng nor chi\033[0m") ap2.add_argument("--theme", metavar="NUM", type=int, default=0, help="default theme to use (0..7)") - ap2.add_argument("--themes", metavar="NUM", type=int, default=8, help="number of themes installed") + ap2.add_argument("--themes", metavar="NUM", type=int, default=10, help="number of themes installed") ap2.add_argument("--au-vol", metavar="0-100", type=int, default=50, choices=range(0, 101), help="default audio/video volume percent") ap2.add_argument("--sort", metavar="C,C,C", type=u, default="href", help="default sort order, comma-separated column IDs (see header tooltips), prefix with '-' for descending. Examples: \033[32mhref -href ext sz ts tags/Album tags/.tn\033[0m (volflag=sort)") ap2.add_argument("--nsort", action="store_true", help="default-enable natural sort of filenames with leading numbers (volflag=nsort)") diff --git a/copyparty/web/browser.css b/copyparty/web/browser.css index 298de0ea..f2dc8cde 100644 --- a/copyparty/web/browser.css +++ b/copyparty/web/browser.css @@ -3272,3 +3272,790 @@ html.d #treepar { transition: background-color .3s ease, color .3s ease; } } +html.ey { + --negative-space: 0em; /* Use this to change the global spacing of the 95 theme */ + --font-main: consolas; + --font-serif: consolas; + --font-mono: consolas; + --w: #fff; + --w2: #dfdfdf; + --w3: grey; + --fg: #000; + --fg-max: #0000ff; + --fg-weak: #0000ff; + --bg: #c6c3c6; + --bg-d3: #ff0; + --bg-d2: var(--w3); + --bg-d1: var(--bg); + --bg-u2: var(--bg); + --bg-u3: var(--bg); + --bg-u5: var(--shadow-color-2); + --tab-alt: #00f; + --g-fsel-bg: #00f; + --g-sel-bg: #00f; + --g-fsel-b1: #fff; + --row-alt: var(--w); + --scroll: var(--silver); + --f-sel-sh: transparent; + --a: #000; + --a-b: #fff; + --a-hil: #fff; + --a-h-bg: var(--bg); + --a-dark: var(--a); + --a-gray: var(--fg-weak); + --btn-fg: var(--fg); + --btn-bg: var(--bg); + --btn-h-fg: var(--fg); + --btn-h-bg: var(--bg); + --btn-1-fg: var(--fg); + --btn-1-bg: var(--bg); + --btn-1h-bg: var(--bg-d3); + --txt-sh: a; + --txt-bg: var(--white); + --u2-b1-bg: var(--w2); + --u2-b2-bg: var(--w2); + --u2-txt-bg: var(--w2); + --u2-tab-bg: a; + --u2-tab-1-bg: var(--w2); + --sort-1: var(--fg-weak); + --tree-bg: var(--w); + --g-b1: a; + --g-b2: a; + --g-f-bg: var(--w2); + --f-sh1: 0.1; + --f-sh2: 0.02; + --f-sh3: 0.1; + --f-h-b1: a; + --srv-1: var(--w); + --srv-3: var(--a); + --mp-sh: a; + --black: #000; + --white: #fff; + --grey: grey; + --silver: silver; + --transparent: transparent; + --shadow-color-1: #0a0a0a; + --shadow-color-2: #808080; + --border-dashed-black: 1px dashed var(--black); + --radius: 0; + --focus-outline: 1px dashed var(--black); + --hover-outline: 1px dotted var(--black); + --fm-off: var(--w3); + --ttlbar: linear-gradient(90deg, navy, #1084d0); + --inset-bg: var(--white); + --scroll-bkg: var(--white); + + /*All sides*/ + --shadow-outset: inset -1px -1px var(--shadow-color-1), + inset 1px 1px var(--white), inset -2px -2px var(--grey), + inset 2px 2px var(--w2); + + --shadow-inset: inset -1px -1px var(--white), + inset 1px 1px var(--shadow-color-1), inset -2px -2px var(--w2), + inset 2px 2px var(--shadow-color-2); + + --shadow-input: inset -1px -1px var(--white), inset 1px 1px var(--grey), + inset -2px -2px var(--w2), inset 2px 2px var(--shadow-color-1); + + /*Indiv sides*/ + --shadow-outset-bottom: inset 0 -1px var(--shadow-color-1), + inset 0 -2px var(--grey); + --shadow-outset-right: inset -1px 0 var(--shadow-color-1), + inset -2px 0 var(--grey); + --shadow-outset-left: inset 1px 0 var(--white), inset 2px 0 var(--w2); + --shadow-outset-top: inset 0 1px var(--white), inset 0 2px var(--w2); + + --shadow-inset-bottom: inset 0 -1px var(--white), inset 0 -2px var(--w2); + --shadow-inset-right: inset -1px 0 var(--white), inset -2px 0 var(--w2); + --shadow-inset-left: inset 1px 0 var(--shadow-color-1), + inset 2px 0 var(--shadow-color-2); + --shadow-inset-top: inset 0 1px var(--shadow-color-1), + inset 0 2px var(--shadow-color-2); +} +html.ez { + --negative-space: 0em; /* Use this to change the global spacing of your theme :) */ + --font-main: consolas; + --font-serif: consolas; + --font-mono: consolas; + --w: #fff; + --w2: var(--inset-bg); + --w3: grey; + --fg: #cfcfcf; + --fg-max: #47b8ff; + --fg-weak: #47b8ff; + --bg: #383838; + --bg-d3: #600000; + --bg-d2: var(--shadow-color-1); + --bg-d1: var(--bg); + --u2-tab-1-fg: #ff0; + --bg-u2: var(--bg); + --bg-u3: var(--bg); + --bg-u5: var(--shadow-color-2); + --tab-alt: #47b8ff; + --g-fsel-bg: #0000b7; + --g-sel-bg: #00f; + --g-fsel-b1: #fff; + --row-alt: #555555; + --scroll: #555555; + --f-sel-sh: transparent; + --a: var(--fg); + --a-b: var(--fg); + --a-hil: var(--fg); + --btn-1h-bg: var(--bg-d3); + --a-h-bg: var(--bg); + --a-dark: var(--a); + --a-gray: var(--fg-weak); + --btn-fg: var(--white); + --btn-bg: var(--bg); + --btn-h-fg: var(--white); + --btn-h-bg: var(--bg); + --btn-1-fg: var(--white); + --btn-1-bg: var(--bg); + --txt-sh: a; + --u2-b1-bg: var(--w2); + --u2-b2-bg: var(--w2); + --u2-txt-bg: var(--w2); + --u2-tab-bg: a; + --u2-tab-1-bg: var(--w2); + --sort-1: var(--fg-weak); + --g-b1: a; + --g-b2: a; + --g-f-bg: var(--w2); + --f-sh1: 0.1; + --f-sh2: 0.02; + --f-sh3: 0.1; + --f-h-b1: a; + --srv-1: var(--w); + --srv-3: var(--a); + --mp-sh: a; + --black: #000; + --white: #fff; + --grey: grey; + --silver: #858585; + --transparent: transparent; + --shadow-color-1: #101010; + --shadow-color-2: #1f1f1f; + --border-dashed-black: 1px dashed var(--shadow-color-1); + --radius: 0; + --focus-outline: 1px dashed var(--white); + --hover-outline: 1px dotted var(--white); + --fm-off: var(--w3); + --ttlbar: linear-gradient(90deg, var(--shadow-color-1) 20%, #888888); + --inset-bg: #3f3f3f; + --tree-bg: var(--inset-bg); + --txt-bg: var(--inset-bg); + --scroll-bkg: var(--black); + + /*All sides*/ + --shadow-outset: inset -1px -1px var(--shadow-color-1), inset 1px 1px #878787, + inset -2px -2px var(--shadow-color-2), inset 2px 2px #575757; + + --shadow-inset: inset -1px -1px #878787, inset 1px 1px var(--shadow-color-1), + inset -2px -2px #575757, inset 2px 2px var(--shadow-color-2); + + --shadow-input: inset -1px -1px var(--white), + inset 1px 1px var(--shadow-color-2), inset -2px -2px #575757, + inset 2px 2px var(--shadow-color-1); + + --shadow-outset-bottom: inset 0 -1px var(--shadow-color-1), + inset 0 -2px var(--shadow-color-2); + --shadow-outset-right: inset -1px 0 var(--shadow-color-1), + inset -2px 0 var(--shadow-color-2); + --shadow-outset-left: inset 1px 0 #878787, inset 2px 0 #575757; + --shadow-outset-top: inset 0 1px #878787, inset 0 2px #575757; + + --shadow-inset-bottom: inset 0 -1px #878787, inset 0 -2px #575757; + --shadow-inset-right: inset -1px 0 #878787, inset -2px 0 #575757; + --shadow-inset-left: inset 1px 0 var(--shadow-color-1), + inset 2px 0 var(--shadow-color-2); + --shadow-inset-top: inset 0 1px var(--shadow-color-1), + inset 0 2px var(--shadow-color-2); +} + +html.e { + text-shadow: none; +} + +html.e #files, +html.e #u2conf input[type="checkbox"]:hover + label, +html.e .tgl.btn.on:hover, +html.e body { + background: var(--bg); +} +html.e #pctl a, +html.e #repl, +html.e #u2conf a, +html.e #u2conf input[type="checkbox"] + label, +html.e #wfp a, +html.e .btn, +html.e .eq_step, +html.e input[type="submit"] { + box-shadow: var(--shadow-outset); + border-radius: var(--radius); + background: var(--bg); + border: 0; +} +a.s0r, +html.e #ghead a.s0, +html.e #u2conf input[type="checkbox"]:checked + label, +html.e .tgl.btn.on, +html.e input[type="submit"]:active { + box-shadow: var(--shadow-inset) !important; +} +html.e #ops a:hover, +html.e #pctl a:hover, +html.e #repl:hover, +html.e #u2conf a:hover, +html.e #u2conf input[type="checkbox"]:hover + label, +html.e #wfp a:hover, +html.e .btn:hover, +html.e .eq_step:hover, +html.e input[type="submit"]:hover { + outline: var(--hover-outline); + outline-offset: -4px; +} +html.e .ntree a:hover, +html.e :focus, +html.e :focus + label, +html.e a:active, +html.e tr:focus, +input[type="text"]:focus { + outline: var(--focus-outline) !important; +} +html.e tr:focus { + box-shadow: none; +} +html.e #pctl a:focus, +html.e #repl:hover, +html.e #u2conf input[type="checkbox"]:focus + label, +html.e #wfp a:focus, +html.e .btn:focus, +html.e .eq_step:focus { + border: 0 !important; + outline: var(--focus-outline) !important; + outline-offset: 2px; + box-shadow: var(--shadow-outset) !important; +} +html.e #files tbody, +html.e #u2cards a.act { + box-shadow: var(--shadow-inset); +} +html.e #files { + border: 2px groove var(--transparent); + box-sizing: border-box; + width: 100%; + padding: 0.3em; + top: 0; + border: 0; +} +html.e #files tbody tr td, +html.e #files thead th { + border-radius: var(--radius); +} +#files td { + background: var(--w2); +} +html.e #files tr { + background-color: var(--black); +} +html.e #srv_info span, +html.e label { + color: var(--btn-fg) !important; +} +html.e #acc_info { + background: var(--transparent); + color: var(--white); + height: 2em; + left: 1em; + width: fit-content; +} +html.e #acc_info, +html.e #ops, +html.e #srv_info { + display: flex; + align-items: center; +} +html.e #flogout:before { + padding-left: 0.2em; + padding-right: 0.4em; + content: " | "; +} +html.e #blogout { + color: var(--w); + box-shadow: none; + background: transparent; +} +html.e .opwide > div { + border-left: 1px solid var(--fg); +} +html.e #srv_info { + background: var(--transparent); + color: var(--white); + height: fit-content; + top: 3.2em; + left: 1em; + gap: 0.2em; +} +html.e #u2cards a.act { + padding: 0.2em 1em; +} +html.e #u2btn { + border: var(--border-dashed-black); + border-radius: var(--border-radius); + transform: translateY(30%); +} +html.e #ops, +html.e #ops a { + border-radius: var(--radius); +} +@media only screen and (max-width: 600px) { + html.e #acc_info { + background: var(--transparent); + color: var(--white); + height: fit-content; + align-items: center; + top: 3.2em; + right: 1em; + left: auto; + display: flex; + gap: 0.2em; + } + html.e #u2btn { + transform: none; + } +} +html.e #ops { + background: var(--ttlbar); + /*HC*/ + box-shadow: inset 0-1px grey, inset 0-2px var(--shadow-color-1); + height: 2em; + gap: 0.6em; + padding: 0.2em; + flex-direction: row-reverse; + margin-bottom: 1.2em; +} +html.e #srch_form, +html.e .opbox { + padding-bottom: 1em; + padding-top: 1em; + max-width: 100vw; +} +html.e #ghead, +html.e #ops a { + align-items: center; + display: flex; +} +html.e #ops a { + text-shadow: 1px 1px 0 rgba(0, 0, 0, 0.5); + height: 1.4em; + padding: 0; + box-shadow: var(--shadow-outset); + background: var(--bg); + aspect-ratio: 1/1; + justify-content: center; + font-size: 1.25em; + z-index: 4; +} +html.e #blogout:focus, +html.e #ops a:focus { + outline: 1px dashed var(--w) !important; +} + +html.e #blogout:hover { + text-decoration: underline; +} + +html.e #ops > a:not(:first-child).act { + height: 1.4em; + width: 1.4em; + padding-bottom: 0.3em; + margin-top: 0.3em; + border-top-left-radius: 3px; + border-top-right-radius: 3px; + box-shadow: var(--shadow-inset-left), var(--shadow-inset-top), + var(--shadow-inset-right); + z-index: 6; +} +html.e #ops a.act { + box-shadow: var(--shadow-inset); + border-bottom: 0; +} +html.e a:active { + border: 0; +} +html.e :focus, +html.e :focus + label { + border: 0 !important; + outline-offset: 1px; + border-radius: var(--radius) !important; + box-shadow: inherit; +} +html.e #opa_x { + text-shadow: 0 0 0 var(--transparent) !important; + color: var(--bg) !important; + display: flex; +} +html.e #opa_x:before { + content: "⨯"; + color: var(--fg) !important; + margin-top: -0.1em; + font-size: 1.75em; + position: absolute; +} +html.e .opbox { + margin: -1.2em 0 0; + box-shadow: var(--shadow-inset-bottom), var(--shadow-inset-left), + var(--shadow-inset-right); + border-radius: var(--radius); + z-index: 5; + background: var(--bg); +} +html.e #srch_form { + margin: 0; + border-radius: var(--radius); +} +html.e #op_unpost { + max-width: 100vw; + margin: 0; +} +html.e label:focus { + box-shadow: 0 0; +} +html.e #tree { + box-shadow: none; + padding-right: 5px; +} +html.e #tt { + background: var(--w2); +} +html.e .mdo a { + background: 0 0; + text-decoration: underline; +} +html.e .mdo code, +html.e .mdo pre { + color: var(--white); + background: var(--w2); + border: 0; +} +html.e .mdo h1, +html.e .mdo h2 { + background: 0 0; + border-color: var(--w2); +} +html.e #tt, +html.e .mdo ol ol, +html.e .mdo ol ul, +html.e .mdo ul ol, +html.e .mdo ul ul { + border-color: var(--w2); +} +html.e .mdo li > em, +html.e .mdo p > em, +html.e .mdo td > em { + color: #fd0; +} +html.e input.txtbox, +html.e input[type="text"], +html.e select { + background-color: var(--txt-bg); + box-shadow: var(--shadow-input) !important; + box-sizing: border-box; + padding: 3px 4px; + border-radius: var(--radius); + border: 0; +} +html.e #gfiles { + box-shadow: var(--shadow-outset); + background: var(--bg); + padding: 0.4em; + display: flex; + flex-direction: column; + gap: 0.3em; +} +html.e #ggrid { + background-color: var(--inset-bg); + box-shadow: var(--shadow-input); + padding: 1.5em; + margin: 0; + overflow-x: scroll; +} +html.e #ghead { + margin: 0; + justify-content: flex-end; + gap: 0.4em; + padding: 0; + overflow: auto; + top: 0px; + border-radius: 0px; +} +html.e #ghead a { + margin: 0; + border-radius: var(--radius); +} +html.e ::-webkit-scrollbar, +html.e::-webkit-scrollbar { + width: 16px !important; + height: 16px !important; + background: var(--transparent) !important; +} +html.e ::-webkit-scrollbar-button, +html.e ::-webkit-scrollbar-thumb, +html.e::-webkit-scrollbar-button, +html.e::-webkit-scrollbar-thumb { + width: 16px !important; + height: 16px !important; + background: var(--scroll) !important; + /*HC*/ + box-shadow: var(--shadow-outset); + border: 1px solid !important; + border-color: var(--silver) var(--black) var(--black) var(--silver) !important; +} +html.e ::-webkit-scrollbar-track, +html.e::-webkit-scrollbar-track { + image-rendering: optimize-contrast !important; + background-image: url(data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgLTAuNSAyIDIiIHNoYXBlLXJlbmRlcmluZz0iY3Jpc3BFZGdlcyI+CjxtZXRhZGF0YT5NYWRlIHdpdGggUGl4ZWxzIHRvIFN2ZyBodHRwczovL2NvZGVwZW4uaW8vc2hzaGF3L3Blbi9YYnh2Tmo8L21ldGFkYXRhPgo8cGF0aCBzdHJva2U9IiNjMGMwYzAiIGQ9Ik0wIDBoMU0xIDFoMSIgLz4KPC9zdmc+) !important; + background-position: 0 0 !important; + background-repeat: repeat !important; + background-size: 2px !important; + background: var(--scroll-bkg); +} +#tree::-webkit-scrollbar, +#tree::-webkit-scrollbar-track { + background: var(--scroll-bkg); +} +html.e ::-webkit-scrollbar-button, +html.e::-webkit-scrollbar-button { + background-repeat: no-repeat !important; + background-size: 16px !important; +} +html.e ::-webkit-scrollbar-button:single-button:vertical:decrement, +html.e::-webkit-scrollbar-button:single-button:vertical:decrement { + background-image: url(data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgLTAuNSAxNiAxNiIgc2hhcGUtcmVuZGVyaW5nPSJjcmlzcEVkZ2VzIj4KPG1ldGFkYXRhPk1hZGUgd2l0aCBQaXhlbHMgdG8gU3ZnIGh0dHBzOi8vY29kZXBlbi5pby9zaHNoYXcvcGVuL1hieHZOajwvbWV0YWRhdGE+CjxwYXRoIHN0cm9rZT0iIzAwMDAwMCIgZD0iTTcgNWgxTTYgNmgzTTUgN2g1TTQgOGg3IiAvPgo8L3N2Zz4=) !important; +} +html.e ::-webkit-scrollbar-button:single-button:vertical:increment, +html.e::-webkit-scrollbar-button:single-button:vertical:increment { + background-image: url(data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgLTAuNSAxNiAxNiIgc2hhcGUtcmVuZGVyaW5nPSJjcmlzcEVkZ2VzIj4KPG1ldGFkYXRhPk1hZGUgd2l0aCBQaXhlbHMgdG8gU3ZnIGh0dHBzOi8vY29kZXBlbi5pby9zaHNoYXcvcGVuL1hieHZOajwvbWV0YWRhdGE+CjxwYXRoIHN0cm9rZT0iIzAwMDAwMCIgZD0iTTQgNWg3TTUgNmg1TTYgN2gzTTcgOGgxIiAvPgo8L3N2Zz4=) !important; +} +html.e ::-webkit-scrollbar-button:single-button:horizontal:decrement, +html.e::-webkit-scrollbar-button:single-button:horizontal:decrement { + background-image: url(data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgLTAuNSAxNiAxNiIgc2hhcGUtcmVuZGVyaW5nPSJjcmlzcEVkZ2VzIj4KPG1ldGFkYXRhPk1hZGUgd2l0aCBQaXhlbHMgdG8gU3ZnIGh0dHBzOi8vY29kZXBlbi5pby9zaHNoYXcvcGVuL1hieHZOajwvbWV0YWRhdGE+CjxwYXRoIHN0cm9rZT0iIzAwMDAwMCIgZD0iTTggM2gxTTcgNGgyTTYgNWgzTTUgNmg0TTYgN2gzTTcgOGgyTTggOWgxIiAvPgo8L3N2Zz4=) !important; +} +html.e ::-webkit-scrollbar-button:single-button:horizontal:increment, +html.e::-webkit-scrollbar-button:single-button:horizontal:increment { + background-image: url(data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgLTAuNSAxNiAxNiIgc2hhcGUtcmVuZGVyaW5nPSJjcmlzcEVkZ2VzIj4KPG1ldGFkYXRhPk1hZGUgd2l0aCBQaXhlbHMgdG8gU3ZnIGh0dHBzOi8vY29kZXBlbi5pby9zaHNoYXcvcGVuL1hieHZOajwvbWV0YWRhdGE+CjxwYXRoIHN0cm9rZT0iIzAwMDAwMCIgZD0iTTYgM2gxTTYgNGgyTTYgNWgzTTYgNmg0TTYgN2gzTTYgOGgyTTYgOWgxIiAvPgo8L3N2Zz4=) !important; +} +html.e ::-webkit-scrollbar-corner, +html.e::-webkit-scrollbar-corner { + background: var(--silver) !important; +} +html, +html.e #tree { + scrollbar-color: inherit !important; +} +html.e #tree { + background: var(--bg); + padding-left: 0.4em; + padding-top: 0; + margin-left: var(--negative-space); +} +html.e.noscroll #tree { + /*HC*/ + box-shadow: 1px 1px var(--grey), 2px 2px var(--shadow-color-1), + var(--shadow-outset-bottom); +} +html.e #treeh { + background: var(--bg); + box-shadow: var(--shadow-outset-top), var(--shadow-outset-bottom); + width: calc(1.5em + var(--nav-sz) - var(--sbw)); + height: 2.4em; + border: none; + top: -2px; + display: flex; + align-items: center; + gap: 0.6em; +} +html.e #treeh .btn { + margin: 0px; + top: auto; +} +html.e #tree ul { + border-left: var(--border-dashed-black); + margin-left: 2.15em; +} +html.e .ntree a:first-child { + font-family: scp, monospace, monospace; + font-size: 1.2em; + line-height: 0; + background: var(--inset-bg); + aspect-ratio: 1/1; + text-align: center; + align-content: center; + border-radius: var(--radius) !important; + padding: 0.057em; + border: 1px solid var(--black); +} +html.e .ntree a:first-child:after { + content: "."; + position: absolute; + border-top: var(--border-dashed-black); + color: var(--transparent); + font-size: 0.9em; + margin-left: 0.13em; +} +html.e #treeul { + border: 0 !important; + position: static; + margin: 0 !important; + min-height: 100%; + height: max-content; +} +html.e .ntree a:last-of-type:before { + content: "📁"; + margin-left: 0.3em; +} +html.e .ntree { + padding-left: 1em !important; + padding-top: 0.3em !important; + background: var(--inset-bg); + box-shadow: var(--shadow-inset-left), var(--shadow-inset-bottom); +} +html.e #tree li { + margin-left: -0.5em; + border-top: 0; +} +html.e .ntree a:hover { + outline-offset: -2px; + color: var(--fg); + border-radius: var(--radius) !important; +} +html.e #treepar { + width: calc(-1em + var(--nav-sz) - var(--sbw)); + overflow: hidden; + left: -0.7em; + box-shadow: var(--shadow-inset-left), var(--shadow-inset-top); + border-left: 0 !important; + border-bottom: var(--border-dashed-black); + margin-left: calc(2.1em - (1em - var(--negative-space))) !important; +} +html.e #path, +html.e #widgeti, +html.e #wtoggle, +html.e #wtoggle a, +html.e #files, +html.e #files thead th, +html.e #ghead a, +html.e #tree { + box-shadow: var(--shadow-outset); +} +html.e.noscroll #treepar { + width: calc(var(--nav-sz) - 1em); +} +html.e #docul { + border-left: 0 !important; + margin-left: 0 !important; +} +html.e #wrap { + transform: translateX(calc((var(--negative-space) * 2) - 1.2em)); + padding-right: var(--negative-space); + position: relative; + margin-right: calc((var(--negative-space) * 2) - 1.2em); + margin-top: var(--negative-space); + margin-left: 1.2em; + /*overflow-x: auto; fix for OOB table when screen space is limited (mobile), but removes sticky header*/ +} +html.e input[type="radio"] { + accent-color: #232323; +} +html.e #path { + width: calc(100% - 0.4em); + display: flex; + align-items: center; + margin: 0; + padding: 0.2em; + overflow-x: auto; +} +html.e #path i { + border: 1px solid var(--w); + border-color: var(--w); + margin: 0; + border-width: 0.1em 0.1em 0 0; + height: 0.5em; + width: 0.5em; +}/* +html.e #hovertree:after { + color: red; + content: "BUGGY"; +html.ez #hovertree:after { + color: rgb(255 98 98); + content: "BUGGY"; +} +}*/ +html.e #widget { + box-shadow: 0 0; + border: 0 !important; +} +html.e #wtico, +html.e #zip1 { + box-shadow: 0 0 !important; +} +html.e #wtgrid { + top: -0.09em; +} +html.e #wfs, +html.e #wm3u, +html.e #wnp, +html.e #wzip { + border-width: 0 1px 0 0; +} +html.e #wfm.act + #wzip1 + #wzip, +html.e #wfm.act + #wzip1 + #wzip + #wnp { + border-left-width: 1px; +} +html.e #barpos { + /* border-radius: var(--radius); */ + box-shadow: var(--shadow-inset); +} +html.e #goh + span { + border-left: 0.1em solid var(--bg-u5); +} +html.e #wfp { + margin: var(--negative-space); + font-size: 0; + display: inline-block; +} +html.e #wfp a { + font-size: large; + display: inline-block; +} +html.e #repl { + font-size: large; + padding: 0.33em; + right: calc(var(--negative-space) * 0.89); + position: absolute; +} +html.e #epi { + text-align: center; + text-wrap-mode: nowrap; + margin: 0px; +} + +html.e #epi.logue:not(.mdo) { + padding: 0.8em; + box-shadow: var(--shadow-outset); +} + +html.e #epi.logue.mdo { + padding-left: 3px; +} + +html.e #doc { + box-shadow: var(--shadow-inset); + background: var(--inset-bg); + margin: 0.2em; + border-radius: var(--radius); +} + +html.e #detree { + padding: 0px; +} diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 9739e55a..9734090d 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -16746,7 +16746,7 @@ var settheme = (function () { var html = [], cb = ebi('themes'), itheme = ax.indexOf(theme[0]) * 2 + (light ? 1 : 0), - names = ['classic dark', 'classic light', 'pm-monokai', 'flat light', 'vice', 'hotdog stand', 'hacker', 'hi-con']; + names = ['classic dark', 'classic light', 'pm-monokai', 'flat light', 'vice', 'hotdog stand', 'hacker', 'hi-con', 'win95 dark', 'win95']; for (var a = 0; a < themes; a++) html.push('<option value="{0}">{0} ┃ {1}</option>'.format(a, names[a] || 'custom')); From 69d9878acd06b8a9423512580e843dc73247a2f3 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Tue, 12 Aug 2025 19:03:34 +0000 Subject: [PATCH 150/174] use the good name --- copyparty/web/browser.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 9734090d..6c79f3ef 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -16746,7 +16746,7 @@ var settheme = (function () { var html = [], cb = ebi('themes'), itheme = ax.indexOf(theme[0]) * 2 + (light ? 1 : 0), - names = ['classic dark', 'classic light', 'pm-monokai', 'flat light', 'vice', 'hotdog stand', 'hacker', 'hi-con', 'win95 dark', 'win95']; + names = ['classic dark', 'classic light', 'pm-monokai', 'flat light', 'vice', 'hotdog stand', 'hacker', 'hi-con', 'phi95 dark', 'phi95']; for (var a = 0; a < themes; a++) html.push('<option value="{0}">{0} ┃ {1}</option>'.format(a, names[a] || 'custom')); From c32a672a6891961a7c61d2c6d38bfaa99716397e Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Tue, 12 Aug 2025 19:34:44 +0000 Subject: [PATCH 151/174] fix transcoding tooltips; closes #580 --- copyparty/web/browser.js | 93 +++++++++++++++++++++------------------- 1 file changed, 48 insertions(+), 45 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 6c79f3ef..a0a1a21c 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -300,9 +300,9 @@ var Ls = { "mt_mloop": "loop the open folder\">🔁 loop", "mt_mnext": "load the next folder and continue\">📂 next", "mt_mstop": "stop playback\">⏸ stop", - "mt_cflac": "convert flac / wav to opus\">flac", - "mt_caac": "convert aac / m4a to opus\">aac", - "mt_coth": "convert all others (not mp3) to opus\">oth", + "mt_cflac": "convert flac / wav to {0}\">flac", + "mt_caac": "convert aac / m4a to {0}\">aac", + "mt_coth": "convert all others (not mp3) to {0}\">oth", "mt_c2opus": "best choice for desktops, laptops, android\">opus", "mt_c2owa": "opus-weba, for iOS 17.5 and newer\">owa", "mt_c2caf": "opus-caf, for iOS 11 through 17\">caf", @@ -930,9 +930,9 @@ var Ls = { "mt_mloop": "repeter hele mappen\">🔁 gjenta", "mt_mnext": "hopp til neste mappe og fortsett\">📂 neste", "mt_mstop": "stopp avspilling\">⏸ stopp", - "mt_cflac": "konverter flac / wav-filer til opus\">flac", - "mt_caac": "konverter aac / m4a-filer til to opus\">aac", - "mt_coth": "konverter alt annet (men ikke mp3) til opus\">andre", + "mt_cflac": "konverter flac / wav-filer til {0}\">flac", + "mt_caac": "konverter aac / m4a-filer til to {0}\">aac", + "mt_coth": "konverter alt annet (men ikke mp3) til {0}\">andre", "mt_c2opus": "det beste valget for alle PCer og Android\">opus", "mt_c2owa": "opus-weba, for iOS 17.5 og nyere\">owa", "mt_c2caf": "opus-caf, for iOS 11 tilogmed 17\">caf", @@ -1559,9 +1559,9 @@ var Ls = { "mt_mloop": "循环打开的文件夹\">🔁 循环", "mt_mnext": "加载下一个文件夹并继续\">📂 下一首", "mt_mstop": "停止播放\">⏸ 停止", //m - "mt_cflac": "将 flac / wav 转换为 opus\">flac", - "mt_caac": "将 aac / m4a 转换为 opus\">aac", - "mt_coth": "将所有其他(不是 mp3)转换为 opus\">oth", + "mt_cflac": "将 flac / wav 转换为 {0}\">flac", + "mt_caac": "将 aac / m4a 转换为 {0}\">aac", + "mt_coth": "将所有其他(不是 mp3)转换为 {0}\">oth", "mt_c2opus": "适合桌面电脑、笔记本电脑和安卓设备的最佳选择\">opus", //m "mt_c2owa": "opus-weba(适用于 iOS 17.5 及更新版本)\">owa", //m "mt_c2caf": "opus-caf(适用于 iOS 11 到 iOS 17)\">caf", //m @@ -2192,9 +2192,9 @@ var Ls = { "mt_mloop": "opakovat otevřenou složku\">🔁 loop", "mt_mnext": "načíst další složku a pokračovat\">📂 next", "mt_mstop": "zastavit přehrávání\">⏸ stop", - "mt_cflac": "převést flac / wav na opus\">flac", - "mt_caac": "převést aac / m4a na opus\">aac", - "mt_coth": "převést všechny ostatní (ne mp3) na opus\">oth", + "mt_cflac": "převést flac / wav na {0}\">flac", + "mt_caac": "převést aac / m4a na {0}\">aac", + "mt_coth": "převést všechny ostatní (ne mp3) na {0}\">oth", "mt_c2opus": "nejlepší volba pro desktopy, laptopy, android\">opus", "mt_c2owa": "opus-weba, pro iOS 17.5 a novější\">owa", "mt_c2caf": "opus-caf, pro iOS 11 až 17\">caf", @@ -2821,9 +2821,9 @@ var Ls = { "mt_mloop": "offenen Ordner wiederholen\">🔁 Schleife", "mt_mnext": "nächsten Ordner laden und fortfahren\">📂 nächster", "mt_mstop": "Wiedergabe beenden\">⏸ Stop", - "mt_cflac": "FLAC / WAV zu OPUS konvertierebn\">flac", - "mt_caac": "AAC / M4A zu OPUS konvertieren\">aac", - "mt_coth": "Convertiere alle Dateien (die nicht MP3 sind) zu OPUS\">oth", + "mt_cflac": "FLAC / WAV zu {0} konvertierebn\">flac", + "mt_caac": "AAC / M4A zu {0} konvertieren\">aac", + "mt_coth": "Convertiere alle Dateien (die nicht MP3 sind) zu {0}\">oth", "mt_c2opus": "Beste Wahl für Desktops, Laptops, Android\">opus", "mt_c2owa": "opus-weba, für iOS 17.5 und neuer\">owa", "mt_c2caf": "opus-caf, für iOS 11 bis 17\">caf", @@ -3450,9 +3450,9 @@ var Ls = { "mt_mloop": "toista avoinna olevaa hakemistoa loputtomasti\">🔁 alkuun", "mt_mnext": "lataa seuraava hakemisto ja jatka\">📂 seuraava", "mt_mstop": "pysäytä toisto\">⏸ pysäytä", - "mt_cflac": "muunna flac / wav opus-muotoon\">flac", - "mt_caac": "muunna aac / m4a opus-muotoon\">aac", - "mt_coth": "muunna kaikki muut paitsi mp3 opus-muotoon\">muut", + "mt_cflac": "muunna flac / wav {0}-muotoon\">flac", + "mt_caac": "muunna aac / m4a {0}-muotoon\">aac", + "mt_coth": "muunna kaikki muut paitsi mp3 {0}-muotoon\">muut", "mt_c2opus": "paras valinta pöytäkoneille, kannettaville, androidille\">opus", "mt_c2owa": "opus-weba, iOS 17.5:lle ja uudemmille\">owa", "mt_c2caf": "opus-caf, iOS 11:lle - 17:lle\">caf", @@ -4079,9 +4079,9 @@ var Ls = { "mt_mloop": "lire en boucle le dossier ouvert\">🔁 loop", "mt_mnext": "charger le dossier suivant et continuer\">📂 next", "mt_mstop": "arrêter la lecture\">⏸ stop", - "mt_cflac": "convertir flac / wav en opus\">flac", - "mt_caac": "convertir aac / m4a en opus\">aac", - "mt_coth": "convertir tout les autres (pas mp3) en opus\">oth", + "mt_cflac": "convertir flac / wav en {0}\">flac", + "mt_caac": "convertir aac / m4a en {0}\">aac", + "mt_coth": "convertir tout les autres (pas mp3) en {0}\">oth", "mt_c2opus": "meilleur choix pour PC fixe, PC portable, android\">opus", "mt_c2owa": "opus-weba, pour iOS 17.5 et supérieur\">owa", "mt_c2caf": "opus-caf, pour iOS 11 à 17\">caf", @@ -4708,9 +4708,9 @@ var Ls = { "mt_mloop": "τυχαία αναπαραγωγή στον ανοικτό φάκελο\">🔁 τυχαία αναπαραγωγή", "mt_mnext": "φόρτωση επόμενου φακέλου και συνέχιση\">📂 επόμενο", "mt_mstop": "σταμάτησε την αναπαραγωγή\">⏸ σταμάτημα", - "mt_cflac": "μετατροπή flac / wav σε opus\">flac", - "mt_caac": "μετατροπή aac / m4a σε opus\">aac", - "mt_coth": "μετατροπή όλων των άλλων (εκτός των mp3) σε opus\">άλλο", + "mt_cflac": "μετατροπή flac / wav σε {0}\">flac", + "mt_caac": "μετατροπή aac / m4a σε {0}\">aac", + "mt_coth": "μετατροπή όλων των άλλων (εκτός των mp3) σε {0}\">άλλο", "mt_c2opus": "καλύτερη επιλογή για desktop, laptop, android\">opus", "mt_c2owa": "opus-weba, για iOS 17.5 και νεότερα\">owa", "mt_c2caf": "opus-caf, για iOS 11 έως 17\">caf", @@ -5337,9 +5337,9 @@ var Ls = { "mt_mloop": "loop della cartella aperta\">🔁 loop", "mt_mnext": "carica la prossima cartella e continua\">📂 succ", "mt_mstop": "ferma riproduzione\">⏸ stop", - "mt_cflac": "converti flac / wav in opus\">flac", - "mt_caac": "converti aac / m4a in opus\">aac", - "mt_coth": "converti tutti gli altri (non mp3) in opus\">oth", + "mt_cflac": "converti flac / wav in {0}\">flac", + "mt_caac": "converti aac / m4a in {0}\">aac", + "mt_coth": "converti tutti gli altri (non mp3) in {0}\">oth", "mt_c2opus": "scelta migliore per desktop, laptop, android\">opus", "mt_c2owa": "opus-weba, per iOS 17.5 e più recenti\">owa", "mt_c2caf": "opus-caf, per iOS 11 fino a 17\">caf", @@ -5966,9 +5966,9 @@ var Ls = { "mt_mloop": "De open map herhalen\">🔁 loop", "mt_mnext": "Laad de volgende map en ga verder\">📂 next", "mt_mstop": "Stoppen met afspelen\">⏸ stop", - "mt_cflac": "flac / wav omzetten naar opus\">flac", - "mt_caac": "aac / m4a omzetten naar opus\">aac", - "mt_coth": "Alle andere bestanden (geen mp3) converteren naar opus\">oth", + "mt_cflac": "flac / wav omzetten naar {0}\">flac", + "mt_caac": "aac / m4a omzetten naar {0}\">aac", + "mt_coth": "Alle andere bestanden (geen mp3) converteren naar {0}\">oth", "mt_c2opus": "Beste keuze voor computers, laptops, android\">opus", "mt_c2owa": "opus-weba, voor iOS 17.5 en nieuwer\">owa", "mt_c2caf": "opus-caf, voor iOS 11 tot en met iOS 17\">caf", @@ -6596,9 +6596,9 @@ var Ls = { "mt_mloop": "repetér heile mappa\">🔁 gjenta", "mt_mnext": "hopp åt neste mappe og fortsett\">📂 neste", "mt_mstop": "stopp avspeling\">⏸ stopp", - "mt_cflac": "konvertér flac / wav-filer åt opus\">flac", - "mt_caac": "konvertér aac / m4a-filer åt to opus\">aac", - "mt_coth": "konvertér alt anna (men ikkje mp3) åt opus\">andre", + "mt_cflac": "konvertér flac / wav-filer åt {0}\">flac", + "mt_caac": "konvertér aac / m4a-filer åt to {0}\">aac", + "mt_coth": "konvertér alt anna (men ikkje mp3) åt {0}\">andre", "mt_c2opus": "det beste valget for alle PCar og Android\">opus", "mt_c2owa": "opus-weba, for iOS 17.5 og nyare\">owa", "mt_c2caf": "opus-caf, for iOS 11 åt og med 17\">caf", @@ -7224,9 +7224,9 @@ var Ls = { "mt_mloop": "odtwarzaj utwory w folderze w pętli\">🔁 loop", "mt_mnext": "wczytaj następny folder i kontynuuj\">📂 next", "mt_mstop": "zatrzymaj odtwarzanie\">⏸ stop", - "mt_cflac": "przekonwertuj format flac / wav na opus\">flac", - "mt_caac": "przekonwertuj format aac / m4a na opus\">aac", - "mt_coth": "przekonwertuj wszystkie inne formaty (nie będące mp3) na opus\">oth", + "mt_cflac": "przekonwertuj format flac / wav na {0}\">flac", + "mt_caac": "przekonwertuj format aac / m4a na {0}\">aac", + "mt_coth": "przekonwertuj wszystkie inne formaty (nie będące mp3) na {0}\">oth", "mt_c2opus": "najlepszy wybór dla komputerów, laptopów i urządzeń z androidem\">opus", "mt_c2owa": "opus-weba, dla iOS 17.5 i nowszych\">owa", "mt_c2caf": "opus-caf, dla iOS 11 do 17\">caf", @@ -7851,9 +7851,9 @@ var Ls = { "mt_mloop": "повторять треки в папке\">🔁 цикл", "mt_mnext": "загрузить следующую папку и продолжить в ней\">📂 след.", "mt_mstop": "приостановить воспроизведение\">⏸ стоп", - "mt_cflac": "конвертировать flac / wav в opus\">flac", - "mt_caac": "конвертировать aac / m4a в opus\">aac", - "mt_coth": "конвертировать всё остальное (кроме mp3) в opus\">др.", + "mt_cflac": "конвертировать flac / wav в {0}\">flac", + "mt_caac": "конвертировать aac / m4a в {0}\">aac", + "mt_coth": "конвертировать всё остальное (кроме mp3) в {0}\">др.", "mt_c2opus": "лучший вариант для компьютеров и устройств на Android\">opus", "mt_c2owa": "opus-weba, для iOS 17.5 и выше\">owa", "mt_c2caf": "opus-caf, для iOS 11-17\">caf", @@ -8479,9 +8479,9 @@ var Ls = { "mt_mloop": 'repetir la carpeta actual">🔁 bucle', "mt_mnext": 'cargar la siguiente carpeta y continuar">📂 sig', "mt_mstop": 'detener reproducción">⏸ parar', - "mt_cflac": 'convertir flac / wav a opus">flac', - "mt_caac": 'convertir aac / m4a a opus">aac', - "mt_coth": 'convertir todos los demás (no mp3) a opus">oth', + "mt_cflac": 'convertir flac / wav a {0}">flac', + "mt_caac": 'convertir aac / m4a a {0}">aac', + "mt_coth": 'convertir todos los demás (no mp3) a {0}">oth', "mt_c2opus": 'la mejor opción para ordenadores, portátiles, android">opus', "mt_c2owa": 'opus-weba, para iOS 17.5 y superior">owa', "mt_c2caf": 'opus-caf, para iOS 11 a 17">caf', @@ -9108,9 +9108,9 @@ var Ls = { "mt_mloop": "зациклити відкриту папку\">🔁 loop", "mt_mnext": "завантажити наступну папку і продовжити\">📂 next", "mt_mstop": "зупинити відтворення\">⏸ stop", - "mt_cflac": "конвертувати flac / wav в opus\">flac", - "mt_caac": "конвертувати aac / m4a в opus\">aac", - "mt_coth": "конвертувати всі інші (не mp3) в opus\">oth", + "mt_cflac": "конвертувати flac / wav в {0}\">flac", + "mt_caac": "конвертувати aac / m4a в {0}\">aac", + "mt_coth": "конвертувати всі інші (не mp3) в {0}\">oth", "mt_c2opus": "найкращий вибір для робочих столів, ноутбуків, android\">opus", "mt_c2owa": "opus-weba, для iOS 17.5 і новіших\">owa", "mt_c2caf": "opus-caf, для iOS 11 до 17\">caf", @@ -10157,6 +10157,9 @@ var mpl = (function () { clmod(btns[a], 'on', fmts[a] == v) r.ac2 = v; + ebi('ac_flac').setAttribute('tt', L.mt_cflac.split('"')[0].format(v)); + ebi('ac_aac').setAttribute('tt', L.mt_caac.split('"')[0].format(v)); + ebi('ac_oth').setAttribute('tt', L.mt_coth.split('"')[0].format(v)); }; r.pp = function () { From c9fd608732914002b14e0961bf216f26cb46ca46 Mon Sep 17 00:00:00 2001 From: Danny Piper <djpiper28@gmail.com> Date: Tue, 12 Aug 2025 20:41:16 +0100 Subject: [PATCH 152/174] feat: added cr3 to raw formats list Signed-off-by: Danny Piper <djpiper28@gmail.com> --- copyparty/__main__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 97392c87..ddfdd8a2 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1447,7 +1447,7 @@ def add_thumbnail(ap): # ffmpeg -hide_banner -demuxers | awk '/^ D /{print$2}' | while IFS= read -r x; do ffmpeg -hide_banner -h demuxer=$x; done | grep -E '^Demuxer |extensions:' ap2.add_argument("--th-r-pil", metavar="T,T", type=u, default="avif,avifs,blp,bmp,cbz,dcx,dds,dib,emf,eps,epub,fits,flc,fli,fpx,gif,heic,heics,heif,heifs,icns,ico,im,j2p,j2k,jp2,jpeg,jpg,jpx,pbm,pcx,pgm,png,pnm,ppm,psd,qoi,sgi,spi,tga,tif,tiff,webp,wmf,xbm,xpm", help="image formats to decode using pillow") ap2.add_argument("--th-r-vips", metavar="T,T", type=u, default="avif,exr,fit,fits,fts,gif,hdr,heic,jp2,jpeg,jpg,jpx,jxl,nii,pfm,pgm,png,ppm,svg,tif,tiff,webp", help="image formats to decode using pyvips") - ap2.add_argument("--th-r-raw", metavar="T,T", type=u, default="arw,cr2,crw,dcr,dng,erf,k25,kdc,mrw,nef,orf,pef,raf,raw,sr2,srf,x3f", help="image formats to decode using rawpy") + ap2.add_argument("--th-r-raw", metavar="T,T", type=u, default="arw,cr2,cr3,crw,dcr,dng,erf,k25,kdc,mrw,nef,orf,pef,raf,raw,sr2,srf,x3f", help="image formats to decode using rawpy") ap2.add_argument("--th-r-ffi", metavar="T,T", type=u, default="apng,avif,avifs,bmp,cbz,dds,dib,epub,fit,fits,fts,gif,hdr,heic,heics,heif,heifs,icns,ico,jp2,jpeg,jpg,jpx,jxl,pbm,pcx,pfm,pgm,png,pnm,ppm,psd,qoi,sgi,tga,tif,tiff,webp,xbm,xpm", help="image formats to decode using ffmpeg") ap2.add_argument("--th-r-ffv", metavar="T,T", type=u, default="3gp,asf,av1,avc,avi,flv,h264,h265,hevc,m4v,mjpeg,mjpg,mkv,mov,mp4,mpeg,mpeg2,mpegts,mpg,mpg2,mts,nut,ogm,ogv,rm,ts,vob,webm,wmv", help="video formats to decode using ffmpeg") ap2.add_argument("--th-r-ffa", metavar="T,T", type=u, default="aac,ac3,aif,aiff,alac,alaw,amr,apac,ape,au,bonk,dfpwm,dts,flac,gsm,ilbc,it,itgz,itxz,itz,m4a,mdgz,mdxz,mdz,mo3,mod,mp2,mp3,mpc,mptm,mt2,mulaw,oga,ogg,okt,opus,ra,s3m,s3gz,s3xz,s3z,tak,tta,ulaw,wav,wma,wv,xm,xmgz,xmxz,xmz,xpk", help="audio formats to decode using ffmpeg") From 715d374ee4d849ef4e7da2aec84f68f5dad87458 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Tue, 12 Aug 2025 21:46:42 +0000 Subject: [PATCH 153/174] button to abort copy/move; closes #572 --- copyparty/__main__.py | 1 + copyparty/httpcli.py | 20 +++++++++++++++-- copyparty/svchub.py | 2 +- copyparty/up2k.py | 12 ++++++++-- copyparty/web/browser.css | 5 +++++ copyparty/web/browser.js | 46 +++++++++++++++++++++++++++++++++++---- tests/util.py | 2 +- 7 files changed, 78 insertions(+), 10 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index ddfdd8a2..86d8cdd5 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1319,6 +1319,7 @@ def add_optouts(ap): ap2.add_argument("--no-del", action="store_true", help="disable delete operations") ap2.add_argument("--no-mv", action="store_true", help="disable move/rename operations") ap2.add_argument("--no-cp", action="store_true", help="disable copy operations") + ap2.add_argument("--no-fs-abrt", action="store_true", help="disable ability to abort ongoing copy/move") ap2.add_argument("-nth", action="store_true", help="no title hostname; don't show \033[33m--name\033[0m in <title>") ap2.add_argument("-nih", action="store_true", help="no info hostname -- don't show in UI") ap2.add_argument("-nid", action="store_true", help="no info disk-usage -- don't show in UI") diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 72bff45e..a7ff303d 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -1987,6 +1987,9 @@ class HttpCli(object): if "eshare" in self.uparam: return self.handle_eshare() + if "fs_abrt" in self.uparam: + return self.handle_fs_abrt() + if "application/octet-stream" in ctype: return self.handle_post_binary() @@ -5958,7 +5961,9 @@ class HttpCli(object): self.asrv.vfs.get(vdst, self.uname, False, True, False, True) wunlink(self.log, dabs, dvn.flags) - x = self.conn.hsrv.broker.ask("up2k.handle_mv", self.uname, self.ip, vsrc, vdst) + x = self.conn.hsrv.broker.ask( + "up2k.handle_mv", self.ouparam.get("akey"), self.uname, self.ip, vsrc, vdst + ) self.loud_reply(x.get(), status=201) return True @@ -5988,10 +5993,21 @@ class HttpCli(object): self.asrv.vfs.get(vdst, self.uname, False, True, False, True) wunlink(self.log, dabs, dvn.flags) - x = self.conn.hsrv.broker.ask("up2k.handle_cp", self.uname, self.ip, vsrc, vdst) + x = self.conn.hsrv.broker.ask( + "up2k.handle_cp", self.ouparam.get("akey"), self.uname, self.ip, vsrc, vdst + ) self.loud_reply(x.get(), status=201) return True + def handle_fs_abrt(self): + if self.args.no_fs_abrt: + t = "aborting an ongoing copy/move is disabled in server config" + raise Pebkac(403, t) + + self.conn.hsrv.broker.say("up2k.handle_fs_abrt", self.uparam["fs_abrt"]) + self.loud_reply("aborting", status=200) + return True + def tx_ls(self, ls: dict[str, Any]) -> bool: dirs = ls["dirs"] files = ls["files"] diff --git a/copyparty/svchub.py b/copyparty/svchub.py index a6f87a85..950122d0 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -39,9 +39,9 @@ from .th_srv import ( HAVE_FFPROBE, HAVE_HEIF, HAVE_PIL, + HAVE_RAW, HAVE_VIPS, HAVE_WEBP, - HAVE_RAW, ThumbSrv, ) from .up2k import Up2k diff --git a/copyparty/up2k.py b/copyparty/up2k.py index b233cecb..993d5b82 100644 --- a/copyparty/up2k.py +++ b/copyparty/up2k.py @@ -144,6 +144,7 @@ class Up2k(object): self.salt = self.args.warksalt self.r_hash = re.compile("^[0-9a-zA-Z_-]{44}$") + self.abrt_key = "" self.gid = 0 self.gt0 = 0 @@ -3988,6 +3989,9 @@ class Up2k(object): except: pass + def handle_fs_abrt(self, akey: str) -> None: + self.abrt_key = akey + def handle_rm( self, uname: str, @@ -4197,7 +4201,7 @@ class Up2k(object): return n_files, ok + ok2, ng + ng2 - def handle_cp(self, uname: str, ip: str, svp: str, dvp: str) -> str: + def handle_cp(self, abrt: str, uname: str, ip: str, svp: str, dvp: str) -> str: if svp == dvp or dvp.startswith(svp + "/"): raise Pebkac(400, "cp: cannot copy parent into subfolder") @@ -4244,6 +4248,8 @@ class Up2k(object): dvpf = dvp + svpf[len(svp) :] self._cp_file(uname, ip, svpf, dvpf, curs) + if abrt and abrt == self.abrt_key: + raise Pebkac(400, "filecopy aborted by http-api") for v in curs: v.connection.commit() @@ -4411,7 +4417,7 @@ class Up2k(object): return "k" - def handle_mv(self, uname: str, ip: str, svp: str, dvp: str) -> str: + def handle_mv(self, abrt: str, uname: str, ip: str, svp: str, dvp: str) -> str: if svp == dvp or dvp.startswith(svp + "/"): raise Pebkac(400, "mv: cannot move parent into subfolder") @@ -4466,6 +4472,8 @@ class Up2k(object): dvpf = dvp + svpf[len(svp) :] self._mv_file(uname, ip, svpf, dvpf, curs) + if abrt and abrt == self.abrt_key: + raise Pebkac(400, "filemove aborted by http-api") for v in curs: v.connection.commit() diff --git a/copyparty/web/browser.css b/copyparty/web/browser.css index f2dc8cde..00e42cae 100644 --- a/copyparty/web/browser.css +++ b/copyparty/web/browser.css @@ -1930,6 +1930,11 @@ html.y #tree.nowrap .ntree a+a:hover { padding: 0; font-size: 1.5em; } +#fs_abrt { + margin-top: 1em; + text-shadow: 0; + box-shadow: 1px 1px 0 var(--bg-d3); +} #doc { overflow: visible; background: #fff; diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index a0a1a21c..112e7da0 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -427,6 +427,7 @@ var Ls = { "fcp_ok": "copy OK", "fp_busy": "moving {0} items...\n\n{1}", "fcp_busy": "copying {0} items...\n\n{1}", + "fp_abrt": "aborting...", "fp_err": "move failed:\n", "fcp_err": "copy failed:\n", "fp_confirm": "move these {0} items here?", @@ -1057,6 +1058,7 @@ var Ls = { "fcp_ok": "kopiering OK", "fp_busy": "flytter {0} filer...\n\n{1}", "fcp_busy": "kopierer {0} filer...\n\n{1}", + "fp_abrt": "avbryter...", "fp_err": "flytting feilet:\n", "fcp_err": "kopiering feilet:\n", "fp_confirm": "flytt disse {0} filene hit?", @@ -1686,6 +1688,7 @@ var Ls = { "fcp_ok": "复制成功", //m "fp_busy": "正在移动 {0} 项...\n\n{1}", "fcp_busy": "正在复制 {0} 项...\n\n{1}", //m + "fp_abrt": "正在中止...", //m "fp_err": "移动失败:\n", "fcp_err": "复制失败:\n", //m "fp_confirm": "将这些 {0} 项移动到这里?", @@ -2319,6 +2322,7 @@ var Ls = { "fcp_ok": "kopírování OK", "fp_busy": "přesouvám {0} položek...\n\n{1}", "fcp_busy": "kopíruji {0} položek...\n\n{1}", + "fp_abrt": "přerušuji...", //m "fp_err": "přesun selhal:\n", "fcp_err": "kopírování selhalo:\n", "fp_confirm": "přesunout těchto {0} položek sem?", @@ -2948,6 +2952,7 @@ var Ls = { "fcp_ok": "Kopieren OK", "fp_busy": "Verschiebe {0} Elemente...\n\n{1}", "fcp_busy": "Kopiere {0} Elemente...\n\n{1}", + "fp_abrt": "Abbrechen...", //m "fp_err": "Verschieben fehlgeschlagen:\n", "fcp_err": "Kopieren fehlgeschlagen:\n", "fp_confirm": "Diese {0} Elemente hierher verschieben?", @@ -3577,6 +3582,7 @@ var Ls = { "fcp_ok": "kopiointi OK", "fp_busy": "siirretään {0} kohdetta...\n\n{1}", "fcp_busy": "kopioidaan {0} kohdetta...\n\n{1}", + "fp_abrt": "keskeytetään...", //m "fp_err": "siirto epäonnistui:\n", "fcp_err": "kopiointi epäonnistui:\n", "fp_confirm": "siirrä nämä {0} kohdetta tänne?", @@ -4206,6 +4212,7 @@ var Ls = { "fcp_ok": "copie OK", "fp_busy": "déplacement de {0} éléments…\n\n{1}", "fcp_busy": "copie de {0} éléments…\n\n{1}", + "fp_abrt": "abandon en cours...", //m "fp_err": "deplacement échoué:\n", "fcp_err": "copie échouée:\n", "fp_confirm": "déplacer ces {0} éléments ici ?", @@ -4835,6 +4842,7 @@ var Ls = { "fcp_ok": "αντιγραφή OK", "fp_busy": "μετακίνηση {0} αντικειμένων...\n\n{1}", "fcp_busy": "αντιγραφή {0} αντικειμένων...\n\n{1}", + "fp_abrt": "γίνεται ακύρωση...", //m "fp_err": "αποτυχία μετακίνησης:\n", "fcp_err": "αποτυχία αντιγραφής:\n", "fp_confirm": "να μετακινηθούν αυτά τα {0} αντικείμενα εδώ;", @@ -5464,6 +5472,7 @@ var Ls = { "fcp_ok": "copia OK", "fp_busy": "spostando {0} elementi...\n\n{1}", "fcp_busy": "copiando {0} elementi...\n\n{1}", + "fp_abrt": "annullamento in corso...", //m "fp_err": "spostamento fallito:\n", "fcp_err": "copia fallita:\n", "fp_confirm": "spostare questi {0} elementi qui?", @@ -6093,6 +6102,7 @@ var Ls = { "fcp_ok": "Kopiëren OK", "fp_busy": "{0} items verplaatsen...\n\n{1}", "fcp_busy": "{0} items kopiëren...\n\n{1}", + "fp_abrt": "afbreken...", //m "fp_err": "Verplaatsen mislukt:\n", "fcp_err": "Kopieëren mislukt:\n", "fp_confirm": "Verplaats deze {0} items hierheen?", @@ -6723,6 +6733,7 @@ var Ls = { "fcp_ok": "kopiering OK", "fp_busy": "flyttar {0} filer...\n\n{1}", "fcp_busy": "kopierar {0} filer...\n\n{1}", + "fp_abrt": "avbryt...", "fp_err": "flytting feila:\n", "fcp_err": "kopiering feila:\n", "fp_confirm": "flytt disse {0} filene hit?", @@ -7349,6 +7360,7 @@ var Ls = { "fcp_ok": "przekopiowano", "fp_busy": "przenoszenie {0} elementów...\n\n{1}", "fcp_busy": "kopiowanie {0} elementów...\n\n{1}", + "fp_abrt": "przerywanie...", //m "fp_err": "nie udało się przenieść:\n", "fcp_err": "nie udało się skopiować:\n", "fp_confirm": "przenieść tutaj {0} elementy(ów)?", @@ -7978,6 +7990,7 @@ var Ls = { "fcp_ok": "успешно скопировано", "fp_busy": "перемещаю {0} файлов...\n\n{1}", "fcp_busy": "копирую {0} файлов...\n\n{1}", + "fp_abrt": "прерывание...", //m "fp_err": "ошибка перемещения:\n", "fcp_err": "ошибка копирования:\n", "fp_confirm": "переместить эти {0} файлов сюда?", @@ -8606,6 +8619,7 @@ var Ls = { "fcp_ok": "copia correcta", "fp_busy": "moviendo {0} elementos...\n\n{1}", "fcp_busy": "copiando {0} elementos...\n\n{1}", + "fp_abrt": "cancelando...", //m "fp_err": "fallo al mover:\n", "fcp_err": "fallo al copiar:\n", "fp_confirm": "¿mover estos {0} elementos aquí?", @@ -9235,6 +9249,7 @@ var Ls = { "fcp_ok": "копіювання OK", "fp_busy": "переміщення {0} елементів...\n\n{1}", "fcp_busy": "копіювання {0} елементів...\n\n{1}", + "fp_abrt": "переривання...", //m "fp_err": "переміщення невдале:\n", "fcp_err": "копіювання невдале:\n", "fp_confirm": "перемістити ці {0} елементи сюди?", @@ -9882,6 +9897,7 @@ var ACtx = !IPHONE && (window.AudioContext || window.webkitAudioContext), hash0 = location.hash, sloc0 = '' + location, noih = /[?&]v\b/.exec(sloc0), + abrt_key = "", rtt = null, ldks = [], dks = {}, @@ -12414,6 +12430,15 @@ function fmt_ren(re, md, fmt) { } +function fs_abrt() { + toast.inf(30, L.fp_abrt); + fileman.f.length = 0; + var xhr = new XHR(); + xhr.open('POST', '/?fs_abrt=' + abrt_key, true); + xhr.send(); +} + + var fileman = (function () { var bren = ebi('fren'), bdel = ebi('fdel'), @@ -12424,6 +12449,7 @@ var fileman = (function () { t_paste, r = {}; + r.f = []; r.clip = null; try { r.bus = new BroadcastChannel("fileman_bus"); @@ -12711,6 +12737,8 @@ var fileman = (function () { base = vsplit(sel[0].vp)[0], mkeys; + r.f = f; + for (var a = 0; a < sel.length; a++) { var vp = sel[a].vp; if (vp.endsWith('/')) @@ -12990,7 +13018,9 @@ var fileman = (function () { return rn_cancel(); } - toast.show('inf r', 0, esc(L.fr_busy.format(f.length, f[0].ofn))); + var msg = esc(L.fr_busy.format(f.length, f[0].ofn)); + msg += '\n<a id="fs_abrt" class="btn" href="#" onclick="fs_abrt()">' + L.fs_abrt + '</a>'; + toast.show('inf r', 0, msg); var dst = base + uricom_enc(f[0].inew.value, false); function rename_cb() { @@ -13004,8 +13034,10 @@ var fileman = (function () { return rn_apply_loop(); } + abrt_key = randstr(9); + var xhr = new XHR(); - xhr.open('POST', f[0].src + '?move=' + dst, true); + xhr.open('POST', f[0].src + '?move=' + dst + '&akey=' + abrt_key, true); xhr.onload = xhr.onerror = rename_cb; xhr.send(); } @@ -13267,6 +13299,8 @@ var fileman = (function () { srcdir = vsplit(r.clip[0])[0], links = QSA('#files tbody td:nth-child(2) a'); + r.f = f; + for (var a = 0, aa = links.length; a < aa; a++) indir.push(uricom_dec(vsplit(noq_href(links[a]))[1])); @@ -13298,13 +13332,17 @@ var fileman = (function () { if (!t.dst) return paster(); - toast.show('inf r', 0, esc((r.ccp ? L.fcp_busy : L.fp_busy).format(f.length + 1, uricom_dec(t.src)))); + var msg = esc((r.ccp ? L.fcp_busy : L.fp_busy).format(f.length + 1, uricom_dec(t.src))); + msg += '\n<a id="fs_abrt" class="btn" href="#" onclick="fs_abrt()">' + L.fs_abrt + '</a>'; + toast.show('inf r', 0, msg); var xhr = new XHR(), act = r.ccp ? '?copy=' : '?move=', dst = get_evpath() + uricom_enc(t.dst); - xhr.open('POST', t.src + act + dst, true); + abrt_key = randstr(9); + + xhr.open('POST', t.src + act + dst + '&akey=' + abrt_key, true); xhr.onload = xhr.onerror = paste_cb; xhr.send(); } diff --git a/tests/util.py b/tests/util.py index f02f81ba..bbd0b215 100644 --- a/tests/util.py +++ b/tests/util.py @@ -143,7 +143,7 @@ class Cfg(Namespace): def __init__(self, a=None, v=None, c=None, **ka0): ka = {} - ex = "allow_flac allow_wav chpw cookie_lax daw dav_auth dav_mac dav_rt e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp early_ban ed emp exp force_js getmod grid gsel hardlink hardlink_only ih ihead localtime magic nid nih no_acode no_athumb no_bauth no_clone no_cp no_dav no_db_ip no_del no_dirsz no_dupe no_fnugg no_lifetime no_logues no_mv no_pipe no_poll no_readme no_robots no_sb_md no_sb_lg no_scandir no_tail no_tarcmp no_thumb no_vthumb no_zip nrand nsort nw og og_no_head og_s_title ohead q rand re_dirsz reflink rmagic rss smb srch_dbg srch_excl stats uqe usernames vague_403 vc ver wo_up_readme write_uplog xdev xlink xvol zipmaxu zs" + ex = "allow_flac allow_wav chpw cookie_lax daw dav_auth dav_mac dav_rt e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp early_ban ed emp exp force_js getmod grid gsel hardlink hardlink_only ih ihead localtime magic nid nih no_acode no_athumb no_bauth no_clone no_cp no_dav no_db_ip no_del no_dirsz no_dupe no_fnugg no_lifetime no_logues no_mv no_pipe no_poll no_readme no_robots no_sb_md no_sb_lg no_scandir no_tail no_tarcmp no_thumb no_vthumb no_u2abrt no_zip nrand nsort nw og og_no_head og_s_title ohead q rand re_dirsz reflink rmagic rss smb srch_dbg srch_excl stats uqe usernames vague_403 vc ver wo_up_readme write_uplog xdev xlink xvol zipmaxu zs" ka.update(**{k: False for k in ex.split()}) ex = "dav_inf dedup dotpart dotsrch hook_v no_dhash no_fastboot no_fpool no_htp no_rescan no_sendfile no_ses no_snap no_up_list no_voldump re_dhash see_dots plain_ip" From d676a86f3fa5b7379ea2c2c495abe3e77f5ef1e7 Mon Sep 17 00:00:00 2001 From: Bevinsky <vin@r-a-d.io> Date: Thu, 14 Aug 2025 00:50:15 +0200 Subject: [PATCH 154/174] Add Swedish translation (#551) --- copyparty/web/browser.js | 632 ++++++++++++++++++++++++++++++++++++++- copyparty/web/splash.js | 42 +++ 2 files changed, 673 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 112e7da0..047c6446 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -8828,6 +8828,636 @@ var Ls = { "lang_set": "¿refrescar para que el cambio surta efecto?" }, + "swe": { + "tt": "Svenska", + + "cols": { + "c": "aktion", + "dur": "längd", + "q": "kvalitet / bitrate", + "Ac": "ljudkodek", + "Vc": "videokodek", + "Fmt": "format / container", + "Ahash": "ljudchecksumma", + "Vhash": "videochecksumma", + "Res": "upplösning", + "T": "filtyp", + "aq": "ljudkvalitet / bitrate", + "vq": "videokvalitet / bitrate", + "pixfmt": "subsampling / pixelstruktur", + "resw": "horisontell upplösning", + "resh": "vertikal upplösning", + "chs": "antal ljudkanaler", + "hz": "samplingsfrekvens" + }, + + "hks": [ + [ + "övrigt", + ["ESC", "stäng diverse paneler"], + + "filhanterare", + ["G", "växla mellan listvy / rutnät"], + ["T", "växla mellan miniatyrer / ikoner"], + ["⇧ A/D", "miniatyrstorlek"], + ["ctrl-K", "radera urval"], + ["ctrl-X", "klipp urval till urklipp"], + ["ctrl-C", "kopiera urval till urklipp"], + ["ctrl-V", "klistra in (kopiera/flytta) hit"], + ["Y", "ladda ner urval"], + ["F2", "byt namn på urval"], + + "välja filer", + ["Blanksteg", "växla val av fil"], + ["↑/↓", "flytta filvalsmarkör"], + ["ctrl ↑/↓", "flytta markör och vy"], + ["⇧ ↑/↓", "välj föregående/nästa fil"], + ["ctrl-A", "välj alla filer / mappar"], + ], [ + "navigation", + ["B", "växla mellan brödsmulor / trädvy"], + ["I/K", "föregående/nästa mapp"], + ["M", "hoppa till överordnad mapp (eller kollapsa närvarande)"], + ["V", "växla mellan mappar / textfiler i trädvy"], + ["A/D", "trädvystorlek"], + ], [ + "ljudspelare", + ["J/L", "föregående/nästa låt"], + ["U/O", "hoppa 10sek bakåt/framåt"], + ["0..9", "hoppa till 0%..90%"], + ["P", "play/paus (startar även uppspelning)"], + ["S", "välj låten som spelas"], + ["Y", "ladda ner låt"], + ], [ + "bildvisare", + ["J/L, ←/→", "föregående/nästa bild"], + ["Hem/End", "första/sista bilden"], + ["F", "helskärm"], + ["R", "rotera medsols"], + ["⇧ R", "rotera motsols"], + ["S", "välj bild"], + ["Y", "ladda ner bild"], + ], [ + "videospelare", + ["U/O", "hoppa 10sek bakåt/framåt"], + ["P/K/Blanksteg", "play/paus"], + ["C", "fortsätt med nästa"], + ["V", "loopa"], + ["M", "stäng av ljud"], + ["[ och ]", "ställ in loopintervall"], + ], [ + "textfilsvisare", + ["I/K", "föregående/nästa fil"], + ["M", "stäng textfil"], + ["E", "redigera textfil"], + ["S", "välj fil"], + ] + ], + + "m_ok": "OK", + "m_ng": "Avbryt", + + "enable": "Aktivera", + "danger": "VARNING", + "clipped": "kopierat till urklipp", + + "ht_s1": "sekund", + "ht_s2": "sekunder", + "ht_m1": "minut", + "ht_m2": "minuter", + "ht_h1": "timme", + "ht_h2": "timmar", + "ht_d1": "dag", + "ht_d2": "dagar", + "ht_and": " och ", + + "goh": "kontrollpanel", + "gop": 'föregående mapp">föreg.', + "gou": 'överordnad mapp">upp', + "gon": 'nästa mapp">nästa', + "logout": "Logga ut ", + "access": "-rättighet", + "ot_close": "stäng undermeny", + "ot_search": "sök efter filer via attribut, sökväg / namn, musiktaggar, eller någon kombination av dessa$N$N<code>foo bar</code> = måste innehålla både «foo» och «bar»,$N<code>foo -bar</code> = måste innehålla «foo» men inte «bar»,$N<code>^yana .opus$</code> = måste börja med «yana» och vara en «opus»-fil$N<code>"try unite"</code> = måste innehålla exakt «try unite»$N$Ndatumformatet är iso-8601, t.ex.$N<code>2009-12-31</code> eller <code>2020-09-12 23:30:00</code>", + "ot_unpost": "unpost: radera dina senaste uppladdningar, eller avbryt pågående sådana", + "ot_bup": "bup: enkel uppladdare, stödjer t o m netscape 4.0", + "ot_mkdir": "mkdir: skapa en ny mapp", + "ot_md": "new-md: skapa ett nytt markdown-dokument", + "ot_msg": "msg: skicka ett meddelande till serverloggen", + "ot_mp": "mediaspelarinställningar", + "ot_cfg": "konfigurationsinställningar", + "ot_u2i": 'up2k: ladda upp filer (om du har skrivrättigheter) eller byt till sökläge för att se om de finns någonstans på servern$N$Nuppladdningarna är återupptagbara, multitrådade och filernas tidsstämpel bevaras, men den använder mer CPU än [🎈]  (den enkla uppladdaren)<br /><br />under uppladdningens gång blir denna ikon en förloppsindikator!', + "ot_u2w": 'up2k: ladda upp filer med stöd för återupptagning (stäng din webbläsare och dra in samma filer senare)$N$Nmultitrådad och filernas tidsstämpel bevaras, men den använder mer CPU än [🎈]  (den enkla uppladdaren)<br /><br />under uppladdningens gång blir denna ikon en förloppsindikator!', + "ot_noie": 'Var vänlig använd Chrome / Firefox / Edge', + + "ab_mkdir": "skapa mapp", + "ab_mkdoc": "nytt markdown-dokument", + "ab_msg": "skicka medd. till serverlogg", + + "ay_path": "hoppa till mappar", + "ay_files": "hoppa till filer", + + "wt_ren": "byt namn på urval$NSnabbtangent: F2", + "wt_del": "radera urval$NSnabbtangent: ctrl-K", + "wt_cut": "klipp urval<small>(för att klistra in någonstans)</small>$NSnabbtangent: ctrl-X", + "wt_cpy": "kopiera urval till urklipp$N(för att klistra in någonstans)$NSnabbtangent: ctrl-C", + "wt_pst": "klistra in tidigare urval$NSnabbtangent: ctrl-V", + "wt_selall": "välj alla filer$NSnabbtangent: ctrl-A (när en fil har fokus)", + "wt_selinv": "invertera urval", + "wt_zip1": "ladda ner denna mapp som ett arkiv", + "wt_selzip": "ladda ner urval som ett arkiv", + "wt_seldl": "ladda ner urval som separata filer$NSnabbtangent: Y", + "wt_npirc": "kopiera IRC-formatterad låtinfo", + "wt_nptxt": "kopiera låtinfo i klartext", + "wt_m3ua": "lägg till i m3u-spellista (klicka på <code>📻copy</code> senare)", + "wt_m3uc": "kopiera m3u-spellista till urklipp", + "wt_grid": "växla mellan rutnät och listvy$NSnabbtangent: G", + "wt_prev": "föregående låt$NSnabbtangent: J", + "wt_play": "play / paus$NSnabbtangent: P", + "wt_next": "nästa låt$NSnabbtangent: L", + + "ul_par": "samtidiga uppladdningar:", + "ut_rand": "slumpa filnamn", + "ut_u2ts": "bevara tidsstämpeln för senaste ändring$Nfrån ditt filsystem till servern\">📅", + "ut_ow": "skriv över existerande filer på servern?$N🛡️: aldrig (skapar ett nytt filnamn istället)$N🕒: skriv över om serverns fil är äldre än din$N♻️: skriv alltid över om filerna skiljer sig", + "ut_mt": "fortsätt hasha filer under uppladdningens gång$N$Nstäng av om din CPU eller disk är en flaskhals", + "ut_ask": 'bekräfta innan uppladdningar påbörjas">💭', + "ut_pot": "förbättra uppladdningshastigheten på långsamma enheter$Ngenom att förenkla användargränssnittet", + "ut_srch": "ladda inte upp; kolla istället om filerna redan existerar på $N servern (detta kommer att skanna alla mappar med läsrättighet)", + "ut_par": "du kan pausa all uppladdning genom att sätta detta till 0$N$Nöka denna om din uppkoppling är långsam eller har hög latens$N$Nsätt till 1 över lokala nätverk eller om serverns disk är en flaskhals", + "ul_btn": "släpp filer / mappar<br>här (eller klicka)", + "ul_btnu": "L A D D A U P P", + "ul_btns": "S Ö K", + + "ul_hash": "hashar", + "ul_send": "skickar", + "ul_done": "klar", + "ul_idle1": "inga uppladdningar har köats", + "ut_etah": "medelhastighet för <em>hashning</em>, och uppskattad återstående tid", + "ut_etau": "medelhastighet för <em>överföring</em>, och uppskattad återstående tid", + "ut_etat": "<em>total</em> medelhastighet, och uppskattad återstående tid", + + "uct_ok": "lyckade", + "uct_ng": "no-good: misslyckade / avvisade / ej funna", + "uct_done": "ok och ng kombinerat", + "uct_bz": "pågående", + "uct_q": "köade", + + "utl_name": "filnamn", + "utl_ulist": "visa", + "utl_ucopy": "kopiera", + "utl_links": "länkar", + "utl_stat": "status", + "utl_prog": "förlopp", + + // keep short: + "utl_404": "404", + "utl_err": "FEL", + "utl_oserr": "OS-fel", + "utl_found": "hittad", + "utl_defer": "väntar", + "utl_yolo": "YOLO", + "utl_done": "klar", + + "ul_flagblk": "filerna lades till i kön,</b><br>men det finns en upptagen up2k i en annan webbläsarflik,<br>så vi väntar på den först", + "ul_btnlk": "serverkonfigurationen har låst denna inställning", + + "udt_up": "Ladda upp", + "udt_srch": "Sök", + "udt_drop": "släpp här", + + "u_nav_m": '<h6>jaha, vad har du då?</h6><code>Enter</code> = Filer (en eller flera)\n<code>ESC</code> = En mapp (inklusive undermappar)', + "u_nav_b": '<a href="#" id="modal-ok">Filer</a><a href="#" id="modal-ng">En mapp</a>', + + "cl_opts": "växlar", + "cl_themes": "tema", + "cl_langs": "språk", + "cl_ziptype": "mappnedladdning", + "cl_uopts": "up2k-inställningar", + "cl_favico": "favikon", + "cl_bigdir": "stora mappar", + "cl_hsort": "#sort.", + "cl_keytype": "tonartsnotering", + "cl_hiddenc": "dolda kolumner", + "cl_hidec": "dölj", + "cl_reset": "återställ", + "cl_hpick": "tryck på en kolumntitel för att dölja den i filvyn", + "cl_hcancel": "kolumndöljning avbruten", + + "ct_grid": '田 rutnätet', + "ct_ttips": '◔ ◡ ◔">ℹ️ tips', + "ct_thumb": 'växla mellan miniatyrer och ikoner i rutnätsvyn$NSnabbtangent: T">🖼️ miniatyrer', + "ct_csel": 'använd CTRL och SKIFT för urval av filer i rutnätsvyn">val', + "ct_ihop": 'skrolla till den senast visade filen när bildvisaren stängs">g⮯', + "ct_dots": 'visa dolda filer (om servern tillåter detta)">dolda', + "ct_qdel": 'bekräfta endast en gång när filer raderas">srad', + "ct_dir1st": 'sortera mappar före filer">📁 först', + "ct_nsort": 'naturlig sortering (för filnamn med ledande siffror)">nsort', + "ct_utc": 'visa alla datum och tider i UTC">UTC', + "ct_readme": 'visa README.md i listvyn">📜 läsmig', + "ct_idxh": 'visa index.html istället för listvyn">htm', + "ct_sbars": 'visa rullningslister">⟊', + + "cut_umod": "om en fil redan existerar på servern, uppdatera serverns senast modifierade tidsstämpel till att matcha din lokala fil (kräver skriv+radera-rättighet)\">re📅", + + "cut_turbo": "yolo-knappen, du vill förmodligen INTE aktivera denna:$N$Nanvänd denna om du höll på att ladda upp en stor mängd filer och var tvungen att stänga webbläsaren, och du vill fortsätta uppladdningen så fort som möjligt$N$Ndetta ersätter hash-checken med en enkel <em>"har denna fil samma filstorlek på servern?"</em>, så om filinnehållet skiljer sig kommer den INTE att laddas upp$N$Ndu bör stänga av denna när uppladdningen är klar och sedan "ladda upp" samma filer igen för att låta klienten verifiera dem\">turbo", + + "cut_datechk": "har endast effekt med turbo-växeln påslagen$N$Nminskar yolo-faktorn lite grann; kollar om filtidsstämplarna på servern matchar dina$N$Ndetta <em>bör</em> fånga de flesta ofärdiga / korrumperade uppladdningarna, men kan inte ersätta ett fullständigt verifieringspass med turbo avstängt\">date-chk", + + "cut_u2sz": "storlek (i MiB) för varje uppladdnings-chunk; stora värden flyger bättre över atlanten. Prova lägre värden på mycket opålitliga uppkopplingar", + + "cut_flag": "garantera att endast en flik laddar upp samtidigt $N -- andra flikar måste också ha denna påslagen -- $N påverkar endast flikar på samma domän", + + "cut_az": "ladda upp filer i alfabetisk ordning, snarare än mindre filer först$N$Nalfabetisk ordning kan göra det enklare att se var något har gått fel, men uppladdningen blir lite långsammare över fiber / lokala nätverk", + + "cut_nag": "skicka en systemnotifikation när uppladdningar blir klara$N(endast om webbläsaren eller fliken inte är fokuserade)", + "cut_sfx": "ljudnotifikation när uppladdningar blir klara$N(endast om webbläsaren eller fliken inte är fokuserade)", + + "cut_mt": "använd multitrådning för att accelerera filhashningen$N$Ndetta använder web-workers och kräver$Nmer RAM (upp till 512 MiB extra)$N$Nuppladdningar över https blir 30% snabbare, över http 4.5x snabbare\">mt", + + "cut_wasm": "använd wasm istället för webbläsarens inbyggda hashare; förbättrar hastigheten i chrome-baserade webbläsare men ökar CPU-lasten, och många äldre versioner av chrome har buggar som får webbläsaren att konsumera allt RAM-minne och krascha om detta är påslaget\">wasm", + + "cft_text": "favikon-text (låt stå tom och uppdatera sidan för att stänga av)", + "cft_fg": "förgrundsfärg", + "cft_bg": "bakgrundsfärg", + + "cdt_lim": "högsta antal filer att visa in en mapp", + "cdt_ask": "när du når botten av vyn,$Nbe om en åtgärd istället för att ladda fler filer", + "cdt_hsort": "hur många sorteringsregler (<code>,sorthref</code>) att inkludera i media-URL:er. Sätts detta till 0 kommer regler i klickade medialänkar även att ignoreras", + + "tt_entree": "visa trädvy$NSnabbtangent: B", + "tt_detree": "visa brödsmulor$NSnabbtangent: B", + "tt_visdir": "skrolla till öppnad mapp", + "tt_ftree": "växla mellan trädvy och textfiler$NHotkey: V", + "tt_pdock": "visa överordnade mappar i en panel längst upp i vyn", + "tt_dynt": "väx vyn när trädet expanderar", + "tt_wrap": "automatisk radbrytning", + "tt_hover": "visa överlånga rader när muspekaren hovrar över dem$N( skrollhjulet fungerar ej såvida inte pekaren$Nstår till vänster )", + + "ml_pmode": "vid mappens slut...", + "ml_btns": "komm.", + "ml_tcode": "konvertera", + "ml_tcode2": "konvertera till", + "ml_tint": "hy", + "ml_eq": "ljudutjämnare", + "ml_drc": "dynamikkompressor", + + "mt_loop": "upprepa en låt\">🔁", + "mt_one": "stoppa uppspelningen efter en låt\">1️⃣", + "mt_shuf": "blanda låtarna i varje mapp\">🔀", + "mt_aplay": "spela automatiskt om det finns en låt-ID i länkar du har klickat på för att öppna sidan$N$Nom detta är avstängt kommer sidans adress inte att bli uppdaterad med en låt-ID om du spelar musik, för att förhindra automatisk uppspelning om dessa inställningar går förlorade men webbadressen återstår\">a▶", + "mt_preload": "påbörja nedladdning av nästa låt i förväg för gapfri uppspelning\">ladda", + "mt_prescan": "hoppa till nästa mapp i förväg så att webbläsaren$Nförblir glad och inte avbryter uppspelningen\">nav", + "mt_fullpre": "försök att ladda ner hela låten i förväg;$N✅ aktivera på <b>opålitliga</b> uppkopplingar,$N❌ <b>avaktivera</b> kanske på långsamma uppkopplingar\">full", + "mt_fau": "förhindra att uppspelningen avstannar på telefoner om nästa låt inte laddas tillräckligt snabbt i förväg (kan ge upphov till buggiga musiktaggar)\">☕️", + "mt_waves": "vågformsreglage:$Nvisa ljudstyrkan i uppspelningsreglaget\">~s", + "mt_npclip": "visa knappar för att kopiera låtinfo till urklippet\">/np", + "mt_m3u_c": "visa knappar för att kopiera de valda$Nlåtarna som en m3u8-spellista\">📻", + "mt_octl": "systemintegration (mediaknappar / skärmdisplay)\">os-ctl", + "mt_oseek": "tillåt fram- och bakåtspolning via systemintegrationen$N$Nobs.: på vissa enheter (iPhone)$Nersätter detta knappen för nästa låt\">spola", + "mt_oscv": "visa skivomslag i skärmdisplayen\">omslag", + "mt_follow": "skrolla vyn till den spelande låten\">🎯", + "mt_compact": "kompakt kontrollpanel\">⟎", + "mt_uncache": "rensa cachen  (prova detta om din webbläsare har cachat$Nen trasig kopia av en låt och den vägrar spela upp den)\">rensa", + "mt_mloop": "upprepa den öppna mappen\">🔁 upprepa", + "mt_mnext": "ladda nästa mapp och fortsätt\">📂 nästa", + "mt_mstop": "stoppa uppspelningen\">⏸ stopp", + "mt_cflac": "konvertera flac / wav till {0}\">flac", + "mt_caac": "konvertera aac / m4a till {0}\">aac", + "mt_coth": "konvertera allt annat (förutom mp3) till {0}\">annat", + "mt_c2opus": "bäst val för pc, laptop, android\">opus", + "mt_c2owa": "opus-weba, för iOS 17.5 och senare\">owa", + "mt_c2caf": "opus-caf, för iOS 11 till 17\">caf", + "mt_c2mp3": "använd detta på mycket gamla enheter\">mp3", + "mt_c2flac": "bäst ljudkvalitet, men enorma nedladdningar\">flac", + "mt_c2wav": "okomprimerad uppspelning (ännu större)\">wav", + "mt_c2ok": "snyggt, bra val", + "mt_c2nd": "det är inte det rekommenderade formatet för din enhet, men det är lungt", + "mt_c2ng": "din enhet verkar inte stödja det här formatet, men vi provar ändå", + "mt_xowa": "det finns buggar i iOS som hindrar uppspelning i bakgrunden med detta format; vänligen använd caf eller mp3 istället", + "mt_tint": "nivå på bakgrundsfärg (0-100) på uppspelningsreglaget;$Ngör buffring mindre distraherande", + "mt_eq": "aktiverar utjämning och förstärkning;$N$Nboost <code>0</code> = standard 100%-volym (omodifierad)$N$Nwidth <code>1  </code> = standard stereo (omodifierad)$Nwidth <code>0.5</code> = 50% vänster-höger crossfeed$Nwidth <code>0  </code> = mono$N$Nboost <code>-0.8</code> & width <code>10</code> = tar bort sång :^)$N$Nnär utjämningen är aktiverad blir gaplösa album verkligen gaplösa, så låt den stå påslagen med alla värden satta till 0 (förutom width = 1) om du bryr dig om det", + "mt_drc": "aktiverar dynamikkompressorn (volymtillplattning / brickwaller); aktiverar även utjämnaren för att balansera röran, så sätt alla fält i utjämnaren förutom 'width' till 0 om du inte vill ha den$N$Nsänker all volym över THRESHOLD dB; för varje RATIO dB över THRESHOLD blir det 1 dB av output, så standardvärdena tresh = -24 och ratio = 12 innebär att volymen aldrig bör bli högre än -22 dB och det är säkert att höja utjämnarens boost till 0.8, eller t.o.m. 1.8 med ATK 0 och ett högt RLS-värde t.ex. 90 (fungerar endast i firefox; RLS är låst till högst 1 i andra webbläsare)$N$N(se wikipedia för en bättre förklaring)", + + "mb_play": "play", + "mm_hashplay": "spela upp den här ljudfilen?", + "mm_m3u": "tryck <code>Enter/OK</code> för att spela\ntryck <code>ESC/Avbryt</code> to Edit", + "mp_breq": "firefox 82+ eller chrome 73+ eller iOS 15+ krävs", + "mm_bload": "laddar...", + "mm_bconv": "konverterar till {0}, vänligen vänta...", + "mm_opusen": "din webbläsare kan inte spela upp aac- eller m4a-filer;\nkonvertering till opus är nu påslaget", + "mm_playerr": "uppspelning misslyckades: ", + "mm_eabrt": "Uppspelningen avbröts", + "mm_enet": "Din uppkoppling är skum", + "mm_edec": "Filen är korrumperad??", + "mm_esupp": "Din webbläsare förstår inte detta format", + "mm_eunk": "Okänt Fel", + "mm_e404": "Kunde inte spela upp ljudfil; fel 404: Filen hittades inte.", + "mm_e403": "Kunde inte spela upp ljudfil; fel 403: Åtkomst nekad.\n\nProva att ladda om sidan med F5, du kanske blev utloggad", + "mm_e500": "Kunde inte spela upp ljudfil; fel 500: Kolla serverloggen.", + "mm_e5xx": "Kunde inte spela upp ljudfil; serverfel ", + "mm_nof": "hittade inga fler låtar i närheten", + "mm_prescan": "Letar efter fler låtar...", + "mm_scank": "Hittade nästa låt:", + "mm_uncache": "cachen rensad; alla låtar kommer att laddas ner igen vid uppspelning", + "mm_hnf": "den låten finns inte längre", + + "im_hnf": "den bilden finns inte längre", + + "f_empty": 'mappen är tom', + "f_chide": 'detta kommer att dölja kolumnen «{0}»\n\ndu kan visa kolumner igen i inställningarna', + "f_bigtxt": "den här filen är {0} MiB stor -- vill du verkligen visa den som text?", + "f_bigtxt2": "visa endast slutet på filen? detta aktiverar även övervakning, vilket visar nya rader i filen i realtid", + "fbd_more": '<div id="blazy"><code>{0}</code> av <code>{1}</code> filer visas; <a href="#" id="bd_more">visa {2}</a> eller <a href="#" id="bd_all">visa alla</a></div>', + "fbd_all": '<div id="blazy"><code>{0}</code> av <code>{1}</code> filer visas; <a href="#" id="bd_all">visa alla</a></div>', + "f_anota": "endast {0} av {1} objekt valdes;\nför att välja hela mappen, skrolla först till botten av vyn", + + "f_dls": 'fillänkarna i den öppna mappen har\nbytts till nedladdningslänkar', + + "f_partial": "För att säkert ladda ner en fil som för tillfället laddas upp, vänligen klicka på filen som har samma filnamn men utan <code>.PARTIAL</code>-filändelsen. Vänligen tryck Avbryt eller Escape för att göra detta.\n\nOm du bortser från denna varning och trycker OK eller Enter kommer den tillfälliga <code>.PARTIAL</code>-filen istället att laddas ner, vilket är nästan garanterat att ge dig korrumperad data.", + + "ft_paste": "klistra in {0} objekt$NSnabbtangent: ctrl-V", + "fr_eperm": 'kan ej byta namn:\ndu har inte flytträttighet i denna mapp', + "fd_eperm": 'kan ej radera:\ndu har inte raderingsrättighet i denna mapp', + "fc_eperm": 'kan ej klippa:\ndu har inte flytträttighet i denna mapp', + "fp_eperm": 'kan ej klistra in:\ndu har inte skrivrättighet i denna mapp', + "fr_emore": "välj minst en fil att byta namn på", + "fd_emore": "välj minst en fil att radera", + "fc_emore": "välj minst en fil att klippa", + "fcp_emore": "välj minst en fil att kopiera till urklippet", + + "fs_sc": "dela den öppna mappen", + "fs_ss": "dela de urvalda filerna", + "fs_just1d": "du kan inte välja mer än en mapp\neller blanda filer och mappar i samma urval", + "fs_abrt": "❌ avbryt", + "fs_rand": "🎲 slump.namn", + "fs_go": "✅ skapa utdelning", + "fs_name": "namn", + "fs_src": "källa", + "fs_pwd": "lösen", + "fs_exp": "utgång", + "fs_tmin": "min", + "fs_thrs": "timmar", + "fs_tdays": "dagar", + "fs_never": "oändlig", + "fs_pname": "valfritt länknamn; slumpas fram om detta står tomt", + "fs_tsrc": "filen eller mappen att dela", + "fs_ppwd": "valfritt lösenord", + "fs_w8": "skapar utdelning...", + "fs_ok": "tryck <code>Enter/OK</code> för att kopiera länken till urklipp\ntryck <code>ESC/Avbryt</code> för att stänga", + + "frt_dec": "kan laga vissa typer av trasiga filnamn\">avkoda-url", + "frt_rst": "återställ modifierade filnamn till de ursprungliga\">↺ återställ", + "frt_abrt": "avbryt och stäng denna panel\">❌ avbryt", + "frb_apply": "BYT NAMN", + "fr_adv": "batch-, metadata- och mönsteromskrivning\">avancerat", + "fr_case": "skiftlägeskänsligt reguljärt uttryck\">skift", + "fr_win": "windows-säkra namn; ersätt <code><>:"\\|?*</code> med japanska fullbreddtecken\">win", + "fr_slash": "ersätt <code>/</code> med ett tecken som inte skapar nya mappar\">ingen /", + "fr_re": "reguljärt sökuttryck att tillämpa på ursprungliga filnamn; grupper kan hänvisas till i formatfältet nedan via <code>(1)</code> och <code>(2)</code> osv.", + "fr_fmt": "inspirerat av foobar2000:$N<code>(title)</code> ersätts av låttitel,$N<code>[(artist) - ](title)</code> skippar [detta] om artisten är tom$N<code>$lpad((tn),2,0)</code> fyller i spårnumret till 2 siffror", + "fr_pdel": "ta bort", + "fr_pnew": "spara som", + "fr_pname": "ge ett nytt namn på din inställning", + "fr_aborted": "avbrutet", + "fr_lold": "gammalt namn", + "fr_lnew": "nytt namn", + "fr_tags": "taggar för de valda filerna (skrivskyddat, endast som referens):", + "fr_busy": "byter namn på {0} objekt...\n\n{1}", + "fr_efail": "namnbyte misslyckades:\n", + "fr_nchg": "{0} av de nya namnen ändrades p g a <code>win</code> och/eller <code>ingen /</code>\n\nÄr det okej att fortsätta med de nya namnen?", + + "fd_ok": "radering lyckades", + "fd_err": "radering misslyckades:\n", + "fd_none": "inget raderades; kanske blockerat av serverkonfigurationen (xbd)?", + "fd_busy": "raderar {0} objekt...\n\n{1}", + "fd_warn1": "RADERA dessa {0} objekt?", + "fd_warn2": "<b>Sista chansen!</b> Det finns inget sätt att ångra detta. Radera?", + + "fc_ok": "klippte {0} objekt", + "fc_warn": 'klippte {0} objekt, men:\n\nendast <b>denna</b> webbläsarflik kan klistra in dem\n(eftersom urvalet är så enormt stort)', + + "fcc_ok": "kopierade {0} objekt till urklippet", + "fcc_warn": 'kopierade {0} objekt till urklippet, men:\n\nendast <b>denna</b> webbläsarflik kan klistra in dem\n(eftersom urvalet är så enormt stort)', + + "fp_apply": "använd dessa namn", + "fp_ecut": "klipp eller kopiera filer / mappar först för att klistra / flytta dem\n\nobs.: du kan klippa och klistra mellan webbläsarflikar", + "fp_ename": "{0} objekt kan ej flyttas hit eftersom filnamnen redan är tagna. Ge dem nya namn nedan för att fortsätta, eller lämna fältet tomt för att skippa:", + "fcp_ename": "{0} objekt kan ej kopieras hit eftersom filnamnen redan är tagna. Ge dem nya namn nedan för att fortsätta, eller lämna fältet tomt för att skippa:", + "fp_emore": "det finns fortfarande filnamnskrockar att fixa", + "fp_ok": "flytt lyckades", + "fcp_ok": "kopiering lyckades", + "fp_busy": "flyttar {0} objekt...\n\n{1}", + "fcp_busy": "kopierar {0} objekt...\n\n{1}", + "fp_abrt": "avbryter...", + "fp_err": "flytt misslyckades:\n", + "fcp_err": "kopiering misslyckades:\n", + "fp_confirm": "flytta dessa {0} objekt hit?", + "fcp_confirm": "kopiera dessa {0} objekt hit?", + "fp_etab": 'lyckades ej läsa urklippet från en annan webbläsarflik', + "fp_name": "laddar upp en fil från din enhet. Ge den ett namn:", + "fp_both_m": '<h6>välj vad som ska klistras in</h6><code>Enter</code> = Flytta {0} objekt från «{1}»\n<code>ESC</code> = Ladda upp {2} filer från din enhet', + "fcp_both_m": '<h6>välj vad som ska klistras in</h6><code>Enter</code> = Kopiera {0} objekt från «{1}»\n<code>ESC</code> = Ladda upp {2} filer från din enhet', + "fp_both_b": '<a href="#" id="modal-ok">Flytta</a><a href="#" id="modal-ng">Ladda upp</a>', + "fcp_both_b": '<a href="#" id="modal-ok">Kopiera</a><a href="#" id="modal-ng">Ladda upp</a>', + + "mk_noname": "skriv ett namn i fältet till vänster först :p", + + "tv_load": "Laddar textfil:\n\n{0}\n\n{1}% ({2} av {3} MiB laddat)", + "tv_xe1": "kunde ej ladda textfil:\n\nfel ", + "tv_xe2": "404, filen hittades inte", + "tv_lst": "lista av textfiler i", + "tvt_close": "återvänd till mapp$NSnabbtangent: M (eller Esc)\">❌ stäng", + "tvt_dl": "ladda ner denna fil$NSnabbtangent: Y\">💾 ladda ner", + "tvt_prev": "visa föregående fil$NSnabbtangent: i\">⬆ föreg.", + "tvt_next": "visa nästa fil$NSnabbtangent: K\">⬇ nästa", + "tvt_sel": "välj fil   ( för klipp / kopiera / radera / ... )$NSnabbtangent: S\">välj", + "tvt_edit": "öppna fil i textredigerare$NSnabbtangent: E\">✏️ redigera", + "tvt_tail": "övervaka filen; visa nya rader i realtid\">📡 övervaka", + "tvt_wrap": "automatisk radbrytning\">↵", + "tvt_atail": "lås vyn till sidans botten\">⚓", + "tvt_ctail": "avkoda terminalfärger (ansi-escapesekvenser)\">🌈", + "tvt_ntail": "gräns för scrollback (hur många byte ska behållas laddade)", + + "m3u_add1": "låt tillagd till m3u-spellista", + "m3u_addn": "{0} låtar tillagda till m3u-spellista", + "m3u_clip": "m3u-spellista kopierad till urklippet\n\ndu bör skapa en ny textfil som heter någonting.m3u och klistra in spellistan i det dokumentet; detta gör den uppspelbar", + + "gt_vau": "visa inte videor, spela endast ljudet\">🎧", + "gt_msel": "urval av filer; ctrl-klicka en fil för standardbeteende$N$N<em>när detta är aktiverat: dubbelklicka en fil / mapp för att öppna den</em>$N$NSnabbtangent: S\">urval", + "gt_crop": "centrera och beskär miniatyrbilder\">beskär", + "gt_3x": "högupplösta miniatyrbilder\">3x", + "gt_zoom": "zoom", + "gt_chop": "klipp", + "gt_sort": "sortera efter", + "gt_name": "namn", + "gt_sz": "storlek", + "gt_ts": "datum", + "gt_ext": "typ", + "gt_c1": "förkorta filnamn (visa mindre)", + "gt_c2": "förläng filnamn (visa mer)", + + "sm_w8": "söker...", + "sm_prev": "sökresultaten nedan är från en tidigare sökning:\n ", + "sl_close": "stäng sökresultaten", + "sl_hits": "visar {0} träffar", + "sl_moar": "ladda fler", + + "s_sz": "storlek", + "s_dt": "datum", + "s_rd": "sökväg", + "s_fn": "namn", + "s_ta": "taggar", + "s_ua": "uppl.", + "s_ad": "avanc.", + "s_s1": "minimum MiB", + "s_s2": "maximum MiB", + "s_d1": "min. iso8601", + "s_d2": "max. iso8601", + "s_u1": "uppladdad efter", + "s_u2": "och/eller före", + "s_r1": "sökvägen innehåller   (blankstegsseparerat)", + "s_f1": "filnamnet innehåller   (invertera med -intedetta)", + "s_t1": "taggar innehåller   (^=start, slut=$)", + "s_a1": "specifika metadataegenskaper", + + "md_eshow": "kan ej visa ", + "md_off": "[📜<em>läsmig</em>] avstängt i [⚙️] -- dokumentet är dolt", + + "badreply": "Kunde ej tolka svaret från servern", + + "xhr403": "403: Åtkomst nekad\n\nProva att ladda om sidan med F5, du kanske blev utloggad", + "xhr0": "okänt (tappade förmodligen kontakt med servern, eller så är den nere)", + "cf_ok": "ledsen -- DD" + wah + "oS-skyddet har aktiverats\n\nsaker bör fungera igen om 30 sekunder\n\nom inget händer, ladda om sidan med F5", + "tl_xe1": "kunde inte visa undermappar:\n\nfel ", + "tl_xe2": "404: Mappen hittades inte", + "fl_xe1": "kunde inte visa filer i mapp:\n\nfel ", + "fl_xe2": "404: Mappen hittades inte", + "fd_xe1": "kunde inte skapa mapp:\n\nfel ", + "fd_xe2": "404: Överordnad mapp hittades inte", + "fsm_xe1": "kunde inte skicka meddelande:\n\ndel ", + "fsm_xe2": "404: Överordnad mapp hittades inte", + "fu_xe1": "kunde inte ladda unpost-listan från servern:\n\nfel ", + "fu_xe2": "404: Filen hittades inte??", + + "fz_tar": "okomprimerad tar-fil i gnu-format (linux / mac)", + "fz_pax": "okomprimerad tar-fil i pax-format (långsammare)", + "fz_targz": "gnu-tar komprimerad med gzip-nivå 3$N$Ndetta är vanligtvis mycket långsamt,$Nanvänd okomprimerad tar istället", + "fz_tarxz": "gnu-tar komprimerad med xz-nivå 1$N$Ndetta är vanligtvis mycket långsamt,$Nanvänd okomprimerad tar istället", + "fz_zip8": "zip-fil med utf8-filnman (kan vara skum i windows 7 och äldre)", + "fz_zipd": "zip-fil med standard cp437-filnamn, för riktigt gammal mjukvara", + "fz_zipc": "cp437 med crc32 uträknad i förväg,$Nför MS-DOS PKZIP v2.04g (oktober 1993)$N(tar längre tid att behandla innan nedladdningen kan påbörjas)", + + "un_m1": "du kan radera dina senaste uppladdningar (eller avbryta pågående sådana) nedan", + "un_upd": "uppdatera", + "un_m4": "eller, dela filerna som syns nedan:", + "un_ulist": "visa", + "un_ucopy": "kopiera", + "un_flt": "filter:  sökvägen måste innehålla", + "un_fclr": "rensa filtret", + "un_derr": 'unpost-radering misslyckades:\n', + "un_f5": 'något gick sönder, prova att uppdatera eller tryck på F5', + "un_uf5": "ledsen men du måste uppdatera sidan (t.ex. genom att trycka på F5 eller CTRL-R) innan du kan avbryta den här uppladdningen", + "un_nou": '<b>varning:</b> servern är för upptagen för att visa pågående uppladdningar; klicka på "uppdatera" om en stund', + "un_noc": '<b>varning:</b> serverkonfigurationen tillåter inte unpost:ning av uppladdade filer', + "un_max": "visar de första 2000 filerna (använd filtret)", + "un_avail": "{0} av de senaste uppladdningarna kan raderas<br />{1} pågående uppladdningar kan avbrytas", + "un_m2": "sorterat efter uppladdningstid; senast uppladdad först:", + "un_no1": "tjosan! inga uppladdningar är tillräckligt nya", + "un_no2": "tjosan! inga uppladdningar som matchar filtret är tillräckligt nya", + "un_next": "radera de {0} nästkommande filerna", + "un_abrt": "avbryt", + "un_del": "radera", + "un_m3": "laddar dina senaste uppladdningar...", + "un_busy": "raderar {0} filer...", + "un_clip": "{0} länkar kopierade till urklippet", + + "u_https1": "du bör", + "u_https2": "byta till https", + "u_https3": "för bättre prestanda", + "u_ancient": 'din webbläsare är imponerande uråldrig -- du kanske borde <a href="#" onclick="goto(\'bup\')">använda bup istället</a>', + "u_nowork": "firefox 53+ eller chrome 57+ eller iOS 11+ krävs", + "tail_2old": "firefox 105+ eller chrome 71+ eller iOS 14.5+ krävs", + "u_nodrop": 'din webbläsare är för gammal för dra-och-släpp-uppladdning', + "u_notdir": "det där är ingen mapp!\n\ndin webbläsare är för gammal,\nprova dra-och-släpp istället", + "u_uri": "släpp bilder från andra webbläsarfönster på den stora\nuppladdningsknappen för att ladda upp dem", + "u_enpot": 'byt till <a href="#">potatisgränssnittet</a> (kan förbättra uppladdningshastigheten)', + "u_depot": 'byt till <a href="#">det snygga gränssnittet</a> (kan försämra uppladdningshastigheten)', + "u_gotpot": 'byter till potatisgränssnittet för förbättrad uppladdningshastighet,\n\nbyt gärna tillbaka om du vill!', + "u_pott": "<p>filer:   <b>{0}</b> färdiga,   <b>{1}</b> misslyckade,   <b>{2}</b> pågående,   <b>{3}</b> köade</p>", + "u_ever": "detta är den enkla uppladdaren; up2k kräver minst<br>chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'detta är den enkla uppladdaren; <a href="#" id="u2yea">up2k</a> är bättre', + "u_uput": 'optimera hastigheten (skippa checksumman)', + "u_ewrite": 'du har inte skrivrättighet i denna mapp', + "u_eread": 'du har inte läsrättighet i denna mapp', + "u_enoi": 'serverkonfigurationen har inte slagit på sökning', + "u_enoow": "du kan inte skriva över här; raderingsrättighet krävs", + "u_badf": 'Dessa {0} filer (av totalt {1}) skippades, möjligtvis p.g.a. filsystemsrättigheter:\n\n', + "u_blankf": 'Dessa {0} filer (av totalt {1}) är tomma; ladda upp dem ändå?\n\n', + "u_applef": 'Dessa {0} filer (av totalt {1}) är förmodligen oönskade;\nTryck <code>OK/Enter</code> för att SKIPPA de följande filerna,\nTryck <code>Avbryt/ESC</code> för att INKLUDERA och LADDA UPP dem:\n\n', + "u_just1": '\nDet kanske fungerar om du endast väljer en fil', + "u_ff_many": "om du använder <b>Linux / MacOS / Android,</b> så <em>kan</em> denna mängd filer <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\">krascha Firefox!</a>\nom detta händer, vänligen försök igen (eller använd Chrome).", + "u_up_life": "Denna uppladdning kommer att raderas från servern om\n{0} efter att den har blivit uppladdad", + "u_asku": 'ladda upp dessa {0} filer till <code>{1}</code>', + "u_unpt": "du kan ångra / radera denna uppladdning med 🧯 uppe till vänster", + "u_bigtab": 'försöker att visa {0} filer\n\ndetta kan krascha din webbläsare, är du säker?', + "u_scan": 'Scannar filer...', + "u_dirstuck": 'katalogskannern fastnade när den försökte komma åt de följande {0} objekten; dessa kommer att skippas:', + "u_etadone": 'Klar ({0}, {1} filer)', + "u_etaprep": '(förbereder uppladdning)', + "u_hashdone": 'hashning klar', + "u_hashing": 'hashar', + "u_hs": 'skakar hand...', + "u_started": "filerna laddas nu upp; se [🚀]", + "u_dupdefer": "duplikat; kommer att behandlas efter alla andra filer", + "u_actx": "klicka här för att undvika prestandaförlust<br />när du byter till andra fönster/flikar", + "u_fixed": "Okej!  Fixat 👍", + "u_cuerr": "misslyckades att ladda upp chunk {0} av {1};\nförmodligen harmlöst, fortsätter\n\nfil: {2}", + "u_cuerr2": "servern avvisade uppladdningen (chunk {0} av {1});\nprovar igen senare\n\nfil: {2}\n\nfel ", + "u_ehstmp": "provar igen; see nedåt till höger", + "u_ehsfin": "servern avvisade förfrågan att färdigställa uppladdningen; provar igen...", + "u_ehssrch": "servern avvisade förfrågan att söka; provar igen...", + "u_ehsinit": "servern avvisade förfrågan att påbörja uppladdningen; provar igen...", + "u_eneths": "nätverksfel vid handskakning; provar igen...", + "u_enethd": "nätverksfel när destinationens existens testades; provar igen...", + "u_cbusy": "väntar på att servern ska lita på oss igen efter nätverksfel...", + "u_ehsdf": "servern fick slut på diskutrymme!\n\nprovar igen, ifall någon rensar upp\ntillräckligt med utrymme för att fortsätta", + "u_emtleak1": "det verkar som att din webbläsare kanske har en minnesläcka;\nvänligen", + "u_emtleak2": ' <a href="{0}">byt till https (rekommenderat)</a> eller ', + "u_emtleak3": ' ', + "u_emtleakc": 'prova följande:\n<ul><li>tryck <code>F5</code> för att uppdatera sidan</li><li>avaktivera sedan  <code>mt</code> -växeln i  <code>⚙️-inställningarna</code></li><li>och prova att ladda upp igen</li></ul>Uppladdningar kommer att vara lite långsammare, men aja.\nBeklagar problemet!\n\nPS: chrome v107 <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">har en buggfix</a> för detta', + "u_emtleakf": 'prova följande:\n<ul><li>tryck <code>F5</code> för att uppdatera sidan</li><li>aktivera sedan <code>🥔</code> (potatis) i uppladdningsgränssnittet<li>och prova att ladda upp igen</li></ul>\nPS: firefox <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">kommer förhoppningsvis få en buggfix</a> vid något tillfälle', + "u_s404": "hittades ej på servern", + "u_expl": "förklara", + "u_maxconn": "de flesta webbläsare begränsar detta till 6, men firefox låter dig höja gränsen med <code>connections-per-server</code> i <code>about:config</code>", + "u_tu": '<p class="warn">VARNING: turbo är aktiverat, <span> det är möjligt att klienten inte upptäcker och återupptar ofärdiga uppladdningar; se tipset för turbo-växeln</span></p>', + "u_ts": '<p class="warn">VARNING: turbo är aktiverat, <span> sökresultat kan vara felaktiga; se tipset för turbo-växeln</span></p>', + "u_turbo_c": "serverkonfigurationen har avaktiverat turbo", + "u_turbo_g": "avaktiverar turbo eftersom du inte har rättigheten\natt se mappars innehåll i den här volymen", + "u_life_cfg": 'radera automatiskt efter <input id="lifem" p="60" /> min (eller <input id="lifeh" p="3600" /> timmar)', + "u_life_est": 'uppladdningen kommer att raderas vid <span id="lifew" tt="local time">---</span>', + "u_life_max": 'denna mapp tvingar en\nhögsta livstid på {0}', + "u_unp_ok": 'unpost är tillåten för {0}', + "u_unp_ng": 'unpost är INTE tillåten', + "ue_ro": 'du har endast läsrättighet till denna mapp\n\n', + "ue_nl": 'du är inte inloggad', + "ue_la": 'du är inloggad som "{0}"', + "ue_sr": 'du är i filsökläge\n\nbyt till uppladdningsläge genom att klicka på förstoringsglaset 🔎 (bredvid den stora SÖK-knappen), och försök ladda upp igen\n\nledsen', + "ue_ta": 'prova att ladda upp igen nu, det bör fungera', + "ue_ab": "denna fil laddas redan upp till en annan mapp, och den uppladdningen måste färdigställas innan filen kan laddas upp någon annanstans.\n\nDu kan avbryta och glömma bort den uppladdningen med 🧯 uppe till vänster", + "ur_1uo": "Okej: Filen laddades upp med framgång", + "ur_auo": "Okej: Alla {0} filer laddades upp med framgång", + "ur_1so": "Okej: Filen fanns på servern", + "ur_aso": "Okej: Alla {0} filer fanns på servern", + "ur_1un": "Uppladdningen misslyckades, ledsen", + "ur_aun": "Alla {0} uppladdningar misslyckades, ledsen", + "ur_1sn": "Filen hittades INTE på servern", + "ur_asn": "De {0} filerna hittades INTE på servern", + "ur_um": "Klar;\n{0} uppladdningar gick okej,\n{1} uppladdningar misslyckades, ledsen", + "ur_sm": "Klar;\n{0} filer hittades på servern,\n{1} filer hittades INTE på servern", + + "lang_set": "uppdatera för att ändringen ska ta effekt?", + }, "ukr": { "tt": "Українська", @@ -9460,7 +10090,7 @@ var Ls = { }, }; -var LANGS = ["eng", "nor", "nno", "chi", "cze", "deu", "fin", "fra", "grc", "ita", "nld", "rus", "spa", "ukr"]; +var LANGS = ["eng", "nor", "nno", "chi", "cze", "deu", "fin", "fra", "grc", "ita", "nld", "rus", "spa", "swe", "ukr"]; if (window.langmod) langmod(); diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 8cbca1e6..3229b9a2 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -510,6 +510,48 @@ var Ls = { "af1": "mostrar subidas recientes", "ag1": "mostrar usuarios IdP conocidos" }, + "swe": { + "a1": "uppdatera", + "b1": "tjena främling   <small>(du är inte inloggad)</small>", + "c1": "logga ut", + "d1": "dumpa stacken", + "d2": "visar tillståndet på alla aktiva trådar", + "e1": "ladda om konfig.", + "e2": "ladda om konfigurationsfiler (konton/volymer/volflaggor),$Noch skanna om alla e2ds-volymer$N$Nobs.: ändrade globala inställningar$Nkräver en fullständig omstart", + "f1": "du kan bläddra:", + "g1": "du kan ladda upp till:", + "cc1": "annat:", + "h1": "avaktivera k304", + "i1": "aktivera k304", + "j1": "med k304 aktiverad kommer klienten att koppla bort sig vid varje HTTP 304-fel, vilket kan hindra vissa buggiga proxyservrar från att fastna (sidor slutar ladda), <em>men</em> saker kommer också att bli långsammare i allmänhet", + "k1": "återställ klientinställningar", + "l1": "logga in för att se mer:", + "m1": "välkommen tillbaka,", + "n1": "404 hittades inte  ┐( ´ -`)┌", + "o1": 'eller så har du kanske inte tillgång -- prova ett lösenord eller <a href="' + SR + '/?h">åk hem</a>', + "p1": "403 nekat  ~┻━┻", + "q1": 'använd ett lösenord eller <a href="' + SR + '/?h">åk hem</a>', + "r1": "åk hem", + ".s1": "skanna om", + "t1": "åtgärd", + "u2": "tid sedan senaste serverskrivning$N( uppladdning / namnbyte / ... )$N$N17d = 17 dagar$N1h23 = 1 timme 23 minuter$N4m56 = 4 minuter 56 sekunder", + "v1": "koppla upp", + "v2": "använd denna server som en lokal disk", + "w1": "byt till https", + "x1": "byt lösenord", + "y1": "redigera utdelningar", + "z1": "lås upp denna utdelning:", + "ta1": "fyll i ditt nya lösenord", + "ta2": "upprepa det nya lösenordet:", + "ta3": "det blev fel; vänligen försök igen", + "aa1": "inkommande filer:", + "ab1": "avaktivera no304", + "ac1": "aktivera no304", + "ad1": "detta stänger av all cachning; prova detta om k304 inte räckte till. Detta kommer att slösa enorma mängder nätverkstrafik!", + "ae1": "aktiva nedladdningar:", + "af1": "visa senaste uppladdningar", + "ag1": "visa idp-cache" + }, "ukr": { "a1": "оновити", "b1": "привітик, незнайомцю   <small>(ви не авторизовані)</small>", From 659f351c65237ce0a30319dacc6ad127308cc09f Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 16:42:48 +0000 Subject: [PATCH 155/174] support pillow-heif; closes #607 --- README.md | 4 ++-- copyparty/th_srv.py | 5 ++++- docs/devnotes.md | 2 +- 3 files changed, 7 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index b3a2d134..c8520202 100644 --- a/README.md +++ b/README.md @@ -2793,7 +2793,7 @@ enable [music tags](#metadata-from-audio-files): enable [thumbnails](#thumbnails) of... * **images:** `Pillow` and/or `pyvips` and/or `ffmpeg` (requires py2.7 or py3.5+) * **videos/audio:** `ffmpeg` and `ffprobe` somewhere in `$PATH` -* **HEIF pictures:** `pyvips` or `ffmpeg` or `pyheif-pillow-opener` (requires Linux or a C compiler) +* **HEIF pictures:** `pyvips` or `ffmpeg` or `pillow-heif` * **AVIF pictures:** `pyvips` or `ffmpeg` or `pillow-avif-plugin` or pillow v11.3+ * **JPEG XL pictures:** `pyvips` or `ffmpeg` * **RAW images:** `rawpy`, plus one of `pyvips` or `Pillow` (for some formats) @@ -2827,7 +2827,7 @@ set any of the following environment variables to disable its associated optiona | `PRTY_NO_PIL` | disable all [Pillow](https://pypi.org/project/pillow/)-based thumbnail support; will fallback to libvips or ffmpeg | | `PRTY_NO_PILF` | disable Pillow `ImageFont` text rendering, used for folder thumbnails | | `PRTY_NO_PIL_AVIF` | disable Pillow avif support (internal and/or [plugin](https://pypi.org/project/pillow-avif-plugin/)) | -| `PRTY_NO_PIL_HEIF` | disable 3rd-party Pillow plugin for [HEIF support](https://pypi.org/project/pyheif-pillow-opener/) | +| `PRTY_NO_PIL_HEIF` | disable 3rd-party Pillow plugin for [HEIF support](https://pypi.org/project/pillow-heif/) | | `PRTY_NO_PIL_WEBP` | disable use of native webp support in Pillow | | `PRTY_NO_PSUTIL` | do not use [psutil](https://pypi.org/project/psutil/) for reaping stuck hooks and plugins on Windows | | `PRTY_NO_RAW` | disable all [rawpy](https://pypi.org/project/rawpy/)-based thumbnail support for RAW images | diff --git a/copyparty/th_srv.py b/copyparty/th_srv.py index 945c6d56..537c748a 100644 --- a/copyparty/th_srv.py +++ b/copyparty/th_srv.py @@ -86,7 +86,10 @@ try: if os.environ.get("PRTY_NO_PIL_HEIF"): raise Exception() - from pyheif_pillow_opener import register_heif_opener + try: + from pillow_heif import register_heif_opener + except ImportError: + from pyheif_pillow_opener import register_heif_opener register_heif_opener() HAVE_HEIF = True diff --git a/docs/devnotes.md b/docs/devnotes.md index 9fbb29df..e2fcc008 100644 --- a/docs/devnotes.md +++ b/docs/devnotes.md @@ -354,7 +354,7 @@ pip install mutagen # audio metadata pip install pyftpdlib # ftp server pip install partftpy # tftp server pip install impacket # smb server -- disable Windows Defender if you REALLY need this on windows -pip install Pillow pyheif-pillow-opener # thumbnails +pip install Pillow pillow-heif # thumbnails pip install pyvips # faster thumbnails pip install psutil # better cleanup of stuck metadata parsers on windows pip install black==21.12b0 click==8.0.2 bandit pylint flake8 isort mypy # vscode tooling From 6303effe59fb4004a075332f9487ae1b6d4c8b3a Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 17:49:48 +0000 Subject: [PATCH 156/174] configurable max num cookies --- copyparty/__main__.py | 2 ++ copyparty/httpcli.py | 8 ++++++-- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 86d8cdd5..22e219a7 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1363,6 +1363,8 @@ def add_safety(ap): ap2.add_argument("--sus-urls", metavar="R", type=u, default=r"\.php$|(^|/)wp-(admin|content|includes)/", help="URLs which are considered sus / eligible for banning; disable with blank or [\033[32mno\033[0m]") ap2.add_argument("--nonsus-urls", metavar="R", type=u, default=r"^(favicon\.ico|robots\.txt)$|^apple-touch-icon|^\.well-known", help="harmless URLs ignored from 404-bans; disable with blank or [\033[32mno\033[0m]") ap2.add_argument("--early-ban", action="store_true", help="if a client is banned, reject its connection as soon as possible; not a good idea to enable when proxied behind cloudflare since it could ban your reverse-proxy") + ap2.add_argument("--cookie-nmax", metavar="N", type=int, default=50, help="reject HTTP-request from client if they send more than N cookies") + ap2.add_argument("--cookie-cmax", metavar="N", type=int, default=8192, help="reject HTTP-request from client if more than N characters in Cookie header") ap2.add_argument("--aclose", metavar="MIN", type=int, default=10, help="if a client maxes out the server connection limit, downgrade it from connection:keep-alive to connection:close for \033[33mMIN\033[0m minutes (and also kill its active connections) -- disable with 0") ap2.add_argument("--loris", metavar="B", type=int, default=60, help="if a client maxes out the server connection limit without sending headers, ban it for \033[33mB\033[0m minutes; disable with [\033[32m0\033[0m]") ap2.add_argument("--acao", metavar="V[,V]", type=u, default="*", help="Access-Control-Allow-Origin; list of origins (domains/IPs without port) to accept requests from; [\033[32mhttps://1.2.3.4\033[0m]. Default [\033[32m*\033[0m] allows requests from all sites but removes cookies and http-auth; only ?pw=hunter2 survives") diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index a7ff303d..479c4f07 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -562,7 +562,7 @@ class HttpCli(object): zso = self.headers.get("cookie") if zso: - if len(zso) > 8192: + if len(zso) > self.args.cookie_cmax: self.loud_reply("cookie header too big", status=400) return False zsll = [x.split("=", 1) for x in zso.split(";") if "=" in x] @@ -570,11 +570,15 @@ class HttpCli(object): cookie_pw = cookies.get("cppws") or cookies.get("cppwd") or "" if "b" in cookies and "b" not in uparam: uparam["b"] = cookies["b"] + if len(cookies) > self.args.cookie_nmax: + self.loud_reply("too many cookies", status=400) else: cookies = {} cookie_pw = "" - if len(uparam) > 10 or len(cookies) > 50: + if len(uparam) > 12: + t = "http-request rejected; num.params: %d %r" + self.log(t % (len(uparam), self.req), 3) self.loud_reply("u wot m8", status=400) return False From 554cc2f3eebe8ff4ad1fa3488d78a30510d5e7df Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 18:09:23 +0000 Subject: [PATCH 157/174] support xdev/xvol in rootless vfs; closes #603 --- copyparty/httpcli.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 479c4f07..c262dc2e 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -704,7 +704,7 @@ class HttpCli(object): cookies["b"] = "" vn, rem = self.asrv.vfs.get(self.vpath, self.uname, False, False) - if "xdev" in vn.flags or "xvol" in vn.flags: + if vn.realpath and ("xdev" in vn.flags or "xvol" in vn.flags): ap = vn.canonical(rem) avn = vn.chk_ap(ap) else: From 5b627425125e0c3c372e009859d251d889871f72 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 18:10:55 +0000 Subject: [PATCH 158/174] run from src with py3.9 --- copyparty/mtag.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/copyparty/mtag.py b/copyparty/mtag.py index 660f8180..43e9dd75 100644 --- a/copyparty/mtag.py +++ b/copyparty/mtag.py @@ -368,7 +368,7 @@ def parse_ffprobe(txt: str) -> tuple[dict[str, tuple[int, Any]], dict[str, list[ return zd, md -def get_cover_from_epub(log: "NamedLogger", abspath: str) -> IO[bytes] | None: +def get_cover_from_epub(log: "NamedLogger", abspath: str) -> Optional[IO[bytes]]: import zipfile from .dxml import parse_xml @@ -418,7 +418,9 @@ def get_cover_from_epub(log: "NamedLogger", abspath: str) -> IO[bytes] | None: return z.open(adjusted_cover_path) -def _get_cover_from_epub2(log: "NamedLogger", package_root, package_ns) -> str | None: +def _get_cover_from_epub2( + log: "NamedLogger", package_root, package_ns +) -> Optional[str]: # <meta name="cover" content="id-to-cover-image"> in <metadata>, then # <item> in <manifest> cover_id = package_root.find("./metadata/meta[@name='cover']", package_ns).get( From c4a4fddd272f31944a7f3615c4f3c62da5265756 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 18:36:31 +0000 Subject: [PATCH 159/174] translations: OK/Cancel; closes #599 --- copyparty/web/browser.js | 42 ++++++++++++++++++++-------------------- 1 file changed, 21 insertions(+), 21 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 047c6446..000edbb7 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -1613,7 +1613,7 @@ var Ls = { "f_dls": '当前文件夹中的文件链接已\n更改为下载链接', - "f_partial": "要安全下载正在上传的文件,请点击没有 <code>.PARTIAL</code> 文件扩展名的同名文件。请按取消或 Escape 执行此操作。\n\n按 OK / Enter 将忽略此警告并继续下载 <code>.PARTIAL</code> 临时文件,这几乎肯定会导致数据损坏。", + "f_partial": "要安全下载正在上传的文件,请点击没有 <code>.PARTIAL</code> 文件扩展名的同名文件。请按取消或 Escape 执行此操作。\n\n按 确定 / Enter 将忽略此警告并继续下载 <code>.PARTIAL</code> 临时文件,这几乎肯定会导致数据损坏。", "ft_paste": "粘贴 {0} 项$N快捷键: ctrl-V", "fr_eperm": '无法重命名:\n你在此文件夹中没有 “移动” 权限', @@ -2214,7 +2214,7 @@ var Ls = { "mb_play": "přehrát", "mm_hashplay": "přehrát tento audio soubor?", - "mm_m3u": "stiskněte <code>Enter/OK</code> pro Přehrání\nstiskněte <code>ESC/Cancel</code> pro Úpravu", + "mm_m3u": "stiskněte <code>Enter/OK</code> pro Přehrání\nstiskněte <code>ESC/Zrušit</code> pro Úpravu", "mp_breq": "potřebujete firefox 82+ nebo chrome 73+ nebo iOS 15+", "mm_bload": "nyní se načítá...", "mm_bconv": "převádí se na {0}, čekejte prosím...", @@ -2247,7 +2247,7 @@ var Ls = { "f_dls": 'odkazy na soubory v aktuální složce byly\nzměněny na odkazy ke stažení', - "f_partial": "Pro bezpečné stažení souboru, který se aktuálně nahrává, klikněte prosím na soubor se stejným názvem, ale bez přípony <code>.PARTIAL</code>. Stiskněte prosím CANCEL nebo Escape.\n\nStisknutím OK / Enter ignorujete toto varování a pokračujete ve stahování <code>.PARTIAL</code> dočasného souboru, což téměř jistě vyústí jako poškozená data.", + "f_partial": "Pro bezpečné stažení souboru, který se aktuálně nahrává, klikněte prosím na soubor se stejným názvem, ale bez přípony <code>.PARTIAL</code>. Stiskněte prosím Zrušit nebo Escape.\n\nStisknutím OK / Enter ignorujete toto varování a pokračujete ve stahování <code>.PARTIAL</code> dočasného souboru, což téměř jistě vyústí jako poškozená data.", "ft_paste": "vložit {0} položek$NKlávesová zkratka: ctrl-V", "fr_eperm": 'nelze přejmenovat:\nnemáte oprávnění “přesunout” v této složce', @@ -2467,7 +2467,7 @@ var Ls = { "u_enoow": "přepsání zde nebude fungovat; je vyžadováno oprávnění k mazání", "u_badf": "Těchto {0} souborů (z celkem {1}) bylo přeskočeno, pravděpodobně kvůli oprávněním v souborovém systému:\n\n", "u_blankf": "Těchto {0} souborů (z celkem {1}) je prázdných; přesto je nahrát?\n\n", - "u_applef": "Těchto {0} souborů (z celkem {1}) je pravděpodobně nežádoucích;\nStiskněte <code>OK/Enter</code> pro PŘESKOČENÍ následujících souborů,\nStiskněte <code>Cancel/ESC</code> pro Zahrnutí a NAHRÁNÍ i těchto souborů:\n\n", + "u_applef": "Těchto {0} souborů (z celkem {1}) je pravděpodobně nežádoucích;\nStiskněte <code>OK/Enter</code> pro PŘESKOČENÍ následujících souborů,\nStiskněte <code>Zrušit/ESC</code> pro Zahrnutí a NAHRÁNÍ i těchto souborů:\n\n", "u_just1": "\nMožná to bude fungovat lépe, když vyberete pouze jeden soubor", "u_ff_many": "pokud používáte <b>Linux / MacOS / Android,</b> takové množství souborů <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>může</em> shodit Firefox!</a>\npokud se to stane, zkuste to prosím znovu (nebo použijte Chrome).", "u_up_life": "Tento upload bude smazán ze serveru\n{0} po jeho dokončení", @@ -3537,7 +3537,7 @@ var Ls = { "fs_tsrc": "jaettava tiedosto tai hakemisto", "fs_ppwd": "valinnainen salasana", "fs_w8": "luodaan sharea...", - "fs_ok": "paina <code>Enter/OK</code> lisätäksesi leikepöydälle\npaina <code>ESC/Cancel</code> sulkeaksesi", + "fs_ok": "paina <code>Enter/OK</code> lisätäksesi leikepöydälle\npaina <code>ESC/Peruuta</code> sulkeaksesi", "frt_dec": "saattaa korjata joitakin rikkinäisiä tiedostonimiä\">url-decode", "frt_rst": "palauta muokatut tiedostonimet takaisin alkuperäisiksi\">↺ palauta", @@ -3727,7 +3727,7 @@ var Ls = { "u_enoow": "ylikirjoitus ei toimi täällä; tarvitaan “Delete”-oikeus", "u_badf": 'Nämä {0} tiedostoa ({1} yhteensä) ohitettiin, mahdollisesti tiedostojärjestelmän oikeuksien vuoksi:\n\n', "u_blankf": 'Nämä {0} tiedostoa ({1} yhteensä) ovat tyhjiä; ladataanko ne silti?\n\n', - "u_applef": 'Nämä {0} tiedostoa ({1} yhteensä) ovat todennäköisesti ei-toivottuja;\nPaina <code>OK/Enter</code> OHITTAAKSESI seuraavat tiedostot,\nPaina <code>Cancel/ESC</code> jos ET halua sulkea pois, ja LATAA nekin:\n\n', + "u_applef": 'Nämä {0} tiedostoa ({1} yhteensä) ovat todennäköisesti ei-toivottuja;\nPaina <code>OK/Enter</code> OHITTAAKSESI seuraavat tiedostot,\nPaina <code>Peruuta/ESC</code> jos ET halua sulkea pois, ja LATAA nekin:\n\n', "u_just1": '\nEhkä toimii paremmin jos valitset vain yhden tiedoston', "u_ff_many": "jos käytät <b>Linux / MacOS / Android,</b> niin tämä määrä tiedostoja <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>saattaa</em> kaataa Firefoxin!</a>\njos niin käy, kokeile uudelleen (tai käytä Chromea).", "u_up_life": "Tämä lataus poistetaan palvelimelta\n{0} sen valmistumisen jälkeen", @@ -4137,7 +4137,7 @@ var Ls = { "f_dls": 'le lien de fichier dans le répertoire actuel\nà été changé en lien de téléchargement', - "f_partial": "Pour télécharger de façon sécurisée un fichier qui est entrain de se faire téléverser, cliquez sur le fichier qui a le même nom, mais sans l'extension de fichier <code>.PARTIAL</code>. Choisissez CANCEL ou appuiez sur la touche Échap pour faire cela.\n\nAppuyer sur OK / Entrée ignorera cet avertissement et continuera à télécharger le fichier temporaire <code>.PARTIAL</code> à la place, ce qui donnera presque certainement des données corrompues.", + "f_partial": "Pour télécharger de façon sécurisée un fichier qui est entrain de se faire téléverser, cliquez sur le fichier qui a le même nom, mais sans l'extension de fichier <code>.PARTIAL</code>. Choisissez ANNULER ou appuiez sur la touche Échap pour faire cela.\n\nAppuyer sur OK / Entrée ignorera cet avertissement et continuera à télécharger le fichier temporaire <code>.PARTIAL</code> à la place, ce qui donnera presque certainement des données corrompues.", "ft_paste": "coller {0} éléments$NHotkey: ctrl-V", "fr_eperm": 'impossible de renommer:\n vous n\'avez pas la permission “move” dans ce dossier', @@ -4357,7 +4357,7 @@ var Ls = { "u_enoow": "l'écrasage ne fonctionnera pas ici; besoin de permissions de suppression", "u_badf": 'Ces {0} fichiers (sur {1} au total) ont été ignorés, probablement en raison de permissions système de fichiers:\n\n', "u_blankf": 'Ces {0} fichiers (sur {1} au total) sont vides; les téléverser quand même ?\n\n', - "u_applef": 'Ces {0} fichiers (sur {1} au total) sont probablement indésirables;\nAppuyez sur <code>OK/Enter</code> pour IGNORER les fichiers suivants,\nAppuyez sur <code>Cancel/Échap</code> pour NE PAS exclure, et TÉLÉVERSER ceux-ci également:\n\n', + "u_applef": 'Ces {0} fichiers (sur {1} au total) sont probablement indésirables;\nAppuyez sur <code>OK/Enter</code> pour IGNORER les fichiers suivants,\nAppuyez sur <code>Annuler/Échap</code> pour NE PAS exclure, et TÉLÉVERSER ceux-ci également:\n\n', "u_just1": '\nPeut-être que cela fonctionne mieux si vous sélectionnez juste un fichier', "u_ff_many": "si vous utilisez <b>Linux / MacOS / Android,</b> alors ce nombre de fichiers <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>peut</em> faire planter Firefox!</a>\nSi cela se produit, veuillez réessayer (ou utiliser Chrome).", "u_up_life": "Ce téléversement va être supprimé du serveur\n{0} après son achèvement", @@ -4734,7 +4734,7 @@ var Ls = { "mb_play": "παίξε", "mm_hashplay": "να παίξω αυτό το αρχείο ήχου;", - "mm_m3u": "πάτα <code>Enter/OK</code> για Αναπαραγωγή\nπάτα <code>ESC/Άκυρο</code> για Επεξεργασία", + "mm_m3u": "πάτα <code>Enter/Εντάξει</code> για Αναπαραγωγή\nπάτα <code>ESC/Άκυρο</code> για Επεξεργασία", "mp_breq": "χρειάζεται firefox 82+ ή chrome 73+ ή iOS 15+", "mm_bload": "φορτώνει...", "mm_bconv": "μετατροπή σε {0}, περίμενε...", @@ -4767,7 +4767,7 @@ var Ls = { "f_dls": 'οι σύνδεσμοι αρχείων στον τρέχοντα φάκελο έχουν\nμετατραπεί σε συνδέσμους λήψης', - "f_partial": "Για να κατεβάσεις με ασφάλεια ένα αρχείο που ανεβαίνει, κλίκαρε το αρχείο με το ίδιο όνομα, αλλά χωρίς την κατάληξη <code>.PARTIAL</code>. Πάτα Άκυρο ή Escape για να σταματήσεις.\n\nΠάτα OK / Enter αν αγνοείς την προειδοποίηση και κατέβασε το <code>.PARTIAL</code> αρχείο, που σχεδόν σίγουρα θα είναι κατεστραμμένο.", + "f_partial": "Για να κατεβάσεις με ασφάλεια ένα αρχείο που ανεβαίνει, κλίκαρε το αρχείο με το ίδιο όνομα, αλλά χωρίς την κατάληξη <code>.PARTIAL</code>. Πάτα Άκυρο ή Escape για να σταματήσεις.\n\nΠάτα Εντάξει / Enter αν αγνοείς την προειδοποίηση και κατέβασε το <code>.PARTIAL</code> αρχείο, που σχεδόν σίγουρα θα είναι κατεστραμμένο.", "ft_paste": "επικόλλησε {0} αντικείμενα$NΠλήκτρο συντόμευσης: ctrl-V", "fr_eperm": 'δεν μπορεί να μετονομαστεί:\nδεν έχεις δικαίωμα “μετακίνησης” σε αυτόν το φάκελο', @@ -4797,7 +4797,7 @@ var Ls = { "fs_tsrc": "το αρχείο ή ο φάκελος προς κοινή χρήση", "fs_ppwd": "προαιρετικός κωδικός", "fs_w8": "δημιουργία κοινής χρήσης...", - "fs_ok": "πάτα <code>Enter/OK</code> για Πρόχειρο\nπάτα <code>ESC/Άκυρο</code> για Κλείσιμο", + "fs_ok": "πάτα <code>Enter/Εντάξει</code> για Πρόχειρο\nπάτα <code>ESC/Άκυρο</code> για Κλείσιμο", "frt_dec": "μπορεί να διορθώσει μερικές περιπτώσεις κατεστραμμένων ονομάτων αρχείων\">αποκωδικοποίηση url", "frt_rst": "επανέφερε τα ονόματα αρχείων στα αρχικά τους\">↺ επαναφορά", @@ -4987,7 +4987,7 @@ var Ls = { "u_enoow": "δεν μπορείς να κάνεις αντικατάσταση εδώ· χρειάζεται δικαίωμα Διαγραφής", "u_badf": "Αυτά τα {0} αρχεία (από {1} συνολικά) παραλείφθηκαν, πιθανώς λόγω δικαιωμάτων συστήματος αρχείων:\n\n", "u_blankf": "Αυτά τα {0} αρχεία (από {1} συνολικά) είναι άδεια / κενά· να τα μεταφορτώσω έτσι κι αλλιώς;\n\n", - "u_applef": "Αυτά τα {0} αρχεία (από {1} συνολικά) πιθανώς δεν είναι επιθυμητά;\nΠάτα <code>OK/Enter</code> για ΝΑ ΑΓΝΟΗΘΟΥΝ τα παρακάτω αρχεία,\nΠάτα <code>Cancel/ESC</code> για ΝΑ ΜΗΝ ΑΠΟΚΛΕΙΣΤΟΥΝ και να ΜΕΤΑΦΟΡΤΩΘΟΎΝ κι αυτά:\n\n", + "u_applef": "Αυτά τα {0} αρχεία (από {1} συνολικά) πιθανώς δεν είναι επιθυμητά;\nΠάτα <code>Εντάξει/Enter</code> για ΝΑ ΑΓΝΟΗΘΟΥΝ τα παρακάτω αρχεία,\nΠάτα <code>Άκυρο/ESC</code> για ΝΑ ΜΗΝ ΑΠΟΚΛΕΙΣΤΟΥΝ και να ΜΕΤΑΦΟΡΤΩΘΟΎΝ κι αυτά:\n\n", "u_just1": "\nΊσως δουλέψει καλύτερα αν επιλέξεις μόνο ένα αρχείο", "u_ff_many": "αν χρησιμοποιείς <b>Linux / MacOS / Android,</b> τότε τόσα αρχεία <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>μπορεί</em> να κατάρρευση του Firefox!</a>\nαν γίνει αυτό, δοκίμασε ξανά (ή χρησιμοποίησε τον Chrome).", "u_up_life": "Αυτή η μεταφόρτωση θα διαγραφεί από το διακομιστή\n{0} μετά την ολοκλήρωσή της", @@ -5994,7 +5994,7 @@ var Ls = { "mb_play": "Afspelen", "mm_hashplay": "Deze audio bestand afspelen?", - "mm_m3u": "Druk op <code>Enter/OK</code> om af te spelen\nDruk op <code>ESC/Cancel</code> om te bewerken", + "mm_m3u": "Druk op <code>Enter/OK</code> om af te spelen\nDruk op <code>ESC/Annuleren</code> om te bewerken", "mp_breq": "Heeft firefox 82+ of chrome 73+ of iOS 15+", "mm_bload": "Aan het laden...", "mm_bconv": "Opmzetten naar {0}, even geduld...", @@ -6057,7 +6057,7 @@ var Ls = { "fs_tsrc": "Het bestand of de map die u wilt delen", "fs_ppwd": "Optioneel wachtwoord", "fs_w8": "Delen...", - "fs_ok": "Druk op <code>Enter/OK</code> naar klembord te zetten\Druk op <code>ESC/Cancel</code> om te sluiten", + "fs_ok": "Druk op <code>Enter/OK</code> naar klembord te zetten\Druk op <code>ESC/Annuleren</code> om te sluiten", "frt_dec": "Kan sommige gevallen van gebroken bestandsnamen oplossen\">url-decode", "frt_rst": "Gewijzigde bestandsnamen terugzetten naar de oorspronkelijke namen\">↺ reset", @@ -6247,7 +6247,7 @@ var Ls = { "u_enoow": "Overschrijven zal hier niet werken; je heb verwijder toestemming nodig", "u_badf": 'Deze {0} bestanden (van {1} totaal) zijn overgeslagen, mogelijk door bestandssysteemmachtigingen:\n\n', "u_blankf": 'Deze {0} bestanden (van {1} totaal) zijn leeg; alsnog uploaden?\n\n', - "u_applef": 'Deze {0} bestanden (van {1} totaal) zijn waarschijnlijk ongewenst;\nKlik op <code>OK/Enter</code> om de volgende bestanden over te slaan,\Klik op <code>Cancel/ESC</code> niet uit te sluiten en deze ook te uploaden:\n\n', + "u_applef": 'Deze {0} bestanden (van {1} totaal) zijn waarschijnlijk ongewenst;\nKlik op <code>OK/Enter</code> om de volgende bestanden over te slaan,\Klik op <code>Annuleren/ESC</code> niet uit te sluiten en deze ook te uploaden:\n\n', "u_just1": '\nMisschien werkt het beter als je slechts één bestand selecteert', "u_ff_many": "Als je <b>Linux / MacOS / Android,</b> gebruikt dan <em>kan</em> deze hoeveelheid bestanden <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\">Firefox crashen!</a>\nals dat gebeurt, probeer het dan opnieuw (of gebruik Chrome).", "u_up_life": "Deze upload wordt verwijderd van de server\n{0} nadat het is voltooid", @@ -7252,7 +7252,7 @@ var Ls = { "mb_play": "odtwórz", "mm_hashplay": "odtworzyć ten plik audio?", - "mm_m3u": "naciśnij <code>Enter/OK</code>, aby odtworzyć\nnaciśnij <code>ESC/Cancel</code>, aby edytować", + "mm_m3u": "naciśnij <code>Enter/OK</code>, aby odtworzyć\nnaciśnij <code>ESC/Anuluj</code>, aby edytować", "mp_breq": "wymagany jest Firefox 82+, Chrome 73+ lub iOS 15+", "mm_bload": "wczytywanie...", "mm_bconv": "konwertowanie do {0}, proszę czekać...", @@ -7945,7 +7945,7 @@ var Ls = { "fs_tsrc": "путь к файлу или папке, которыми нужно поделиться", "fs_ppwd": "пароль (необязательно)", "fs_w8": "создаю доступ...", - "fs_ok": "нажмите <code>Enter/OK</code>, чтобы скопировать\nнажмите <code>ESC/Cancel</code>, чтобы закрыть", + "fs_ok": "нажмите <code>Enter/OK</code>, чтобы скопировать\nнажмите <code>ESC/Отмена</code>, чтобы закрыть", "frt_dec": "может исправить некоторые случаи с некорректными именами файлов\">декодировать url", "frt_rst": "сбросить изменённые имена обратно к оригинальным\">↺ сброс", @@ -9771,7 +9771,7 @@ var Ls = { "mb_play": "відтворити", "mm_hashplay": "відтворити цей аудіо файл?", - "mm_m3u": "натисніть <code>Enter/OK</code> для відтворення\nнатисніть <code>ESC/Cancel</code> для редагування", + "mm_m3u": "натисніть <code>Enter/Гаразд</code> для відтворення\nнатисніть <code>ESC/Скасувати</code> для редагування", "mp_breq": "потрібен firefox 82+ або chrome 73+ або iOS 15+", "mm_bload": "зараз завантажується...", "mm_bconv": "конвертується в {0}, будь ласка, зачекайте...", @@ -9804,7 +9804,7 @@ var Ls = { "f_dls": 'посилання на файли в поточній папці були\nзмінені на посилання для завантаження', - "f_partial": "Щоб безпечно завантажити файл, який зараз завантажується, будь ласка, клацніть на файл, який має таке саме ім'я, але без розширення <code>.PARTIAL</code>. Будь ласка, натисніть CANCEL або Escape, щоб зробити це.\n\nНатиснення OK / Enter проігнорує це попередження і продовжить завантаження <code>.PARTIAL</code> робочого файлу замість цього, що майже напевно дасть вам пошкоджені дані.", + "f_partial": "Щоб безпечно завантажити файл, який зараз завантажується, будь ласка, клацніть на файл, який має таке саме ім'я, але без розширення <code>.PARTIAL</code>. Будь ласка, натисніть Скасувати або Escape, щоб зробити це.\n\nНатиснення Гаразд / Enter проігнорує це попередження і продовжить завантаження <code>.PARTIAL</code> робочого файлу замість цього, що майже напевно дасть вам пошкоджені дані.", "ft_paste": "вставити {0} елементів$NГаряча клавіша: ctrl-V", "fr_eperm": 'не можу перейменувати:\nу вас немає дозволу “переміщення“ в цій папці', @@ -9834,7 +9834,7 @@ var Ls = { "fs_tsrc": "файл або папка для спільного доступу", "fs_ppwd": "необов'язковий пароль", "fs_w8": "створення спільного доступу...", - "fs_ok": "натисніть <code>Enter/OK</code> для копіювання до буфера\nнатисніть <code>ESC/Cancel</code> для закриття", + "fs_ok": "натисніть <code>Enter/Гаразд</code> для копіювання до буфера\nнатисніть <code>ESC/Скасувати</code> для закриття", "frt_dec": "може виправити деякі випадки пошкоджених імен файлів\">url-decode", "frt_rst": "скинути змінені імена файлів назад до оригінальних\">↺ reset", @@ -10024,7 +10024,7 @@ var Ls = { "u_enoow": "перезапис не працюватиме тут; потрібен дозвіл на видалення", "u_badf": 'Ці {0} файли (з {1} загальних) були пропущені, можливо, через дозволи файлової системи:\n\n', "u_blankf": 'Ці {0} файли (з {1} загальних) порожні; все одно завантажити їх?\n\n', - "u_applef": 'Ці {0} файли (з {1} загальних), ймовірно, небажані;\nНатисніть <code>OK/Enter</code> щоб ПРОПУСТИТИ наступні файли,\nНатисніть <code>Cancel/ESC</code> щоб НЕ виключати, і ЗАВАНТАЖИТИ їх також:\n\n', + "u_applef": 'Ці {0} файли (з {1} загальних), ймовірно, небажані;\nНатисніть <code>Гаразд/Enter</code> щоб ПРОПУСТИТИ наступні файли,\nНатисніть <code>Скасувати/ESC</code> щоб НЕ виключати, і ЗАВАНТАЖИТИ їх також:\n\n', "u_just1": '\nМожливо, це спрацює краще, якщо ви виберете лише один файл', "u_ff_many": "якщо ви використовуєте <b>Linux / MacOS / Android,</b> то така кількість файлів <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>може</em> завісити Firefox!</a>\nякщо це станеться, будь ласка, спробуйте знову (або використовуйте Chrome).", "u_up_life": "Це завантаження буде видалено з сервера\n{0} після його завершення", From 68907eaf487ff83b07daea9f25d600fdd6573628 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 19:11:57 +0000 Subject: [PATCH 160/174] add "@acct", a group with all authed users; closes #604 --- README.md | 3 +++ copyparty/__main__.py | 4 ++++ copyparty/authsrv.py | 4 ++++ tests/util.py | 3 +++ 4 files changed, 14 insertions(+) diff --git a/README.md b/README.md index c8520202..16bfb15f 100644 --- a/README.md +++ b/README.md @@ -513,6 +513,8 @@ examples: * replacing the `g` permission with `wg` would let anonymous users upload files, but not see the required filekey to access it * replacing the `g` permission with `wG` would let anonymous users upload files, receiving a working direct link in return +if you want to grant access to all users who are logged in, the group `acct` will always contain all known users, so for example `-v /mnt/music:music:r,@acct` + anyone trying to bruteforce a password gets banned according to `--ban-pw`; default is 24h ban for 9 failed attempts in 1 hour and if you want to use config files instead of commandline args (good!) then here's the same examples as a configfile; save it as `foobar.conf` and use it like this: `python copyparty-sfx.py -c foobar.conf` @@ -538,6 +540,7 @@ and if you want to use config files instead of commandline args (good!) then her accs: r: u1, u2 # only these accounts can read, r: @g1 # (exactly the same, just with a group instead) + r: @acct # (alternatively, ALL users who are logged in) rw: u3 # and only u3 can read-write [/inc] diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 22e219a7..38f786d8 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -609,6 +609,9 @@ def get_sects(): if no accounts or volumes are configured, current folder will be read/write for everyone + the group @acct will always have every user with an account + (the name of that group can be changed with --grp-all) + consider the config file for more flexible account/volume management, including dynamic reload at runtime (and being more readable w) """ @@ -1163,6 +1166,7 @@ def add_auth(ap): ap2.add_argument("--ses-db", metavar="PATH", type=u, default=ses_db, help="where to store the sessions database (if you run multiple copyparty instances, make sure they use different DBs)") ap2.add_argument("--ses-len", metavar="CHARS", type=int, default=20, help="session key length; default is 120 bits ((20//4)*4*6)") ap2.add_argument("--no-ses", action="store_true", help="disable sessions; use plaintext passwords in cookies") + ap2.add_argument("--grp-all", metavar="NAME", type=u, default="acct", help="the name of the auto-generated group which contains every username which is known") ap2.add_argument("--ipu", metavar="CIDR=USR", type=u, action="append", help="\033[34mREPEATABLE:\033[0m users with IP matching \033[33mCIDR\033[0m are auto-authenticated as username \033[33mUSR\033[0m; example: [\033[32m172.16.24.0/24=dave]") diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 0f1c94f9..3bea55c8 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -1099,6 +1099,9 @@ class AuthSrv(object): if rejected: continue + if gn == self.args.grp_all: + gn = "" + # if ap/vp has a user/group placeholder, make sure to keep # track so the same user/group is mapped when setting perms; # otherwise clear un/gn to indicate it's a regular volume @@ -1208,6 +1211,7 @@ class AuthSrv(object): self.load_idp_db(bool(self.idp_accs)) ret = {un: gns[:] for un, gns in self.idp_accs.items()} ret.update({zs: [""] for zs in acct if zs not in ret}) + grps[self.args.grp_all] = list(ret.keys()) for gn, uns in grps.items(): for un in uns: try: diff --git a/tests/util.py b/tests/util.py index bbd0b215..16c75e92 100644 --- a/tests/util.py +++ b/tests/util.py @@ -185,9 +185,12 @@ class Cfg(Namespace): E=E, bup_ck="sha512", chmod_d="755", + cookie_cmax=8192, + cookie_nmax=50, dbd="wal", dk_salt="b" * 16, fk_salt="a" * 16, + grp_all="acct", idp_gsep=re.compile("[|:;+,]"), iobuf=256 * 1024, lang="eng", From 7f448750610426740c39da5db4e64184899a9fd2 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 19:22:04 +0000 Subject: [PATCH 161/174] autogen pw for blank-pw users (closes #596); if a user is defined with a blank password, generate a strong password for that user --- copyparty/authsrv.py | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 3bea55c8..63613cc8 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -1885,6 +1885,16 @@ class AuthSrv(object): if LEELOO_DALLAS in all_users: raise Exception("sorry, reserved username: " + LEELOO_DALLAS) + zsl = [] + for usr in list(acct)[:]: + zs = acct[usr].strip() + if not zs: + zs = ub64enc(os.urandom(48)).decode("ascii") + zsl.append(usr) + acct[usr] = zs + if zsl: + self.log("generated random passwords for users %r" % (zsl,), 6) + seenpwds = {} for usr, pwd in acct.items(): if pwd in seenpwds: From 4e878d2f1e2231027040de65cdb1cd97878f670c Mon Sep 17 00:00:00 2001 From: nyqui <67160376+nyqui@users.noreply.github.com> Date: Fri, 15 Aug 2025 04:30:41 +0900 Subject: [PATCH 162/174] initial Korean translation (#583) Signed-off-by: nyqui <67160376+nyqui@users.noreply.github.com> --- copyparty/web/browser.js | 631 ++++++++++++++++++++++++++++++++++++++- copyparty/web/splash.js | 42 +++ 2 files changed, 672 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 000edbb7..a387495e 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -5681,6 +5681,635 @@ var Ls = { "lang_set": "aggiornare per rendere effettivo il cambiamento?", }, + "kor": { + "tt": "한국어", + + "cols": { + "c": "작업 버튼", + "dur": "길이", + "q": "품질/비트레이트", + "Ac": "오디오 코덱", + "Vc": "비디오 코덱", + "Fmt": "형식/컨테이너", + "Ahash": "오디오 체크섬", + "Vhash": "비디오 체크섬", + "Res": "해상도", + "T": "파일 유형", + "aq": "오디오 품질/비트레이트", + "vq": "비디오 품질/비트레이트", + "pixfmt": "서브샘플링/픽셀 구조", + "resw": "가로 해상도", + "resh": "세로 해상도", + "chs": "오디오 채널", + "hz": "샘플레이트" + }, + + "hks": [ + [ + "기타", + ["ESC", "다양한 창 닫기"], + + "파일 관리자", + ["G", "목록/그리드 보기 전환"], + ["T", "썸네일/아이콘 전환"], + ["⇧ A/D", "썸네일 이미지 크기"], + ["ctrl-K", "선택 항목 삭제"], + ["ctrl-X", "선택 항목 잘라내기"], + ["ctrl-C", "선택 항목 복사"], + ["ctrl-V", "여기에 붙여넣기 (이동/복사)"], + ["Y", "선택 항목 다운로드"], + ["F2", "선택 항목 이름 바꾸기"], + + "파일 목록 선택", + ["space", "파일 선택/해제"], + ["↑/↓", "선택 커서 이동"], + ["ctrl ↑/↓", "커서와 뷰포트 동시 이동"], + ["⇧ ↑/↓", "이전/다음 파일 선택"], + ["ctrl-A", "모든 파일/폴더 선택"], + ], [ + "탐색", + ["B", "브레드크럼/탐색창 전환"], + ["I/K", "이전/다음 폴더"], + ["M", "상위 폴더 (또는 현재 항목 닫기)"], + ["V", "탐색창에 폴더/텍스트 파일 표시 전환"], + ["A/D", "탐색창 크기"], + ], [ + "오디오 플레이어", + ["J/L", "이전/다음 곡"], + ["U/O", "10초 뒤로/앞으로 건너뛰기"], + ["0..9", "0%..90% 지점으로 이동"], + ["P", "재생/일시정지 (시작 포함)"], + ["S", "재생 중인 곡 선택"], + ["Y", "곡 다운로드"], + ], [ + "이미지 뷰어", + ["J/L, ←/→", "이전/다음 이미지"], + ["Home/End", "첫/마지막 이미지"], + ["F", "전체 화면"], + ["R", "시계 방향으로 회전"], + ["⇧ R", "반시계 방향으로 회전"], + ["S", "이미지 선택"], + ["Y", "이미지 다운로드"], + ], [ + "비디오 플레이어", + ["U/O", "10초 뒤로/앞으로 건너뛰기"], + ["P/K/Space", "재생/일시정지"], + ["C", "다음 파일 계속 재생"], + ["V", "반복"], + ["M", "음소거"], + ["[ 와 ]", "반복 구간 설정"], + ], [ + "텍스트 파일 뷰어", + ["I/K", "이전/다음 파일"], + ["M", "텍스트 파일 닫기"], + ["E", "텍스트 파일 편집"], + ["S", "파일 선택 (잘라내기/복사/이름 바꾸기용)"], + ] + ], + + "m_ok": "확인", + "m_ng": "취소", + + "enable": "활성화", + "danger": "위험", + "clipped": "클립보드에 복사되었습니다", + + "ht_s1": "초", + "ht_s2": "초", + "ht_m1": "분", + "ht_m2": "분", + "ht_h1": "시간", + "ht_h2": "시간", + "ht_d1": "일", + "ht_d2": "일", + "ht_and": " ", + + "goh": "제어판", + "gop": '이전 형제 폴더">이전', + "gou": '상위 폴더">위로', + "gon": '다음 폴더">다음', + "logout": "로그아웃 ", + "access": " 액세스", + "ot_close": "하위 메뉴 닫기", + "ot_search": "속성, 경로/이름, 음악 태그 또는 이들의 조합으로 파일을 검색합니다.$N$N<code>foo bar</code> = «foo»와 «bar»를 모두 포함해야 함,$N<code>foo -bar</code> = «foo»는 포함하지만 «bar»는 포함하지 않아야 함,$N<code>^yana .opus$</code> = «yana»로 시작하고 «opus» 파일이어야 함$N<code>"try unite"</code> = 정확히 «try unite»를 포함해야 함$N$N날짜 형식은 ISO-8601입니다. 예:$N<code>2009-12-31</code> 또는 <code>2020-09-12 23:30:00</code>", + "ot_unpost": "주워담기: 최근 업로드한 항목을 삭제하거나 미완료된 업로드를 중단합니다", + "ot_bup": "bup: 기본 업로더. 넷스케이프 4.0도 지원합니다", + "ot_mkdir": "mkdir: 새 디렉터리를 만듭니다", + "ot_md": "new-md: 새 마크다운 문서를 만듭니다", + "ot_msg": "msg: 서버 로그에 메시지를 보냅니다", + "ot_mp": "미디어 플레이어 옵션", + "ot_cfg": "구성 옵션", + "ot_u2i": 'up2k: (쓰기 권한이 있는 경우) 파일을 업로드하거나, 검색 모드로 전환하여 서버 어딘가에 파일이 있는지 확인합니다.$N$N업로드는 재개 가능하고, 멀티스레드로 작동하며, 파일 타임스탬프가 보존되지만, [🎈] (기본 업로더)보다 CPU를 더 많이 사용합니다.<br /><br />업로드 중에는 이 아이콘이 진행률 표시창이 됩니다!', + "ot_u2w": 'up2k: 이어올리기 기능을 지원하는 파일 업로더입니다 (브라우저를 닫았다가 나중에 동일한 파일을 끌어다 놓으세요).$N$N멀티스레드로 작동하며, 파일 타임스탬프가 보존되지만, [🎈] (기본 업로더)보다 CPU를 더 많이 사용합니다.<br /><br />업로드 중에는 이 아이콘이 진행률 표시창이 됩니다!', + "ot_noie": 'Chrome / Firefox / Edge를 사용해주세요', + + "ab_mkdir": "디렉터리 만들기", + "ab_mkdoc": "새 마크다운 문서", + "ab_msg": "서버 로그에 메시지 보내기", + + "ay_path": "폴더로 건너뛰기", + "ay_files": "파일로 건너뛰기", + + "wt_ren": "선택한 항목 이름 바꾸기$N단축키: F2", + "wt_del": "선택한 항목 삭제$N단축키: ctrl-K", + "wt_cut": "선택한 항목 잘라내기 <small>(다른 곳에 붙여넣기용)</small>$N단축키: ctrl-X", + "wt_cpy": "선택한 항목 클립보드에 복사$N(다른 곳에 붙여넣기용)$N단축키: ctrl-C", + "wt_pst": "이전에 잘라내거나 복사한 항목 붙여넣기$N단축키: ctrl-V", + "wt_selall": "모든 파일 선택$N단축키: ctrl-A (파일에 포커스된 경우)", + "wt_selinv": "선택 반전", + "wt_zip1": "이 폴더를 압축 파일로 다운로드", + "wt_selzip": "선택 항목을 압축 파일로 다운로드", + "wt_seldl": "선택 항목을 개별 파일로 다운로드$N단축키: Y", + "wt_npirc": "IRC 형식 트랙 정보 복사", + "wt_nptxt": "일반 텍스트 트랙 정보 복사", + "wt_m3ua": "m3u 재생 목록에 추가 (나중에 <code>📻복사</code> 클릭)", + "wt_m3uc": "m3u 재생 목록을 클립보드에 복사", + "wt_grid": "그리드/목록 보기 전환$N단축키: G", + "wt_prev": "이전 트랙$N단축키: J", + "wt_play": "재생/일시정지$N단축키: P", + "wt_next": "다음 트랙$N단축키: L", + + "ul_par": "동시 업로드:", + "ut_rand": "파일명 무작위로 만들기", + "ut_u2ts": "사용자 파일 시스템의 마지막 수정 타임스탬프를$N서버에 복사\">📅", + "ut_ow": "서버에 있는 기존 파일을 덮어쓸까요?$N🛡️: 안 함 (대신 새 파일 이름 생성)$N🕒: 서버 파일이 더 오래된 경우 덮어쓰기$N♻️: 파일이 다르면 항상 덮어쓰기", + "ut_mt": "업로드 중 다른 파일 해싱 계속하기$N$NCPU 또는 HDD가 병목 현상을 일으키는 경우 비활성화하세요", + "ut_ask": '업로드 시작 전 확인 요청">💭', + "ut_pot": "느린 기기에서 UI를 단순화하여$N업로드 속도 향상", + "ut_srch": "실제로 업로드하는 대신, 파일이 이미 서버에 있는지 확인합니다$N(읽을 수 있는 모든 폴더를 스캔합니다)", + "ut_par": "0으로 설정하여 업로드 일시정지$N$N연결이 느리거나 지연 시간이 길면 늘리세요$N$NLAN 환경이거나 서버 HDD가 병목 현상을 일으키면 1로 유지하세요", + "ul_btn": "파일/폴더를 여기에<br>끌어다 놓거나 클릭하세요", + "ul_btnu": "업 로 드", + "ul_btns": "검 색", + + "ul_hash": "해싱", + "ul_send": "전송", + "ul_done": "완료", + "ul_idle1": "대기 중인 업로드가 없습니다", + "ut_etah": "평균 <em>해싱</em> 속도 및 예상 완료 시간", + "ut_etau": "평균 <em>업로드</em> 속도 및 예상 완료 시간", + "ut_etat": "평균 <em>총</em> 속도 및 예상 완료 시간", + + "uct_ok": "성공적으로 완료됨", + "uct_ng": "문제 발생: 실패/거부/찾을 수 없음", + "uct_done": "완료됨 (성공 및 문제 발생 포함)", + "uct_bz": "해싱 또는 업로드 중", + "uct_q": "대기 중, 보류 중", + + "utl_name": "파일명", + "utl_ulist": "목록", + "utl_ucopy": "복사", + "utl_links": "링크", + "utl_stat": "상태", + "utl_prog": "진행률", + + // keep short: + "utl_404": "404", + "utl_err": "오류", + "utl_oserr": "OS 오류", + "utl_found": "찾음", + "utl_defer": "보류", + "utl_yolo": "YOLO", + "utl_done": "완료", + + "ul_flagblk": "파일이 대기열에 추가되었습니다.</b><br>하지만 다른 브라우저 탭에서 up2k가 실행 중이므로,<br>해당 작업이 끝날 때까지 기다립니다.", + "ul_btnlk": "서버 구성에서 이 스위치를 현재 상태로 잠갔습니다.", + + "udt_up": "업로드", + "udt_srch": "검색", + "udt_drop": "여기에 놓으세요", + + "u_nav_m": '<h6>자, 갖고 있는 게 무엇인가?</h6><code>Enter</code> = 파일 (하나 이상)\n<code>ESC</code> = 폴더 하나 (하위 폴더 포함)', + "u_nav_b": '<a href="#" id="modal-ok">파일</a><a href="#" id="modal-ng">폴더 하나</a>', + + "cl_opts": "스위치", + "cl_themes": "테마", + "cl_langs": "언어", + "cl_ziptype": "폴더 다운로드", + "cl_uopts": "up2k 스위치", + "cl_favico": "파비콘", + "cl_bigdir": "큰 디렉터리", + "cl_hsort": "#sort", + "cl_keytype": "조성 표기법", + "cl_hiddenc": "숨겨진 열", + "cl_hidec": "숨기기", + "cl_reset": "초기화", + "cl_hpick": "아래 테이블에서 숨기고 싶은 열의 헤더를 탭하세요", + "cl_hcancel": "열 숨기기가 중단되었습니다", + + "ct_grid": "田 그리드", + "ct_ttips": '◔ ◡ ◔">ℹ️ 도움말', + "ct_thumb": '그리드 보기에서 아이콘 또는 미리보기 이미지 전환$N단축키: T">🖼️ 미리보기', + "ct_csel": '그리드 보기에서 CTRL과 SHIFT를 사용하여 파일 선택">선택', + "ct_ihop": '이미지 뷰어를 닫으면 마지막으로 본 파일로 스크롤">g⮯', + "ct_dots": '숨김 파일 표시 (서버가 허용하는 경우)">숨김파일', + "ct_qdel": '파일 삭제 시 한 번만 확인 요청">빠른삭제', + "ct_dir1st": '폴더를 파일보다 먼저 정렬">📁 먼저', + "ct_nsort": '자연어 정렬 (파일명의 숫자를 인식)">자연어정렬', + "ct_utc": '모든 날짜/시간을 UTC로 표시">UTC', + "ct_readme": '폴더 목록에 README.md 표시">📜 readme', + "ct_idxh": '폴더 목록 대신 index.html 표시">htm', + "ct_sbars": '스크롤바 표시">⟊', + + "cut_umod": '파일이 서버에 이미 있는 경우, 서버의 마지막 수정 타임스탬프를 로컬 파일과 일치하도록 업데이트합니다 (쓰기+삭제 권한 필요).\">re📅', + + "cut_turbo": 'YOLO 버튼. 아마 활성화하고 싶지 않으실 겁니다.$N$N대량의 파일을 업로드하다가 어떤 이유로 재시작해야 할 때, 최대한 빨리 업로드를 계속하고 싶을 때 사용하세요.$N$N이 옵션은 해시 확인을 단순히 <em>"서버에 동일한 파일 크기를 가진 파일이 있는가?"</em>로 대체하므로, 파일 내용만 다를 경우 업로드되지 않습니다.$N$N업로드가 끝나면 이 옵션을 끄고, 동일한 파일을 다시 \"업로드\"하여 클라이언트가 검증하도록 해야 합니다.\">turbo', + + "cut_datechk": '터보 버튼이 활성화되어 있지 않으면 효과가 없습니다.$N$NYOLO의 위험성을 약간 줄여줍니다. 서버의 파일 타임스탬프가 사용자의 것과 일치하는지 확인합니다.$N$N<em>이론적으로는</em> 대부분의 미완료/손상된 업로드를 잡아내지만, 터보를 비활성화하고 검증 과정을 거치는 것을 대체할 수는 없습니다.\">날짜확인', + + "cut_u2sz": "각 업로드 청크의 크기 (MiB)입니다. 큰 값은 태평양을 건너는 데 더 유리합니다. 매우 불안정한 연결에서는 낮은 값을 시도해보세요.", + + "cut_flag": '한 번에 하나의 탭만 업로드하도록 보장합니다.$N-- 다른 탭도 이 옵션을 활성화해야 합니다.$N-- 동일한 도메인의 탭에만 영향을 미칩니다.', + + "cut_az": '가장 작은 파일 우선이 아닌 알파벳 순서로 파일을 업로드합니다.$N$N알파벳 순서는 서버에서 문제가 발생했는지 눈으로 확인하기 쉽게 해주지만, 광랜/LAN 환경에서는 업로드 속도가 약간 느려집니다.', + + "cut_nag": '업로드 완료 시 OS 알림$N(브라우저나 탭이 활성화되지 않은 경우에만)', + "cut_sfx": '업로드 완료 시 소리 알림$N(브라우저나 탭이 활성화되지 않은 경우에만)', + + "cut_mt": '멀티스레딩을 사용하여 파일 해싱 속도를 높입니다.$N$N이 기능은 웹 워커를 사용하며$N더 많은 RAM이 필요합니다 (추가적으로 최대 512 MiB).$N$Nhttps는 30% 더 빠르게, http는 4.5배 더 빠르게 만듭니다.\">mt', + + "cut_wasm": '브라우저 내장 해셔 대신 wasm을 사용합니다. 크롬 기반 브라우저에서 속도를 향상시키지만 CPU 부하를 증가시키며, 많은 구버전 크롬에는 이 기능을 활성화하면 모든 RAM을 소모하고 충돌하는 버그가 있습니다.\">wasm', + + "cft_text": "파비콘 텍스트 (비워두고 새로고침하면 비활성화됨)", + "cft_fg": "전경색", + "cft_bg": "배경색", + + "cdt_lim": "폴더에 표시할 최대 파일 수", + "cdt_ask": "맨 아래로 스크롤할 때$N더 많은 파일을 불러오는 대신$N무엇을 할지 묻기", + "cdt_hsort": "미디어 URL에 포함할 정렬 규칙 (<code>,sorthref</code>)의 수. 0으로 설정하면 미디어 링크를 클릭할 때 포함된 정렬 규칙도 무시됩니다.", + + "tt_entree": "탐색 창 (디렉터리 트리 사이드바) 표시$N단축키: B", + "tt_detree": "이동 경로 표시$N단축키: B", + "tt_visdir": "선택한 폴더로 스크롤하기", + "tt_ftree": "폴더 트리/텍스트 파일 전환$N단축키: V", + "tt_pdock": "상위 폴더를 상단에 고정된 창에 표시", + "tt_dynt": "트리가 확장될 때 자동으로 너비 증가", + "tt_wrap": "자동 줄 바꿈", + "tt_hover": "마우스를 올리면 넘어가는 줄 표시$N(마우스 커서가 왼쪽 여백에$N  있지 않으면 스크롤이 깨짐)", + + "ml_pmode": "폴더 끝에서...", + "ml_btns": "명령", + "ml_tcode": "트랜스코딩", + "ml_tcode2": "다음으로 트랜스코딩", + "ml_tint": "틴트", + "ml_eq": "오디오 이퀄라이저", + "ml_drc": "다이내믹 레인지 압축기", + + "mt_loop": "한 곡 반복 재생\">🔁", + "mt_one": "한 곡 재생 후 중지\">1️⃣", + "mt_shuf": "각 폴더의 곡을 무작위 재생\">🔀", + "mt_aplay": "서버에 접속한 링크에 곡 ID가 있으면 자동 재생$N$N이것을 비활성화하면 음악 재생 시 페이지 URL이 곡 ID로 업데이트되지 않아, 이 설정이 손실되고 URL이 남아있을 경우 자동 재생되는 것을 방지합니다.\">a▶", + "mt_preload": "끊김 없는 재생을 위해 다음 곡을 미리 불러오기 시작\">미리로드", + "mt_prescan": "마지막 곡이 끝나기 전에 다음 폴더로 이동하여$N웹브라우저가 재생을 멈추지 않도록 합니다.\">탐색", + "mt_fullpre": "전체 곡을 미리 불러오기 시도;$N✅ <b>불안정한</b> 연결에서 활성화,$N❌ <b>느린</b> 연결에서는 아마도 비활성화\">전체", + "mt_fau": "폰에서 다음 곡이 충분히 빨리 미리 불러오지 않아 음악이 멈추는 것을 방지합니다 (태그 표시가 불안정해질 수 있음).\">☕️", + "mt_waves": "파형 탐색 바:$N탐색 바에 오디오 진폭 표시\">~s", + "mt_npclip": "현재 재생 중인 곡을 클립보드에 복사하는 버튼 표시\">/np", + "mt_m3u_c": "선택한 곡을 m3u8 재생 목록 항목으로$N클립보드에 복사하는 버튼 표시\">📻", + "mt_octl": "OS 통합 (미디어 단축키/OSD)\">os-ctl", + "mt_oseek": "OS 통합을 통해 탐색 허용$N$N참고: 일부 기기 (iPhone)에서는$N이것이 다음 곡 버튼을 대체합니다.\">탐색", + "mt_oscv": "OSD에 앨범 커버 표시\">아트", + "mt_follow": "재생 중인 트랙이 보이도록 스크롤 유지\">🎯", + "mt_compact": "컴팩트 컨트롤\">⟎", + "mt_uncache": "캐시 지우기 (브라우저가 곡의 깨진 사본을 캐시하여$N재생이 안되는 경우 시도해보세요)\">캐시삭제", + "mt_mloop": "열린 폴더 반복\">🔁 반복", + "mt_mnext": "다음 폴더 불러오고 계속\">📂 다음", + "mt_mstop": "재생 중지\">⏸ 중지", + "mt_cflac": "flac/wav를 opus로 변환\">flac", + "mt_caac": "aac/m4a를 opus로 변환\">aac", + "mt_coth": "다른 모든 것 (mp3 제외)을 opus로 변환\">기타", + "mt_c2opus": "데스크톱, 노트북, 안드로이드 환경에 최적\">opus", + "mt_c2owa": "iOS 17.5 이상용 opus-weba\">owa", + "mt_c2caf": "iOS 11부터 17까지용 opus-caf\">caf", + "mt_c2mp3": "매우 오래된 기기에서 사용\">mp3", + "mt_c2flac": "최고 음질이지만 다운로드 용량이 큼\">flac", + "mt_c2wav": "비압축 재생 (더 큼)\">wav", + "mt_c2ok": "네, 좋은 선택입니다", + "mt_c2nd": "기기에 권장되는 출력 형식이 아니지만 괜찮습니다", + "mt_c2ng": "기기가 이 출력 형식을 지원하지 않는 것 같지만, 시도해 보겠습니다", + "mt_xowa": "iOS에서 이 형식의 백그라운드 재생이 안되는 버그가 있습니다. 대신 caf나 mp3를 사용해주세요.", + "mt_tint": "탐색 바의 배경 레벨 (0-100)$N버퍼링이 덜 눈시리게 만듦", + "mt_eq": "이퀄라이저 및 게인 제어 활성화;$N$Nboost <code>0</code> = 표준 100% 볼륨 (수정 없음)$N$Nwidth <code>1  </code> = 표준 스테레오 (수정 없음)$Nwidth <code>0.5</code> = 50% 좌우 크로스피드$Nwidth <code>0  </code> = 모노$N$Nboost <code>-0.8</code> & width <code>10</code> = 보컬 제거 :^)$N$N이퀄라이저를 활성화하면 끊김 없는 앨범이 온전히 끊김 없이 재생되므로, 그 점이 중요하다면 모든 값을 0으로 두고 (width=1 제외) 켜두세요.", + "mt_drc": "다이내믹 레인지 컴프레서(볼륨 평탄화/벽돌화)를 활성화합니다. 스파게티의 균형을 맞추기 위해 EQ도 활성화되므로, 원하지 않으면 'width'를 제외한 모든 EQ 필드를 0으로 설정하세요.$N$NTHRESHOLD dB 이상의 오디오 볼륨을 낮춥니다. THRESHOLD를 초과하는 모든 RATIO dB에 대해 1dB의 출력이 있으므로, 기본값인 tresh -24 및 ratio 12는 볼륨이 -22dB보다 커지지 않음을 의미하며, 이퀄라이저 부스트를 0.8 또는 ATK 0과 큰 RLS (예: 90)를 사용하여 1.8까지 안전하게 높일 수 있습니다 (firefox에서만 작동, 다른 브라우저에서는 RLS 최대 1).$N$N(위키백과를 참조하세요, 훨씬 더 잘 설명되어 있습니다)", + + "mb_play": "재생", + "mm_hashplay": "이 오디오 파일을 재생할까요?", + "mm_m3u": "<code>Enter/확인</code>을 눌러 재생\n<code>ESC/취소</code>를 눌러 편집", + "mp_breq": "Firefox 82+, Chrome 73+ 또는 iOS 15+ 필요", + "mm_bload": "불러오는 중...", + "mm_bconv": "{0}(으)로 변환 중, 잠시만 기다려주세요...", + "mm_opusen": "브라우저가 aac/m4a 파일을 재생할 수 없습니다.\nopus로의 트랜스코딩이 활성화되었습니다.", + "mm_playerr": "재생 실패: ", + "mm_eabrt": "재생 시도가 취소되었습니다", + "mm_enet": "인터넷 연결이 불안정합니다", + "mm_edec": "이 파일이 손상된 것 같습니다??", + "mm_esupp": "브라우저가 이 오디오 형식을 이해하지 못합니다", + "mm_eunk": "알 수 없는 오류", + "mm_e404": "오디오를 재생할 수 없습니다; 오류 404: 파일을 찾을 수 없습니다.", + "mm_e403": "오디오를 재생할 수 없습니다; 오류 403: 접근이 거부되었습니다.\n\nF5를 눌러 새로고침 해보세요, 로그아웃되었을 수 있습니다", + "mm_e500": "오디오를 재생할 수 없습니다; 오류 500: 서버 로그를 확인하세요.", + "mm_e5xx": "오디오를 재생할 수 없습니다; 서버 오류 ", + "mm_nof": "주변에서 더 이상 오디오 파일을 찾을 수 없습니다", + "mm_prescan": "다음에 재생할 음악을 찾는 중...", + "mm_scank": "다음 곡을 찾았습니다:", + "mm_uncache": "캐시가 지워졌습니다. 모든 곡은 다음 재생 시 다시 다운로드됩니다.", + "mm_hnf": "그 곡이 더 이상 존재하지 않습니다", + + "im_hnf": "그 이미지가 더 이상 존재하지 않습니다", + + "f_empty": '이 폴더는 비어 있습니다', + "f_chide": '«{0}» 열을 숨깁니다.\n\n설정 탭에서 열을 다시 표시할 수 있습니다.', + "f_bigtxt": "이 파일은 {0} MiB입니다 -- 정말 텍스트로 보시겠습니까?", + "f_bigtxt2": "대신 파일의 끝부분만 보시겠습니까? 이렇게 하면 실시간으로 새로 추가되는 텍스트 줄을 보여주는 팔로잉/테일링 기능도 활성화됩니다.", + "fbd_more": '<div id="blazy"><code>{1}</code>개 파일 중 <code>{0}</code>개 표시 중; <a href="#" id="bd_more">{2}개 더 보기</a> 또는 <a href="#" id="bd_all">모두 보기</a></div>', + "fbd_all": '<div id="blazy"><code>{1}</code>개 파일 중 <code>{0}</code>개 표시 중; <a href="#" id="bd_all">모두 보기</a></div>', + "f_anota": "{1}개 항목 중 {0}개만 선택되었습니다.\n전체 폴더를 선택하려면 먼저 맨 아래로 스크롤하세요.", + + "f_dls": '현재 폴더의 파일 링크가\n다운로드 링크로 변경되었습니다', + + "f_partial": "현재 업로드 중인 파일을 안전하게 다운로드하려면, 파일 이름이 같지만 <code>.PARTIAL</code> 확장자가 없는 파일을 클릭하세요. 이 경고를 무시하려면 \"취소\" 또는 ESC를 누르세요.\n\n\"확인\"/Enter를 누르면 이 경고를 무시하고 <code>.PARTIAL</code> 임시 파일을 계속 다운로드하며, 이 경우 거의 확실히 손상된 데이터를 받게 됩니다.", + + "ft_paste": "{0}개 항목 붙여넣기$N단축키: ctrl-V", + "fr_eperm": "이름을 바꿀 수 없습니다:\n이 폴더에 \"이동\" 권한이 없습니다", + "fd_eperm": "삭제할 수 없습니다:\n이 폴더에 \"삭제\" 권한이 없습니다", + "fc_eperm": "잘라낼 수 없습니다:\n이 폴더에 \"이동\" 권한이 없습니다", + "fp_eperm": "붙여넣을 수 없습니다\n이 폴더에 \"쓰기\" 권한이 없습니다", + "fr_emore": "이름을 바꿀 항목을 하나 이상 선택하세요", + "fd_emore": "삭제할 항목을 하나 이상 선택하세요", + "fc_emore": "잘라낼 항목을 하나 이상 선택하세요", + "fcp_emore": "클립보드에 복사할 항목을 하나 이상 선택하세요", + + "fs_sc": "현재 폴더 공유", + "fs_ss": "선택한 파일 공유", + "fs_just1d": "하나 이상의 폴더를 선택하거나,\n파일과 폴더를 한 번에 섞어 선택할 수 없습니다", + "fs_abrt": "❌ 중단", + "fs_rand": "🎲 무작위 이름", + "fs_go": "✅ 공유 생성", + "fs_name": "이름", + "fs_src": "소스", + "fs_pwd": "비밀번호", + "fs_exp": "만료", + "fs_tmin": "분", + "fs_thrs": "시간", + "fs_tdays": "일", + "fs_never": "영원", + "fs_pname": "선택적 링크 이름; 비워두면 무작위로 생성", + "fs_tsrc": "공유할 파일 또는 폴더", + "fs_ppwd": "비밀번호 (선택사항)", + "fs_w8": "공유 생성 중...", + "fs_ok": "<code>Enter/OK</code>를 눌러 클립보드에 복사\n<code>ESC/Cancel</code>를 눌러 닫기", + + "frt_dec": "깨진 파일 이름의 일부 경우를 수정할 수 있습니다\">url-디코드", + "frt_rst": "수정된 파일 이름을 원래대로 되돌립니다\">↺ 초기화", + "frt_abrt": "이 창을 중단하고 닫습니다\">❌ 취소", + "frb_apply": "이름 바꾸기 적용", + "fr_adv": "배치/메타데이터/패턴 이름 바꾸기\">고급", + "fr_case": "대소문자 구분 정규식\">대소문자", + "fr_win": "Windows 안전 이름; <code><>:"\\|?*</code>를 일본어 전각 문자로 바꿉니다\">win", + "fr_slash": "<code>/</code>를 새 폴더를 만들지 않는 문자로 바꿉니다\">/ 없음", + "fr_re": "원본 파일 이름에 적용할 정규식 검색 패턴; 캡처링 그룹은 아래 형식 필드에서 <code>(1)</code>, <code>(2)</code> 등으로 참조할 수 있습니다", + "fr_fmt": "foobar2000에서 영감을 받음:$N<code>(title)</code>은(는) 곡 제목으로 대체됨,$N<code>[(artist) - ](title)</code>은(는) 아티스트가 비어 있으면 [이] 부분을 건너뜀$N<code>$lpad((tn),2,0)</code>은(는) 트랙 번호를 2자리로 채움", + "fr_pdel": "삭제", + "fr_pnew": "다른 이름으로 저장", + "fr_pname": "새 프리셋의 이름을 입력하세요", + "fr_aborted": "중단됨", + "fr_lold": "이전 이름", + "fr_lnew": "새 이름", + "fr_tags": "선택한 파일의 태그 (읽기 전용, 참조용):", + "fr_busy": "{0}개 항목 이름 바꾸는 중...\n\n{1}", + "fr_efail": "이름 바꾸기 실패:\n", + "fr_nchg": "<code>win</code> 및/또는 <code>/ 없음</code>으로 인해 새 이름 중 {0}개가 변경되었습니다.\n\n이 변경된 새 이름으로 계속하시겠습니까?", + + "fd_ok": "삭제 확인", + "fd_err": "삭제 실패:\n", + "fd_none": "아무것도 삭제되지 않았습니다. 서버 구성 (xbd)에 의해 차단되었을 수 있습니다.", + "fd_busy": "삭제 중 {0}개 항목...\n\n{1}", + "fd_warn1": "이 {0}개 항목을 삭제하시겠습니까?", + "fd_warn2": "<b>마지막 기회입니다!</b> 되돌릴 수 없습니다. 삭제하시겠습니까?", + + "fc_ok": "{0}개 항목 잘라내기 완료", + "fc_warn": "{0}개 항목 잘라내기 완료\n\n하지만: 선택 항목이 너무 커서 <b>이</b> 브라우저 탭에서만 붙여넣을 수 있습니다", + + "fcc_ok": "{0}개 항목을 클립보드에 복사했습니다", + "fcc_warn": "{0}개 항목을 클립보드에 복사했습니다\n\n하지만: 선택 항목이 너무 커서 <b>이</b> 브라우저 탭에서만 붙여넣을 수 있습니다", + + "fp_apply": "이 이름 사용", + "fp_ecut": "붙여넣거나 이동하려면 먼저 파일/폴더를 잘라내거나 복사하세요\n\n참고: 다른 브라우저 탭 간에 잘라내기/붙여넣기를 할 수 있습니다", + "fp_ename": "이름이 이미 사용 중이므로 {0}개 항목을 여기로 이동할 수 없습니다. 계속하려면 아래에 새 이름을 지정하거나, 이름을 비워두면 건너뜁니다:", + "fcp_ename": "이름이 이미 사용 중이므로 {0}개 항목을 여기로 복사할 수 없습니다. 계속하려면 아래에 새 이름을 지정하거나, 이름을 비워두면 건너뜁니다:", + "fp_emore": "아직 해결해야 할 파일 이름 충돌이 남아 있습니다", + "fp_ok": "이동 완료", + "fcp_ok": "복사 완료", + "fp_busy": "{0}개 항목 이동 중...\n\n{1}", + "fcp_busy": "{0}개 항목 복사 중...\n\n{1}", + "fp_err": "이동 실패:\n", + "fcp_err": "복사 실패:\n", + "fp_confirm": "이 {0}개 항목을 여기로 이동하시겠습니까?", + "fcp_confirm": "이 {0}개 항목을 여기로 복사하시겠습니까?", + "fp_etab": '다른 브라우저 탭에서 클립보드를 읽지 못했습니다', + "fp_name": "기기에서 파일을 업로드합니다. 이름을 지정하세요:", + "fp_both_m": '<h6>붙여넣을 항목 선택</h6><code>Enter</code> = «{1}»에서 파일 {0}개 이동\n<code>ESC</code> = 기기에서 파일 {2}개 업로드', + "fcp_both_m": '<h6>붙여넣을 항목 선택</h6><code>Enter</code> = «{1}»에서 파일 {0}개 복사\n<code>ESC</code> = 기기에서 파일 {2}개 업로드', + "fp_both_b": '<a href="#" id="modal-ok">이동</a><a href="#" id="modal-ng">업로드</a>', + "fcp_both_b": '<a href="#" id="modal-ok">복사</a><a href="#" id="modal-ng">업로드</a>', + + "mk_noname": "왼쪽 텍스트 필드에 이름을 먼저 입력해주세요 :p", + + "tv_load": "텍스트 문서 불러오는 중:\n\n{0}\n\n{1}% ({3} MiB 중 {2} MiB 로드됨)", + "tv_xe1": "텍스트 파일을 불러올 수 없습니다:\n\n오류 ", + "tv_xe2": "404, 파일을 찾을 수 없음", + "tv_lst": "텍스트 파일 목록", + "tvt_close": "폴더 보기로 돌아가기$N단축키: M (또는 Esc)\">❌ 닫기", + "tvt_dl": "이 파일 다운로드$N단축키: Y\">💾 다운로드", + "tvt_prev": "이전 문서 보기$N단축키: i\">⬆ 이전", + "tvt_next": "다음 문서 보기$N단축키: K\">⬇ 다음", + "tvt_sel": "파일 선택   (잘라내기/복사/삭제/...용)$N단축키: S\">선택", + "tvt_edit": "텍스트 편집기에서 파일 열기$N단축키: E\">✏️ 편집", + "tvt_tail": "파일 변경 사항 모니터링; 실시간으로 새 줄 표시\">📡 팔로우", + "tvt_wrap": "자동 줄 바꿈\">↵", + "tvt_atail": "페이지 하단으로 스크롤 고정\">⚓", + "tvt_ctail": "터미널 색상 디코딩 (ANSI 이스케이프 코드)\">🌈", + "tvt_ntail": "스크롤백 제한 (불러온 상태로 유지할 텍스트 바이트 수)", + + "m3u_add1": "m3u 재생 목록에 곡이 추가되었습니다", + "m3u_addn": "{0}개의 곡이 m3u 재생 목록에 추가되었습니다", + "m3u_clip": "m3u 재생 목록이 클립보드에 복사되었습니다\n\n something.m3u와 같은 이름의 새 텍스트 파일을 만들고 그 문서에 재생 목록을 붙여넣으면 재생할 수 있습니다.", + + "gt_vau": "비디오를 표시하지 않고 오디오만 재생\">🎧", + "gt_msel": "파일 선택 활성화; ctrl-클릭하여 파일 재정의$N$N<em>활성 시: 파일/폴더를 두 번 클릭하여 열기</em>$N$N단축키: S\">다중선택", + "gt_crop": "썸네일 중앙 자르기\">자르기", + "gt_3x": "고해상도 썸네일\">3x", + "gt_zoom": "확대/축소", + "gt_chop": "자르기", + "gt_sort": "정렬 기준", + "gt_name": "이름", + "gt_sz": "크기", + "gt_ts": "날짜", + "gt_ext": "유형", + "gt_c1": "파일명 더 많이 생략하기 (더 적게 표시)", + "gt_c2": "파일명 덜 생략하기 (더 많이 표시)", + + "sm_w8": "검색 중...", + "sm_prev": "아래 검색 결과는 이전 검색어에 대한 결과입니다:\n  ", + "sl_close": "검색 결과 닫기", + "sl_hits": "{0}개 결과 표시 중", + "sl_moar": "더 불러오기", + + "s_sz": "크기", + "s_dt": "날짜", + "s_rd": "경로", + "s_fn": "이름", + "s_ta": "태그", + "s_ua": "업로드 시점", + "s_ad": "고급", + "s_s1": "최소 MiB", + "s_s2": "최대 MiB", + "s_d1": "최소 ISO-8601", + "s_d2": "최대 ISO-8601", + "s_u1": "이후", + "s_u2": "이전", + "s_r1": "경로에 포함   (공백으로 구분)", + "s_f1": "이름에 포함   (-로 제외)", + "s_t1": "태그에 포함   (^=시작, 끝=$)", + "s_a1": "특정 메타데이터 속성", + + "md_eshow": "렌더링할 수 없음 ", + "md_off": "[📜<em>readme</em>]가 [⚙️]에서 비활성화됨 -- 문서 숨김", + + "badreply": "서버로부터의 응답을 구문 분석하지 못했습니다", + + "xhr403": "403: 접근 거부됨\n\nF5를 눌러보세요, 로그아웃되었을 수 있습니다", + "xhr0": "알 수 없음 (서버와의 연결이 끊겼거나 서버가 오프라인일 수 있습니다)", + "cf_ok": "죄송합니다 -- DD" + wah + "oS 보호 기능이 작동했습니다\n\n약 30초 후에 다시 정상적으로 작동할 것입니다\n\n아무 일도 일어나지 않으면 F5를 눌러 페이지를 새로고침하세요", + "tl_xe1": "하위 폴더를 나열할 수 없습니다:\n\n오류 ", + "tl_xe2": "404: 폴더를 찾을 수 없음", + "fl_xe1": "폴더의 파일을 나열할 수 없습니다:\n\n오류 ", + "fl_xe2": "404: 폴더를 찾을 수 없음", + "fd_xe1": "하위 폴더를 만들 수 없습니다:\n\n오류 ", + "fd_xe2": "404: 상위 폴더를 찾을 수 없음", + "fsm_xe1": "메시지를 보낼 수 없습니다:\n\n오류 ", + "fsm_xe2": "404: 상위 폴더를 찾을 수 없음", + "fu_xe1": "서버에서 주워담기 목록을 불러오지 못했습니다:\n\n오류 ", + "fu_xe2": "404: 파일을 찾을 수 없음??", + + "fz_tar": "압축되지 않은 gnu-tar 파일 (linux / mac)", + "fz_pax": "압축되지 않은 pax 형식 tar (느림)", + "fz_targz": "gzip 레벨 3 압축이 적용된 gnu-tar$N$N이것은 보통 매우 느리므로$N압축되지 않은 tar를 대신 사용하세요", + "fz_tarxz": "xz 레벨 1 압축이 적용된 gnu-tar$N$N이것은 보통 매우 느리므로$N압축되지 않은 tar를 대신 사용하세요", + "fz_zip8": "utf8 파일 이름이 포함된 zip (windows 7 및 이전 버전에서 문제가 있을 수 있음)", + "fz_zipd": "정말 오래된 소프트웨어를 위한 전통적인 cp437 파일 이름이 포함된 zip", + "fz_zipc": "MS-DOS PKZIP v2.04g (1993년 10월)용으로$Ncrc32가 미리 계산된 cp437$N(다운로드 시작 전 처리 시간이 더 걸림)", + + "un_m1": "아래에서 최근 업로드를 삭제하거나 미완료된 업로드를 중단할 수 있습니다", + "un_upd": "새로고침", + "un_m4": "또는 아래에 보이는 파일을 공유할 수 있습니다:", + "un_ulist": "보기", + "un_ucopy": "복사", + "un_flt": "선택적 필터:  URL에 포함되어야 함", + "un_fclr": "필터 지우기", + "un_derr": '주워담기-삭제 실패:\n', + "un_f5": '문제가 발생했습니다, 새로고침하거나 F5를 눌러보세요', + "un_uf5": "죄송하지만, 이 업로드를 중단하기 전에 페이지를 새로고침해야 합니다 (예: F5 또는 CTRL-R 누르기).", + "un_nou": '<b>경고:</b> 서버가 너무 바빠서 미완료 업로드를 표시할 수 없습니다; 잠시 후 "새로고침" 링크를 클릭하세요', + "un_noc": '<b>경고:</b> 완전히 업로드된 파일의 주워담기가 서버 구성에서 활성화/허용되지 않았습니다', + "un_max": "처음 2000개 파일을 표시합니다 (필터를 사용하세요)", + "un_avail": "{0}개의 최근 업로드를 삭제할 수 있습니다<br />{1}개의 미완료 업로드를 중단할 수 있습니다", + "un_m2": "업로드 시간순으로 정렬됨. 가장 최근 항목이 먼저 표시:", + "un_no1": "아쉽다! 충분히 최근인 업로드가 없습니다.", + "un_no2": "아쉽다! 해당 필터와 일치하는 최근 업로드가 없습니다.", + "un_next": "아래의 다음 {0}개 파일 삭제", + "un_abrt": "중단", + "un_del": "삭제", + "un_m3": "최근 업로드 로드 중...", + "un_busy": "{0}개 파일 삭제 중...", + "un_clip": "{0}개의 링크가 클립보드에 복사되었습니다", + + "u_https1": "더 나은 성능을 위해", + "u_https2": "https로 전환", + "u_https3": "하는 것이 좋습니다", + "u_ancient": '브라우저가 정말 오래되었네요 -- 아마도 <a href="#" onclick="goto(\'bup\')">bup을 대신 사용</a>해야 할 것 같습니다', + "u_nowork": "Firefox 53+, Chrome 57+ 또는 iOS 11+가 필요합니다", + "tail_2old": "Firefox 105+, Chrome 71+ 또는 iOS 14.5+가 필요합니다", + "u_nodrop": '브라우저가 너무 오래되어 드래그 앤 드롭 업로드를 지원하지 않습니다', + "u_notdir": '폴더가 아닙니다!\n\n브라우저가 너무 오래되었습니다,\n대신 드래그드롭을 시도해보세요', + "u_uri": '다른 브라우저 창에서 이미지를 드래그드롭하려면,\n큰 업로드 버튼 위로 떨어뜨려주세요', + "u_enpot": '<a href="#">단순 UI로 전환</a> (업로드 속도가 향상될 수 있음)', + "u_depot": '<a href="#">화려한 UI로 전환</a> (업로드 속도가 감소할 수 있음)', + "u_gotpot": '업로드 속도 향상을 위해 단순 UI로 전환합니다,\n\n언제든지 다시 전환하셔도 좋습니다!', + "u_pott": "<p>파일:   <b>{0}</b> 완료,   <b>{1}</b> 실패,   <b>{2}</b> 처리 중,   <b>{3}</b> 대기 중</p>", + "u_ever": "이것은 기본 업로더입니다. up2k는 최소한 다음 버전이 필요합니다:<br>Chrome 21 // Firefox 13 // Edge 12 // Opera 12 // Safari 5.1", + "u_su2k": '이것은 기본 업로더입니다. <a href="#" id="u2yea">up2k</a>가 더 좋습니다', + "u_uput": '속도 최적화 (체크섬 건너뛰기)', + "u_ewrite": '이 폴더에 쓰기 권한이 없습니다', + "u_eread": '이 폴더에 읽기 권한이 없습니다', + "u_enoi": '파일 검색이 서버 구성에서 활성화되지 않았습니다', + "u_enoow": '여기서는 덮어쓰기가 작동하지 않습니다. 삭제 권한이 필요합니다', + "u_badf": '총 {1}개 중 다음 {0}개의 파일은 파일 시스템 권한 문제 등으로 건너뛰었습니다:\n\n', + "u_blankf": '총 {1}개 중 다음 {0}개의 파일은 비어있습니다. 그래도 업로드하시겠습니까?\n\n', + "u_applef": '총 {1}개 중 다음 {0}개의 파일은 아마도 불필요한 파일일 것입니다.\n다음 파일을 건너뛰려면 <code>확인/Enter</code>를 누르세요,\n해당 파일도 업로드하려면 <code>취소/ESC</code>를 누르세요:\n\n', + "u_just1": '\n파일을 하나만 선택하면 더 잘 작동할 수 있습니다', + "u_ff_many": '<b>리눅스/macOS/안드로이드</b>를 사용 중이라면, 이 정도의 파일 수는 <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">Firefox를 <em>충돌시킬 수 있습니다!</em></a>\n만약 그런 일이 발생하면 다시 시도하거나 Chrome을 사용해주세요.', + "u_up_life": '이 업로드는 완료 후 {0} 뒤에\n서버에서 삭제됩니다', + "u_asku": '이 {0}개의 파일을 <code>{1}</code>(으)로 업로드하시겠습니까?', + "u_unpt": '왼쪽 상단의 🧯를 사용하여 이 업로드를 취소/삭제할 수 있습니다', + "u_bigtab": '{0}개의 파일을 표시하려고 합니다\n\n브라우저가 충돌할 수 있습니다, 계속 진행합니까?', + "u_scan": '파일 스캔 중...', + "u_dirstuck": '디렉터리 반복자가 다음 {0}개 항목에 접근하는 데 실패하여 건너뜁니다:', + "u_etadone": '완료 ({0}, {1}개 파일)', + "u_etaprep": '(업로드 준비 중)', + "u_hashdone": '해싱 완료', + "u_hashing": '해시', + "u_hs": '핸드셰이킹 중...', + "u_started": '파일이 현재 업로드 중입니다. [🚀] 참조', + "u_dupdefer": '중복됨. 다른 모든 파일 처리 후 처리됩니다', + "u_actx": "다른 창/탭으로 전환 시 성능 저하를<br />방지하려면 이 텍스트를 클릭하세요", + "u_fixed": "OK!  해결됐습니다 👍", + "u_cuerr": "{1} 중 청크 {0} 업로드 실패;\n아마 문제 없을 겁니다. 계속 진행합니다\n\n파일: {2}", + "u_cuerr2": "서버가 업로드를 거부했습니다 (청크 {0}/{1});\n나중에 다시 시도합니다\n\n파일: {2}\n\n오류 ", + "u_ehstmp": "다시 시도합니다; 오른쪽 하단 참조", + "u_ehsfin": "서버가 업로드 완료 요청을 거부했습니다. 재시도 중...", + "u_ehssrch": "서버가 검색 수행 요청을 거부했습니다. 재시도 중...", + "u_ehsinit": "서버가 업로드 시작 요청을 거부했습니다. 재시도 중...", + "u_eneths": "업로드 핸드셰이크 중 네트워크 오류 발생. 재시도 중...", + "u_enethd": "대상 존재 여부 테스트 중 네트워크 오류 발생; 재시도 중...", + "u_cbusy": '네트워크 문제 후 서버가 다시 우리를 신뢰할 때까지 기다리는 중...', + "u_ehsdf": '서버 디스크 공간이 부족합니다!\n\n누군가 계속할 수 있을 만큼의 공간을\n비워줄 경우를 대비해 계속 재시도합니다', + "u_emtleak1": "웹 브라우저에 메모리 누수가 있는 것 같습니다.\n", + "u_emtleak2": ' <a href="{0}">https로 전환 (권장)</a>하거나 ', + "u_emtleak3": ' ', + "u_emtleakc": '다음을 시도해보세요:\n<ul><li><code>F5</code>를 눌러 페이지를 새로고침하세요</li><li>그런 다음 <code>⚙️ 설정</code>에서  <code>mt</code>  버튼을 비활성화하세요</li><li>그리고 다시 그 업로드를 시도해보세요</li></ul>업로드가 조금 느려지겠지만, 어쩔 수 없죠.\n불편을 드려 죄송합니다!\n\nPS: 이 버그는 Chrome v107에서 <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">수정되었습니다</a>.', + "u_emtleakf": '다음을 시도해보세요:\n<ul><li><code>F5</code>를 눌러 페이지를 새로고침하세요</li><li>그런 다음 업로드 UI에서 <code>🥔</code>(단순 UI)를 활성화하세요<li>그리고 다시 그 업로드를 시도해보세요</li></ul>\nPS: Firefox에서 <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">언젠가 이 버그가 수정될 거라 믿습니다</a>.', + "u_s404": '서버에서 찾을 수 없음', + "u_expl": '설명', + "u_maxconn": "대부분의 브라우저는 이를 6으로 제한하지만, Firefox에서는 <code>about:config</code>에서 <code>connections-per-server</code> 설정값으로 높일 수 있습니다.", + "u_tu": '<p class="warn">경고: 터보가 활성화되어 클라이언트가 불완전한 업로드를 감지하고 재개하지 못할 수 있습니다. 터보 버튼의 툴팁을 참조하세요</span></p>', + "u_ts": '<p class="warn">경고: 터보가 활성화되어 검색 결과가 부정확할 수 있습니다. 터보 버튼의 툴팁을 참조하세요</span></p>', + "u_turbo_c": "터보가 서버 구성에서 비활성화되었습니다", + "u_turbo_g": '이 볼륨 내에서 디렉터리 목록 권한이 없으므로\n터보를 비활성화합니다', + "u_life_cfg": '자동 삭제 시간 <input id="lifem" p="60" /> 분 (또는 <input id="lifeh" p="3600" /> 시간)', + "u_life_est": '업로드가 <span id="lifew" tt="현지 시간">---</span>에 삭제됩니다', + "u_life_max": '이 폴더는 최대 수명을\n{0}(으)로 강제합니다', + "u_unp_ok": '주워담기는 {0} 동안 허용됩니다', + "u_unp_ng": '주워담기는 허용되지 않습니다', + "ue_ro": '이 폴더에 대한 접근은 읽기 전용입니다\n\n', + "ue_nl": '현재 로그인되어 있지 않습니다', + "ue_la": '현재 \'{0}\'(으)로 로그인되어 있습니다', + "ue_sr": '현재 파일 검색 모드입니다\n\n큰 "검색" 버튼 옆의 돋보기 🔎를 클릭하여 업로드 모드로 전환한 후 다시 업로드해보세요\n\n죄송합니다', + "ue_ta": '다시 업로드해보세요, 이제 작동할 겁니다', + "ue_ab": '이 파일은 이미 다른 폴더로 업로드 중이며, 파일이 다른 곳에 업로드되기 전에 해당 업로드가 완료되어야 합니다.\n\n왼쪽 상단의 🧯를 사용하여 초기 업로드를 중단하고 잊을 수 있습니다.', + "ur_1uo": "OK: 파일이 성공적으로 업로드되었습니다", + "ur_auo": "OK: 모든 {0}개의 파일이 성공적으로 업로드되었습니다", + "ur_1so": "OK: 서버에서 파일을 찾았습니다", + "ur_aso": "OK: 서버에서 모든 {0}개의 파일을 찾았습니다", + "ur_1un": "업로드에 실패했습니다, 죄송", + "ur_aun": "모든 {0}개의 업로드에 실패했습니다, 죄송", + "ur_1sn": "서버에서 파일을 찾지 못했습니다", + "ur_asn": "서버에서 {0}개의 파일을 찾지 못했습니다", + "ur_um": "완료;\n{0}개 업로드 성공,\n{1}개 업로드 실패, 죄송", + "ur_sm": "완료;\n서버에서 {0}개 파일 찾음,\n서버에서 {1}개 파일 찾지 못함", + + "lang_set": '변경 사항을 적용하기 위해 새로고침하시겠습니까?' + }, "nld": { "tt": "Nederlands", @@ -10090,7 +10719,7 @@ var Ls = { }, }; -var LANGS = ["eng", "nor", "nno", "chi", "cze", "deu", "fin", "fra", "grc", "ita", "nld", "rus", "spa", "swe", "ukr"]; +var LANGS = ["eng", "nor", "nno", "chi", "cze", "deu", "fin", "fra", "grc", "ita", "kor", "nld", "rus", "spa", "swe", "ukr"]; if (window.langmod) langmod(); diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index 3229b9a2..aad6ad2a 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -342,6 +342,48 @@ var Ls = { "af1": "mostra i file caricati di recente", "ag1": "mostra utenti IdP conosciuti" }, + "kor": { + "a1": "새로고침", + "b1": "어이 친구! 처음 보는 얼굴인데?   <small>(로그인되어 있지 않습니다)</small>", + "c1": "로그아웃", + "d1": "스택 덤프하기", + "d2": "모든 활성 스레드의 상태를 표시합니다", + "e1": "설정 다시 불러오기", + "e2": "설정 파일(계정/볼륨/볼륨 플래그)을 다시 불러오고,$N모든 e2ds 볼륨을 다시 스캔합니다$N$N참고: 전역 설정에 대한 변경 사항은$N적용하려면 전체 재시작이 필요합니다", + "f1": "탐색 가능한 곳:", + "g1": "업로드 가능한 곳:", + "cc1": "기타 항목:", + "h1": "k304 비활성화", + "i1": "k304 활성화", + "j1": "k304를 활성화하면 모든 HTTP 304 응답 시 클라이언트 연결이 끊어집니다. 이는 일부 프록시가 멈추는 현상(갑자기 페이지가 로드되지 않음)을 방지할 수 있지만, <em>대신 전반적인 속도는 느려집니다.</em>", + "k1": "클라이언트 설정 초기화", + "l1": "로그인하기:", + "m1": "또 오셨네요,", + "n1": "404 찾을 수 없음  ┐( ´ -`)┌", + "o1": "또는 접근 권한이 없을 수 있습니다. 비밀번호를 입력하거나 <a href=\"' + SR + '/?h\">홈으로 이동</a>하세요", + "p1": "403 접근 금지  ~┻━┻", + "q1": "비밀번호를 입력하거나 <a href=\"' + SR + '/?h\">홈으로 이동</a>하세요", + "r1": "홈으로 이동", + ".s1": "다시 스캔", + "t1": "작업", + "u2": "서버에 마지막으로 쓰기 작업을 한 후 경과된 시간$N(업로드 / 이름 변경 / 등등...)$N$N17d = 17일$N1h23 = 1시간 23분$N4m56 = 4분 56초", + "v1": "연결", + "v2": "이 서버를 로컬 하드디스크처럼 사용하기", + "w1": "HTTPS로 전환", + "x1": "비밀번호 변경", + "y1": "공유 설정", + "z1": "이 공유 잠금해제:", + "ta1": "새 비밀번호를 먼저 입력하세요", + "ta2": "새 비밀번호 확인을 위해 다시 입력하세요:", + "ta3": "오타가 있습니다. 다시 시도해주세요", + "aa1": "수신 중인 파일:", + "ab1": "no304 비활성화", + "ac1": "no304 활성화", + "ad1": "no304를 활성화하면 모든 캐싱이 비활성화됩니다. k304로 충분하지 않은 경우 시도해보세요. 네트워크 트래픽이 대량으로 낭비됩니다!", + "ae1": "활성 다운로드:", + "af1": "최근 업로드 보기", + "ag1": "IdP 캐시 보기" + }, "nld": { "a1": "Update", "b1": "Hallo, hoe gaat het met jou?   <small>(Je bent niet ingelogd)</small>", From 2961dea5bb46036de9849061b95ef63a24dda217 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 19:36:20 +0000 Subject: [PATCH 163/174] tl cleanup --- copyparty/web/browser.js | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index a387495e..92bba2ba 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -6102,6 +6102,7 @@ var Ls = { "fcp_ok": "복사 완료", "fp_busy": "{0}개 항목 이동 중...\n\n{1}", "fcp_busy": "{0}개 항목 복사 중...\n\n{1}", + "fp_abrt": "취소 중...", "fp_err": "이동 실패:\n", "fcp_err": "복사 실패:\n", "fp_confirm": "이 {0}개 항목을 여기로 이동하시겠습니까?", @@ -6282,8 +6283,8 @@ var Ls = { "u_s404": '서버에서 찾을 수 없음', "u_expl": '설명', "u_maxconn": "대부분의 브라우저는 이를 6으로 제한하지만, Firefox에서는 <code>about:config</code>에서 <code>connections-per-server</code> 설정값으로 높일 수 있습니다.", - "u_tu": '<p class="warn">경고: 터보가 활성화되어 클라이언트가 불완전한 업로드를 감지하고 재개하지 못할 수 있습니다. 터보 버튼의 툴팁을 참조하세요</span></p>', - "u_ts": '<p class="warn">경고: 터보가 활성화되어 검색 결과가 부정확할 수 있습니다. 터보 버튼의 툴팁을 참조하세요</span></p>', + "u_tu": '<p class="warn">경고: 터보가 활성화되어 <span>클라이언트가 불완전한 업로드를 감지하고 재개하지 못할 수 있습니다. 터보 버튼의 툴팁을 참조하세요</span></p>', + "u_ts": '<p class="warn">경고: 터보가 활성화되어 <span>검색 결과가 부정확할 수 있습니다. 터보 버튼의 툴팁을 참조하세요</span></p>', "u_turbo_c": "터보가 서버 구성에서 비활성화되었습니다", "u_turbo_g": '이 볼륨 내에서 디렉터리 목록 권한이 없으므로\n터보를 비활성화합니다', "u_life_cfg": '자동 삭제 시간 <input id="lifem" p="60" /> 분 (또는 <input id="lifeh" p="3600" /> 시간)', From af8620da924f48513997381904323065bc6d4785 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 19:50:50 +0000 Subject: [PATCH 164/174] fix `--lang` helptext; closes #594 --- copyparty/__main__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 38f786d8..b538b6a9 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1575,7 +1575,7 @@ def add_ui(ap, retry): ap2.add_argument("--grid", action="store_true", help="show grid/thumbnails by default (volflag=grid)") ap2.add_argument("--gsel", action="store_true", help="select files in grid by ctrl-click (volflag=gsel)") ap2.add_argument("--localtime", action="store_true", help="default to local timezone instead of UTC") - ap2.add_argument("--lang", metavar="LANG", type=u, default="eng", help="language; one of the following: \033[32meng nor chi\033[0m") + ap2.add_argument("--lang", metavar="LANG", type=u, default="eng", help="language, for example \033[32meng\033[0m / \033[32mnor\033[0m / ...") ap2.add_argument("--theme", metavar="NUM", type=int, default=0, help="default theme to use (0..7)") ap2.add_argument("--themes", metavar="NUM", type=int, default=10, help="number of themes installed") ap2.add_argument("--au-vol", metavar="0-100", type=int, default=50, choices=range(0, 101), help="default audio/video volume percent") From d56230573dc63f9aa1ee5f3100112e3f00886894 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 20:02:32 +0000 Subject: [PATCH 165/174] separate audio-transcode timeout (#598) --- copyparty/__main__.py | 3 ++- copyparty/authsrv.py | 2 +- copyparty/cfg.py | 4 +++- copyparty/th_srv.py | 26 +++++++++++++------------- tests/util.py | 2 +- 5 files changed, 20 insertions(+), 17 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index b538b6a9..4ae60e61 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1435,7 +1435,8 @@ def add_thumbnail(ap): ap2.add_argument("--no-athumb", action="store_true", help="disable audio thumbnails (spectrograms) (volflag=dathumb)") ap2.add_argument("--th-size", metavar="WxH", default="320x256", help="thumbnail res (volflag=thsize)") ap2.add_argument("--th-mt", metavar="CORES", type=int, default=CORES, help="num cpu cores to use for generating thumbnails") - ap2.add_argument("--th-convt", metavar="SEC", type=float, default=60.0, help="conversion timeout in seconds (volflag=convt)") + ap2.add_argument("--th-convt", metavar="SEC", type=float, default=60.0, help="convert-to-image timeout in seconds (volflag=convt)") + ap2.add_argument("--ac-convt", metavar="SEC", type=float, default=150.0, help="convert-to-audio timeout in seconds (volflag=aconvt)") ap2.add_argument("--th-ram-max", metavar="GB", type=float, default=th_ram, help="max memory usage (GiB) permitted by thumbnailer; not very accurate") ap2.add_argument("--th-crop", metavar="TXT", type=u, default="y", help="crop thumbnails to 4:3 or keep dynamic height; client can override in UI unless force. [\033[32my\033[0m]=crop, [\033[32mn\033[0m]=nocrop, [\033[32mfy\033[0m]=force-y, [\033[32mfn\033[0m]=force-n (volflag=crop)") ap2.add_argument("--th-x3", metavar="TXT", type=u, default="n", help="show thumbs at 3x resolution; client can override in UI unless force. [\033[32my\033[0m]=yes, [\033[32mn\033[0m]=no, [\033[32mfy\033[0m]=force-yes, [\033[32mfn\033[0m]=force-no (volflag=th3x)") diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 63613cc8..d65169ed 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -2220,7 +2220,7 @@ class AuthSrv(object): if k in vol.flags: vol.flags[k] = int(vol.flags[k]) - zs = "convt tail_fd tail_rate tail_tmax" + zs = "aconvt convt tail_fd tail_rate tail_tmax" for k in zs.split(): if k in vol.flags: vol.flags[k] = float(vol.flags[k]) diff --git a/copyparty/cfg.py b/copyparty/cfg.py index 8a07f956..5b1651c4 100644 --- a/copyparty/cfg.py +++ b/copyparty/cfg.py @@ -68,6 +68,7 @@ def vf_bmap() -> dict[str, str]: def vf_vmap() -> dict[str, str]: """argv-to-volflag: simple values""" ret = { + "ac_convt": "aconvt", "no_hash": "nohash", "no_idx": "noidx", "re_maxage": "scan", @@ -260,7 +261,8 @@ flagcats = { "thsize": "thumbnail res; WxH", "crop": "center-cropping (y/n/fy/fn)", "th3x": "3x resolution (y/n/fy/fn)", - "convt": "conversion timeout in seconds", + "convt": "convert-to-image timeout in seconds", + "aconvt": "convert-to-audio timeout in seconds", "ext_th=s=/b.png": "use /b.png as thumbnail for file-extension s", }, "handlers\n(better explained in --help-handlers)": { diff --git a/copyparty/th_srv.py b/copyparty/th_srv.py index 537c748a..2522f3a6 100644 --- a/copyparty/th_srv.py +++ b/copyparty/th_srv.py @@ -662,11 +662,11 @@ class ThumbSrv(object): ] cmd += [fsenc(tpath)] - self._run_ff(cmd, vn) + self._run_ff(cmd, vn, "convt") - def _run_ff(self, cmd: list[bytes], vn: VFS, oom: int = 400) -> None: + def _run_ff(self, cmd: list[bytes], vn: VFS, kto: str, oom: int = 400) -> None: # self.log((b" ".join(cmd)).decode("utf-8")) - ret, _, serr = runcmd(cmd, timeout=vn.flags["convt"], nice=True, oom=oom) + ret, _, serr = runcmd(cmd, timeout=vn.flags[kto], nice=True, oom=oom) if not ret: return @@ -748,7 +748,7 @@ class ThumbSrv(object): # fmt: on cmd += [fsenc(tpath)] - self._run_ff(cmd, vn) + self._run_ff(cmd, vn, "convt") if "pngquant" in vn.flags: wtpath = tpath + ".png" @@ -809,7 +809,7 @@ class ThumbSrv(object): b"-y", fsenc(infile), ] # fmt: on - self._run_ff(cmd, vn) + self._run_ff(cmd, vn, "convt") fc = "[0:a:0]aresample=48000{},showspectrumpic=s=" if "3" in fmt: @@ -851,7 +851,7 @@ class ThumbSrv(object): ] cmd += [fsenc(tpath)] - self._run_ff(cmd, vn) + self._run_ff(cmd, vn, "convt") def conv_mp3(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: quality = self.args.q_mp3.lower() @@ -890,7 +890,7 @@ class ThumbSrv(object): fsenc(tpath) ] # fmt: on - self._run_ff(cmd, vn, oom=300) + self._run_ff(cmd, vn, "aconvt", oom=300) def conv_flac(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: if self.args.no_acode or not self.args.allow_flac: @@ -915,7 +915,7 @@ class ThumbSrv(object): fsenc(tpath) ] # fmt: on - self._run_ff(cmd, vn, oom=300) + self._run_ff(cmd, vn, "aconvt", oom=300) def conv_wav(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: if self.args.no_acode or not self.args.allow_wav: @@ -950,7 +950,7 @@ class ThumbSrv(object): fsenc(tpath) ] # fmt: on - self._run_ff(cmd, vn, oom=300) + self._run_ff(cmd, vn, "aconvt", oom=300) def conv_opus(self, abspath: str, tpath: str, fmt: str, vn: VFS) -> None: if self.args.no_acode or not self.args.q_opus: @@ -1006,7 +1006,7 @@ class ThumbSrv(object): fsenc(tpath) ] # fmt: on - self._run_ff(cmd, vn, oom=300) + self._run_ff(cmd, vn, "aconvt", oom=300) def _conv_caf( self, @@ -1046,7 +1046,7 @@ class ThumbSrv(object): fsenc(tmp_opus) ] # fmt: on - self._run_ff(cmd, vn, oom=300) + self._run_ff(cmd, vn, "aconvt", oom=300) # iOS fails to play some "insufficiently complex" files # (average file shorter than 8 seconds), so of course we @@ -1073,7 +1073,7 @@ class ThumbSrv(object): fsenc(tpath) ] # fmt: on - self._run_ff(cmd, vn, oom=300) + self._run_ff(cmd, vn, "aconvt", oom=300) else: # simple remux should be safe @@ -1092,7 +1092,7 @@ class ThumbSrv(object): fsenc(tpath) ] # fmt: on - self._run_ff(cmd, vn, oom=300) + self._run_ff(cmd, vn, "aconvt", oom=300) try: wunlink(self.log, tmp_opus, vn.flags) diff --git a/tests/util.py b/tests/util.py index 16c75e92..80e3067e 100644 --- a/tests/util.py +++ b/tests/util.py @@ -158,7 +158,7 @@ class Cfg(Namespace): ex = "hash_mt hsortn qdel safe_dedup srch_time tail_fd tail_rate u2abort u2j u2sz" ka.update(**{k: 1 for k in ex.split()}) - ex = "au_vol dl_list mtab_age reg_cap s_thead s_tbody tail_tmax tail_who th_convt ups_who zip_who" + ex = "ac_convt au_vol dl_list mtab_age reg_cap s_thead s_tbody tail_tmax tail_who th_convt ups_who zip_who" ka.update(**{k: 9 for k in ex.split()}) ex = "ctl_re db_act forget_ip idp_cookie idp_store k304 loris no304 nosubtle qr_pin re_maxage rproxy rsp_jtr rsp_slp s_wr_slp snap_wri theme themes turbo u2ow zipmaxn zipmaxs" From 3aa8b7aa2d75091cc988e864fb89f344124ebbce Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 20:31:58 +0000 Subject: [PATCH 166/174] ftp: reject uploads nicely; closes #573 if a client tries to upload where it does not have write-access, rather than kicking the client with an exception, reply properly --- copyparty/ftpd.py | 15 +++++++++++---- 1 file changed, 11 insertions(+), 4 deletions(-) diff --git a/copyparty/ftpd.py b/copyparty/ftpd.py index 004fb492..99b37d26 100644 --- a/copyparty/ftpd.py +++ b/copyparty/ftpd.py @@ -285,9 +285,12 @@ class FtpFs(AbstractedFS): # returning 550 is library-default and suitable raise FSE("No such file or directory") - avfs = vfs.chk_ap(ap, st) - if not avfs: - raise FSE("Permission denied", 1) + if vfs.realpath: + avfs = vfs.chk_ap(ap, st) + if not avfs: + raise FSE("Permission denied", 1) + else: + avfs = vfs self.cwd = nwd ( @@ -492,7 +495,11 @@ class FtpHandler(FTPHandler): def ftp_STOR(self, file: str, mode: str = "w") -> Any: # Optional[str] vp = join(self.fs.cwd, file).lstrip("/") - ap, vfs, rem = self.fs.v2a(vp, w=True) + try: + ap, vfs, rem = self.fs.v2a(vp, w=True) + except Exception as ex: + self.respond("550 %s" % (ex,), logging.info) + return self.vfs_map[ap] = vp xbu = vfs.flags.get("xbu") if xbu and not runhook( From f4a3fba29c9e58ca76d92e8e0a70d3083406b048 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Thu, 14 Aug 2025 21:53:57 +0000 Subject: [PATCH 167/174] add global-logout button --- copyparty/web/splash.css | 4 ++ copyparty/web/splash.html | 9 +++- copyparty/web/splash.js | 88 +++++++++++++++++++++++++++++++++++++++ 3 files changed, 100 insertions(+), 1 deletion(-) diff --git a/copyparty/web/splash.css b/copyparty/web/splash.css index 5ccebdc7..3b4d0f68 100644 --- a/copyparty/web/splash.css +++ b/copyparty/web/splash.css @@ -24,6 +24,7 @@ h1 { li { margin: 1em 0; } +#lo, a { color: #047; background: #fff; @@ -47,6 +48,7 @@ td a { float: right; margin: -.2em 0 0 .8em; } +#lo, .logout, a.r { color: #c04; @@ -176,12 +178,14 @@ html.z { html.z h1 { border-color: #777; } +html.z #lo, html.z a { color: #fff; background: #057; border-color: #37a; } html.z .logout, +html.z #lo, html.z a.r { background: #804; border-color: #c28; diff --git a/copyparty/web/splash.html b/copyparty/web/splash.html index 4363efd2..0a886ad4 100644 --- a/copyparty/web/splash.html +++ b/copyparty/web/splash.html @@ -22,7 +22,7 @@ <p id="b">howdy stranger   <small>(you're not logged in)</small></p> {%- else %} <a id="c" href="{{ r }}/?pw=x" class="logout">logout</a> - <p><span id="m">welcome back,</span> <strong>{{ this.uname|e }}</strong></p> + <p><span id="m">welcome back,</span> <strong id="un">{{ this.uname|e }}</strong></p> {%- endif %} {%- endif %} @@ -168,6 +168,13 @@ <li><a id="af" href="{{ r }}/?ru">show recent uploads</a></li> <li><a id="k" href="{{ r }}/?reset" class="r" onclick="localStorage.clear();return true">reset client settings</a></li> + + {%- if this.uname != '*' %} + <li><form method="post" enctype="multipart/form-data"> + <input type="hidden" name="act" value="logout" /> + <input type="submit" id="lo" value="logout “{{ this.uname|e }}” everywhere" /> + </form></li> + {% endif %} </ul> </div> diff --git a/copyparty/web/splash.js b/copyparty/web/splash.js index aad6ad2a..ad2d81db 100644 --- a/copyparty/web/splash.js +++ b/copyparty/web/splash.js @@ -17,6 +17,11 @@ var Ls = { "j1": "k304 bryter tilkoplingen for hver HTTP 304. Dette hjelper mot visse mellomtjenere som kan sette seg fast / plutselig slutter å laste sider, men det reduserer også ytelsen betydelig", "k1": "nullstill innstillinger", "l1": "logg inn:", + "ls3": "logg inn", + "lu4": "brukernavn", + "lp4": "passord", + "lo3": "logg ut “{0}” overalt", + "lo2": "avslutter økten på alle nettlesere", "m1": "velkommen tilbake,", "n1": "404: filen finnes ikke  ┐( ´ -`)┌", "o1": 'eller kanskje du ikke har tilgang? prøv et passord eller <a href="' + SR + '/?h">gå hjem</a>', @@ -46,6 +51,7 @@ var Ls = { "eng": { "d2": "shows the state of all active threads", "e2": "reload config files (accounts/volumes/volflags),$Nand rescan all e2ds volumes$N$Nnote: any changes to global settings$Nrequire a full restart to take effect", + "lo2": "ends the session on all browsers", "u2": "time since the last server write$N( upload / rename / ... )$N$N17d = 17 days$N1h23 = 1 hour 23 minutes$N4m56 = 4 minutes 56 seconds", "v2": "use this server as a local HDD", "ta1": "fill in your new password first", @@ -68,6 +74,11 @@ var Ls = { "j1": "k304 会在每个 HTTP 304 时断开连接。这有助于避免某些代理服务器卡住或突然停止加载页面,但也会显著降低性能。", "k1": "重置设置", "l1": "登录:", + "ls3": "登录", //m + "lu4": "用户名", //m + "lp4": "密码", //m + "lo3": "在所有地方注销 {0}", //m + "lo2": "这将结束在所有浏览器中的会话", //m "m1": "欢迎回来,", "n1": "404: 文件不存在  ┐( ´ -`)┌", "o1": '或者你可能没有权限?尝试输入密码或 <a href="' + SR + '/?h">回家</a>', @@ -110,6 +121,11 @@ var Ls = { "j1": "povolení k304 odpojí vašeho klienta při každém HTTP 304, což může zabránit některým chybovým proxy serverům, aby se zasekly (náhle nenačítaly stránky), <em>ale</em> také to obecně zpomalí věci", "k1": "resetovat nastavení klienta", "l1": "přihlaste se pro více:", + "ls3": "přihlásit se", //m + "lu4": "uživatelské jméno", //m + "lp4": "heslo", //m + "lo3": "odhlásit “{0}” všude", //m + "lo2": "tímto ukončíte relaci ve všech prohlížečích", //m "m1": "vítej zpět,", "n1": "404 nenalezeno  ┐( ´ -`)┌", "o1": 'nebo možná nemáš přístup -- zkus heslo nebo <a href="' + SR + '/?h">jdi domů</a>', @@ -151,6 +167,11 @@ var Ls = { "j1": "k304 trennt die Clientverbindung bei jedem HTTP 304, was Bugs mit problematischen Proxies vorbeugen kann (z.B. nicht ladenden Seiten), macht Dinge aber generell langsamer", "k1": "Client-Einstellungen zurücksetzen", "l1": "Melde dich an für mehr:", + "ls3": "Anmelden", //m + "lu4": "Benutzername", //m + "lp4": "Passwort", //m + "lo3": "“{0}” überall abmelden", //m + "lo2": "Dies beendet die Sitzung in allen Browsern", //m "m1": "Willkommen zurück,", "n1": "404 Nicht gefunden  ┐( ´ -`)┌", "o1": 'or maybe you don\'t have access -- try a password or <a href="' + SR + '/?h">go home</a>', @@ -192,6 +213,11 @@ var Ls = { "j1": "k304 katkaisee yhteytesi jokaisella HTTP 304:llä, mikä voi estää joitain bugisia välityspalvelimia jumittumasta/lopettamasta sivujen lataamista, <em>mutta</em> se myös vähentää suorituskykyä", "k1": "nollaa asetukset", "l1": "kirjaudu sisään:", + "ls3": "kirjaudu sisään", //m + "lu4": "käyttäjätunnus", //m + "lp4": "salasana", //m + "lo3": "kirjaa “{0}” ulos kaikkialta", //m + "lo2": "tämä lopettaa istunnon kaikissa selaimissa", //m "m1": "tervetuloa takaisin,", "n1": "404: ei löytynyt mitään  ┐( ´ -`)┌", "o1": 'tai ehkä sinulla ei vain ole käyttöoikeuksia? kokeile salasanaa tai <a href="' + SR + '/?h">mene kotiin</a>', @@ -234,6 +260,11 @@ var Ls = { "j1": "activer k304 va déconnecter votre client sur chaque HTTP 304, ce qui peut éviter à certains proxies défectueux de rester bloqués (les pages ne se chargent soudainement plus), <em>mais</em> cela ralentira également les choses en général", "k1": "réinitialiser les paramètres du client", "l1": "connectez-vous pour en savoir plus :", + "ls3": "se connecter", //m + "lu4": "nom d'utilisateur", //m + "lp4": "mot de passe", //m + "lo3": "déconnecter “{0}” partout", //m + "lo2": "cela mettra fin à la session sur tous les navigateurs", //m "m1": "heureux de vous revoir,", "n1": "404 introuvable  ┐( ´ -`)┌", "o1": 'ou peut-être que vous n\'y avez pas accès -- essayer un mot de passe ou <a href="' + SR + '/?h">aller à la page d\'accueil</a>', @@ -275,6 +306,11 @@ var Ls = { "j1": "η ενεργοποίηση του k304 θα αποσυνδέσει το πρόγραμμα πελάτη σου σε κάθε HTTP 304, κάτι που μπορεί να αποτρέψει κάποια προβληματικά proxies από το να κολλάνε (να μην φορτώνουν ξαφνικά σελίδες), <em>αλλά</em> θα κάνει τα πράγματα, γενικά πιο αργά", "k1": "επαναφορά ρυθμίσεων στο πρόγραμμα πελάτη", "l1": "συνδέσου για περισσότερα:", + "ls3": "σύνδεση", //m + "lu4": "όνομα χρήστη", //m + "lp4": "κωδικός πρόσβασης", //m + "lo3": "αποσύνδεση του “{0}” από παντού", //m + "lo2": "αυτό θα τερματίσει τη συνεδρία σε όλους τους περιηγητές", //m "m1": "καλώς ήρθες,", "n1": "404 δεν βρέθηκε  ┐( ´ -`)┌", "o1": '´η μήπως δεν έχεις πρόσβαση -- δοκίμασε έναν κωδικό <a href="' + SR + '/?h">πήγαινε στην αρχική</a>', @@ -316,6 +352,11 @@ var Ls = { "j1": "k304 interrompe la connessione per ogni HTTP 304. Questo aiuta contro alcuni proxy difettosi che possono bloccarsi o smettere improvvisamente di caricare pagine, ma riduce notevolmente le prestazioni", "k1": "resetta impostazioni", "l1": "accedi:", + "ls3": "accedi", //m + "lu4": "nome utente", //m + "lp4": "password", //m + "lo3": "disconnetti “{0}” ovunque", //m + "lo2": "questo terminerà la sessione su tutti i browser", //m "m1": "bentornato,", "n1": "404: file non trovato  ┐( ´ -`)┌", "o1": "oppure forse non hai accesso? prova una password o <a href=\"SR/?h\">torna alla home</a>", @@ -358,6 +399,11 @@ var Ls = { "j1": "k304를 활성화하면 모든 HTTP 304 응답 시 클라이언트 연결이 끊어집니다. 이는 일부 프록시가 멈추는 현상(갑자기 페이지가 로드되지 않음)을 방지할 수 있지만, <em>대신 전반적인 속도는 느려집니다.</em>", "k1": "클라이언트 설정 초기화", "l1": "로그인하기:", + "ls3": "로그인", //m + "lu4": "사용자 이름", //m + "lp4": "비밀번호", //m + "lo3": "{0}을(를) 모든 곳에서 로그아웃", //m + "lo2": "이 작업은 모든 브라우저에서 세션을 종료합니다", //m "m1": "또 오셨네요,", "n1": "404 찾을 수 없음  ┐( ´ -`)┌", "o1": "또는 접근 권한이 없을 수 있습니다. 비밀번호를 입력하거나 <a href=\"' + SR + '/?h\">홈으로 이동</a>하세요", @@ -400,6 +446,11 @@ var Ls = { "j1": "k304 verbreekt de verbinding voor elke HTTP 304. Dit helpt tegen bepaalde proxy servers die kunnen vastlopen/plotseling stoppen met het laden van pagina's, maar het vermindert ook de prestaties aanzienlijk", "k1": "Instellingen resetten", "l1": "Inloggen:", + "ls3": "inloggen", //m + "lu4": "gebruikersnaam", //m + "lp4": "wachtwoord", //m + "lo3": "“{0}” overal afmelden", //m + "lo2": "dit zal de sessie in alle browsers beëindigen", //m "m1": "Welkom terug,", "n1": "404: bestand bestaat niet  ┐( ´ -`)┌", "o1": 'of misschien heb je geen toegang? probeer een wachtwoord of <a href="' + SR + '/?h">ga naar startscherm</a>', @@ -442,6 +493,11 @@ var Ls = { "j1": "k304 bryt tilkoplinga for kvar HTTP 304. Dette hjelp mot visse mellomtjenarar som kan sette seg fast / plutselig sluttar å laste sider, men det sett óg ytinga ned betydelig", "k1": "nullstill innstillinger", "l1": "logg inn:", + "ls3": "logg inn", + "lu4": "brukarnamn", + "lp4": "passord", + "lo3": "logg ut “{0}” overalt", + "lo2": "avslutt økta på alle nettlesarar", "m1": "velkomen attende,", "n1": "404: filen finnast ikkje  ┐( ´ -`)┌", "o1": 'eller kanskje du ikkje har høve? prøv eit passord eller <a href="' + SR + '/?h">gå heim</a>', @@ -484,6 +540,11 @@ var Ls = { "j1": "włączenie k304 będzie odłączało klienta przy każdorazowym otrzymaniu kodu HTTP 304, co może zapobiec wieszaniu się wadliwych proxy, <em>ale</em> spowolni ogólne działanie", "k1": "zresetuj ustawienia klienta", "l1": "zaloguj się po więcej:", + "ls3": "zaloguj się", //m + "lu4": "nazwa użytkownika", //m + "lp4": "hasło", //m + "lo3": "wyloguj “{0}” wszędzie", //m + "lo2": "spowoduje to zakończenie sesji we wszystkich przeglądarkach", //m "m1": "Witaj,", "n1": "404 nie znaleziono  ┐( ´ -`)┌", "o1": 'lub możesz nie mieć dostępu -- spróbuj wprowadzić hasło lub <a href="' + SR + '/?h">przejdź do strony głównej</a>', @@ -526,6 +587,11 @@ var Ls = { "j1": "activar k304 desconectará tu cliente en cada HTTP 304, lo que puede evitar que algunos proxies con errores se atasquen (dejando de cargar páginas de repente), <em>pero</em> también ralentizará las cosas en general", "k1": "restablecer config. de cliente", "l1": "inicia sesión para más:", + "ls3": "iniciar sesión", //m + "lu4": "nombre de usuario", //m + "lp4": "contraseña", //m + "lo3": "cerrar sesión de “{0}” en todas partes", //m + "lo2": "esto finalizará la sesión en todos los navegadores", //m "m1": "bienvenido de nuevo,", "n1": "404 no encontrado  ┐( ´ -`)┌", "o1": '¿o quizás no tienes acceso? -- prueba con una contraseña o <a href=\"' + SR + '/?h\">vuelve al inicio</a>', @@ -568,6 +634,11 @@ var Ls = { "j1": "med k304 aktiverad kommer klienten att koppla bort sig vid varje HTTP 304-fel, vilket kan hindra vissa buggiga proxyservrar från att fastna (sidor slutar ladda), <em>men</em> saker kommer också att bli långsammare i allmänhet", "k1": "återställ klientinställningar", "l1": "logga in för att se mer:", + "ls3": "logga in", //m + "lu4": "användarnamn", //m + "lp4": "lösenord", //m + "lo3": "logga ut “{0}” överallt", //m + "lo2": "avsluta sessionen i alla webbläsare", //m "m1": "välkommen tillbaka,", "n1": "404 hittades inte  ┐( ´ -`)┌", "o1": 'eller så har du kanske inte tillgång -- prova ett lösenord eller <a href="' + SR + '/?h">åk hem</a>', @@ -610,6 +681,11 @@ var Ls = { "j1": "увімкнення k304 буде відключати ваш клієнт при кожному HTTP 304, що може запобігти зависанню деяких глючних проксі (раптово перестають завантажувати сторінки), <em>але</em> це також зробить усе повільнішим загалом", "k1": "скинути налаштування клієнта", "l1": "авторизуйтесь для інших опцій:", + "ls3": "увійти", //m + "lu4": "ім'я користувача", //m + "lp4": "пароль", //m + "lo3": "вийти з облікового запису “{0}” всюди", //m + "lo2": "це завершить сеанс у всіх браузерах", //m "m1": "з поверненням,", "n1": "404 не знайдено  ┐( ´ -`)┌", "o1": 'або у вас немає доступу -- спробуйте авторизуватися або <a href="' + SR + '/?h">повернутися на головну</a>', @@ -652,6 +728,11 @@ var Ls = { "j1": "включённый k304 будет отключать вас при получении HTTP 304, что может помочь при работе с некоторыми глючными прокси (перестают загружаться страницы), <em>но</em> это также сделает работу клиента медленнее", "k1": "сбросить локальные настройки", "l1": "авторизуйтесь для других опций:", + "ls3": "войти", //m + "lu4": "имя пользователя", //m + "lp4": "пароль", //m + "lo3": "выйти из “{0}” везде", //m + "lo2": "это завершит сеанс во всех браузерах", //m "m1": "с возвращением,", "n1": "404 не найдено  ┐( ´ -`)┌", "o1": 'или у вас нет доступа -- попробуйте авторизоваться или <a href="' + SR + '/?h">вернуться на главную</a>', @@ -696,7 +777,14 @@ for (var k in (d || {})) { o[a].innerHTML = d[k]; else if (f == 2) o[a].setAttribute("tt", d[k]); + else if (f == 3) + o[a].setAttribute("value", d[k]); + else if (f == 4) + o[a].setAttribute("placeholder", " " + d[k]); } +var o1 = ebi('lo'), o2 = ebi('un'); +if (o1 && o2 && d.lo3) + o1.setAttribute("value", d.lo3.format(o2.textContent)); try { if (is_idp) { From a4649d1e71308f3b663e95f1a26762f63d9d5a81 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Fri, 15 Aug 2025 19:19:21 +0000 Subject: [PATCH 168/174] generic header auth (closes #504); extends idp-auth to also accept a collection of headers (and expected values of those headers) and map those to certain users useful for Tailscale-User-Login and similar --- README.md | 15 ++++++++++ copyparty/__main__.py | 64 ++++++++++++++++++++++++++++++++++++++++- copyparty/authsrv.py | 12 ++++---- copyparty/httpcli.py | 18 ++++++++++-- copyparty/svchub.py | 19 ++++++++++-- copyparty/up2k.py | 2 +- copyparty/web/svcs.html | 2 +- tests/test_idp.py | 2 +- tests/util.py | 4 +-- 9 files changed, 122 insertions(+), 16 deletions(-) diff --git a/README.md b/README.md index 16bfb15f..f6ed75c6 100644 --- a/README.md +++ b/README.md @@ -91,6 +91,7 @@ made in Norway 🇳🇴 * [handlers](#handlers) - redefine behavior with plugins ([examples](./bin/handlers/)) * [ip auth](#ip-auth) - autologin based on IP range (CIDR) * [identity providers](#identity-providers) - replace copyparty passwords with oauth and such + * [generic header auth](#generic-header-auth) - other ways to auth by header * [user-changeable passwords](#user-changeable-passwords) - if permitted, users can change their own passwords * [using the cloud as storage](#using-the-cloud-as-storage) - connecting to an aws s3 bucket and similar * [hiding from google](#hiding-from-google) - tell search engines you don't wanna be indexed @@ -1915,6 +1916,20 @@ a more complete example of the copyparty configuration options [look like this]( but if you just want to let users change their own passwords, then you probably want [user-changeable passwords](#user-changeable-passwords) instead +### generic header auth + +other ways to auth by header + +if you have a middleware which adds a header with a user identifier, for example tailscale's `Tailscale-User-Login: alice.m@forest.net` then you can automatically auth as `alice` by defining that mapping with `--idp-hm-usr '^Tailscale-User-Login^alice.m@forest.net^alice'` or the following config file: + +```yaml +[global] + idp-hm-usr: ^Tailscale-User-Login^alice.m@forest.net^alice +``` + +repeat the whole `idp-hm-usr` option to add more mappings + + ## user-changeable passwords if permitted, users can change their own passwords in the control-panel diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 4ae60e61..3c89af99 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -614,6 +614,36 @@ def get_sects(): consider the config file for more flexible account/volume management, including dynamic reload at runtime (and being more readable w) + + see \033[32m--help-auth\033[0m for ways to provide the password in requests; + see \033[32m--help-idp\033[0m for replacing it with SSO and auth-middlewares + """ + ), + ], + [ + "auth", + "how to login from a client", + dedent( + """ + different ways to provide the password so you become authenticated: + + login with the ui: + go to \033[36mhttp://127.0.0.1:3923/?h\033[0m and login there + + send the password in the '\033[36mPW\033[0m' http-header: + \033[36mPW: \033[35mhunter2\033[0m + or if you have \033[33m--accounts\033[0m enabled, + \033[36mPW: \033[35med:hunter2\033[0m + + send the password in the URL itself: + \033[36mhttp://127.0.0.1:3923/\033[35m?pw=hunter2\033[0m + or if you have \033[33m--accounts\033[0m enabled, + \033[36mhttp://127.0.0.1:3923/\033[35m?pw=ed:hunter2\033[0m + + use basic-authentication: + \033[36mhttp://\033[35med:hunter2\033[36m@127.0.0.1:3923/\033[0m + which should be the same as this header: + \033[36mAuthorization: Basic \033[35mZWQ6aHVudGVyMg==\033[0m """ ), ], @@ -765,6 +795,36 @@ def get_sects(): the upload speed can easily drop to 10% for small files)""" ), ], + [ + "idp", + "replacing the login system with fancy middleware", + dedent( + """ + if you already have a centralized service which handles + user-authentication for other services already, you can + integrate copyparty with that for automatic login + + if the middleware is providing the username in an http-header + named '\033[35mtheUsername\033[0m' then do this: \033[36m--idp-h-usr theUsername\033[0m + + if the middleware is providing a list of groups in the header + named '\033[35mtheGroups\033[0m' then do this: \033[36m--idp-h-grp theGroup\033[0m + + if the list of groups is separated by '\033[35m%\033[0m' then \033[36m--idp-gsep %\033[0m + + if the middleware is providing a header named '\033[35mAccount\033[0m' + and the value is '\033[35malice@forest.net\033[0m' but the username is + actually '\033[35mmarisa\033[0m' then do this for each user: + \033[36m--idp-hm-usr ^Account^alice@forest.net^marisa\033[0m + (the separator '\033[35m^\033[0m' can be any character) + + make ABSOLUTELY SURE that the header can only be set by your + middleware and not by clients! and, as an extra precaution, + send a header named '\033[36mfinalmasterspark\033[0m' (a secret keyword) + and then \033[36m--idp-h-key finalmasterspark\033[0m to require that + """ + ), + ], [ "urlform", "how to handle url-form POSTs", @@ -1153,7 +1213,8 @@ def add_auth(ap): idp_db = os.path.join(E.cfg, "idp.db") ses_db = os.path.join(E.cfg, "sessions.db") ap2 = ap.add_argument_group("IdP / identity provider / user authentication options") - ap2.add_argument("--idp-h-usr", metavar="HN", type=u, default="", help="bypass the copyparty authentication checks if the request-header \033[33mHN\033[0m contains a username to associate the request with (for use with authentik/oauth/...)\n\033[1;31mWARNING:\033[0m if you enable this, make sure clients are unable to specify this header themselves; must be washed away and replaced by a reverse-proxy") + ap2.add_argument("--idp-h-usr", metavar="HN", type=u, action="append", help="\033[34mREPEATABLE:\033[0m bypass the copyparty authentication checks if the request-header \033[33mHN\033[0m contains a username to associate the request with (for use with authentik/oauth/...)\n\033[1;31mWARNING:\033[0m if you enable this, make sure clients are unable to specify this header themselves; must be washed away and replaced by a reverse-proxy") + ap2.add_argument("--idp-hm-usr", metavar="TXT", type=u, action="append", help="\033[34mREPEATABLE:\033[0m bypass the copyparty authentication checks if the request-header \033[33mHN\033[0m is provided, and its value exists in a mapping defined by this option; see --help-idp") ap2.add_argument("--idp-h-grp", metavar="HN", type=u, default="", help="assume the request-header \033[33mHN\033[0m contains the groupname of the requesting user; can be referenced in config files for group-based access control") ap2.add_argument("--idp-h-key", metavar="HN", type=u, default="", help="optional but recommended safeguard; your reverse-proxy will insert a secret header named \033[33mHN\033[0m into all requests, and the other IdP headers will be ignored if this header is not present") ap2.add_argument("--idp-gsep", metavar="RE", type=u, default="|:;+,", help="if there are multiple groups in \033[33m--idp-h-grp\033[0m, they are separated by one of the characters in \033[33mRE\033[0m") @@ -1168,6 +1229,7 @@ def add_auth(ap): ap2.add_argument("--no-ses", action="store_true", help="disable sessions; use plaintext passwords in cookies") ap2.add_argument("--grp-all", metavar="NAME", type=u, default="acct", help="the name of the auto-generated group which contains every username which is known") ap2.add_argument("--ipu", metavar="CIDR=USR", type=u, action="append", help="\033[34mREPEATABLE:\033[0m users with IP matching \033[33mCIDR\033[0m are auto-authenticated as username \033[33mUSR\033[0m; example: [\033[32m172.16.24.0/24=dave]") + ap2.add_argument("--have-idp-hdrs", type=u, default="", help=argparse.SUPPRESS) def add_chpw(ap): diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index d65169ed..09c845f8 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -1689,6 +1689,8 @@ class AuthSrv(object): self.log("\n{0}\n{1}{0}".format(t, "\n".join(slns))) raise + self.args.have_idp_hdrs = bool(self.args.idp_h_usr or self.args.idp_hm_usr) + self.setup_pwhash(acct) defpw = acct.copy() self.setup_chpw(acct) @@ -1701,7 +1703,7 @@ class AuthSrv(object): mount = cased - if not mount and not self.args.idp_h_usr: + if not mount and not self.args.have_idp_hdrs: # -h says our defaults are CWD at root and read/write for everyone axs = AXS(["*"], ["*"], None, None) ehint = "" @@ -1874,7 +1876,7 @@ class AuthSrv(object): if missing_users: zs = ", ".join(k for k in sorted(missing_users)) - if self.args.idp_h_usr: + if self.args.have_idp_hdrs: t = "the following users are unknown, and assumed to come from IdP: " self.log(t + zs, c=6) else: @@ -2551,7 +2553,7 @@ class AuthSrv(object): if not self.args.no_voldump: self.log(t) - if have_e2d or self.args.idp_h_usr: + if have_e2d or self.args.have_idp_hdrs: t = self.chk_sqlite_threadsafe() if t: self.log("\n\033[{}\033[0m\n".format(t)) @@ -2841,7 +2843,7 @@ class AuthSrv(object): def load_idp_db(self, quiet=False) -> None: # mutex me level = self.args.idp_store - if level < 2 or not self.args.idp_h_usr: + if level < 2 or not self.args.have_idp_hdrs: return assert sqlite3 # type: ignore # !rm @@ -2898,7 +2900,7 @@ class AuthSrv(object): n = [] q = "insert into us values (?,?,?)" accs = list(self.acct) - if self.args.idp_h_usr and self.args.idp_cookie: + if self.args.have_idp_hdrs and self.args.idp_cookie: accs.extend(self.idp_accs.keys()) for uname in accs: if uname not in ases: diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index c262dc2e..451f6afe 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -624,8 +624,22 @@ class HttpCli(object): or "*" ) - if self.args.idp_h_usr: - idp_usr = self.headers.get(self.args.idp_h_usr) or "" + if self.args.have_idp_hdrs: + idp_usr = "" + if self.args.idp_hm_usr: + for hn, hmv in self.args.idp_hm_usr_p.items(): + zs = self.headers.get(hn) + if zs: + for zs1, zs2 in hmv.items(): + if zs == zs1: + idp_usr = zs2 + break + if idp_usr: + break + for hn in self.args.idp_h_usr: + if idp_usr: + break + idp_usr = self.headers.get(hn) if idp_usr: idp_grp = ( self.headers.get(self.args.idp_h_grp) or "" diff --git a/copyparty/svchub.py b/copyparty/svchub.py index 950122d0..f2737b06 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -243,7 +243,7 @@ class SvcHub(object): t = "WARNING: --th-ram-max is very small (%.2f GiB); will not be able to %s" self.log("root", t % (args.th_ram_max, zs), 3) - if args.chpw and args.idp_h_usr: + if args.chpw and args.have_idp_hdrs: t = "ERROR: user-changeable passwords is incompatible with IdP/identity-providers; you must disable either --chpw or --idp-h-usr" self.log("root", t, 1) raise Exception(t) @@ -268,7 +268,7 @@ class SvcHub(object): args.no_ses = True args.shr = "" - if args.idp_store and args.idp_h_usr: + if args.idp_store and args.have_idp_hdrs: self.setup_db("idp") if not self.args.no_ses: @@ -1011,10 +1011,23 @@ class SvcHub(object): al.sus_urls = None al.xff_hdr = al.xff_hdr.lower() - al.idp_h_usr = al.idp_h_usr.lower() + al.idp_h_usr = [x.lower() for x in al.idp_h_usr or []] al.idp_h_grp = al.idp_h_grp.lower() al.idp_h_key = al.idp_h_key.lower() + al.idp_hm_usr_p = {} + for zs0 in al.idp_hm_usr or []: + try: + sep = zs0[:1] + hn, zs1, zs2 = zs0[1:].split(sep) + hn = hn.lower() + if hn in al.idp_hm_usr_p: + al.idp_hm_usr_p[hn][zs1] = zs2 + else: + al.idp_hm_usr_p[hn] = {zs1: zs2} + except: + raise Exception("invalid --idp-hm-usr [%s]" % (zs0,)) + al.ftp_ipa_nm = build_netmap(al.ftp_ipa or al.ipa, True) al.tftp_ipa_nm = build_netmap(al.tftp_ipa or al.ipa, True) diff --git a/copyparty/up2k.py b/copyparty/up2k.py index 993d5b82..f4d6d66f 100644 --- a/copyparty/up2k.py +++ b/copyparty/up2k.py @@ -903,7 +903,7 @@ class Up2k(object): self.iacct = self.asrv.iacct self.grps = self.asrv.grps - have_e2d = self.args.idp_h_usr or self.args.chpw or self.args.shr + have_e2d = self.args.have_idp_hdrs or self.args.chpw or self.args.shr vols = list(all_vols.values()) t0 = time.time() diff --git a/copyparty/web/svcs.html b/copyparty/web/svcs.html index 26fd5267..9d139a7b 100644 --- a/copyparty/web/svcs.html +++ b/copyparty/web/svcs.html @@ -42,7 +42,7 @@ - {% if args.idp_h_usr %} + {% if args.have_idp_hdrs %} <p style="line-height:2em"><b>WARNING:</b> this server is using IdP-based authentication, so this stuff may not work as advertised. Depending on server config, these commands can probably only be used to access areas which don't require authentication, unless you auth using any non-IdP accounts defined in the copyparty config. Please see <a href="https://github.com/9001/copyparty/blob/hovudstraum/docs/idp.md#connecting-webdav-clients">the IdP docs</a></p> {% endif %} diff --git a/tests/test_idp.py b/tests/test_idp.py index bd81378f..21d46758 100644 --- a/tests/test_idp.py +++ b/tests/test_idp.py @@ -63,7 +63,7 @@ class TestVFS(unittest.TestCase): cfgdir = os.path.join(here, "res", "idp") # globals are applied by main so need to cheat a little - xcfg = {"idp_h_usr": "x-idp-user", "idp_h_grp": "x-idp-group"} + xcfg = {"idp_h_usr": ["x-idp-user"], "idp_h_grp": "x-idp-group"} return here, cfgdir, xcfg diff --git a/tests/util.py b/tests/util.py index 80e3067e..8f4cd0e9 100644 --- a/tests/util.py +++ b/tests/util.py @@ -164,13 +164,13 @@ class Cfg(Namespace): ex = "ctl_re db_act forget_ip idp_cookie idp_store k304 loris no304 nosubtle qr_pin re_maxage rproxy rsp_jtr rsp_slp s_wr_slp snap_wri theme themes turbo u2ow zipmaxn zipmaxs" ka.update(**{k: 0 for k in ex.split()}) - ex = "ah_alg bname chmod_f chpw_db doctitle df exit favico idp_h_usr ipa html_head lg_sba lg_sbf log_fk md_sba md_sbf name og_desc og_site og_th og_title og_title_a og_title_v og_title_i shr tcolor textfiles txt_eol unlist vname xff_src zipmaxt R RS SR" + ex = "ah_alg bname chmod_f chpw_db doctitle df exit favico ipa html_head lg_sba lg_sbf log_fk md_sba md_sbf name og_desc og_site og_th og_title og_title_a og_title_v og_title_i shr tcolor textfiles txt_eol unlist vname xff_src zipmaxt R RS SR" ka.update(**{k: "" for k in ex.split()}) ex = "ban_403 ban_404 ban_422 ban_pw ban_pwc ban_url spinner" ka.update(**{k: "no" for k in ex.split()}) - ex = "ext_th grp on403 on404 xac xad xar xau xban xbc xbd xbr xbu xiu xm" + ex = "ext_th grp idp_h_usr idp_hm_usr on403 on404 xac xad xar xau xban xbc xbd xbr xbu xiu xm" ka.update(**{k: [] for k in ex.split()}) ex = "exp_lg exp_md" From 62e072a2ed14a96386509581616fde1545454480 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Fri, 15 Aug 2025 20:12:17 +0000 Subject: [PATCH 169/174] restrict account to ip/subnet; closes #397 --- README.md | 15 +++++++++++++++ copyparty/__main__.py | 2 ++ copyparty/authsrv.py | 1 + copyparty/ftpd.py | 4 ++++ copyparty/httpcli.py | 10 ++++++++-- copyparty/httpsrv.py | 6 ++++++ copyparty/svchub.py | 8 ++++++++ copyparty/util.py | 21 +++++++++++++++++++++ tests/util.py | 2 +- 9 files changed, 66 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index f6ed75c6..1d143b6d 100644 --- a/README.md +++ b/README.md @@ -90,6 +90,7 @@ made in Norway 🇳🇴 * [upload events](#upload-events) - the older, more powerful approach ([examples](./bin/mtag/)) * [handlers](#handlers) - redefine behavior with plugins ([examples](./bin/handlers/)) * [ip auth](#ip-auth) - autologin based on IP range (CIDR) + * [restrict to ip](#restrict-to-ip) - limit a user to certain IP ranges (CIDR) * [identity providers](#identity-providers) - replace copyparty passwords with oauth and such * [generic header auth](#generic-header-auth) - other ways to auth by header * [user-changeable passwords](#user-changeable-passwords) - if permitted, users can change their own passwords @@ -1897,6 +1898,20 @@ repeat the option to map additional subnets **be careful with this one!** if you have a reverseproxy, then you definitely want to make sure you have [real-ip](#real-ip) configured correctly, and it's probably a good idea to nullmap the reverseproxy's IP just in case; so if your reverseproxy is sending requests from `172.24.27.9` then that would be `--ipu=172.24.27.9/32=` +### restrict to ip + +limit a user to certain IP ranges (CIDR) , using the global-option `--ipr` + +for example, if the user `spartacus` should get rejected if they're not connecting from an IP that starts with `192.168.123` or `172.16`, then you can either specify `--ipr=192.168.123.0/24,172.16.0.0/16=spartacus` as a commandline option, or put this in a config file: + +```yaml +[global] + ipr: 192.168.123.0/24,172.16.0.0/16=spartacus +``` + +repeat the option to map additional users + + ## identity providers replace copyparty passwords with oauth and such diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 3c89af99..e732369b 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1229,7 +1229,9 @@ def add_auth(ap): ap2.add_argument("--no-ses", action="store_true", help="disable sessions; use plaintext passwords in cookies") ap2.add_argument("--grp-all", metavar="NAME", type=u, default="acct", help="the name of the auto-generated group which contains every username which is known") ap2.add_argument("--ipu", metavar="CIDR=USR", type=u, action="append", help="\033[34mREPEATABLE:\033[0m users with IP matching \033[33mCIDR\033[0m are auto-authenticated as username \033[33mUSR\033[0m; example: [\033[32m172.16.24.0/24=dave]") + ap2.add_argument("--ipr", metavar="CIDR=USR", type=u, action="append", help="\033[34mREPEATABLE:\033[0m username \033[33mUSR\033[0m can only connect from an IP matching one or more \033[33mCIDR\033[0m (comma-sep.); example: [\033[32m192.168.123.0/24,172.16.0.0/16=dave]") ap2.add_argument("--have-idp-hdrs", type=u, default="", help=argparse.SUPPRESS) + ap2.add_argument("--have-ipu-or-ipr", type=u, default="", help=argparse.SUPPRESS) def add_chpw(ap): diff --git a/copyparty/authsrv.py b/copyparty/authsrv.py index 09c845f8..5329ede1 100644 --- a/copyparty/authsrv.py +++ b/copyparty/authsrv.py @@ -1690,6 +1690,7 @@ class AuthSrv(object): raise self.args.have_idp_hdrs = bool(self.args.idp_h_usr or self.args.idp_hm_usr) + self.args.have_ipu_or_ipr = bool(self.args.ipu or self.args.ipr) self.setup_pwhash(acct) defpw = acct.copy() diff --git a/copyparty/ftpd.py b/copyparty/ftpd.py index 99b37d26..14f8d5b6 100644 --- a/copyparty/ftpd.py +++ b/copyparty/ftpd.py @@ -96,6 +96,10 @@ class FtpAuth(DummyAuthorizer): if args.ipu and uname == "*": uname = args.ipu_iu[args.ipu_nm.map(ip)] + if args.ipr and uname in args.ipr_u: + if not args.ipr_u[uname].map(ip): + logging.warning("username [%s] rejected by --ipr", uname) + uname = "*" if not uname or not (asrv.vfs.aread.get(uname) or asrv.vfs.awrite.get(uname)): g = self.hub.gpwd diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 451f6afe..4710feaf 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -695,8 +695,14 @@ class HttpCli(object): else: self.log("unknown username: %r" % (idp_usr,), 1) - if self.args.ipu and self.uname == "*": - self.uname = self.conn.ipu_iu[self.conn.ipu_nm.map(self.ip)] + if self.args.have_ipu_or_ipr: + if self.args.ipu and self.uname == "*": + self.uname = self.conn.ipu_iu[self.conn.ipu_nm.map(self.ip)] + ipr = self.conn.hsrv.ipr + if ipr and self.uname in ipr: + if not ipr[self.uname].map(self.ip): + self.log("username [%s] rejected by --ipr" % (self.uname,), 3) + self.uname = "*" self.rvol = self.asrv.vfs.aread[self.uname] self.wvol = self.asrv.vfs.awrite[self.uname] diff --git a/copyparty/httpsrv.py b/copyparty/httpsrv.py index ddf5e7bc..77492d56 100644 --- a/copyparty/httpsrv.py +++ b/copyparty/httpsrv.py @@ -70,6 +70,7 @@ from .util import ( build_netmap, has_resource, ipnorm, + load_ipr, load_ipu, load_resource, min_ex, @@ -193,6 +194,11 @@ class HttpSrv(object): else: self.ipu_iu = self.ipu_nm = None + if self.args.ipr: + self.ipr = load_ipr(self.log, self.args.ipr) + else: + self.ipr = None + self.ipa_nm = build_netmap(self.args.ipa) self.xff_nm = build_netmap(self.args.xff_src) self.xff_lan = build_netmap("lan") diff --git a/copyparty/svchub.py b/copyparty/svchub.py index f2737b06..7b06d96c 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -66,6 +66,7 @@ from .util import ( build_netmap, expat_ver, gzip, + load_ipr, load_ipu, lock_file, min_ex, @@ -259,6 +260,10 @@ class SvcHub(object): setattr(args, "ipu_iu", iu) setattr(args, "ipu_nm", nm) + if args.ipr: + ipr = load_ipr(self.log, args.ipr, True) + setattr(args, "ipr_u", ipr) + for zs in "ah_salt fk_salt dk_salt".split(): if getattr(args, "show_%s" % (zs,)): self.log("root", "effective %s is %s" % (zs, getattr(args, zs))) @@ -432,6 +437,9 @@ class SvcHub(object): getattr(args, zs).mutex = threading.Lock() except: pass + if args.ipr: + for nm in args.ipr_u.values(): + nm.mutex = threading.Lock() def _db_onfail_ses(self) -> None: self.args.no_ses = True diff --git a/copyparty/util.py b/copyparty/util.py index 1578d291..03517785 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -2954,6 +2954,27 @@ def load_ipu( return ip_u, nm +def load_ipr( + log: "RootLogger", iprs: list[str], defer_mutex: bool = False +) -> dict[str, NetMap]: + ret = {} + for ipr in iprs: + try: + zs, uname = ipr.split("=") + cidrs = zs.split(",") + except: + t = "\n invalid value %r for argument --ipr; must be CIDR[,CIDR[,...]]=UNAME (192.168.0.0/16=amelia)" + raise Exception(t % (ipr,)) + try: + nm = NetMap(["::"], cidrs, True, True, defer_mutex) + except Exception as ex: + t = "failed to translate --ipr into netmap, probably due to invalid config: %r" + log("root", t % (ex,), 1) + raise + ret[uname] = nm + return ret + + def yieldfile(fn: str, bufsz: int) -> Generator[bytes, None, None]: readsz = min(bufsz, 128 * 1024) with open(fsenc(fn), "rb", bufsz) as f: diff --git a/tests/util.py b/tests/util.py index 8f4cd0e9..16a0132a 100644 --- a/tests/util.py +++ b/tests/util.py @@ -170,7 +170,7 @@ class Cfg(Namespace): ex = "ban_403 ban_404 ban_422 ban_pw ban_pwc ban_url spinner" ka.update(**{k: "no" for k in ex.split()}) - ex = "ext_th grp idp_h_usr idp_hm_usr on403 on404 xac xad xar xau xban xbc xbd xbr xbu xiu xm" + ex = "ext_th grp idp_h_usr idp_hm_usr ipr on403 on404 xac xad xar xau xban xbc xbd xbr xbu xiu xm" ka.update(**{k: [] for k in ex.split()}) ex = "exp_lg exp_md" From 1228b5510b518bacf10baf07ed15d33ce5817f2e Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Fri, 15 Aug 2025 20:14:35 +0000 Subject: [PATCH 170/174] option -ss requires webdav login; closes #613 --- copyparty/__main__.py | 2 +- copyparty/svchub.py | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index e732369b..dcd78d8c 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1409,7 +1409,7 @@ def add_optouts(ap): def add_safety(ap): ap2 = ap.add_argument_group("safety options") ap2.add_argument("-s", action="count", default=0, help="increase safety: Disable thumbnails / potentially dangerous software (ffmpeg/pillow/vips), hide partial uploads, avoid crawlers.\n └─Alias of\033[32m --dotpart --no-thumb --no-mtag-ff --no-robots --force-js") - ap2.add_argument("-ss", action="store_true", help="further increase safety: Prevent js-injection, accidental move/delete, broken symlinks, webdav, 404 on 403, ban on excessive 404s.\n └─Alias of\033[32m -s --unpost=0 --no-del --no-mv --hardlink --vague-403 -nih") + ap2.add_argument("-ss", action="store_true", help="further increase safety: Prevent js-injection, accidental move/delete, broken symlinks, webdav requires login, 404 on 403, ban on excessive 404s.\n └─Alias of\033[32m -s --unpost=0 --no-del --no-mv --hardlink --dav-auth --vague-403 -nih") ap2.add_argument("-sss", action="store_true", help="further increase safety: Enable logging to disk, scan for dangerous symlinks.\n └─Alias of\033[32m -ss --no-dav --no-logues --no-readme -lo=cpp-%%Y-%%m%%d-%%H%%M%%S.txt.xz --ls=**,*,ln,p,r") ap2.add_argument("--ls", metavar="U[,V[,F]]", type=u, default="", help="do a sanity/safety check of all volumes on startup; arguments \033[33mUSER\033[0m,\033[33mVOL\033[0m,\033[33mFLAGS\033[0m (see \033[33m--help-ls\033[0m); example [\033[32m**,*,ln,p,r\033[0m]") ap2.add_argument("--xvol", action="store_true", help="never follow symlinks leaving the volume root, unless the link is into another volume where the user has similar access (volflag=xvol)") diff --git a/copyparty/svchub.py b/copyparty/svchub.py index 7b06d96c..8661bc29 100644 --- a/copyparty/svchub.py +++ b/copyparty/svchub.py @@ -156,6 +156,7 @@ class SvcHub(object): args.no_del = True args.no_mv = True args.hardlink = True + args.dav_auth = True args.vague_403 = True args.nih = True From 4df033ecc3888de1fe17cba39e28b6dee98f2242 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Fri, 15 Aug 2025 21:33:13 +0000 Subject: [PATCH 171/174] [DB-V6]: store usernames; closes #530 --- copyparty/__main__.py | 1 + copyparty/cfg.py | 2 + copyparty/httpcli.py | 39 +++++++++++++--- copyparty/u2idx.py | 2 +- copyparty/up2k.py | 105 +++++++++++++++++++++++++++--------------- copyparty/util.py | 2 +- copyparty/web/rups.js | 3 +- 7 files changed, 108 insertions(+), 46 deletions(-) diff --git a/copyparty/__main__.py b/copyparty/__main__.py index dcd78d8c..01216722 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -1120,6 +1120,7 @@ def add_upload(ap): ap2.add_argument("--put-ck", metavar="ALG", type=u, default="sha512", help="default checksum-hasher for PUT/WebDAV uploads: no / md5 / sha1 / sha256 / sha512 / b2 / blake2 / b2s / blake2s (volflag=put_ck)") ap2.add_argument("--bup-ck", metavar="ALG", type=u, default="sha512", help="default checksum-hasher for bup/basic-uploader: no / md5 / sha1 / sha256 / sha512 / b2 / blake2 / b2s / blake2s (volflag=bup_ck)") ap2.add_argument("--unpost", metavar="SEC", type=int, default=3600*12, help="grace period where uploads can be deleted by the uploader, even without delete permissions; 0=disabled, default=12h") + ap2.add_argument("--unp-who", metavar="NUM", type=int, default=1, help="clients can undo recent uploads by using the unpost tab (requires \033[33m-e2d\033[0m). [\033[32m0\033[0m] = never allowed (disable feature), [\033[32m1\033[0m] = allow if client has the same IP as the upload AND is using the same account, [\033[32m2\033[0m] = just check the IP, [\033[32m3\033[0m] = just check account-name (volflag=unp_who)") ap2.add_argument("--u2abort", metavar="NUM", type=int, default=1, help="clients can abort incomplete uploads by using the unpost tab (requires \033[33m-e2d\033[0m). [\033[32m0\033[0m] = never allowed (disable feature), [\033[32m1\033[0m] = allow if client has the same IP as the upload AND is using the same account, [\033[32m2\033[0m] = just check the IP, [\033[32m3\033[0m] = just check account-name (volflag=u2abort)") ap2.add_argument("--blank-wt", metavar="SEC", type=int, default=300, help="file write grace period (any client can write to a blank file last-modified more recently than \033[33mSEC\033[0m seconds ago)") ap2.add_argument("--reg-cap", metavar="N", type=int, default=38400, help="max number of uploads to keep in memory when running without \033[33m-e2d\033[0m; roughly 1 MiB RAM per 600") diff --git a/copyparty/cfg.py b/copyparty/cfg.py index 5b1651c4..3a7a205b 100644 --- a/copyparty/cfg.py +++ b/copyparty/cfg.py @@ -118,6 +118,7 @@ def vf_vmap() -> dict[str, str]: "u2ts", "uid", "gid", + "unp_who", "ups_who", "zip_who", "zipmaxn", @@ -343,6 +344,7 @@ flagcats = { "dky": 'allow seeing files (not folders) inside a specific folder\nwith "g" perm, and does not require a valid dirkey to do so', "rss": "allow '?rss' URL suffix (experimental)", "rmagic": "expensive analysis for mimetype accuracy", + "unp_who=2": "unpost only if same... 1=ip+name, 2=ip, 3=name", "ups_who=2": "restrict viewing the list of recent uploads", "zip_who=2": "restrict access to download-as-zip/tar", "zipmaxn=9k": "reject download-as-zip if more than 9000 files", diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 4710feaf..de3e6ede 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -5501,6 +5501,10 @@ class HttpCli(object): and ("*" in x.axs.uwrite or self.uname in x.axs.uwrite or x == shr_dbv) ] + q = "" + qp = (0,) + q_c = -1 + for vol in allvols: cur = idx.get_cur(vol) if not cur: @@ -5508,9 +5512,23 @@ class HttpCli(object): nfk, fk_alg = fk_vols.get(vol) or (0, 0) + zi = vol.flags["unp_who"] + if q_c != zi: + q_c = zi + q = "select sz, rd, fn, at from up where " + if zi == 1: + q += "ip=? and un=?" + qp = (self.ip, self.uname, lim) + elif zi == 2: + q += "ip=?" + qp = (self.ip, lim) + if zi == 3: + q += "un=?" + qp = (self.uname, lim) + q += " and at>? order by at desc" + n = 2000 - q = "select sz, rd, fn, at from up where ip=? and at>? order by at desc" - for sz, rd, fn, at in cur.execute(q, (self.ip, lim)): + for sz, rd, fn, at in cur.execute(q, qp): vp = "/" + "/".join(x for x in [vol.vpath, rd, fn] if x) if nfi == 0 or (nfi == 1 and vfi in vp.lower()): pass @@ -5635,8 +5653,8 @@ class HttpCli(object): continue n = 1000 - q = "select sz, rd, fn, ip, at from up where at>0 order by at desc" - for sz, rd, fn, ip, at in cur.execute(q): + q = "select sz, rd, fn, ip, at, un from up where at>0 order by at desc" + for sz, rd, fn, ip, at, un in cur.execute(q): vp = "/" + "/".join(x for x in [vol.vpath, rd, fn] if x) if nfi == 0 or (nfi == 1 and vfi in vp.lower()): pass @@ -5657,6 +5675,7 @@ class HttpCli(object): "sz": sz, "ip": ip, "at": at, + "un": un, "nfk": nfk, "adm": adm, } @@ -5701,12 +5720,16 @@ class HttpCli(object): adm = rv.pop("adm") if not adm: rv["ip"] = "(You)" if rv["ip"] == self.ip else "(?)" + if rv["un"] not in ("*", self.uname): + rv["un"] = "(?)" else: for rv in ret: adm = rv.pop("adm") if not adm: rv["ip"] = "(You)" if rv["ip"] == self.ip else "(?)" rv["at"] = 0 + if rv["un"] not in ("*", self.uname): + rv["un"] = "(?)" if self.is_vproxied: for v in ret: @@ -6628,13 +6651,15 @@ class HttpCli(object): tags = {k: v for k, v in r} if is_admin: - q = "select ip, at from up where rd=? and fn=?" + q = "select ip, at, un from up where rd=? and fn=?" try: - zs1, zs2 = icur.execute(q, erd_efn).fetchone() + zs1, zs2, zs3 = icur.execute(q, erd_efn).fetchone() if zs1: tags["up_ip"] = zs1 if zs2: tags[".up_at"] = zs2 + if zs3: + tags["up_by"] = zs3 except: pass elif add_up_at: @@ -6655,7 +6680,7 @@ class HttpCli(object): lmte = list(mte) if self.can_admin: - lmte.extend(("up_ip", ".up_at")) + lmte.extend(("up_by", "up_ip", ".up_at")) if "nodirsz" not in vf: tagset.add(".files") diff --git a/copyparty/u2idx.py b/copyparty/u2idx.py index ff0e329f..45b479c5 100644 --- a/copyparty/u2idx.py +++ b/copyparty/u2idx.py @@ -391,7 +391,7 @@ class U2idx(object): fk_alg = 2 if "fka" in flags else 1 c = cur.execute(uq, tuple(vuv)) for hit in c: - w, ts, sz, rd, fn, ip, at = hit[:7] + w, ts, sz, rd, fn = hit[:5] if rd.startswith("//") or fn.startswith("//"): rd, fn = s3dec(rd, fn) diff --git a/copyparty/up2k.py b/copyparty/up2k.py index f4d6d66f..7ae6441b 100644 --- a/copyparty/up2k.py +++ b/copyparty/up2k.py @@ -77,7 +77,7 @@ except: if HAVE_SQLITE3: import sqlite3 -DB_VER = 5 +DB_VER = 6 if True: # pylint: disable=using-constant-test from typing import Any, Optional, Pattern, Union @@ -1655,7 +1655,7 @@ class Up2k(object): abspath = cdirs + fn nohash = reh.search(abspath) if reh else False - sql = "select w, mt, sz, ip, at from up where rd = ? and fn = ?" + sql = "select w, mt, sz, ip, at, un from up where rd = ? and fn = ?" try: c = db.c.execute(sql, (rd, fn)) except: @@ -1664,7 +1664,7 @@ class Up2k(object): in_db = list(c.fetchall()) if in_db: self.pp.n -= 1 - dw, dts, dsz, ip, at = in_db[0] + dw, dts, dsz, ip, at, un = in_db[0] if len(in_db) > 1: t = "WARN: multiple entries: %r => %r |%d|\n%r" rep_db = "\n".join([repr(x) for x in in_db]) @@ -1677,6 +1677,9 @@ class Up2k(object): if dts == lmod and dsz == sz and (nohash or dw[0] != "#" or not sz): continue + if un is None: + un = "" + t = "reindex %r => %r mtime(%s/%s) size(%s/%s)" self.log(t % (top, rp, dts, lmod, dsz, sz)) self.db_rm(db.c, rd, fn, 0) @@ -1687,6 +1690,7 @@ class Up2k(object): dw = "" ip = "" at = 0 + un = "" self.pp.msg = "a%d %s" % (self.pp.n, abspath) @@ -1712,9 +1716,10 @@ class Up2k(object): if dw and dw != wark: ip = "" at = 0 + un = "" # skip upload hooks by not providing vflags - self.db_add(db.c, {}, rd, fn, lmod, sz, "", "", wark, wark, "", "", ip, at) + self.db_add(db.c, {}, rd, fn, lmod, sz, "", "", wark, wark, "", un, ip, at) db.n += 1 db.nf += 1 tfa += 1 @@ -2151,8 +2156,8 @@ class Up2k(object): with self.mutex: try: - q = "select rd, fn, ip, at from up where substr(w,1,16)=? and +w=?" - rd, fn, ip, at = cur.execute(q, (w16, w)).fetchone() + q = "select rd, fn, ip, at, un from up where substr(w,1,16)=? and +w=?" + rd, fn, ip, at, un = cur.execute(q, (w16, w)).fetchone() except: # file modified/deleted since spooling continue @@ -2171,12 +2176,15 @@ class Up2k(object): abspath = djoin(ptop, rd, fn) self.pp.msg = "c%d %s" % (nq, abspath) if not mpool: - n_tags = self._tagscan_file(cur, entags, w, abspath, ip, at) + n_tags = self._tagscan_file(cur, entags, w, abspath, ip, at, un) else: + oth_tags = {} if ip: - oth_tags = {"up_ip": ip, "up_at": at} - else: - oth_tags = {} + oth_tags["up_ip"] = ip + if at: + oth_tags["up_at"] = at + if un: + oth_tags["up_by"] = un mpool.put(Mpqe({}, entags, w, abspath, oth_tags)) with self.mutex: @@ -2332,8 +2340,8 @@ class Up2k(object): if w in in_progress: continue - q = "select rd, fn, ip, at from up where substr(w,1,16)=? limit 1" - rd, fn, ip, at = cur.execute(q, (w,)).fetchone() + q = "select rd, fn, ip, at, un from up where substr(w,1,16)=? limit 1" + rd, fn, ip, at, un = cur.execute(q, (w,)).fetchone() rd, fn = s3dec(rd, fn) abspath = djoin(ptop, rd, fn) @@ -2357,7 +2365,10 @@ class Up2k(object): if ip: oth_tags["up_ip"] = ip + if at: oth_tags["up_at"] = at + if un: + oth_tags["up_by"] = un jobs.append(Mpqe(parsers, set(), w, abspath, oth_tags)) in_progress[w] = True @@ -2546,6 +2557,7 @@ class Up2k(object): abspath: str, ip: str, at: float, + un: Optional[str], ) -> int: """will mutex(main)""" assert self.mtag # !rm @@ -2566,7 +2578,10 @@ class Up2k(object): if ip: tags["up_ip"] = ip + if at: tags["up_at"] = at + if un: + tags["up_by"] = un with self.mutex: return self._tag_file(write_cur, entags, wark, abspath, tags) @@ -2670,16 +2685,19 @@ class Up2k(object): if not existed and ver is None: return self._try_create_db(db_path, cur) - if ver == 4: + for upver in (4, 5): + if ver != upver: + continue try: t = "creating backup before upgrade: " cur = self._backup_db(db_path, cur, ver, t) - self._upgrade_v4(cur) - ver = 5 + getattr(self, "_upgrade_v%d" % (upver,))(cur) + ver += 1 # type: ignore except: - self.log("WARN: failed to upgrade from v4", 3) + self.log("WARN: failed to upgrade from v%d" % (ver,), 3) if ver == DB_VER: + # these no longer serve their intended purpose but they're great as additional sanchks self._add_dhash_tab(cur) self._add_xiu_tab(cur) self._add_cv_tab(cur) @@ -2781,7 +2799,7 @@ class Up2k(object): idx = r"create index up_w on up(w)" for cmd in [ - r"create table up (w text, mt int, sz int, rd text, fn text, ip text, at int)", + r"create table up (w text, mt int, sz int, rd text, fn text, ip text, at int, un text)", r"create index up_vp on up(rd, fn)", r"create index up_fn on up(fn)", r"create index up_ip on up(ip)", @@ -2814,6 +2832,15 @@ class Up2k(object): cur.connection.commit() + def _upgrade_v5(self, cur: "sqlite3.Cursor") -> None: + for cmd in [ + r"alter table up add column un text", + r"update kv set v=6 where k='sver'", + ]: + cur.execute(cmd) + + cur.connection.commit() + def _add_dhash_tab(self, cur: "sqlite3.Cursor") -> None: # v5 -> v5a try: @@ -3011,7 +3038,7 @@ class Up2k(object): argv = [dwark[:16], dwark] c2 = cur.execute(q, tuple(argv)) - for _, dtime, dsize, dp_dir, dp_fn, ip, at in c2: + for _, dtime, dsize, dp_dir, dp_fn, ip, at, _ in c2: if dp_dir.startswith("//") or dp_fn.startswith("//"): dp_dir, dp_fn = s3dec(dp_dir, dp_fn) @@ -3433,7 +3460,7 @@ class Up2k(object): try: vrel = vjoin(job["prel"], fname) xlink = bool(vf.get("xlink")) - cur, wark, _, _, _, _ = self._find_from_vpath(ptop, vrel) + cur, wark, _, _, _, _, _ = self._find_from_vpath(ptop, vrel) self._forget_file(ptop, vrel, cur, wark, True, st.st_size, xlink) except Exception as ex: self.log("skipping replace-relink: %r" % (ex,)) @@ -3890,14 +3917,14 @@ class Up2k(object): # plugins may expect this to look like an actual IP db_ip = "1.1.1.1" if "no_db_ip" in vflags else ip - sql = "insert into up values (?,?,?,?,?,?,?)" - v = (dwark, int(ts), sz, rd, fn, db_ip, int(at or 0)) + sql = "insert into up values (?,?,?,?,?,?,?,?)" + v = (dwark, int(ts), sz, rd, fn, db_ip, int(at or 0), usr) try: db.execute(sql, v) except: assert self.mem_cur # !rm rd, fn = s3enc(self.mem_cur, rd, fn) - v = (dwark, int(ts), sz, rd, fn, db_ip, int(at or 0)) + v = (dwark, int(ts), sz, rd, fn, db_ip, int(at or 0), usr) db.execute(sql, v) self.volsize[db] += sz @@ -4038,7 +4065,7 @@ class Up2k(object): vn, rem = vn0.get_dbv(rem0) ptop = vn.realpath with self.mutex, self.reg_mutex: - abrt_cfg = self.flags.get(ptop, {}).get("u2abort", 1) + abrt_cfg = vn.flags.get("u2abort", 1) addr = (ip or "\n") if abrt_cfg in (1, 2) else "" user = ((uname or "\n"), "*") if abrt_cfg in (1, 3) else None reg = self.registry.get(ptop, {}) if abrt_cfg else {} @@ -4059,17 +4086,22 @@ class Up2k(object): if partial: dip = ip dat = time.time() + dun = uname + un_cfg = 1 else: - if not self.args.unpost: + un_cfg = vn.flags["unp_who"] + if not self.args.unpost or not un_cfg: t = "the unpost feature is disabled in server config" raise Pebkac(400, t) - _, _, _, _, dip, dat = self._find_from_vpath(ptop, rem) + _, _, _, _, dip, dat, dun = self._find_from_vpath(ptop, rem) t = "you cannot delete this: " if not dip: t += "file not found" - elif dip != ip: + elif dip != ip and un_cfg in (1, 2): + t += "not uploaded by (You)" + elif dun != uname and un_cfg in (1, 3): t += "not uploaded by (You)" elif dat < time.time() - self.args.unpost: t += "uploaded too long ago" @@ -4158,7 +4190,7 @@ class Up2k(object): try: ptop = dbv.realpath xlink = bool(dbv.flags.get("xlink")) - cur, wark, _, _, _, _ = self._find_from_vpath(ptop, volpath) + cur, wark, _, _, _, _, _ = self._find_from_vpath(ptop, volpath) self._forget_file( ptop, volpath, cur, wark, True, st.st_size, xlink ) @@ -4319,7 +4351,7 @@ class Up2k(object): bos.makedirs(os.path.dirname(dabs), vf=dvn.flags) - c1, w, ftime_, fsize_, ip, at = self._find_from_vpath( + c1, w, ftime_, fsize_, ip, at, un = self._find_from_vpath( svn_dbv.realpath, srem_dbv ) c2 = self.cur.get(dvn.realpath) @@ -4344,7 +4376,7 @@ class Up2k(object): w, w, "", - "", + un or "", ip or "", at or 0, ) @@ -4605,7 +4637,7 @@ class Up2k(object): return "k" - c1, w, ftime_, fsize_, ip, at = self._find_from_vpath(svn.realpath, srem) + c1, w, ftime_, fsize_, ip, at, un = self._find_from_vpath(svn.realpath, srem) c2 = self.cur.get(dvn.realpath) has_dupes = False @@ -4639,7 +4671,7 @@ class Up2k(object): w, w, "", - "", + un or "", ip or "", at or 0, ) @@ -4739,13 +4771,14 @@ class Up2k(object): Optional[int], str, Optional[int], + str, ]: cur = self.cur.get(ptop) if not cur: - return None, None, None, None, "", None + return None, None, None, None, "", None, "" rd, fn = vsplit(vrem) - q = "select w, mt, sz, ip, at from up where rd=? and fn=? limit 1" + q = "select w, mt, sz, ip, at, un from up where rd=? and fn=? limit 1" try: c = cur.execute(q, (rd, fn)) except: @@ -4754,9 +4787,9 @@ class Up2k(object): hit = c.fetchone() if hit: - wark, ftime, fsize, ip, at = hit - return cur, wark, ftime, fsize, ip, at - return cur, None, None, None, "", None + wark, ftime, fsize, ip, at, un = hit + return cur, wark, ftime, fsize, ip, at, un + return cur, None, None, None, "", None, "" def _forget_file( self, diff --git a/copyparty/util.py b/copyparty/util.py index 03517785..7e9d4bb1 100644 --- a/copyparty/util.py +++ b/copyparty/util.py @@ -3602,7 +3602,7 @@ def runihook( verbose: bool, cmd: str, vol: "VFS", - ups: list[tuple[str, int, int, str, str, str, int]], + ups: list[tuple[str, int, int, str, str, str, int, str]], ) -> bool: _, chk, fork, jtxt, wait, sp_ka, acmd = _parsehook(log, cmd) bcmd = [sfsenc(x) for x in acmd] diff --git a/copyparty/web/rups.js b/copyparty/web/rups.js index 17a44556..e4e8b3ce 100644 --- a/copyparty/web/rups.js +++ b/copyparty/web/rups.js @@ -1,5 +1,5 @@ function render() { - var html = ['<table id="tab"><thead><tr><th>size</th><th>who</th><th>when</th><th>age</th><th>dir</th><th>file</th></tr></thead><tbody>']; + var html = ['<table id="tab"><thead><tr><th>size</th><th>who</th><th>ip</th><th>when</th><th>age</th><th>dir</th><th>file</th></tr></thead><tbody>']; var ups = V.ups, now = V.now; ebi('filter').value = V.filter; ebi('hits').innerHTML = 'showing ' + ups.length + ' files'; @@ -16,6 +16,7 @@ function render() { sz = ('' + f.sz).replace(/\B(?=(\d{3})+(?!\d))/g, " "); html.push('<tr><td>' + sz + + '</td><td>' + (f.un || '') + '</td><td>' + f.ip + '</td><td>' + ts + '</td><td>' + sa + From e3c7d6776ee401673a9c34e4fd8a6e6fb2e86dad Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Fri, 15 Aug 2025 21:34:56 +0000 Subject: [PATCH 172/174] fix test --- tests/util.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/util.py b/tests/util.py index 16a0132a..14031829 100644 --- a/tests/util.py +++ b/tests/util.py @@ -155,7 +155,7 @@ class Cfg(Namespace): ex = "gid uid" ka.update(**{k: -1 for k in ex.split()}) - ex = "hash_mt hsortn qdel safe_dedup srch_time tail_fd tail_rate u2abort u2j u2sz" + ex = "hash_mt hsortn qdel safe_dedup srch_time tail_fd tail_rate u2abort u2j u2sz unp_who" ka.update(**{k: 1 for k in ex.split()}) ex = "ac_convt au_vol dl_list mtab_age reg_cap s_thead s_tbody tail_tmax tail_who th_convt ups_who zip_who" From 23ea1c8a14ae2c8adb7a38bc5f51bb42bbaa636b Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Fri, 15 Aug 2025 21:35:10 +0000 Subject: [PATCH 173/174] better dropdown color --- copyparty/web/browser.css | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/copyparty/web/browser.css b/copyparty/web/browser.css index 00e42cae..f5f472a5 100644 --- a/copyparty/web/browser.css +++ b/copyparty/web/browser.css @@ -1377,7 +1377,7 @@ html.y #ops svg circle { .opview select, .opview input[type=text] { color: var(--fg); - background: var(--bg-u5); + background: var(--txt-bg); border: none; box-shadow: 0 0 2px var(--txt-sh); border-bottom: 1px solid #999; @@ -1388,6 +1388,7 @@ html.y #ops svg circle { .opview select { padding: .3em; margin: .2em .4em; + background: var(--bg-u3); } .opview input.err { color: var(--err-fg); From 43a19779c181eeb22734e2417b662e667d1a0ad7 Mon Sep 17 00:00:00 2001 From: ed <s@ocv.me> Date: Fri, 15 Aug 2025 21:36:02 +0000 Subject: [PATCH 174/174] ftp: fix potential utime issue; closes #539 --- copyparty/ftpd.py | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/copyparty/ftpd.py b/copyparty/ftpd.py index 14f8d5b6..b7246291 100644 --- a/copyparty/ftpd.py +++ b/copyparty/ftpd.py @@ -409,8 +409,12 @@ class FtpFs(AbstractedFS): return st def utime(self, path: str, timeval: float) -> None: - ap = self.rv2a(path, w=True)[0] - return bos.utime(ap, (timeval, timeval)) + try: + ap = self.rv2a(path, w=True)[0] + return bos.utime(ap, (int(time.time()), int(timeval))) + except Exception as ex: + logging.error("ftp.utime: %s, %r", ex, ex) + raise def lstat(self, path: str) -> os.stat_result: ap = self.rv2a(path)[0]
timesizedonefile
' + (done ? L.un_del : L.un_abrt) + '' + unix2iso(res[a].at) + '
{{ f.lead }}{{ f.name|e }}{{ f.sz }}{{ f.tags[k]|e }}{{ f.ext|e }}{{ f.dt }}
{{ f.ext }}{{ f.dt }}
'); + ln = ln.concat([tn.ext, unix2ui(tn.ts)]).join(''); html.push(ln + '
' + (done ? L.un_del : L.un_abrt) + '' + unix2iso(res[a].at) + '' + unix2ui(res[a].at) + '' + ('' + res[a].sz).replace(/\B(?=(\d{3})+(?!\d))/g, " ") + '100%' + res[a].pd + '%' + linksplit(res[a].vp).join(' / ') + '