|
|
@ -9,6 +9,7 @@ import itertools |
|
|
|
import functools |
|
|
|
import functools |
|
|
|
import logging |
|
|
|
import logging |
|
|
|
import argparse |
|
|
|
import argparse |
|
|
|
|
|
|
|
import collections |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
logger = logging.getLogger("AnimeImport") |
|
|
|
logger = logging.getLogger("AnimeImport") |
|
|
@ -65,6 +66,7 @@ def make_symlink_in(target, in_directory, linkname, anchor=None): |
|
|
|
os.symlink(transport_path(target, anchor if anchor else os.curdir, in_directory), |
|
|
|
os.symlink(transport_path(target, anchor if anchor else os.curdir, in_directory), |
|
|
|
os.path.join(in_directory, linkname)) |
|
|
|
os.path.join(in_directory, linkname)) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def clean_links(path): |
|
|
|
def clean_links(path): |
|
|
|
for f in os.listdir(path): |
|
|
|
for f in os.listdir(path): |
|
|
|
if not f.startswith(".") and os.path.islink(os.path.join(path, f)): |
|
|
|
if not f.startswith(".") and os.path.islink(os.path.join(path, f)): |
|
|
@ -85,10 +87,14 @@ def opt_value_str(o): |
|
|
|
return str(o) |
|
|
|
return str(o) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def natural_sort_key(s, _nsre=re.compile(r'(\d+)')): |
|
|
|
def natural_sort_key(s, *, _nsre=re.compile(r'(\d+)')): |
|
|
|
return [int(text) if text.isdigit() else text.lower() for text in _nsre.split(s)] |
|
|
|
return [int(text) if text.isdigit() else text.lower() for text in _nsre.split(s)] |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def natural_name_sort_key(f, *, _nsre=re.compile(r'(\d+)')): |
|
|
|
|
|
|
|
return [int(text) if text.isdigit() else text.lower() for text in _nsre.split(f.name)] |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
############################################################################### |
|
|
|
############################################################################### |
|
|
|
## Specials patterns ## |
|
|
|
## Specials patterns ## |
|
|
|
############################################################################### |
|
|
|
############################################################################### |
|
|
@ -231,9 +237,11 @@ class Importer: |
|
|
|
Special.simple("Ending", r"[_ ]((NC|TV)?ED|Ending|Closing)[ _]?(?P<ep>\d+)"), |
|
|
|
Special.simple("Ending", r"[_ ]((NC|TV)?ED|Ending|Closing)[ _]?(?P<ep>\d+)"), |
|
|
|
Special.simple("Special", r"[_ ](Special|OVA|SP)[ _]?(?P<ep>\d+)?[ _]?"), |
|
|
|
Special.simple("Special", r"[_ ](Special|OVA|SP)[ _]?(?P<ep>\d+)?[ _]?"), |
|
|
|
# .nfo files |
|
|
|
# .nfo files |
|
|
|
Special({"$custom": lambda m, i: ("link", os.path.join(i.destination, i.main_name + ".nfo"))}, re.compile(r".nfo$")) |
|
|
|
Special({"$custom": lambda m, i: ("link", i.destination, i.main_name + ".nfo")}, re.compile(r".nfo$")) |
|
|
|
] |
|
|
|
] |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
log_skip = False |
|
|
|
|
|
|
|
|
|
|
|
link_template = "{series} S{season:d}E{episode:03d}" |
|
|
|
link_template = "{series} S{season:d}E{episode:03d}" |
|
|
|
until_template = "-E{episode:03d}" |
|
|
|
until_template = "-E{episode:03d}" |
|
|
|
|
|
|
|
|
|
|
@ -260,6 +268,8 @@ class Importer: |
|
|
|
self.destination = os.path.abspath(destination) |
|
|
|
self.destination = os.path.abspath(destination) |
|
|
|
self.main_name = os.path.basename(self.destination) |
|
|
|
self.main_name = os.path.basename(self.destination) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
self.library_path = os.path.normpath(os.path.join(self.destination, "..")) |
|
|
|
|
|
|
|
|
|
|
|
def get_source_loc(fn): |
|
|
|
def get_source_loc(fn): |
|
|
|
source = os.readlink(fn) |
|
|
|
source = os.readlink(fn) |
|
|
|
if not os.path.isabs(source): |
|
|
|
if not os.path.isabs(source): |
|
|
@ -374,7 +384,44 @@ class Importer: |
|
|
|
def option(self, name): |
|
|
|
def option(self, name): |
|
|
|
return self.options[name] if name in self.options else self.default_options.get(name, None) |
|
|
|
return self.options[name] if name in self.options else self.default_options.get(name, None) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def reset(self, *things): |
|
|
|
|
|
|
|
if "pattern" in things: |
|
|
|
|
|
|
|
self.pattern = self.default_pattern |
|
|
|
|
|
|
|
if "exclude" in things: |
|
|
|
|
|
|
|
self.exclude = None |
|
|
|
|
|
|
|
if "specials" in things: |
|
|
|
|
|
|
|
self.specials.clear() |
|
|
|
|
|
|
|
if "options" in things: |
|
|
|
|
|
|
|
self.options.clear() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def print_info(self): |
|
|
|
|
|
|
|
print("Series Info for %s:" % self.main_name) |
|
|
|
|
|
|
|
print("-" * (17 + len(self.main_name))) |
|
|
|
|
|
|
|
print("Sources | %s" % ("\n | ".join(self.sources))) |
|
|
|
|
|
|
|
print("Pattern | r'%s'%s" % (self.pattern.pattern, " (default)" if self.pattern is self.default_pattern else "")) |
|
|
|
|
|
|
|
print("Exclude | %s" % (("r'%s'" % self.exclude.pattern) if self.exclude else "None")) |
|
|
|
|
|
|
|
print("Options | %s" % self.options) |
|
|
|
|
|
|
|
print("Specials | %s" % ("None" |
|
|
|
|
|
|
|
if not self.specials else |
|
|
|
|
|
|
|
"\n | ".join("%-35s :: %s" % ("r'%s'" % special.pattern.pattern, special._properties) |
|
|
|
|
|
|
|
for special in self.specials))) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@property |
|
|
|
|
|
|
|
def flags(self): |
|
|
|
|
|
|
|
return "".join((f for c, f in [ |
|
|
|
|
|
|
|
(self.pattern is not self.default_pattern, "p"), |
|
|
|
|
|
|
|
(self.exclude, "e"), |
|
|
|
|
|
|
|
#(self.option("exclude_parts"), "d"), |
|
|
|
|
|
|
|
(not self.option("exclude_parts"), "D"), |
|
|
|
|
|
|
|
(self.option("auto_specials"), "i"), |
|
|
|
|
|
|
|
(not self.option("auto_specials"), "I"), |
|
|
|
|
|
|
|
(self.specials, str(len(self.specials))), |
|
|
|
|
|
|
|
] if c)) |
|
|
|
|
|
|
|
|
|
|
|
def process_file(self, filename, subdir=None): |
|
|
|
def process_file(self, filename, subdir=None): |
|
|
|
|
|
|
|
""" |
|
|
|
|
|
|
|
This is the magic that decides if and where to link to a specific source file. |
|
|
|
|
|
|
|
""" |
|
|
|
# Exclude |
|
|
|
# Exclude |
|
|
|
if self.option("exclude_parts") and filename.endswith(".part"): |
|
|
|
if self.option("exclude_parts") and filename.endswith(".part"): |
|
|
|
return "skip", "partial download" |
|
|
|
return "skip", "partial download" |
|
|
@ -403,7 +450,7 @@ class Importer: |
|
|
|
|
|
|
|
|
|
|
|
if special.is_extern: |
|
|
|
if special.is_extern: |
|
|
|
name = special.name |
|
|
|
name = special.name |
|
|
|
linkpath = os.path.join(self.destination, "..", name) |
|
|
|
linkpath = os.path.join(self.library_path, name) |
|
|
|
else: |
|
|
|
else: |
|
|
|
name = self.main_name |
|
|
|
name = self.main_name |
|
|
|
|
|
|
|
|
|
|
@ -440,7 +487,7 @@ class Importer: |
|
|
|
linkname = self.format_linkname(**data) |
|
|
|
linkname = self.format_linkname(**data) |
|
|
|
self.last_episode = (data["until_ep"] if "until_ep" in data else data["episode"]) |
|
|
|
self.last_episode = (data["until_ep"] if "until_ep" in data else data["episode"]) |
|
|
|
|
|
|
|
|
|
|
|
return "link", os.path.join(linkpath, linkname + os.path.splitext(filename)[1]) |
|
|
|
return "link", linkpath, linkname + os.path.splitext(filename)[1] |
|
|
|
|
|
|
|
|
|
|
|
def clean_all(self): |
|
|
|
def clean_all(self): |
|
|
|
clean_links(self.destination) |
|
|
|
clean_links(self.destination) |
|
|
@ -449,77 +496,104 @@ class Importer: |
|
|
|
if special.is_extern: |
|
|
|
if special.is_extern: |
|
|
|
clean_links(os.path.join(self.destination, "..", special.name)) |
|
|
|
clean_links(os.path.join(self.destination, "..", special.name)) |
|
|
|
|
|
|
|
|
|
|
|
def run(self, *, dry=False): |
|
|
|
def collect(self): |
|
|
|
if not dry: |
|
|
|
""" |
|
|
|
self.clean_all() |
|
|
|
Collect operations |
|
|
|
|
|
|
|
""" |
|
|
|
self.last_episode = 0 # FIXME: global state is bad |
|
|
|
self.last_episode = 0 # FIXME: global state is bad |
|
|
|
self.last_special = {} |
|
|
|
self.last_special = {} |
|
|
|
|
|
|
|
|
|
|
|
for source in self.sources: |
|
|
|
for source in self.sources: |
|
|
|
for f in sorted(os.listdir(source), key=natural_sort_key): |
|
|
|
for f in sorted(os.scandir(source), key=natural_name_sort_key): |
|
|
|
path = os.path.join(source, f) |
|
|
|
if f.is_dir(): |
|
|
|
|
|
|
|
for ff in sorted(os.scandir(f.path), key=natural_name_sort_key): |
|
|
|
if os.path.isdir(path): |
|
|
|
yield (ff.path, *self.process_file(ff.name, subdir=f.name)) |
|
|
|
if self.specials: |
|
|
|
|
|
|
|
for ff in sorted(os.listdir(path), key=natural_sort_key): |
|
|
|
|
|
|
|
if self.handle_file(source, os.path.join(f, ff), *self.process_file(ff, subdir=f), dry=dry) != 0: |
|
|
|
|
|
|
|
return 1 |
|
|
|
|
|
|
|
else: |
|
|
|
else: |
|
|
|
if self.handle_file(source, f, *self.process_file(f), dry=dry) != 0: |
|
|
|
yield (f.path, *self.process_file(f.name)) |
|
|
|
return 1 |
|
|
|
|
|
|
|
|
|
|
|
def build_vtree(self): |
|
|
|
def handle_file(self, source, f, what, where, *, dry=False): |
|
|
|
""" |
|
|
|
if what == "link": |
|
|
|
Build a virtual tree of links |
|
|
|
if os.path.exists(where) and not dry: |
|
|
|
""" |
|
|
|
logger.error("LINK %s => %s exists!" % (f, os.path.basename(where))) |
|
|
|
dirs = collections.defaultdict(dict) |
|
|
|
return 1 |
|
|
|
|
|
|
|
|
|
|
|
self.skipped = [] |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for src, op, *args in self.collect(): |
|
|
|
|
|
|
|
if op == "skip": |
|
|
|
|
|
|
|
if self.log_skip: |
|
|
|
|
|
|
|
logger.info("Skipped %s (%s)" % (os.path.basename(src), args[0])) |
|
|
|
|
|
|
|
self.skipped.append(src) |
|
|
|
|
|
|
|
elif op == "link": |
|
|
|
|
|
|
|
dirname, linkname = args |
|
|
|
|
|
|
|
dirs[dirname][linkname] = os.path.relpath(src, dirname) |
|
|
|
else: |
|
|
|
else: |
|
|
|
logger.info("LINK %s => %s" % (f, os.path.basename(where))) |
|
|
|
raise ValueError("Collected unknown operation '%s'" % op) |
|
|
|
|
|
|
|
|
|
|
|
if not dry: |
|
|
|
return dirs |
|
|
|
make_symlink(os.path.join(source, f), where) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
elif what == "skip": |
|
|
|
def build_fstree(self, dirpaths=None): |
|
|
|
logger.info("SKIP %s (%s)" % (f, where)) |
|
|
|
""" |
|
|
|
|
|
|
|
Build a tree from existing links |
|
|
|
|
|
|
|
""" |
|
|
|
|
|
|
|
if dirpaths is None: |
|
|
|
|
|
|
|
dirpaths = [self.destination] + [os.path.join(self.library_path, special.name) |
|
|
|
|
|
|
|
for special in self.effective_specials |
|
|
|
|
|
|
|
if special.is_extern] |
|
|
|
|
|
|
|
dirs = {} |
|
|
|
|
|
|
|
|
|
|
|
else: |
|
|
|
for dirpath in dirpaths: |
|
|
|
assert(False and "Should not be reached") |
|
|
|
dc = dirs[dirpath] = {} |
|
|
|
|
|
|
|
for f in os.scandir(dirpath): |
|
|
|
|
|
|
|
if f.name[0] != "." and f.is_symlink(): |
|
|
|
|
|
|
|
dc[f.name] = os.readlink(f.path) |
|
|
|
|
|
|
|
|
|
|
|
return 0 |
|
|
|
return dirs |
|
|
|
|
|
|
|
|
|
|
|
def reset(self, *things): |
|
|
|
DIFF_SAME = 0 |
|
|
|
if "pattern" in things: |
|
|
|
DIFF_MINUS = 1 |
|
|
|
self.pattern = self.default_pattern |
|
|
|
DIFF_PLUS = 2 |
|
|
|
if "exclude" in things: |
|
|
|
|
|
|
|
self.exclude = None |
|
|
|
|
|
|
|
if "specials" in things: |
|
|
|
|
|
|
|
self.specials.clear() |
|
|
|
|
|
|
|
if "options" in things: |
|
|
|
|
|
|
|
self.options.clear() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def print_info(self): |
|
|
|
def diff(self): |
|
|
|
print("Series Info for %s:" % self.main_name) |
|
|
|
diff = {} |
|
|
|
print("-" * (17 + len(self.main_name))) |
|
|
|
|
|
|
|
print("Sources | %s" % ("\n | ".join(self.sources))) |
|
|
|
|
|
|
|
print("Pattern | r'%s'%s" % (self.pattern.pattern, " (default)" if self.pattern is self.default_pattern else "")) |
|
|
|
|
|
|
|
print("Exclude | %s" % (("r'%s'" % self.exclude.pattern) if self.exclude else "None")) |
|
|
|
|
|
|
|
print("Options | %s" % self.options) |
|
|
|
|
|
|
|
print("Specials | %s" % ("None" |
|
|
|
|
|
|
|
if not self.specials else |
|
|
|
|
|
|
|
"\n | ".join("%-35s :: %s" % ("r'%s'" % special.pattern.pattern, special._properties) |
|
|
|
|
|
|
|
for special in self.specials))) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def flags(self): |
|
|
|
wtree = self.build_vtree() |
|
|
|
return "".join((f for c, f in [ |
|
|
|
htree = self.build_fstree() |
|
|
|
(self.pattern is not self.default_pattern, "p"), |
|
|
|
|
|
|
|
(self.exclude, "e"), |
|
|
|
for dirpath in set(wtree.keys()) | set(htree.keys()): |
|
|
|
#(self.option("exclude_parts"), "d"), |
|
|
|
wdir = wtree[dirpath] if dirpath in wtree else {} |
|
|
|
(not self.option("exclude_parts"), "D"), |
|
|
|
hdir = htree[dirpath] if dirpath in htree else {} |
|
|
|
(self.option("auto_specials"), "i"), |
|
|
|
|
|
|
|
(not self.option("auto_specials"), "I"), |
|
|
|
wfiles = set(wdir.keys()) |
|
|
|
(self.specials, str(len(self.specials))), |
|
|
|
hfiles = set(hdir.keys()) |
|
|
|
] if c)) |
|
|
|
bfiles = wfiles & hfiles |
|
|
|
|
|
|
|
sfiles = {f for f in bfiles if wdir[f] == hdir[f]} # Anchors must be the same in wtree and htree fir relatve paths!! |
|
|
|
|
|
|
|
cfiles = (bfiles - sfiles) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
ddir = diff[dirpath] = {} |
|
|
|
|
|
|
|
ddir[self.DIFF_SAME] = {f: wdir[f] for f in sfiles} |
|
|
|
|
|
|
|
ddir[self.DIFF_MINUS] = {f: hdir[f] for f in (cfiles | (hfiles - wfiles))} |
|
|
|
|
|
|
|
ddir[self.DIFF_PLUS] = {f: wdir[f] for f in (cfiles | (wfiles - hfiles))} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return diff |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def run(self, *, dry=False): |
|
|
|
|
|
|
|
""" |
|
|
|
|
|
|
|
Update this library entry |
|
|
|
|
|
|
|
""" |
|
|
|
|
|
|
|
for path, diff in self.diff().items(): |
|
|
|
|
|
|
|
for f, target in diff[self.DIFF_MINUS].items(): |
|
|
|
|
|
|
|
# Check if target still matches? |
|
|
|
|
|
|
|
logger.info("Remove %s (%s)" % (f, os.path.basename(target))) |
|
|
|
|
|
|
|
if not dry: |
|
|
|
|
|
|
|
os.unlink(os.path.join(path, f)) |
|
|
|
|
|
|
|
for f, target in diff[self.DIFF_PLUS].items(): |
|
|
|
|
|
|
|
logger.info("Link %s => %s" % (f, os.path.basename(target))) |
|
|
|
|
|
|
|
lpath = os.path.join(path, f) |
|
|
|
|
|
|
|
if os.path.exists(lpath): |
|
|
|
|
|
|
|
raise FileExistsError("File %s already exists" % f) |
|
|
|
|
|
|
|
if not dry: |
|
|
|
|
|
|
|
os.symlink(target, lpath) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
############################################################################### |
|
|
|
############################################################################### |
|
|
@ -529,6 +603,7 @@ class HelpFormatter(argparse.RawTextHelpFormatter): |
|
|
|
def __init__(self, prog): |
|
|
|
def __init__(self, prog): |
|
|
|
super().__init__(prog, max_help_position=16) |
|
|
|
super().__init__(prog, max_help_position=16) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def parse_args(argv): |
|
|
|
def parse_args(argv): |
|
|
|
parser = argparse.ArgumentParser(prog=argv[0], formatter_class=HelpFormatter) |
|
|
|
parser = argparse.ArgumentParser(prog=argv[0], formatter_class=HelpFormatter) |
|
|
|
|
|
|
|
|
|
|
@ -689,6 +764,7 @@ def do_interactive_import(args, sources): |
|
|
|
if not args.no_update: |
|
|
|
if not args.no_update: |
|
|
|
run_update(i, args) |
|
|
|
run_update(i, args) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Command Mains |
|
|
|
# Command Mains |
|
|
|
def import_main(args): |
|
|
|
def import_main(args): |
|
|
|
i = get_series_importer(args) |
|
|
|
i = get_series_importer(args) |
|
|
@ -782,7 +858,7 @@ def update_main(args): |
|
|
|
fin_dirs = set() |
|
|
|
fin_dirs = set() |
|
|
|
|
|
|
|
|
|
|
|
for i in get_series_importers(args, args.series): |
|
|
|
for i in get_series_importers(args, args.series): |
|
|
|
logger.info("Processing '%s' (%s)" % (i.main_name, i.flags())) |
|
|
|
logger.info("Processing '%s' (%s)" % (i.main_name, i.flags)) |
|
|
|
|
|
|
|
|
|
|
|
if i.destination in fin_dirs: |
|
|
|
if i.destination in fin_dirs: |
|
|
|
logger.info("Already processed '%s'. Skipping" % i.main_name) |
|
|
|
logger.info("Already processed '%s'. Skipping" % i.main_name) |
|
|
|