#!/usr/bin/env python # -*- coding: utf-8 -*- """ This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, see . @author: mkaay @version: v0.3 """ SERVER_VERSION = "0.3" import sys from time import sleep, time from PyQt4.QtCore import * from PyQt4.QtGui import * from xmlrpclib import ServerProxy class main(QObject): def __init__(self): """ main setup """ QObject.__init__(self) self.app = QApplication(sys.argv) self.mainWindow = mainWindow() self.connector = connector() self.mainloop = self.Loop(self) self.connector.start() sleep(1) self.mainWindow.show() self.testStuff() self.mainloop.start() def connectSignals(self): """ signal and slot stuff, yay! """ self.connect(self.connector, SIGNAL("error_box"), self.slotErrorBox) def loop(self): """ start exec loop """ sys.exit(self.app.exec_()) def slotErrorBox(self, msg): """ display a nice error box """ QMessageBox(QMessageBox.Warning, "Error", msg) def testStuff(self): """ only for testing ;) """ #test for link collector ids = self.connector.getLinkCollector() for id in ids: data = self.connector.getLinkInfo(id) item = QListWidgetItem() item.setData(Qt.UserRole, QVariant(data)) item.setData(Qt.DisplayRole, QVariant(data["url"])) self.mainWindow.tabs["collector_links"]["listwidget"].addItem(item) #test for package collector packs = self.connector.getPackageCollector() for data in packs: item = QTreeWidgetItem() item.setData(0, Qt.UserRole, QVariant(data)) item.setData(0, Qt.DisplayRole, QVariant(data["package_name"])) files = self.connector.getPackageFiles(data["id"]) for id in files: info = self.connector.getLinkInfo(id) sub = QTreeWidgetItem(item) sub.setData(0, Qt.DisplayRole, QVariant(info["filename"])) self.mainWindow.tabs["collector_packages"]["treewidget"].addTopLevelItem(item) #test for queue """ packs = self.connector.getPackageQueue() for data in packs: item = QTreeWidgetItem() item.setData(0, Qt.UserRole, QVariant(data)) item.setData(0, Qt.DisplayRole, QVariant(data["package_name"])) files = self.connector.getPackageFiles(data["id"]) for id in files: info = self.connector.getLinkInfo(id) sub = QTreeWidgetItem(item) sub.setData(0, Qt.DisplayRole, QVariant(info["filename"])) sub.setData(1, Qt.DisplayRole, QVariant(info["status_type"])) self.mainWindow.tabs["queue"]["treewidget"].addTopLevelItem(item) model = QueueModel(self.connector) model.setView(self.mainWindow.tabs["queue"]["view"]) self.mainWindow.tabs["queue"]["view"].setModel(model) self.mainWindow.tabs["queue"]["view"].setup() model.startLoop() """ view = self.mainWindow.tabs["queue"]["view"] view.setColumnCount(3) view.setHeaderLabels(["Name", "Status", "Fortschritt"]) view.setColumnWidth(0, 300) view.setColumnWidth(1, 200) view.setColumnWidth(2, 100) self.queue = Queue(view, self.connector) delegate = QueueProgressBarDelegate(view, self.queue) view.setItemDelegateForColumn(2, delegate) #view.setup(self.queue) self.queue.start() def refreshServerStatus(self): status = self.connector.getServerStatus() if status["pause"]: status["status"] = "Paused" else: status["status"] = "Running" status["speed"] = int(status["speed"]) text = "Status: %(status)s | Speed: %(speed)s kb/s" % status self.mainWindow.serverStatus.setText(text) class Loop(QThread): def __init__(self, parent): QThread.__init__(self) self.parent = parent self.running = True def run(self): while self.running: sleep(1) self.update() def update(self): self.parent.refreshServerStatus() class connector(QThread): def __init__(self): """ init thread """ QThread.__init__(self) self.mutex = QMutex() self.running = True self.proxy = None def run(self): """ start thread (called from thread.start()) """ self.connectProxy("http://admin:pwhere@localhost:7227/") #@TODO: change me! while self.running: sleep(1) def stop(self): """ stop thread """ self.running = False def connectProxy(self, addr): """ connect to remote server """ self.proxy = ServerProxy(addr, allow_none=True) server_version = self.proxy.get_server_version() if not server_version == SERVER_VERSION: self.emit(SIGNAL("error_box"), "server is version %s client accepts version %s" % (server_version, SERVER_VERSION)) def getLinkCollector(self): """ grab links from collector and return the ids """ self.mutex.lock() try: return self.proxy.get_collector_files() finally: self.mutex.unlock() def getPackageCollector(self): """ grab packages from collector and return the data """ self.mutex.lock() try: return self.proxy.get_collector_packages() finally: self.mutex.unlock() def getLinkInfo(self, id): """ grab file info for the given id and return it """ self.mutex.lock() try: return self.proxy.get_file_info(id) finally: self.mutex.unlock() def getPackageInfo(self, id): """ grab package info for the given id and return it """ self.mutex.lock() try: return self.proxy.get_package_data(id) finally: self.mutex.unlock() def getPackageQueue(self): """ grab queue return the data """ self.mutex.lock() try: return self.proxy.get_queue() finally: self.mutex.unlock() def getPackageFiles(self, id): """ grab package files and return ids """ self.mutex.lock() try: return self.proxy.get_package_files(id) finally: self.mutex.unlock() def getDownloadQueue(self): """ grab files that are currently downloading and return info """ self.mutex.lock() try: return self.proxy.status_downloads() finally: self.mutex.unlock() def getServerStatus(self): """ return server status """ self.mutex.lock() try: return self.proxy.status_server() finally: self.mutex.unlock() class mainWindow(QMainWindow): def __init__(self): """ set up main window """ QMainWindow.__init__(self) #window stuff self.setWindowTitle("pyLoad Client") self.setWindowIcon(QIcon("icons/logo.png")) self.resize(750,500) #central widget, layout self.masterlayout = QVBoxLayout() lw = QWidget() lw.setLayout(self.masterlayout) self.setCentralWidget(lw) #set menubar and statusbar self.menubar = self.menuBar() self.statusbar = self.statusBar() self.serverStatus = QLabel("Status: Not Connected") self.statusbar.addPermanentWidget(self.serverStatus) #menu self.menus = {} self.menus["file"] = self.menubar.addMenu("&File") self.menus["connections"] = self.menubar.addMenu("&Connections") #menu actions self.mactions = {} self.mactions["exit"] = QAction("Exit", self.menus["file"]) self.mactions["manager"] = QAction("Connection manager", self.menus["connections"]) #add menu actions self.menus["file"].addAction(self.mactions["exit"]) self.menus["connections"].addAction(self.mactions["manager"]) #tabs self.tabw = QTabWidget() self.tabs = {} self.tabs["queue"] = {"w":QWidget()} self.tabs["collector_packages"] = {"w":QWidget()} self.tabs["collector_links"] = {"w":QWidget()} self.tabw.addTab(self.tabs["queue"]["w"], "Queue") self.tabw.addTab(self.tabs["collector_packages"]["w"], "Package collector") self.tabw.addTab(self.tabs["collector_links"]["w"], "Link collector") #init tabs self.init_tabs() #layout self.masterlayout.addWidget(self.tabw) def init_tabs(self): """ create tabs """ #queue self.tabs["queue"]["l"] = QGridLayout() self.tabs["queue"]["w"].setLayout(self.tabs["queue"]["l"]) self.tabs["queue"]["view"] = QTreeWidget() self.tabs["queue"]["l"].addWidget(self.tabs["queue"]["view"]) #collector_packages self.tabs["collector_packages"]["l"] = QGridLayout() self.tabs["collector_packages"]["w"].setLayout(self.tabs["collector_packages"]["l"]) self.tabs["collector_packages"]["treewidget"] = QTreeWidget() self.tabs["collector_packages"]["l"].addWidget(self.tabs["collector_packages"]["treewidget"]) #collector_links self.tabs["collector_links"]["l"] = QGridLayout() self.tabs["collector_links"]["w"].setLayout(self.tabs["collector_links"]["l"]) self.tabs["collector_links"]["listwidget"] = QListWidget() self.tabs["collector_links"]["l"].addWidget(self.tabs["collector_links"]["listwidget"]) class Queue(QThread): def __init__(self, view, connector): QThread.__init__(self) self.view = view self.connector = connector self.statusMap = { "finished": 0, "checking": 1, "waiting": 2, "reconnected": 3, "downloading": 4, "failed": 5, "aborted": 6, } self.statusMapReverse = dict((v,k) for k, v in self.statusMap.iteritems()) self.queue = [] self.interval = 2 self.running = True self.mutex = QMutex() def run(self): while self.running: self.update() sleep(self.interval) def update(self): locker = QMutexLocker(self.mutex) packs = self.connector.getPackageQueue() downloading_raw = self.connector.getDownloadQueue() downloading = {} for d in downloading: did = d["id"] del d["id"] del d["name"] del d["status"] downloading[did] = d for data in packs: pack = self.getPack(data["id"]) if not pack: pack = self.QueuePack(self) pack.setData(data) self.addPack(data["id"], pack) files = self.connector.getPackageFiles(data["id"]) for fid in files: info = self.connector.getLinkInfo(fid) child = pack.getChild(fid) if not child: child = self.QueueFile(self, pack) try: info["downloading"] = downloading[data["id"]] except: info["downloading"] = None child.setData(info) pack.addChild(fid, child) def addPack(self, pid, newPack): pos = None try: for k, pack in enumerate(self.queue): if pack.getData()["id"] == pid: pos = k break if pos == None: raise Exception() self.queue[pos] = newPack except: self.queue.append(newPack) pos = self.queue.index(newPack) item = self.view.topLevelItem(pos) if not item: item = QTreeWidgetItem() self.view.insertTopLevelItem(pos, item) item.setData(0, Qt.DisplayRole, QVariant(newPack.getData()["package_name"])) status = -1 for child in newPack.getChildren(): if self.statusMap.has_key(child.data["status_type"]) and self.statusMap[child.data["status_type"]] > status: status = self.statusMap[child.data["status_type"]] if status >= 0: item.setData(1, Qt.DisplayRole, QVariant(self.statusMapReverse[status])) item.setData(0, Qt.UserRole, QVariant(pid)) item.setData(2, Qt.UserRole, QVariant(newPack)) def getPack(self, pid): for k, pack in enumerate(self.queue): if pack.getData()["id"] == pid: return pack return None def getProgress(self, q): locker = QMutexLocker(self.mutex) if isinstance(q, self.QueueFile): data = q.getData() if data["downloading"]: return int(data["downloading"]["percent"]) if data["status_type"] == "finished" or \ data["status_type"] == "failed" or \ data["status_type"] == "aborted": return 100 elif isinstance(q, self.QueuePack): children = q.getChildren() count = len(children) perc_sum = 0 for child in children: val = 0 data = child.getData() if data["downloading"]: val = int(data["downloading"]["percent"]) elif child.data["status_type"] == "finished" or \ child.data["status_type"] == "failed" or \ child.data["status_type"] == "aborted": val = 100 perc_sum += val if count == 0: return 0 return perc_sum/count return 0 def getSpeed(self, q): locker = QMutexLocker(self.mutex) if isinstance(q, self.QueueFile): data = q.getData() if data["downloading"]: return int(data["downloading"]["speed"]) elif isinstance(q, self.QueuePack): children = q.getChildren() count = len(children) speed_sum = 0 for child in children: val = 0 data = child.getData() running = False if data["downloading"]: val = int(data["downloading"]["speed"]) running = True speed_sum += val if count == 0 or not running: return None return speed_sum return None class QueuePack(): def __init__(self, queue): self.queue = queue self.data = [] self.children = [] def addChild(self, cid, newChild): pos = None try: for k, child in enumerate(self.getChildren()): if child.getData()["id"] == cid: pos = k break if pos == None: raise Exception() self.children[pos] = newChild except: self.children.append(newChild) pos = self.children.index(newChild) ppos = self.queue.queue.index(self) parent = self.queue.view.topLevelItem(ppos) item = parent.child(pos) if not item: item = QTreeWidgetItem() parent.insertChild(pos, item) status = "%s (%s)" % (newChild.getData()["status_type"], newChild.getData()["plugin"]) item.setData(0, Qt.DisplayRole, QVariant(newChild.getData()["filename"])) item.setData(1, Qt.DisplayRole, QVariant(status)) item.setData(0, Qt.UserRole, QVariant(cid)) item.setData(2, Qt.UserRole, QVariant(newChild)) def getChildren(self): return self.children def getChild(self, cid): try: return self.children[cid] except: return None def hasChildren(self, data): return (len(self.children) > 0) def setData(self, data): self.data = data def getData(self): return self.data class QueueFile(): def __init__(self, queue, pack): self.queue = queue self.pack = pack def getData(self): return self.data def setData(self, data): self.data = data def getPack(self): return self.pack class QueueProgressBarDelegate(QItemDelegate): def __init__(self, parent, queue): QItemDelegate.__init__(self, parent) self.queue = queue def paint(self, painter, option, index): if index.column() == 2: qe = index.data(Qt.UserRole).toPyObject() progress = self.queue.getProgress(qe) opts = QStyleOptionProgressBarV2() opts.maximum = 100 opts.minimum = 0 opts.progress = progress opts.rect = option.rect opts.rect.setRight(option.rect.right()-1) opts.rect.setHeight(option.rect.height()-1) opts.textVisible = True opts.textAlignment = Qt.AlignCenter speed = self.queue.getSpeed(qe) if speed == None: opts.text = QString.number(opts.progress) + "%" else: opts.text = QString("%s kb/s - %s" % (speed, opts.progress)) + "%" QApplication.style().drawControl(QStyle.CE_ProgressBar, opts, painter) return QItemDelegate.paint(self, painter, option, index) if __name__ == "__main__": app = main() app.loop()