mirror of
https://github.com/mikf/gallery-dl.git
synced 2024-11-25 20:22:36 +01:00
226 lines
7.1 KiB
Python
226 lines
7.1 KiB
Python
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright 2019-2023 Mike Fährmann
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License version 2 as
|
|
# published by the Free Software Foundation.
|
|
|
|
"""Write metadata to external files"""
|
|
|
|
from .common import PostProcessor
|
|
from .. import util, formatter
|
|
import json
|
|
import sys
|
|
import os
|
|
|
|
|
|
class MetadataPP(PostProcessor):
|
|
|
|
def __init__(self, job, options):
|
|
PostProcessor.__init__(self, job)
|
|
|
|
mode = options.get("mode")
|
|
cfmt = options.get("content-format") or options.get("format")
|
|
omode = "w"
|
|
filename = None
|
|
|
|
if mode == "tags":
|
|
self.write = self._write_tags
|
|
ext = "txt"
|
|
elif mode == "modify":
|
|
self.run = self._run_modify
|
|
self.fields = {
|
|
name: formatter.parse(value, None, util.identity).format_map
|
|
for name, value in options.get("fields").items()
|
|
}
|
|
ext = None
|
|
elif mode == "delete":
|
|
self.run = self._run_delete
|
|
self.fields = options.get("fields")
|
|
ext = None
|
|
elif mode == "custom" or not mode and cfmt:
|
|
self.write = self._write_custom
|
|
if isinstance(cfmt, list):
|
|
cfmt = "\n".join(cfmt) + "\n"
|
|
self._content_fmt = formatter.parse(cfmt).format_map
|
|
ext = "txt"
|
|
elif mode == "jsonl":
|
|
self.write = self._write_json
|
|
self._json_encode = self._make_encoder(options).encode
|
|
omode = "a"
|
|
filename = "data.jsonl"
|
|
else:
|
|
self.write = self._write_json
|
|
self._json_encode = self._make_encoder(options, 4).encode
|
|
ext = "json"
|
|
|
|
directory = options.get("directory")
|
|
if directory:
|
|
self._directory = self._directory_custom
|
|
sep = os.sep + (os.altsep or "")
|
|
self._metadir = util.expand_path(directory).rstrip(sep) + os.sep
|
|
|
|
filename = options.get("filename", filename)
|
|
extfmt = options.get("extension-format")
|
|
if filename:
|
|
if filename == "-":
|
|
self.run = self._run_stdout
|
|
else:
|
|
self._filename = self._filename_custom
|
|
self._filename_fmt = formatter.parse(filename).format_map
|
|
elif extfmt:
|
|
self._filename = self._filename_extfmt
|
|
self._extension_fmt = formatter.parse(extfmt).format_map
|
|
else:
|
|
self.extension = options.get("extension", ext)
|
|
|
|
events = options.get("event")
|
|
if events is None:
|
|
events = ("file",)
|
|
elif isinstance(events, str):
|
|
events = events.split(",")
|
|
job.register_hooks({event: self.run for event in events}, options)
|
|
|
|
self._init_archive(job, options, "_MD_")
|
|
self.mtime = options.get("mtime")
|
|
self.omode = options.get("open", omode)
|
|
self.encoding = options.get("encoding", "utf-8")
|
|
self.private = options.get("private", False)
|
|
self.skip = options.get("skip", False)
|
|
|
|
def run(self, pathfmt):
|
|
archive = self.archive
|
|
if archive and archive.check(pathfmt.kwdict):
|
|
return
|
|
|
|
directory = self._directory(pathfmt)
|
|
path = directory + self._filename(pathfmt)
|
|
|
|
if self.skip and os.path.exists(path):
|
|
return
|
|
|
|
try:
|
|
with open(path, self.omode, encoding=self.encoding) as fp:
|
|
self.write(fp, pathfmt.kwdict)
|
|
except FileNotFoundError:
|
|
os.makedirs(directory, exist_ok=True)
|
|
with open(path, self.omode, encoding=self.encoding) as fp:
|
|
self.write(fp, pathfmt.kwdict)
|
|
|
|
if archive:
|
|
archive.add(pathfmt.kwdict)
|
|
|
|
if self.mtime:
|
|
mtime = pathfmt.kwdict.get("_mtime")
|
|
if mtime:
|
|
util.set_mtime(path, mtime)
|
|
|
|
def _run_stdout(self, pathfmt):
|
|
self.write(sys.stdout, pathfmt.kwdict)
|
|
|
|
def _run_modify(self, pathfmt):
|
|
kwdict = pathfmt.kwdict
|
|
for key, func in self.fields.items():
|
|
obj = kwdict
|
|
try:
|
|
if "[" in key:
|
|
obj, key = _traverse(obj, key)
|
|
obj[key] = func(kwdict)
|
|
except Exception:
|
|
pass
|
|
|
|
def _run_delete(self, pathfmt):
|
|
kwdict = pathfmt.kwdict
|
|
for key in self.fields:
|
|
obj = kwdict
|
|
try:
|
|
if "[" in key:
|
|
obj, key = _traverse(obj, key)
|
|
del obj[key]
|
|
except Exception:
|
|
pass
|
|
|
|
def _directory(self, pathfmt):
|
|
return pathfmt.realdirectory
|
|
|
|
def _directory_custom(self, pathfmt):
|
|
return os.path.join(pathfmt.realdirectory, self._metadir)
|
|
|
|
def _filename(self, pathfmt):
|
|
return (pathfmt.filename or "metadata") + "." + self.extension
|
|
|
|
def _filename_custom(self, pathfmt):
|
|
return pathfmt.clean_path(pathfmt.clean_segment(
|
|
self._filename_fmt(pathfmt.kwdict)))
|
|
|
|
def _filename_extfmt(self, pathfmt):
|
|
kwdict = pathfmt.kwdict
|
|
ext = kwdict.get("extension")
|
|
kwdict["extension"] = pathfmt.extension
|
|
kwdict["extension"] = pathfmt.prefix + self._extension_fmt(kwdict)
|
|
filename = pathfmt.build_filename(kwdict)
|
|
kwdict["extension"] = ext
|
|
return filename
|
|
|
|
def _write_custom(self, fp, kwdict):
|
|
fp.write(self._content_fmt(kwdict))
|
|
|
|
def _write_tags(self, fp, kwdict):
|
|
tags = kwdict.get("tags") or kwdict.get("tag_string")
|
|
|
|
if not tags:
|
|
return
|
|
|
|
if isinstance(tags, str):
|
|
taglist = tags.split(", ")
|
|
if len(taglist) < len(tags) / 16:
|
|
taglist = tags.split(" ")
|
|
tags = taglist
|
|
elif isinstance(tags, dict):
|
|
taglists = tags.values()
|
|
tags = []
|
|
extend = tags.extend
|
|
for taglist in taglists:
|
|
extend(taglist)
|
|
tags.sort()
|
|
elif all(isinstance(e, dict) for e in tags):
|
|
taglists = tags
|
|
tags = []
|
|
extend = tags.extend
|
|
for tagdict in taglists:
|
|
extend([x for x in tagdict.values() if isinstance(x, str)])
|
|
tags.sort()
|
|
|
|
fp.write("\n".join(tags) + "\n")
|
|
|
|
def _write_json(self, fp, kwdict):
|
|
if not self.private:
|
|
kwdict = util.filter_dict(kwdict)
|
|
fp.write(self._json_encode(kwdict) + "\n")
|
|
|
|
@staticmethod
|
|
def _make_encoder(options, indent=None):
|
|
return json.JSONEncoder(
|
|
ensure_ascii=options.get("ascii", False),
|
|
sort_keys=options.get("sort", False),
|
|
separators=options.get("separators"),
|
|
indent=options.get("indent", indent),
|
|
check_circular=False,
|
|
default=util.json_default,
|
|
)
|
|
|
|
|
|
def _traverse(obj, key):
|
|
name, _, key = key.partition("[")
|
|
obj = obj[name]
|
|
|
|
while "[" in key:
|
|
name, _, key = key.partition("[")
|
|
obj = obj[name.strip("\"']")]
|
|
|
|
return obj, key.strip("\"']")
|
|
|
|
|
|
__postprocessor__ = MetadataPP
|