X-Git-Url: http://dolda2000.com/gitweb/?a=blobdiff_plain;f=manga%2Fprofile.py;h=081cd6c58ff0af1de389fde728c79a8fba725ff5;hb=3cc7937cd91ec6d3cfb7eebcd4c1afd85c5a615a;hp=e8414a1afbd5548a1fdb23623413e5f99a513e7d;hpb=5997ac775166eef814b46dc8d5df7f9961397c28;p=automanga.git diff --git a/manga/profile.py b/manga/profile.py index e8414a1..081cd6c 100644 --- a/manga/profile.py +++ b/manga/profile.py @@ -6,14 +6,46 @@ if home is None or not os.path.isdir(home): raise Exception("Could not find home directory for profile keeping") basedir = pj(home, ".manga", "profiles") +class txfile(object): + def __init__(self, name, mode): + self.realname = name + self.tempname = name + ".new" + self.bk = open(self.tempname, mode) + + def close(self, abort=False): + self.bk.close() + if abort: + os.unlink(self.tempname) + else: + os.rename(self.tempname, self.realname) + + def read(self, sz=-1): + return self.bk.read(sz) + + def write(self, data): + return self.bk.write(data) + + def __enter__(self): + return self + + def __exit__(self, *exc_info): + if exc_info[0] is not None: + self.close(True) + else: + self.close(False) + def openwdir(nm, mode="r"): + ft = open + if mode == "W": + mode = "w" + ft = txfile if os.path.exists(nm): - return open(nm, mode) + return ft(nm, mode) if mode != "r": d = os.path.dirname(nm) if not os.path.isdir(d): os.makedirs(d) - return open(nm, mode) + return ft(nm, mode) def splitline(line): def bsq(c): @@ -60,6 +92,13 @@ def splitline(line): ret.append(buf) return ret +def splitlines(fp): + for line in fp: + cur = splitline(line) + if len(cur) < 1: + continue + yield cur + def consline(*words): buf = "" for w in words: @@ -86,7 +125,7 @@ class manga(object): self.props = self.loadprops() def open(self): - import lib + from . import lib return lib.findlib(self.libnm).byid(self.id) def save(self): @@ -99,17 +138,72 @@ class memmanga(manga): def loadprops(self): return {} +class tagview(object): + def __init__(self, manga): + self.manga = manga + self.profile = manga.profile + + def add(self, *tags): + mt = self.getall(self.profile) + ctags = mt.setdefault((self.manga.libnm, self.manga.id), set()) + ctags |= set(tags) + self.save(self.profile, mt) + + def remove(self, *tags): + mt = self.getall(self.profile) + ctags = mt.get((self.manga.libnm, self.manga.id), set()) + ctags -= set(tags) + if len(ctags) < 1: + try: + del mt[self.manga.libnm, self.manga.id] + except KeyError: + pass + self.save(self.profile, mt) + + def __iter__(self): + return iter(self.getall(self.profile).get((self.manga.libnm, self.manga.id), set())) + + @staticmethod + def getall(profile): + ret = {} + try: + with profile.file("tags") as fp: + for words in splitlines(fp): + libnm, id = words[0:2] + tags = set(words[2:]) + ret[libnm, id] = tags + except IOError: + pass + return ret + + @staticmethod + def save(profile, m): + with profile.file("tags", "W") as fp: + for (libnm, id), tags in m.items(): + fp.write(consline(libnm, id, *tags) + "\n") + + @staticmethod + def bytag(profile, tag): + try: + with profile.file("tags") as fp: + for words in splitlines(fp): + libnm, id = words[0:2] + tags = words[2:] + if tag in tags: + yield profile.getmanga(libnm, id) + except IOError: + pass + class filemanga(manga): def __init__(self, profile, libnm, id, path): self.path = path super(filemanga, self).__init__(profile, libnm, id) + self.tags = tagview(self) def loadprops(self): ret = {} with openwdir(self.path) as f: - for line in f: - words = splitline(line) - if len(words) < 1: continue + for words in splitlines(f): if words[0] == "set" and len(words) > 2: ret[words[1]] = words[2] elif words[0] == "lset" and len(words) > 1: @@ -117,8 +211,8 @@ class filemanga(manga): return ret def save(self): - with openwdir(self.path, "w") as f: - for key, val in self.props.iteritems(): + with openwdir(self.path, "W") as f: + for key, val in self.props.items(): if isinstance(val, str): f.write(consline("set", key, val) + "\n") else: @@ -134,10 +228,7 @@ class profile(object): ret = {} if os.path.exists(pj(self.dir, "map")): with openwdir(pj(self.dir, "map")) as f: - for ln in f: - words = splitline(ln) - if len(words) < 1: - continue + for words in splitlines(f): if words[0] == "seq" and len(words) > 1: try: seq = int(words[1]) @@ -151,9 +242,9 @@ class profile(object): return seq, ret def savemapping(self, seq, m): - with openwdir(pj(self.dir, "map"), "w") as f: + with openwdir(pj(self.dir, "map"), "W") as f: f.write(consline("seq", str(seq)) + "\n") - for (libnm, id), num in m.iteritems(): + for (libnm, id), num in m.items(): f.write(consline("manga", libnm, id, str(num)) + "\n") def getmanga(self, libnm, id, creat=False): @@ -177,7 +268,7 @@ class profile(object): def setlast(self): if self.name is None: raise ValueError("profile at " + self.dir + " has no name") - with openwdir(pj(basedir, "last"), "w") as f: + with openwdir(pj(basedir, "last"), "W") as f: f.write(self.name + "\n") def getaliases(self): @@ -192,10 +283,13 @@ class profile(object): return ret def savealiases(self, map): - with openwdir(pj(self.dir, "alias"), "w") as f: - for nm, (libnm, id) in map.iteritems(): + with openwdir(pj(self.dir, "alias"), "W") as f: + for nm, (libnm, id) in map.items(): f.write(consline("alias", nm, libnm, id) + "\n") + def file(self, name, mode="r"): + return openwdir(pj(self.dir, name), mode) + def getalias(self, nm): return self.getaliases()[nm] @@ -204,6 +298,9 @@ class profile(object): aliases[nm] = libnm, id self.savealiases(aliases) + def bytag(self, tag): + return tagview.bytag(self, tag) + @classmethod def byname(cls, name): if not name or name == "last" or name[0] == '.':