From 5d0807959de238399fb49de6d0b29b5f4e4b37f1 Mon Sep 17 00:00:00 2001 From: RaNaN Date: Thu, 26 May 2011 20:20:21 +0200 Subject: mirror detection/handling --- module/plugins/Plugin.py | 128 +++++++++++++++++++++++++++-------------------- 1 file changed, 75 insertions(+), 53 deletions(-) (limited to 'module/plugins/Plugin.py') diff --git a/module/plugins/Plugin.py b/module/plugins/Plugin.py index c2e1c3044..cffbb8acb 100644 --- a/module/plugins/Plugin.py +++ b/module/plugins/Plugin.py @@ -40,34 +40,39 @@ from itertools import islice from module.utils import save_join, decode, removeChars def chunks(iterable, size): - it = iter(iterable) - item = list(islice(it, size)) - while item: - yield item + it = iter(iterable) item = list(islice(it, size)) + while item: + yield item + item = list(islice(it, size)) class Abort(Exception): """ raised when aborted """ + class Fail(Exception): """ raised when failed """ - + + class Reconnect(Exception): """ raised when reconnected """ + class Retry(Exception): """ raised when start again from beginning """ + class SkipDownload(Exception): """ raised when download should be skipped """ + class Plugin(object): __name__ = "Plugin" __version__ = "0.4" __pattern__ = None __type__ = "hoster" - __config__ = [ ("name", "type", "desc" , "default") ] + __config__ = [("name", "type", "desc", "default")] __description__ = """Base Plugin""" __author_name__ = ("RaNaN", "spoob", "mkaay") __author_mail__ = ("RaNaN@pyload.org", "spoob@pyload.org", "mkaay@mkaay.de") @@ -101,7 +106,7 @@ class Plugin(object): self.premium = self.account.isPremium(self.user) #premium status else: self.req = pyfile.m.core.requestFactory.getRequest(self.__name__) - + self.log = pyfile.m.core.log self.pyfile = pyfile @@ -115,12 +120,12 @@ class Plugin(object): self.html = None #some plugins store html code here self.init() - + def getChunkCount(self): if self.chunkLimit <= 0: return self.config["download"]["chunks"] return min(self.config["download"]["chunks"], self.chunkLimit) - + def __call__(self): return self.__name__ @@ -151,13 +156,13 @@ class Plugin(object): def process(self, pyfile): """the 'main' method of every plugin""" raise NotImplementedError - + def resetAccount(self): """ dont use account and retry download """ self.account = None self.req = self.core.requestFactory.getRequest(self.__name__) self.retry() - + def checksum(self, local_file=None): """ return codes: @@ -181,14 +186,14 @@ class Plugin(object): return self.config.getPlugin(self.__name__, option) def setConfig(self, option, value): - """ sets a config value """ - self.setConf(option, value) - + """ sets a config value """ + self.setConf(option, value) + def getConfig(self, option): - """ gets a config value """ - return self.getConf(option) - - + """ gets a config value """ + return self.getConf(option) + + def setWait(self, seconds, reconnect=False): """ set the wait time to specified seconds """ if reconnect: @@ -199,16 +204,16 @@ class Plugin(object): """ waits the time previously set """ self.waiting = True self.pyfile.setStatus("waiting") - + while self.pyfile.waitUntil > time(): self.thread.m.reconnecting.wait(2) - + if self.pyfile.abort: raise Abort if self.thread.m.reconnecting.isSet(): self.waiting = False self.wantReconnect = False raise Reconnect - + self.waiting = False self.pyfile.setStatus("starting") @@ -227,28 +232,28 @@ class Plugin(object): def retry(self): """ begin again from the beginning """ raise Retry - + def invalidCaptcha(self): if self.cTask: self.cTask.invalid() - + def correctCaptcha(self): if self.cTask: self.cTask.correct() def decryptCaptcha(self, url, get={}, post={}, cookies=False, forceUser=False, imgtype="jpg"): """ loads the catpcha and decrypt it or ask the user for input """ - + content = self.load(url, get=get, post=post, cookies=cookies) - id = ("%.2f" % time())[-6:].replace(".","") - temp = open(join("tmp","tmpCaptcha_%s_%s.%s" % (self.__name__, id, imgtype)), "wb") - + id = ("%.2f" % time())[-6:].replace(".", "") + temp = open(join("tmp", "tmpCaptcha_%s_%s.%s" % (self.__name__, id, imgtype)), "wb") + temp.write(content) temp.close() has_plugin = self.core.pluginManager.captchaPlugins.has_key(self.__name__) - + if self.core.captcha: Ocr = self.core.pluginManager.getCaptchaPlugin(self.__name__) else: @@ -257,16 +262,15 @@ class Plugin(object): if Ocr and not forceUser: sleep(randint(3000, 5000) / 1000.0) if self.pyfile.abort: raise Abort - + ocr = Ocr() result = ocr.get_captcha(temp.name) else: - captchaManager = self.core.captchaManager task = captchaManager.newTask(content, imgtype, temp.name) self.cTask = task captchaManager.handleCaptcha(task) - + while task.isWaiting(): if self.pyfile.abort: captchaManager.removeTask(task) @@ -282,16 +286,15 @@ class Plugin(object): elif not task.result: self.fail(_("No captcha result obtained in appropiate time by any of the plugins.")) - result = task.result self.log.debug("Received captcha result: %s" % result) if not self.core.debug: - try: - remove(temp.name) - except: - pass - + try: + remove(temp.name) + except: + pass + return result @@ -307,25 +310,27 @@ class Plugin(object): if self.core.debug: from inspect import currentframe + frame = currentframe() if not exists(join("tmp", self.__name__)): makedirs(join("tmp", self.__name__)) - f = open(join("tmp", self.__name__, "%s_line%s.dump.html" % (frame.f_back.f_code.co_name, frame.f_back.f_lineno)), "wb") + f = open( + join("tmp", self.__name__, "%s_line%s.dump.html" % (frame.f_back.f_code.co_name, frame.f_back.f_lineno)) + , "wb") del frame # delete the frame or it wont be cleaned - + try: tmp = res.encode("utf8") except: tmp = res - + f.write(tmp) f.close() - if just_header: #parse header - header = {"code" : self.req.code} + header = {"code": self.req.code} for line in res.splitlines(): line = line.strip() if not line or ":" not in line: continue @@ -338,7 +343,7 @@ class Plugin(object): if type(header[key]) == list: header[key].append(value) else: - header[key] = [header[key],value] + header[key] = [header[key], value] else: header[key] = value res = header @@ -353,11 +358,11 @@ class Plugin(object): self.pyfile.setStatus("downloading") download_folder = self.config['general']['download_folder'] - + location = save_join(download_folder, self.pyfile.package().folder) if not exists(location): - makedirs(location, int(self.core.config["permission"]["folder"],8)) + makedirs(location, int(self.core.config["permission"]["folder"], 8)) if self.core.config["permission"]["change_dl"] and os.name != "nt": try: @@ -365,7 +370,7 @@ class Plugin(object): gid = getgrnam(self.config["permission"]["group"])[2] chown(location, uid, gid) - except Exception,e: + except Exception, e: self.log.warning(_("Setting User and Group failed: %s") % str(e)) name = self.pyfile.name @@ -377,7 +382,9 @@ class Plugin(object): filename = save_join(location, name) try: - newname = self.req.httpDownload(url, filename, get=get, post=post, ref=ref, cookies=cookies, chunks=self.getChunkCount(), resume=self.resumeDownload, progressNotify=self.pyfile.progress.setValue, disposition=disposition) + newname = self.req.httpDownload(url, filename, get=get, post=post, ref=ref, cookies=cookies, + chunks=self.getChunkCount(), resume=self.resumeDownload, + progressNotify=self.pyfile.progress.setValue, disposition=disposition) finally: self.pyfile.size = self.req.size @@ -387,7 +394,7 @@ class Plugin(object): filename = save_join(location, newname) if self.core.config["permission"]["change_file"]: - chmod(filename, int(self.core.config["permission"]["file"],8)) + chmod(filename, int(self.core.config["permission"]["file"], 8)) if self.core.config["permission"]["change_dl"] and os.name != "nt": try: @@ -395,20 +402,20 @@ class Plugin(object): gid = getgrnam(self.config["permission"]["group"])[2] chown(filename, uid, gid) - except Exception,e: + except Exception, e: self.log.warning(_("Setting User and Group failed: %s") % str(e)) self.lastDownload = filename return self.lastDownload - def checkDownload(self, rules, api_size=0 ,max_size=50000, delete=True, read_size=0): + def checkDownload(self, rules, api_size=0, max_size=50000, delete=True, read_size=0): """ checks the content of the last downloaded file rules - dict with names and rules to match(re or strings) size - excpected size @return name of first rule matched or None""" if not exists(self.lastDownload): return None - + size = stat(self.lastDownload) size = size.st_size @@ -441,9 +448,24 @@ class Plugin(object): return password - def checkForSameFiles(self): + def checkForSameFiles(self, starting=False): """ checks if same file was/is downloaded within same package and raise exception """ - pass + + pack = self.pyfile.package() + + cache = self.core.files.cache.values() + + for pyfile in cache: + if pyfile != self.pyfile and pyfile.name == self.pyfile.name and pyfile.package().folder == pack.folder: + if pyfile.status in (0, 12): #finished or downloading + raise SkipDownload(pyfile.pluginname) + elif pyfile.status in (5, 7) and starting: #a download is waiting and was appenrently started before + raise SkipDownload(pyfile.pluginname) + + #TODO check same packagenames + pyfile = self.core.db.findDuplicates(self.pyfile.id, self.pyfile.packageid, self.pyfile.name) + if pyfile: + raise SkipDownload(pyfile[0]) def clean(self): """ clean everything and remove references """ -- cgit v1.2.3