diff options
-rw-r--r-- | module/HookManager.py | 27 | ||||
-rw-r--r-- | module/PluginThread.py | 2 | ||||
-rw-r--r-- | module/PyFile.py | 1 | ||||
-rw-r--r-- | module/database/FileDatabase.py | 164 | ||||
-rw-r--r-- | module/network/HTTPDownload.py | 2 |
5 files changed, 122 insertions, 74 deletions
diff --git a/module/HookManager.py b/module/HookManager.py index 13e8695c2..393db2de6 100644 --- a/module/HookManager.py +++ b/module/HookManager.py @@ -28,7 +28,32 @@ from module.plugins.PluginManager import literal_eval from utils import lock class HookManager: - """Manages hooks and delegates and handles Events""" + """Manages hooks, delegates and handles Events. + + Every plugin can define events, \ + but some very usefull events are called by the Core. + Contrary to overwriting hook methods you can use event listener, + which provides additional entry point in the control flow. + Only use very short tasks or use threads. + + *Known Events:* + All hook methods exists as events. + downloadPreparing: A download was just queued and will be prepared now. + Argument: fid + + downloadStarts: A file will immediately starts the download afterwards. + Argument: fid + + linksAdded: Someone just added links, you are able to modify the links. + Arguments: links, pid + + allDownloadsProcessed: Every link was handled, pyload would idle afterwards. + + allDownloadsFinished: Every download in queue is finished. + + Note: allDownloadsProcessed is *always* called before allDownloadsFinished. + + """ def __init__(self, core): self.core = core diff --git a/module/PluginThread.py b/module/PluginThread.py index d1c139062..228ba4e6a 100644 --- a/module/PluginThread.py +++ b/module/PluginThread.py @@ -276,7 +276,7 @@ class DownloadThread(PluginThread): finally: self.m.core.files.save() - self.m.core.files.resetCount() + self.m.core.files.checkAllLinksProcessed() exc_clear() diff --git a/module/PyFile.py b/module/PyFile.py index b230f9e0b..540933a93 100644 --- a/module/PyFile.py +++ b/module/PyFile.py @@ -200,6 +200,7 @@ class PyFile(object): self.setStatus("finished") self.release() + self.m.checkAllLinksFinished() return True def formatWait(self): diff --git a/module/database/FileDatabase.py b/module/database/FileDatabase.py index 916a53b76..96b505fe0 100644 --- a/module/database/FileDatabase.py +++ b/module/database/FileDatabase.py @@ -34,10 +34,10 @@ except: class FileHandler: - """Handles all request made to obtain information, + """Handles all request made to obtain information, modify status or other request for links or packages""" - + #---------------------------------------------------------------------- def __init__(self, core): """Constructor""" @@ -45,16 +45,16 @@ class FileHandler: # translations self.statusMsg = [_("finished"), _("offline"), _("online"), _("queued"), _("skipped"), _("waiting"), _("temp. offline"), _("starting"), _("failed"), _("aborted"), _("decrypting"), _("custom"), _("downloading"), _("processing"), _("unknown")] - + self.cache = {} #holds instances for files self.packageCache = {} # same for packages #@TODO: purge the cache self.jobCache = {} - + self.lock = RLock() #@TODO should be a Lock w/o R #self.lock._Verbose__verbose = True - + self.filecount = -1 # if an invalid value is set get current value from db self.queuecount = -1 #number of package to be loaded self.unchanged = False #determines if any changes was made since last call @@ -69,7 +69,7 @@ class FileHandler: args[0].jobCache = {} return func(*args) return new - + #---------------------------------------------------------------------- def save(self): """saves all data to backend""" @@ -87,14 +87,14 @@ class FileHandler: pypack.sync() self.db.syncSave() - + @lock def getCompleteData(self, queue=1): """gets a complete data representation""" data = self.db.getAllLinks(queue) packs = self.db.getAllPackages(queue) - + data.update([(str(x.id), x.toDbDict()[x.id]) for x in self.cache.itervalues()]) packs.update([(str(x.id), x.toDict()[x.id]) for x in self.packageCache.itervalues() if x.queue == queue]) @@ -121,7 +121,7 @@ class FileHandler: self.core.hookManager.dispatchEvent("linksAdded", urls, package) data = self.core.pluginManager.parseUrls(urls) - + self.db.addLinks(data, package) self.core.threadManager.createInfoThread(data, package) @@ -144,7 +144,7 @@ class FileHandler: @change def deletePackage(self, id): """delete package and all contained links""" - + p = self.getPackage(id) if not p: @@ -152,9 +152,9 @@ class FileHandler: return e = RemoveEvent("pack", id, "collector" if not p.queue else "queue") - + pyfiles = self.cache.values() - + for pyfile in pyfiles: if pyfile.packageid == id: pyfile.abortDownload() @@ -163,34 +163,34 @@ class FileHandler: self.db.deletePackage(p) self.core.pullManager.addEvent(e) self.core.hookManager.dispatchEvent("packageDeleted", id) - + if self.packageCache.has_key(id): del self.packageCache[id] #---------------------------------------------------------------------- - @lock + @lock @change def deleteLink(self, id): """deletes links""" - + f = self.getFile(id) if not f: return None pid = f.packageid e = RemoveEvent("file", id, "collector" if not f.package().queue else "queue") - - + + if id in self.core.threadManager.processingIds(): self.cache[id].abortDownload() - + if self.cache.has_key(id): del self.cache[id] - + self.db.deleteLink(f) self.core.pullManager.addEvent(e) - + p = self.getPackage(pid) if not len(p.getChildren()): p.delete() @@ -211,7 +211,7 @@ class FileHandler: def updateLink(self, pyfile): """updates link""" self.db.updateLink(pyfile) - + e = UpdateEvent("file", pyfile.id, "collector" if not pyfile.package().queue else "queue") self.core.pullManager.addEvent(e) @@ -219,31 +219,31 @@ class FileHandler: def updatePackage(self, pypack): """updates a package""" self.db.updatePackage(pypack) - + e = UpdateEvent("pack", pypack.id, "collector" if not pypack.queue else "queue") self.core.pullManager.addEvent(e) #---------------------------------------------------------------------- def getPackage(self, id): """return package instance""" - + if self.packageCache.has_key(id): return self.packageCache[id] else: return self.db.getPackage(id) - + #---------------------------------------------------------------------- def getPackageData(self, id): """returns dict with package information""" pack = self.getPackage(id) - + if not pack: return None - + pack = pack.toDict()[id] - + data = self.db.getPackageData(id) - + tmplist = [] cache = self.cache.values() @@ -251,19 +251,19 @@ class FileHandler: if int(x.toDbDict()[x.id]["package"]) == int(id): tmplist.append((str(x.id), x.toDbDict()[x.id])) data.update(tmplist) - + pack["links"] = data - + return pack - + #---------------------------------------------------------------------- def getFileData(self, id): """returns dict with file information""" if self.cache.has_key(id): return self.cache[id].toDbDict() - + return self.db.getLinkData(id) - + #---------------------------------------------------------------------- def getFile(self, id): """returns pyfile instance""" @@ -276,10 +276,10 @@ class FileHandler: @lock def getJob(self, occ): """get suitable job""" - + #@TODO clean mess #@TODO improve selection of valid jobs - + if self.jobCache.has_key(occ): if self.jobCache[occ]: id = self.jobCache[occ].pop() @@ -297,13 +297,13 @@ class FileHandler: else: self.jobCache[occ].extend(jobs) pyfile = self.getFile(self.jobCache[occ].pop()) - + else: self.jobCache = {} #better not caching to much jobs = self.db.getJob(occ) jobs.reverse() self.jobCache[occ] = jobs - + if not jobs: self.jobCache[occ].append("empty") pyfile = None @@ -311,8 +311,8 @@ class FileHandler: pyfile = self.getFile(self.jobCache[occ].pop()) #@TODO: maybe the new job has to be approved... - - + + #pyfile = self.getFile(self.jobCache[occ].pop()) return pyfile @@ -337,7 +337,7 @@ class FileHandler: if self.filecount == -1: self.filecount = self.db.filecount(1) - + return self.filecount def getQueueCount(self, force=False): @@ -347,10 +347,32 @@ class FileHandler: return self.queuecount + def checkAllLinksFinished(self): + """checks if all files are finished and dispatch event""" + + if not self.getQueueCount(True): + #hope its not called twice + self.core.hookManager.dispatchEvent("allDownloadsProcessed") + self.core.hookManager.dispatchEvent("allDownloadsFinished") + return True + + return False + + def checkAllLinksProcessed(self): + """checks if all files was processed and pyload would idle now""" + + self.resetCount() + + if not self.db.processcount(1): + self.core.hookManager.dispatchEvent("allDownloadsProcessed") + return True + + return False + def resetCount(self): self.queuecount = -1 - @lock + @lock @change def restartPackage(self, id): """restart package""" @@ -358,16 +380,16 @@ class FileHandler: for pyfile in pyfiles: if pyfile.packageid == id: self.restartFile(pyfile.id) - + self.db.restartPackage(id) if self.packageCache.has_key(id): self.packageCache[id].setFinished = False - + e = UpdateEvent("pack", id, "collector" if not self.getPackage(id).queue else "queue") self.core.pullManager.addEvent(e) - - @lock + + @lock @change def restartFile(self, id): """ restart file""" @@ -376,38 +398,38 @@ class FileHandler: self.cache[id].name = self.cache[id].url self.cache[id].error = "" self.cache[id].abortDownload() - - + + self.db.restartFile(id) - + e = UpdateEvent("file", id, "collector" if not self.getFile(id).package().queue else "queue") self.core.pullManager.addEvent(e) - - @lock + + @lock @change def setPackageLocation(self, id, queue): """push package to queue""" - + pack = self.db.getPackage(id) - + e = RemoveEvent("pack", id, "collector" if not pack.queue else "queue") self.core.pullManager.addEvent(e) - + self.db.clearPackageOrder(pack) - + pack = self.db.getPackage(id) - + pack.queue = queue self.db.updatePackage(pack) - + self.db.reorderPackage(pack, -1, True) - + self.db.commit() self.releasePackage(id) pack = self.getPackage(id) e = InsertEvent("pack", id, pack.order, "collector" if not pack.queue else "queue") self.core.pullManager.addEvent(e) - + @lock @change def reorderPackage(self, id, position): @@ -432,16 +454,16 @@ class FileHandler: e = ReloadAllEvent("collector" if not p.queue else "queue") self.core.pullManager.addEvent(e) - + @lock @change def reorderFile(self, id, position): f = self.getFileData(id) f = f[str(id)] - + e = RemoveEvent("file", id, "collector" if not self.getPackage(f["package"]).queue else "queue") self.core.pullManager.addEvent(e) - + self.db.reorderLink(f, position) pyfiles = self.cache.values() @@ -456,41 +478,41 @@ class FileHandler: if self.cache.has_key(id): self.cache[id].order = position - + self.db.commit() - + e = ReloadAllEvent("collector" if not self.getPackage(f["package"]).queue else "queue") self.core.pullManager.addEvent(e) - + @change def updateFileInfo(self, data, pid): """ updates file info (name, size, status, url)""" ids = self.db.updateLinkInfo(data) e = UpdateEvent("pack", pid, "collector" if not self.getPackage(pid).queue else "queue") self.core.pullManager.addEvent(e) - + def checkPackageFinished(self, pyfile): """ checks if package is finished and calls hookmanager """ - + ids = self.db.getUnfinished(pyfile.packageid) if not ids or (pyfile.id in ids and len(ids) == 1): if not pyfile.package().setFinished: self.core.log.info(_("Package finished: %s") % pyfile.package().name) self.core.hookManager.packageFinished(pyfile.package()) pyfile.package().setFinished = True - - + + def reCheckPackage(self, pid): """ recheck links in package """ data = self.db.getPackageData(pid) urls = [] - + for pyfile in data.itervalues(): if pyfile["status"] not in (0, 12, 13): urls.append((pyfile["url"], pyfile["plugin"])) - + self.core.threadManager.createInfoThread(urls, pid) @lock @@ -502,7 +524,7 @@ class FileHandler: old_packs.update(self.getInfoData(1)) self.db.deleteFinished() - + new_packs = self.db.getAllPackages(0) new_packs.update(self.db.getAllPackages(1)) #get new packages only from db diff --git a/module/network/HTTPDownload.py b/module/network/HTTPDownload.py index 0f06dcd6f..781929b9d 100644 --- a/module/network/HTTPDownload.py +++ b/module/network/HTTPDownload.py @@ -234,7 +234,7 @@ class HTTPDownload(): remove(self.info.getChunkName(chunk.id)) chunk.fp.flush() - fsync(chunk.fp) #make sure everything was written to disk + fsync(chunk.fp.fileno()) #make sure everything was written to disk chunk.fp.close() #needs to be closed, or merging chunks will fail if failed: raise BadHeader(failed) |