1
0
mirror of https://github.com/spl0k/supysonic.git synced 2025-01-22 06:53:59 +00:00

Daemon can now scan on its own

Scanning threads cannot be stopped
This commit is contained in:
spl0k 2019-04-20 15:47:05 +02:00
parent 22a207c79f
commit 75540bb376
2 changed files with 119 additions and 15 deletions

View File

@ -49,6 +49,6 @@ if __name__ == "__main__":
signal(SIGINT, __terminate)
init_database(config.BASE['database_uri'])
daemon = Daemon(config.DAEMON['socket'])
daemon.run(config)
daemon = Daemon(config)
daemon.run()
release_database()

View File

@ -10,9 +10,13 @@
import logging
from multiprocessing.connection import Client, Listener
from pony.orm import db_session
from threading import Thread
from .db import Folder
from .config import get_current_config
from .py23 import strtype
from .scanner import Scanner
from .utils import get_secret_key
from .watcher import SupysonicWatcher
@ -23,8 +27,11 @@ logger = logging.getLogger(__name__)
class DaemonUnavailableError(Exception):
pass
class ScannerAlreadyRunningError(Exception):
pass
class DaemonCommand(object):
def apply(self, connection, *args):
def apply(self, connection, daemon):
raise NotImplementedError()
class WatcherCommand(DaemonCommand):
@ -32,12 +39,50 @@ class WatcherCommand(DaemonCommand):
self._folder = folder
class AddWatchedFolderCommand(WatcherCommand):
def apply(self, connection, watcher):
watcher.add_folder(self._folder)
def apply(self, connection, daemon):
if daemon.watcher is not None:
daemon.watcher.add_folder(self._folder)
class RemoveWatchedFolder(WatcherCommand):
def apply(self, connection, watcher):
watcher.remove_folder(self._folder)
def apply(self, connection, daemon):
if daemon.watcher is not None:
daemon.watcher.remove_folder(self._folder)
class ScannerCommand(DaemonCommand):
pass
class ScannerProgressCommand(ScannerCommand):
def apply(self, connection, daemon):
scanner = daemon.scanner
rv = scanner.scanned if scanner is not None and scanner.is_alive() else None
connection.send(ScannerProgressResult(rv))
class ScannerStartCommand(ScannerCommand):
def __init__(self, folders = [], force = False):
self.__folders = folders
self.__force = force
def apply(self, connection, daemon):
try:
daemon.start_scan(self.__folders, self.__force)
connection.send(ScannerStartResult(None))
except ScannerAlreadyRunningError as e:
connection.send(ScannerStartResult(e))
class DaemonCommandResult(object):
pass
class ScannerProgressResult(DaemonCommandResult):
def __init__(self, scanned):
self.__scanned = scanned
scanned = property(lambda self: self.__scanned)
class ScannerStartResult(DaemonCommandResult):
def __init__(self, exception):
self.__exception = exception
exception = property(lambda self: self.__exception)
class DaemonClient(object):
def __init__(self, address = None):
@ -64,31 +109,90 @@ class DaemonClient(object):
with self.__get_connection() as c:
c.send(RemoveWatchedFolder(folder))
def get_scanning_progress(self):
with self.__get_connection() as c:
c.send(ScannerProgressCommand())
return c.recv().scanned
def scan(self, folders = [], force = False):
if not isinstance(folders, list):
raise TypeError('Expecting list, got ' + str(type(folders)))
with self.__get_connection() as c:
c.send(ScannerStartCommand(folders, force))
rv = c.recv()
if rv.exception is not None:
raise rv.exception
class Daemon(object):
def __init__(self, address):
self.__address = address
def __init__(self, config):
self.__config = config
self.__listener = None
self.__watcher = None
self.__scanner = None
watcher = property(lambda self: self.__watcher)
scanner = property(lambda self: self.__scanner)
def __handle_connection(self, connection):
cmd = connection.recv()
logger.debug('Received %s', cmd)
if self.__watcher is not None and isinstance(cmd, WatcherCommand):
cmd.apply(connection, self.__watcher)
if isinstance(cmd, DaemonCommand):
cmd.apply(connection, self)
else:
logger.warn('Received unknown command %s', cmd)
def run(self, config):
self.__listener = Listener(address = self.__address, authkey = get_secret_key('daemon_key'))
def run(self):
self.__listener = Listener(address = self.__config.DAEMON['socket'], authkey = get_secret_key('daemon_key'))
logger.info("Listening to %s", self.__listener.address)
if config.DAEMON['run_watcher']:
self.__watcher = SupysonicWatcher(config)
if self.__config.DAEMON['run_watcher']:
self.__watcher = SupysonicWatcher(self.__config)
self.__watcher.start()
while True:
conn = self.__listener.accept()
self.__handle_connection(conn)
def start_scan(self, folders = [], force = False):
if self.__scanner is not None and self.__scanner.is_alive():
raise ScannerAlreadyRunningError()
extensions = self.__config.BASE['scanner_extensions']
if extensions:
extensions = extensions.split(' ')
self.__scanner = ScannerThread(args = folders, kwargs = { 'force': force, 'extensions': extensions })
self.__scanner.start()
def terminate(self):
self.__listener.close()
if self.__watcher is not None:
self.__watcher.stop()
class ScannerThread(Thread):
def __init__(self, *args, **kwargs):
super(ScannerThread, self).__init__(*args, **kwargs)
self.__scanned = {}
def run(self):
force = self._kwargs.get('force', False)
extensions = self._kwargs.get('extensions')
s = Scanner(force = force, extensions = extensions)
with db_session:
if self._args:
folders = Folder.select(lambda f: f.root and f.name in self._args)
else:
folders = Folder.select(lambda f: f.root)
for f in folders:
name = f.name
logger.info('Scanning %s', name)
s.scan(f, lambda x: self.__scanned.update({ name: x }))
s.finish()
@property
def scanned(self):
# This isn't quite thread-safe but locking each time a file is scanned could affect performance
return sum(self.__scanned.values())