more thumbnailer configs available as volflags:
--th-convt = convt --th-no-crop = nocrop --th-size = thsize
This commit is contained in:
		
							parent
							
								
									a14943c8de
								
							
						
					
					
						commit
						b54b7213a7
					
				| @ -1007,10 +1007,10 @@ def add_thumbnail(ap): | |||||||
|     ap2.add_argument("--no-thumb", action="store_true", help="disable all thumbnails (volflag=dthumb)") |     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-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)") |     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") |     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-mt", metavar="CORES", type=int, default=CORES, help="num cpu cores to use for generating thumbnails") | ||||||
|     ap2.add_argument("--th-convt", metavar="SEC", type=int, default=60, help="conversion timeout in seconds") |     ap2.add_argument("--th-convt", metavar="SEC", type=float, default=60, help="conversion timeout in seconds (volflag=convt)") | ||||||
|     ap2.add_argument("--th-no-crop", action="store_true", help="dynamic height; show full image") |     ap2.add_argument("--th-no-crop", action="store_true", help="dynamic height; show full image (volflag=nocrop)") | ||||||
|     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,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-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-no-webp", action="store_true", help="disable webp output") | ||||||
|  | |||||||
| @ -1420,6 +1420,10 @@ class AuthSrv(object): | |||||||
|                 if k in vol.flags: |                 if k in vol.flags: | ||||||
|                     vol.flags[k] = int(vol.flags[k]) |                     vol.flags[k] = int(vol.flags[k]) | ||||||
| 
 | 
 | ||||||
|  |             for k in ("convt",): | ||||||
|  |                 if k in vol.flags: | ||||||
|  |                     vol.flags[k] = float(vol.flags[k]) | ||||||
|  | 
 | ||||||
|             for k1, k2 in IMPLICATIONS: |             for k1, k2 in IMPLICATIONS: | ||||||
|                 if k1 in vol.flags: |                 if k1 in vol.flags: | ||||||
|                     vol.flags[k2] = True |                     vol.flags[k2] = True | ||||||
|  | |||||||
| @ -13,6 +13,7 @@ def vf_bmap() -> dict[str, str]: | |||||||
|         "no_dedup": "copydupes", |         "no_dedup": "copydupes", | ||||||
|         "no_dupe": "nodupe", |         "no_dupe": "nodupe", | ||||||
|         "no_forget": "noforget", |         "no_forget": "noforget", | ||||||
|  |         "th_no_crop": "nocrop", | ||||||
|         "dav_auth": "davauth", |         "dav_auth": "davauth", | ||||||
|         "dav_rt": "davrt", |         "dav_rt": "davrt", | ||||||
|     } |     } | ||||||
| @ -40,8 +41,8 @@ def vf_bmap() -> dict[str, str]: | |||||||
| 
 | 
 | ||||||
| def vf_vmap() -> dict[str, str]: | def vf_vmap() -> dict[str, str]: | ||||||
|     """argv-to-volflag: simple values""" |     """argv-to-volflag: simple values""" | ||||||
|     ret = {} |     ret = {"th_convt": "convt", "th_size": "thsize"} | ||||||
|     for k in ("lg_sbf", "md_sbf", "unlist"): |     for k in ("dbd", "lg_sbf", "md_sbf", "nrand", "unlist"): | ||||||
|         ret[k] = k |         ret[k] = k | ||||||
|     return ret |     return ret | ||||||
| 
 | 
 | ||||||
| @ -49,7 +50,7 @@ def vf_vmap() -> dict[str, str]: | |||||||
| def vf_cmap() -> dict[str, str]: | def vf_cmap() -> dict[str, str]: | ||||||
|     """argv-to-volflag: complex/lists""" |     """argv-to-volflag: complex/lists""" | ||||||
|     ret = {} |     ret = {} | ||||||
|     for k in ("dbd", "html_head", "mte", "mth", "nrand"): |     for k in ("html_head", "mte", "mth"): | ||||||
|         ret[k] = k |         ret[k] = k | ||||||
|     return ret |     return ret | ||||||
| 
 | 
 | ||||||
| @ -124,6 +125,9 @@ flagcats = { | |||||||
|         "dvthumb": "disables video thumbnails", |         "dvthumb": "disables video thumbnails", | ||||||
|         "dathumb": "disables audio thumbnails (spectrograms)", |         "dathumb": "disables audio thumbnails (spectrograms)", | ||||||
|         "dithumb": "disables image thumbnails", |         "dithumb": "disables image thumbnails", | ||||||
|  |         "thsize": "thumbnail res; WxH", | ||||||
|  |         "nocrop": "disable center-cropping", | ||||||
|  |         "convt": "conversion timeout in seconds", | ||||||
|     }, |     }, | ||||||
|     "handlers\n(better explained in --help-handlers)": { |     "handlers\n(better explained in --help-handlers)": { | ||||||
|         "on404=PY": "handle 404s by executing PY file", |         "on404=PY": "handle 404s by executing PY file", | ||||||
|  | |||||||
| @ -13,6 +13,7 @@ import time | |||||||
| from queue import Queue | from queue import Queue | ||||||
| 
 | 
 | ||||||
| from .__init__ import ANYWIN, TYPE_CHECKING | from .__init__ import ANYWIN, TYPE_CHECKING | ||||||
|  | from .authsrv import VFS | ||||||
| from .bos import bos | from .bos import bos | ||||||
| from .mtag import HAVE_FFMPEG, HAVE_FFPROBE, ffprobe | from .mtag import HAVE_FFMPEG, HAVE_FFPROBE, ffprobe | ||||||
| from .util import ( | from .util import ( | ||||||
| @ -110,8 +111,6 @@ class ThumbSrv(object): | |||||||
|         self.args = hub.args |         self.args = hub.args | ||||||
|         self.log_func = hub.log |         self.log_func = hub.log | ||||||
| 
 | 
 | ||||||
|         res = hub.args.th_size.split("x") |  | ||||||
|         self.res = tuple([int(x) for x in res]) |  | ||||||
|         self.poke_cd = Cooldown(self.args.th_poke) |         self.poke_cd = Cooldown(self.args.th_poke) | ||||||
| 
 | 
 | ||||||
|         self.mutex = threading.Lock() |         self.mutex = threading.Lock() | ||||||
| @ -119,7 +118,7 @@ class ThumbSrv(object): | |||||||
|         self.stopping = False |         self.stopping = False | ||||||
|         self.nthr = max(1, self.args.th_mt) |         self.nthr = max(1, self.args.th_mt) | ||||||
| 
 | 
 | ||||||
|         self.q: Queue[Optional[tuple[str, str]]] = Queue(self.nthr * 4) |         self.q: Queue[Optional[tuple[str, str, VFS]]] = Queue(self.nthr * 4) | ||||||
|         for n in range(self.nthr): |         for n in range(self.nthr): | ||||||
|             Daemon(self.worker, "thumb-{}-{}".format(n, self.nthr)) |             Daemon(self.worker, "thumb-{}-{}".format(n, self.nthr)) | ||||||
| 
 | 
 | ||||||
| @ -184,6 +183,10 @@ class ThumbSrv(object): | |||||||
|         with self.mutex: |         with self.mutex: | ||||||
|             return not self.nthr |             return not self.nthr | ||||||
| 
 | 
 | ||||||
|  |     def getres(self, vn: VFS) -> tuple[int, int]: | ||||||
|  |         w, h = vn.flags["thsize"].split("x") | ||||||
|  |         return int(w), int(h) | ||||||
|  | 
 | ||||||
|     def get(self, ptop: str, rem: str, mtime: float, fmt: str) -> Optional[str]: |     def get(self, ptop: str, rem: str, mtime: float, fmt: str) -> Optional[str]: | ||||||
|         histpath = self.asrv.vfs.histtab.get(ptop) |         histpath = self.asrv.vfs.histtab.get(ptop) | ||||||
|         if not histpath: |         if not histpath: | ||||||
| @ -211,7 +214,13 @@ class ThumbSrv(object): | |||||||
|                 do_conv = True |                 do_conv = True | ||||||
| 
 | 
 | ||||||
|         if do_conv: |         if do_conv: | ||||||
|             self.q.put((abspath, tpath)) |             allvols = list(self.asrv.vfs.all_vols.values()) | ||||||
|  |             vn = next((x for x in allvols if x.realpath == ptop), None) | ||||||
|  |             if not vn: | ||||||
|  |                 self.log("ptop [{}] not in {}".format(ptop, allvols), 3) | ||||||
|  |                 vn = self.asrv.vfs.all_aps[0][1] | ||||||
|  | 
 | ||||||
|  |             self.q.put((abspath, tpath, vn)) | ||||||
|             self.log("conv {} \033[0m{}".format(tpath, abspath), c=6) |             self.log("conv {} \033[0m{}".format(tpath, abspath), c=6) | ||||||
| 
 | 
 | ||||||
|         while not self.stopping: |         while not self.stopping: | ||||||
| @ -248,7 +257,7 @@ class ThumbSrv(object): | |||||||
|             if not task: |             if not task: | ||||||
|                 break |                 break | ||||||
| 
 | 
 | ||||||
|             abspath, tpath = task |             abspath, tpath, vn = task | ||||||
|             ext = abspath.split(".")[-1].lower() |             ext = abspath.split(".")[-1].lower() | ||||||
|             png_ok = False |             png_ok = False | ||||||
|             funs = [] |             funs = [] | ||||||
| @ -281,7 +290,7 @@ class ThumbSrv(object): | |||||||
| 
 | 
 | ||||||
|             for fun in funs: |             for fun in funs: | ||||||
|                 try: |                 try: | ||||||
|                     fun(abspath, ttpath) |                     fun(abspath, ttpath, vn) | ||||||
|                     break |                     break | ||||||
|                 except Exception as ex: |                 except Exception as ex: | ||||||
|                     msg = "{} could not create thumbnail of {}\n{}" |                     msg = "{} could not create thumbnail of {}\n{}" | ||||||
| @ -315,9 +324,10 @@ class ThumbSrv(object): | |||||||
|         with self.mutex: |         with self.mutex: | ||||||
|             self.nthr -= 1 |             self.nthr -= 1 | ||||||
| 
 | 
 | ||||||
|     def fancy_pillow(self, im: "Image.Image") -> "Image.Image": |     def fancy_pillow(self, im: "Image.Image", vn: VFS) -> "Image.Image": | ||||||
|         # exif_transpose is expensive (loads full image + unconditional copy) |         # exif_transpose is expensive (loads full image + unconditional copy) | ||||||
|         r = max(*self.res) * 2 |         res = self.getres(vn) | ||||||
|  |         r = max(*res) * 2 | ||||||
|         im.thumbnail((r, r), resample=Image.LANCZOS) |         im.thumbnail((r, r), resample=Image.LANCZOS) | ||||||
|         try: |         try: | ||||||
|             k = next(k for k, v in ExifTags.TAGS.items() if v == "Orientation") |             k = next(k for k, v in ExifTags.TAGS.items() if v == "Orientation") | ||||||
| @ -331,23 +341,23 @@ class ThumbSrv(object): | |||||||
|         if rot in rots: |         if rot in rots: | ||||||
|             im = im.transpose(rots[rot]) |             im = im.transpose(rots[rot]) | ||||||
| 
 | 
 | ||||||
|         if self.args.th_no_crop: |         if "nocrop" in vn.flags: | ||||||
|             im.thumbnail(self.res, resample=Image.LANCZOS) |             im.thumbnail(res, resample=Image.LANCZOS) | ||||||
|         else: |         else: | ||||||
|             iw, ih = im.size |             iw, ih = im.size | ||||||
|             dw, dh = self.res |             dw, dh = res | ||||||
|             res = (min(iw, dw), min(ih, dh)) |             res = (min(iw, dw), min(ih, dh)) | ||||||
|             im = ImageOps.fit(im, res, method=Image.LANCZOS) |             im = ImageOps.fit(im, res, method=Image.LANCZOS) | ||||||
| 
 | 
 | ||||||
|         return im |         return im | ||||||
| 
 | 
 | ||||||
|     def conv_pil(self, abspath: str, tpath: str) -> None: |     def conv_pil(self, abspath: str, tpath: str, vn: VFS) -> None: | ||||||
|         with Image.open(fsenc(abspath)) as im: |         with Image.open(fsenc(abspath)) as im: | ||||||
|             try: |             try: | ||||||
|                 im = self.fancy_pillow(im) |                 im = self.fancy_pillow(im, vn) | ||||||
|             except Exception as ex: |             except Exception as ex: | ||||||
|                 self.log("fancy_pillow {}".format(ex), "90") |                 self.log("fancy_pillow {}".format(ex), "90") | ||||||
|                 im.thumbnail(self.res) |                 im.thumbnail(self.getres(vn)) | ||||||
| 
 | 
 | ||||||
|             fmts = ["RGB", "L"] |             fmts = ["RGB", "L"] | ||||||
|             args = {"quality": 40} |             args = {"quality": 40} | ||||||
| @ -370,12 +380,12 @@ class ThumbSrv(object): | |||||||
| 
 | 
 | ||||||
|             im.save(tpath, **args) |             im.save(tpath, **args) | ||||||
| 
 | 
 | ||||||
|     def conv_vips(self, abspath: str, tpath: str) -> None: |     def conv_vips(self, abspath: str, tpath: str, vn: VFS) -> None: | ||||||
|         crops = ["centre", "none"] |         crops = ["centre", "none"] | ||||||
|         if self.args.th_no_crop: |         if "nocrop" in vn.flags: | ||||||
|             crops = ["none"] |             crops = ["none"] | ||||||
| 
 | 
 | ||||||
|         w, h = self.res |         w, h = self.getres(vn) | ||||||
|         kw = {"height": h, "size": "down", "intent": "relative"} |         kw = {"height": h, "size": "down", "intent": "relative"} | ||||||
| 
 | 
 | ||||||
|         for c in crops: |         for c in crops: | ||||||
| @ -389,8 +399,8 @@ class ThumbSrv(object): | |||||||
| 
 | 
 | ||||||
|         img.write_to_file(tpath, Q=40) |         img.write_to_file(tpath, Q=40) | ||||||
| 
 | 
 | ||||||
|     def conv_ffmpeg(self, abspath: str, tpath: str) -> None: |     def conv_ffmpeg(self, abspath: str, tpath: str, vn: VFS) -> None: | ||||||
|         ret, _ = ffprobe(abspath, int(self.args.th_convt / 2)) |         ret, _ = ffprobe(abspath, int(vn.flags["convt"] / 2)) | ||||||
|         if not ret: |         if not ret: | ||||||
|             return |             return | ||||||
| 
 | 
 | ||||||
| @ -402,12 +412,13 @@ class ThumbSrv(object): | |||||||
|             seek = [b"-ss", "{:.0f}".format(dur / 3).encode("utf-8")] |             seek = [b"-ss", "{:.0f}".format(dur / 3).encode("utf-8")] | ||||||
| 
 | 
 | ||||||
|         scale = "scale={0}:{1}:force_original_aspect_ratio=" |         scale = "scale={0}:{1}:force_original_aspect_ratio=" | ||||||
|         if self.args.th_no_crop: |         if "nocrop" in vn.flags: | ||||||
|             scale += "decrease,setsar=1:1" |             scale += "decrease,setsar=1:1" | ||||||
|         else: |         else: | ||||||
|             scale += "increase,crop={0}:{1},setsar=1:1" |             scale += "increase,crop={0}:{1},setsar=1:1" | ||||||
| 
 | 
 | ||||||
|         bscale = scale.format(*list(self.res)).encode("utf-8") |         res = self.getres(vn) | ||||||
|  |         bscale = scale.format(*list(res)).encode("utf-8") | ||||||
|         # fmt: off |         # fmt: off | ||||||
|         cmd = [ |         cmd = [ | ||||||
|             b"ffmpeg", |             b"ffmpeg", | ||||||
| @ -439,11 +450,11 @@ class ThumbSrv(object): | |||||||
|             ] |             ] | ||||||
| 
 | 
 | ||||||
|         cmd += [fsenc(tpath)] |         cmd += [fsenc(tpath)] | ||||||
|         self._run_ff(cmd) |         self._run_ff(cmd, vn) | ||||||
| 
 | 
 | ||||||
|     def _run_ff(self, cmd: list[bytes]) -> None: |     def _run_ff(self, cmd: list[bytes], vn: VFS) -> None: | ||||||
|         # self.log((b" ".join(cmd)).decode("utf-8")) |         # self.log((b" ".join(cmd)).decode("utf-8")) | ||||||
|         ret, _, serr = runcmd(cmd, timeout=self.args.th_convt) |         ret, _, serr = runcmd(cmd, timeout=vn.flags["convt"]) | ||||||
|         if not ret: |         if not ret: | ||||||
|             return |             return | ||||||
| 
 | 
 | ||||||
| @ -486,8 +497,8 @@ class ThumbSrv(object): | |||||||
|         self.log(t + txt, c=c) |         self.log(t + txt, c=c) | ||||||
|         raise sp.CalledProcessError(ret, (cmd[0], b"...", cmd[-1])) |         raise sp.CalledProcessError(ret, (cmd[0], b"...", cmd[-1])) | ||||||
| 
 | 
 | ||||||
|     def conv_waves(self, abspath: str, tpath: str) -> None: |     def conv_waves(self, abspath: str, tpath: str, vn: VFS) -> None: | ||||||
|         ret, _ = ffprobe(abspath, int(self.args.th_convt / 2)) |         ret, _ = ffprobe(abspath, int(vn.flags["convt"] / 2)) | ||||||
|         if "ac" not in ret: |         if "ac" not in ret: | ||||||
|             raise Exception("not audio") |             raise Exception("not audio") | ||||||
| 
 | 
 | ||||||
| @ -512,10 +523,10 @@ class ThumbSrv(object): | |||||||
|         # fmt: on |         # fmt: on | ||||||
| 
 | 
 | ||||||
|         cmd += [fsenc(tpath)] |         cmd += [fsenc(tpath)] | ||||||
|         self._run_ff(cmd) |         self._run_ff(cmd, vn) | ||||||
| 
 | 
 | ||||||
|     def conv_spec(self, abspath: str, tpath: str) -> None: |     def conv_spec(self, abspath: str, tpath: str, vn: VFS) -> None: | ||||||
|         ret, _ = ffprobe(abspath, int(self.args.th_convt / 2)) |         ret, _ = ffprobe(abspath, int(vn.flags["convt"] / 2)) | ||||||
|         if "ac" not in ret: |         if "ac" not in ret: | ||||||
|             raise Exception("not audio") |             raise Exception("not audio") | ||||||
| 
 | 
 | ||||||
| @ -555,13 +566,13 @@ class ThumbSrv(object): | |||||||
|             ] |             ] | ||||||
| 
 | 
 | ||||||
|         cmd += [fsenc(tpath)] |         cmd += [fsenc(tpath)] | ||||||
|         self._run_ff(cmd) |         self._run_ff(cmd, vn) | ||||||
| 
 | 
 | ||||||
|     def conv_opus(self, abspath: str, tpath: str) -> None: |     def conv_opus(self, abspath: str, tpath: str, vn: VFS) -> None: | ||||||
|         if self.args.no_acode: |         if self.args.no_acode: | ||||||
|             raise Exception("disabled in server config") |             raise Exception("disabled in server config") | ||||||
| 
 | 
 | ||||||
|         ret, _ = ffprobe(abspath, int(self.args.th_convt / 2)) |         ret, _ = ffprobe(abspath, int(vn.flags["convt"] / 2)) | ||||||
|         if "ac" not in ret: |         if "ac" not in ret: | ||||||
|             raise Exception("not audio") |             raise Exception("not audio") | ||||||
| 
 | 
 | ||||||
| @ -597,7 +608,7 @@ class ThumbSrv(object): | |||||||
|                 fsenc(tmp_opus) |                 fsenc(tmp_opus) | ||||||
|             ] |             ] | ||||||
|             # fmt: on |             # fmt: on | ||||||
|             self._run_ff(cmd) |             self._run_ff(cmd, vn) | ||||||
| 
 | 
 | ||||||
|         # iOS fails to play some "insufficiently complex" files |         # iOS fails to play some "insufficiently complex" files | ||||||
|         # (average file shorter than 8 seconds), so of course we |         # (average file shorter than 8 seconds), so of course we | ||||||
| @ -621,7 +632,7 @@ class ThumbSrv(object): | |||||||
|                 fsenc(tpath) |                 fsenc(tpath) | ||||||
|             ] |             ] | ||||||
|             # fmt: on |             # fmt: on | ||||||
|             self._run_ff(cmd) |             self._run_ff(cmd, vn) | ||||||
| 
 | 
 | ||||||
|         elif want_caf: |         elif want_caf: | ||||||
|             # simple remux should be safe |             # simple remux should be safe | ||||||
| @ -639,7 +650,7 @@ class ThumbSrv(object): | |||||||
|                 fsenc(tpath) |                 fsenc(tpath) | ||||||
|             ] |             ] | ||||||
|             # fmt: on |             # fmt: on | ||||||
|             self._run_ff(cmd) |             self._run_ff(cmd, vn) | ||||||
| 
 | 
 | ||||||
|         if tmp_opus != tpath: |         if tmp_opus != tpath: | ||||||
|             try: |             try: | ||||||
|  | |||||||
| @ -24,6 +24,7 @@ from queue import Queue | |||||||
| from .__init__ import ANYWIN, PY2, TYPE_CHECKING, WINDOWS | from .__init__ import ANYWIN, PY2, TYPE_CHECKING, WINDOWS | ||||||
| from .authsrv import LEELOO_DALLAS, VFS, AuthSrv | from .authsrv import LEELOO_DALLAS, VFS, AuthSrv | ||||||
| from .bos import bos | from .bos import bos | ||||||
|  | from .cfg import vf_bmap, vf_vmap | ||||||
| from .fsutil import Fstab | from .fsutil import Fstab | ||||||
| from .mtag import MParser, MTag | from .mtag import MParser, MTag | ||||||
| from .util import ( | from .util import ( | ||||||
| @ -757,8 +758,9 @@ class Up2k(object): | |||||||
|         ff = "\033[0;35m{}{:.0}" |         ff = "\033[0;35m{}{:.0}" | ||||||
|         fv = "\033[0;36m{}:\033[90m{}" |         fv = "\033[0;36m{}:\033[90m{}" | ||||||
|         fx = set(("html_head",)) |         fx = set(("html_head",)) | ||||||
|         fdl = ("dbd", "lg_sbf", "md_sbf", "mte", "mth", "mtp", "nrand", "rand") |         fd = vf_bmap() | ||||||
|         fd = {x: x for x in fdl} |         fd.update(vf_vmap()) | ||||||
|  |         fd = {v: k for k, v in fd.items()} | ||||||
|         fl = { |         fl = { | ||||||
|             k: v |             k: v | ||||||
|             for k, v in flags.items() |             for k, v in flags.items() | ||||||
| @ -769,6 +771,9 @@ class Up2k(object): | |||||||
|             for k, v in fl.items() |             for k, v in fl.items() | ||||||
|             if k not in fx |             if k not in fx | ||||||
|         ] |         ] | ||||||
|  |         if not a: | ||||||
|  |             a = ["\033[90mall-default"] | ||||||
|  | 
 | ||||||
|         if a: |         if a: | ||||||
|             vpath = "?" |             vpath = "?" | ||||||
|             for k, v in self.asrv.vfs.all_vols.items(): |             for k, v in self.asrv.vfs.all_vols.items(): | ||||||
|  | |||||||
| @ -2427,7 +2427,7 @@ def killtree(root: int) -> None: | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def runcmd( | def runcmd( | ||||||
|     argv: Union[list[bytes], list[str]], timeout: Optional[int] = None, **ka: Any |     argv: Union[list[bytes], list[str]], timeout: Optional[float] = None, **ka: Any | ||||||
| ) -> tuple[int, str, str]: | ) -> tuple[int, str, str]: | ||||||
|     kill = ka.pop("kill", "t")  # [t]ree [m]ain [n]one |     kill = ka.pop("kill", "t")  # [t]ree [m]ain [n]one | ||||||
|     capture = ka.pop("capture", 3)  # 0=none 1=stdout 2=stderr 3=both |     capture = ka.pop("capture", 3)  # 0=none 1=stdout 2=stderr 3=both | ||||||
| @ -2480,7 +2480,7 @@ def chkcmd(argv: Union[list[bytes], list[str]], **ka: Any) -> tuple[str, str]: | |||||||
|     return sout, serr |     return sout, serr | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def mchkcmd(argv: Union[list[bytes], list[str]], timeout: int = 10) -> None: | def mchkcmd(argv: Union[list[bytes], list[str]], timeout: float = 10) -> None: | ||||||
|     if PY2: |     if PY2: | ||||||
|         with open(os.devnull, "wb") as f: |         with open(os.devnull, "wb") as f: | ||||||
|             rv = sp.call(argv, stdout=f, stderr=f) |             rv = sp.call(argv, stdout=f, stderr=f) | ||||||
|  | |||||||
| @ -98,7 +98,7 @@ class Cfg(Namespace): | |||||||
|     def __init__(self, a=None, v=None, c=None): |     def __init__(self, a=None, v=None, c=None): | ||||||
|         ka = {} |         ka = {} | ||||||
| 
 | 
 | ||||||
|         ex = "daw dav_auth dav_inf dav_mac dav_rt dotsrch e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp ed emp force_js getmod grid hardlink ih ihead magic never_symlink nid nih no_acode no_athumb no_dav no_dedup no_del no_dupe no_logues no_mv no_readme no_robots no_sb_md no_sb_lg no_scandir no_thumb no_vthumb no_zip nrand nw rand smb vc xdev xlink xvol" |         ex = "daw dav_auth dav_inf dav_mac dav_rt dotsrch e2d e2ds e2dsa e2t e2ts e2tsr e2v e2vu e2vp ed emp force_js getmod grid hardlink ih ihead magic never_symlink nid nih no_acode no_athumb no_dav no_dedup no_del no_dupe no_logues no_mv no_readme no_robots no_sb_md no_sb_lg no_scandir no_thumb no_vthumb no_zip nrand nw rand smb th_no_crop vc xdev xlink xvol" | ||||||
|         ka.update(**{k: False for k in ex.split()}) |         ka.update(**{k: False for k in ex.split()}) | ||||||
| 
 | 
 | ||||||
|         ex = "dotpart no_rescan no_sendfile no_voldump plain_ip" |         ex = "dotpart no_rescan no_sendfile no_voldump plain_ip" | ||||||
| @ -107,7 +107,7 @@ class Cfg(Namespace): | |||||||
|         ex = "css_browser hist js_browser no_forget no_hash no_idx" |         ex = "css_browser hist js_browser no_forget no_hash no_idx" | ||||||
|         ka.update(**{k: None for k in ex.split()}) |         ka.update(**{k: None for k in ex.split()}) | ||||||
| 
 | 
 | ||||||
|         ex = "s_thead s_tbody" |         ex = "s_thead s_tbody th_convt" | ||||||
|         ka.update(**{k: 9 for k in ex.split()}) |         ka.update(**{k: 9 for k in ex.split()}) | ||||||
| 
 | 
 | ||||||
|         ex = "df loris re_maxage rproxy rsp_jtr rsp_slp s_wr_slp theme themes turbo" |         ex = "df loris re_maxage rproxy rsp_jtr rsp_slp s_wr_slp theme themes turbo" | ||||||
| @ -126,6 +126,7 @@ class Cfg(Namespace): | |||||||
|             E=E, |             E=E, | ||||||
|             dbd="wal", |             dbd="wal", | ||||||
|             s_wr_sz=512 * 1024, |             s_wr_sz=512 * 1024, | ||||||
|  |             th_size="320x256", | ||||||
|             unpost=600, |             unpost=600, | ||||||
|             u2sort="s", |             u2sort="s", | ||||||
|             mtp=[], |             mtp=[], | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 ed
						ed