summaryrefslogtreecommitdiffstats
path: root/module/FileDatabase.py
diff options
context:
space:
mode:
Diffstat (limited to 'module/FileDatabase.py')
-rw-r--r--module/FileDatabase.py216
1 files changed, 42 insertions, 174 deletions
diff --git a/module/FileDatabase.py b/module/FileDatabase.py
index 20d67082a..633f210e1 100644
--- a/module/FileDatabase.py
+++ b/module/FileDatabase.py
@@ -18,9 +18,6 @@
"""
from Queue import Queue
-from os import remove
-from os.path import exists
-from shutil import move
from threading import Lock
from threading import RLock
from threading import Thread
@@ -36,13 +33,14 @@ from module.PyPackage import PyPackage
from module.PyFile import PyFile
from module.PyFile import formatSize
+from module.DatabaseBackend import style
+from module.DatabaseBackend import DatabaseBackend
+
try:
from pysqlite2 import dbapi2 as sqlite3
except:
import sqlite3
-DB_VERSION = 3
-
########################################################################
class FileHandler:
"""Handles all request made to obtain information,
@@ -68,8 +66,7 @@ class FileHandler:
self.filecount = -1 # if an invalid value is set get current value from db
self.unchanged = False #determines if any changes was made since last call
- self.db = FileDatabaseBackend(self) # the backend
-
+ self.db = self.core.db
def change(func):
def new(*args):
@@ -548,146 +545,15 @@ class FileHandler:
return default
return value
-#########################################################################
-class FileDatabaseBackend(Thread):
- """underlying backend for the filehandler to save the data"""
-
- def __init__(self, manager):
- Thread.__init__(self)
-
- self.setDaemon(True)
-
- self.manager = manager
-
- self.lock = Lock()
-
- self.jobs = Queue() # queues for jobs
- self.res = Queue()
-
- self.start()
-
-
- def queue(func):
- """use as decorator when fuction directly executes sql commands"""
- def new(*args):
- #print "DataBase: %s args: %s" % (func, args[1:])
- args[0].lock.acquire()
- args[0].jobs.put((func, args, 0))
- res = args[0].res.get()
- args[0].lock.release()
- #print "DataBase: %s return: %s" % (func, res)
- return res
-
-
- return new
-
- def async(func):
- """use as decorator when function does not return anything and asynchron execution is wanted"""
- def new(*args):
- #print "DataBase async: %s args: %s" % (func, args[1:])
- args[0].lock.acquire()
- args[0].jobs.put((func, args, 1))
- args[0].lock.release()
- return True
- return new
-
- def run(self):
- """main loop, which executes commands"""
- convert = self._checkVersion() #returns None or current version
-
- self.conn = sqlite3.connect("files.db")
- self.c = self.conn.cursor()
- #self.c.execute("PRAGMA synchronous = OFF")
-
- if convert is not None:
- self._convertDB(convert)
-
- self._createTables()
-
- self.used = 0
-
- while True:
- try:
- f, args, async = self.jobs.get()
- self.used += 1
- #if f == "quit": return True
- #if self.used > 300: #recycle connection
- # self.recycleConnection()
- res = f(*args)
- if not async: self.res.put(res)
- except Exception, e:
- #@TODO log etc
- print "Database Error @", f.__name__, args[1:], e
- traceback.print_exc()
- if not async: self.res.put(None)
-
- def shutdown(self):
- self.syncSave()
- self.jobs.put(("quit", "", 0))
-
- def recycleConnection(self):
- self.manager.core.log.debug("Recycle sqlite connection")
- self.conn.commit()
- self.c.close()
- self.conn.close()
- del self.c
- del self.conn
- self.conn = sqlite3.connect("files.db")
- self.c = self.conn.cursor()
- self.used = 0
-
- def _checkVersion(self):
- """ check db version and delete it if needed"""
- if not exists("files.version"):
- f = open("files.version", "wb")
- f.write(str(DB_VERSION))
- f.close()
- return
-
- f = open("files.version", "rb")
- v = int(f.read().strip())
- f.close()
- if v < DB_VERSION:
- if v < 2:
- self.manager.core.log.warning(_("Filedatabase was deleted due to incompatible version."))
- remove("files.version")
- move("files.db", "files.backup.db")
- f = open("files.version", "wb")
- f.write(str(DB_VERSION))
- f.close()
- return v
-
- def _convertDB(self, v):
- try:
- getattr(self, "_convertV%i" % v)()
- except:
- self.manager.core.log.error(_("Filedatabase could NOT be converted."))
-
- #--convert scripts start
-
- def _convertV2(self):
- self.c.execute('CREATE TABLE IF NOT EXISTS "storage" ("id" INTEGER PRIMARY KEY AUTOINCREMENT, "identifier" TEXT NOT NULL, "key" TEXT NOT NULL, "value" TEXT DEFAULT "")')
- self.manager.core.log.info(_("Filedatabase was converted from v2 to v3."))
-
- #--convert scripts end
-
- def _createTables(self):
- """create tables for database"""
-
- self.c.execute('CREATE TABLE IF NOT EXISTS "packages" ("id" INTEGER PRIMARY KEY AUTOINCREMENT, "name" TEXT NOT NULL, "folder" TEXT, "password" TEXT DEFAULT "", "site" TEXT DEFAULT "", "queue" INTEGER DEFAULT 0 NOT NULL, "packageorder" INTEGER DEFAULT 0 NOT NULL, "priority" INTEGER DEFAULT 0 NOT NULL)')
- self.c.execute('CREATE TABLE IF NOT EXISTS "links" ("id" INTEGER PRIMARY KEY AUTOINCREMENT, "url" TEXT NOT NULL, "name" TEXT, "size" INTEGER DEFAULT 0 NOT NULL, "status" INTEGER DEFAULT 3 NOT NULL, "plugin" TEXT DEFAULT "BasePlugin" NOT NULL, "error" TEXT DEFAULT "", "linkorder" INTEGER DEFAULT 0 NOT NULL, "package" INTEGER DEFAULT 0 NOT NULL, FOREIGN KEY(package) REFERENCES packages(id))')
- self.c.execute('CREATE INDEX IF NOT EXISTS "pIdIndex" ON links(package)')
- self.c.execute('CREATE TABLE IF NOT EXISTS "storage" ("id" INTEGER PRIMARY KEY AUTOINCREMENT, "identifier" TEXT NOT NULL, "key" TEXT NOT NULL, "value" TEXT DEFAULT "")')
- self.c.execute('VACUUM')
-
- #----------------------------------------------------------------------
- @queue
+class FileMethods():
+ @style.queue
def filecount(self, queue):
"""returns number of files in queue"""
self.c.execute("SELECT l.id FROM links as l INNER JOIN packages as p ON l.package=p.id WHERE p.queue=? ORDER BY l.id", (queue, ))
r = self.c.fetchall()
return len(r)
+ @style.inner
def _nextPackageOrder(self, queue=0):
self.c.execute('SELECT packageorder FROM packages WHERE queue=?', (queue,))
o = -1
@@ -695,6 +561,7 @@ class FileDatabaseBackend(Thread):
if r[0] > o: o = r[0]
return o + 1
+ @style.inner
def _nextFileOrder(self, package):
self.c.execute('SELECT linkorder FROM links WHERE package=?', (package,))
o = -1
@@ -702,13 +569,13 @@ class FileDatabaseBackend(Thread):
if r[0] > o: o = r[0]
return o + 1
- @queue
+ @style.queue
def addLink(self, url, name, plugin, package):
order = self._nextFileOrder(package)
self.c.execute('INSERT INTO links(url, name, plugin, package, linkorder) VALUES(?,?,?,?,?)', (url, name, plugin, package, order))
return self.c.lastrowid
- @queue
+ @style.queue
def addLinks(self, links, package):
""" links is a list of tupels (url,plugin)"""
order = self._nextFileOrder(package)
@@ -716,27 +583,27 @@ class FileDatabaseBackend(Thread):
links = [(x[0], x[0], x[1], package, o) for x, o in zip(links, orders)]
self.c.executemany('INSERT INTO links(url, name, plugin, package, linkorder) VALUES(?,?,?,?,?)', links)
- @queue
+ @style.queue
def addPackage(self, name, folder, queue):
order = self._nextPackageOrder(queue)
self.c.execute('INSERT INTO packages(name, folder, queue, packageorder) VALUES(?,?,?,?)', (name, folder, queue, order))
return self.c.lastrowid
- @queue
+ @style.queue
def deletePackage(self, p):
self.c.execute('DELETE FROM links WHERE package=?', (str(p.id),))
self.c.execute('DELETE FROM packages WHERE id=?', (str(p.id),))
self.c.execute('UPDATE packages SET packageorder=packageorder-1 WHERE packageorder > ? AND queue=?', (p.order, p.queue))
- @queue
+ @style.queue
def deleteLink(self, f):
self.c.execute('DELETE FROM links WHERE id=?', (str(f.id),))
self.c.execute('UPDATE links SET linkorder=linkorder-1 WHERE linkorder > ? AND package=?', (f.order, str(f.packageid)))
- @queue
+ @style.queue
def getAllLinks(self, q):
"""return information about all links in queue q
@@ -770,7 +637,7 @@ class FileDatabaseBackend(Thread):
return data
- @queue
+ @style.queue
def getAllPackages(self, q):
"""return information about packages in queue q
(only useful in get all data)
@@ -802,7 +669,7 @@ class FileDatabaseBackend(Thread):
return data
- @queue
+ @style.queue
def getLinkData(self, id):
"""get link information as dict"""
self.c.execute('SELECT id,url,name,size,status,error,plugin,package,linkorder FROM links WHERE id=?', (str(id), ))
@@ -827,7 +694,7 @@ class FileDatabaseBackend(Thread):
return data
- @queue
+ @style.queue
def getPackageData(self, id):
"""get package data"""
self.c.execute('SELECT id,url,name,size,status,error,plugin,package,linkorder FROM links WHERE package=? ORDER BY linkorder', (str(id), ))
@@ -851,15 +718,15 @@ class FileDatabaseBackend(Thread):
return data
- @async
+ @style.async
def updateLink(self, f):
self.c.execute('UPDATE links SET url=?,name=?,size=?,status=?,error=?,package=? WHERE id=?', (f.url, f.name, f.size, f.status, f.error, str(f.packageid), str(f.id)))
- @queue
+ @style.queue
def updatePackage(self, p):
self.c.execute('UPDATE packages SET name=?,folder=?,site=?,password=?,queue=?,priority=? WHERE id=?', (p.name, p.folder, p.site, p.password, p.queue, p.priority, str(p.id)))
- @queue
+ @style.queue
def updateLinkInfo(self, data):
""" data is list of tupels (name, size, status, url) """
self.c.executemany('UPDATE links SET name=?, size=?, status=? WHERE url=? AND status NOT IN (0,8,12,13)', data)
@@ -869,7 +736,7 @@ class FileDatabaseBackend(Thread):
ids.append(int(r[0]))
return ids
- @queue
+ @style.queue
def reorderPackage(self, p, position, noMove=False):
if position == -1:
position = self._nextPackageOrder(p.queue)
@@ -881,7 +748,7 @@ class FileDatabaseBackend(Thread):
self.c.execute('UPDATE packages SET packageorder=? WHERE id=?', (position, str(p.id)))
- @queue
+ @style.queue
def reorderLink(self, f, position):
""" reorder link with f as dict for pyfile """
if f["order"] > position:
@@ -892,28 +759,24 @@ class FileDatabaseBackend(Thread):
self.c.execute('UPDATE links SET linkorder=? WHERE id=?', (position, f["id"]))
- @queue
+ @style.queue
def clearPackageOrder(self, p):
self.c.execute('UPDATE packages SET packageorder=? WHERE id=?', (-1, str(p.id)))
self.c.execute('UPDATE packages SET packageorder=packageorder-1 WHERE packageorder > ? AND queue=? AND id != ?', (p.order, p.queue, str(p.id)))
- @async
+ @style.async
def restartFile(self, id):
self.c.execute('UPDATE links SET status=3,error="" WHERE id=?', (str(id),))
- @async
+ @style.async
def restartPackage(self, id):
self.c.execute('UPDATE links SET status=3 WHERE package=?', (str(id),))
- @async
- def commit(self):
- self.conn.commit()
-
- @queue
+ @style.async
def syncSave(self):
- self.conn.commit()
+ self.commit()
- @queue
+ @style.queue
def getPackage(self, id):
"""return package instance from id"""
self.c.execute("SELECT name,folder,site,password,queue,packageorder,priority FROM packages WHERE id=?", (str(id), ))
@@ -922,7 +785,7 @@ class FileDatabaseBackend(Thread):
return PyPackage(self.manager, id, * r)
#----------------------------------------------------------------------
- @queue
+ @style.queue
def getFile(self, id):
"""return link instance from id"""
self.c.execute("SELECT url, name, size, status, error, plugin, package, linkorder FROM links WHERE id=?", (str(id), ))
@@ -931,7 +794,7 @@ class FileDatabaseBackend(Thread):
return PyFile(self.manager, id, * r)
- @queue
+ @style.queue
def getJob(self, occ):
"""return pyfile ids, which are suitable for download and dont use a occupied plugin"""
@@ -951,7 +814,7 @@ class FileDatabaseBackend(Thread):
return [x[0] for x in self.c]
- @queue
+ @style.queue
def getPluginJob(self, plugins):
"""returns pyfile ids with suited plugins"""
cmd = "SELECT l.id FROM links as l INNER JOIN packages as p ON l.package=p.id WHERE l.plugin IN %s AND l.status IN (2,3,6,14) ORDER BY p.priority DESC, p.packageorder ASC, l.linkorder ASC LIMIT 5" % plugins
@@ -960,24 +823,27 @@ class FileDatabaseBackend(Thread):
return [x[0] for x in self.c]
- @queue
+ @style.queue
def getUnfinished(self, pid):
"""return list of max length 3 ids with pyfiles in package not finished or processed"""
self.c.execute("SELECT id FROM links WHERE package=? AND status NOT IN (0, 13) LIMIT 3", (str(pid),))
return [r[0] for r in self.c]
- @queue
+ @style.queue
def deleteFinished(self):
self.c.execute("DELETE FROM links WHERE status=0")
self.c.execute("DELETE FROM packages WHERE NOT EXISTS(SELECT 1 FROM links WHERE packages.id=links.package)")
- @queue
+ @style.queue
def restartFailed(self):
self.c.execute("UPDATE links SET status=3,error='' WHERE status IN (8, 9)")
-
- @queue
+
+DatabaseBackend.registerSub(FileMethods)
+
+class StorageMethods():
+ @style.queue
def setStorage(self, identifier, key, value):
self.c.execute("SELECT id FROM storage WHERE identifier=? AND key=?", (identifier, key))
if self.c.fetchone() is not None:
@@ -985,13 +851,15 @@ class FileDatabaseBackend(Thread):
else:
self.c.execute("INSERT INTO storage (identifier, key, value) VALUES (?, ?, ?)", (identifier, key, value))
- @queue
+ @style.queue
def getStorage(self, identifier, key):
self.c.execute("SELECT value FROM storage WHERE identifier=? AND key=?", (identifier, key))
row = self.c.fetchone()
if row is not None:
return row[0]
+DatabaseBackend.registerSub(StorageMethods)
+
if __name__ == "__main__":
pypath = "."