mcg/mcg.py
2013-02-25 17:32:33 +01:00

712 lines
15 KiB
Python

#!/usr/bin/python
# -*- coding: utf-8 -*-
# Author: coderkun <olli@coderkun.de>
import mpd
import os
import threading
import queue
from hashlib import md5
import urllib.request
import configparser
class MCGBase():
def __init__(self):
self._callbacks = {}
def connect_signal(self, signal, callback):
"""Connects a callback function to a signal (event).
"""
self._callbacks[signal] = callback
def disconnect_signal(self, signal):
if self._has_callback(signal):
del self._callbacks[signal]
def _has_callback(self, signal):
"""Checks if there is a registered callback function for a
signal.
"""
return signal in self._callbacks
def _callback(self, signal, *data):
if signal in self._callbacks:
callback = self._callbacks[signal]
callback(*data)
class MCGClient(MCGBase, mpd.MPDClient):
"""Client library for handling the connection to the Music Player Daemon.
This class implements an album-based MPD client.
It offers a non-blocking threaded worker model for use in graphical
environments and is based on python-mpd2.
"""
# Signal: connect/disconnect event
SIGNAL_CONNECT = 'connect'
# Signal: status event
SIGNAL_STATUS = 'status'
# Signal: load albums
SIGNAL_LOAD_ALBUMS = 'load-albums'
# Signal: load playlist
SIGNAL_LOAD_PLAYLIST = 'load-playlist'
def __init__(self):
"""Sets class variables and instantiates the MPDClient.
"""
MCGBase.__init__(self)
mpd.MPDClient.__init__(self)
self._connected = False
self._client_lock = threading.Lock()
self._client_stop = threading.Event()
self._actions = queue.Queue()
self._worker = None
self._albums = {}
self._host = None
self._image_dir = ""
# Connection commands
def connect(self, host="localhost", port="6600", password=None, image_dir=""):
"""Connects to MPD with the given host, port and password or
with standard values.
"""
self._host = host
self._image_dir = image_dir
self._add_action(self._connect, host, port, password)
def is_connected(self):
"""Returns the connection status.
"""
return self._connected
def disconnect(self):
"""Disconnects from the connected MPD.
"""
self._client_stop.set()
self._add_action(self._disconnect)
def join(self):
self._actions.join()
# Status commands
def get_status(self):
"""Determines the current status.
"""
self._add_action(self._get_status)
# Playback option commands
# Playback control commands
def playpause(self):
"""Plays or pauses the current state.
"""
self._add_action(self._playpause)
def play_album(self, album):
"""Plays the given album.
"""
self._add_action(self._play_album, album)
def stop(self):
self._add_action(self._stop)
# Playlist commands
def load_playlist(self):
self._add_action(self._load_playlist)
# Database commands
def load_albums(self):
self._add_action(self._load_albums)
def update(self):
self._add_action(self._update)
# Private methods
def _add_action(self, method, *args):
"""Adds an action to the action list.
"""
action = [method, args]
self._actions.put(action)
self._start_worker()
def _start_worker(self):
"""Starts the worker thread which waits for action to be
performed.
"""
if self._worker is None or not self._worker.is_alive():
self._worker = threading.Thread(target=self._run, name='mcg-worker', args=())
self._worker.setDaemon(True)
self._worker.start()
else:
try:
self.noidle()
except mpd.ConnectionError as e:
self._callback(MCGClient.SIGNAL_CONNECT, False, e)
def _work(self, action):
method = action[0]
params = action[1]
method(*params)
def _call(self, command, *args):
return getattr(super(), command)(*args)
def _run(self):
while not self._client_stop.is_set() or not self._actions.empty():
if self._actions.empty():
self._actions.put([self._idle, ()])
action = self._actions.get()
self._client_lock.acquire()
self._work(action)
self._client_lock.release()
self._actions.task_done()
# Connection commands
def _connect(self, host, port, password):
try:
self._call('connect', host, port)
if password:
self._call('password', password)
self._set_connction_status(True, None)
except mpd.ConnectionError as e:
self._set_connction_status(False, e)
except OSError as e:
self._set_connction_status(False, e)
def _disconnect(self):
try:
self._call('noidle')
self._call('disconnect')
self._set_connction_status(False, None)
except mpd.ConnectionError as e:
self._set_connction_status(False, e)
# Status commands
def _get_status(self):
"""Action: Performs the real status determination
"""
try:
self._call('noidle')
status = self._call('status')
state = status['state']
self._call('noidle')
song = self._call('currentsong')
album = None
pos = None
if song:
hash = MCGAlbum.hash(song['artist'], song['album'])
if hash in self._albums:
album = self._albums[hash]
pos = int(song['pos'])
self._callback(MCGClient.SIGNAL_STATUS, state, album, pos, None)
except mpd.ConnectionError as e:
self._set_connction_status(False, e)
# Playback option commants
# Playback control commands
def _playpause(self):
"""Action: Performs the real play/pause command.
"""
status = self._call('status')
state = status['state']
if state == 'play':
self._call('pause')
else:
self._call('play')
def _play_album(self, album):
"""Action: Performs the real play command.
"""
if not album in self._albums:
# TODO print
print("album not found")
return
try:
self._call('clear')
track_ids = []
for track in self._albums[album].get_tracks():
track_id = self._call('addid', track.get_file())
track_ids.append(track_id)
self._call('moveid', track_id, len(track_ids)-1)
self._call('playid', track_ids[0])
except mpd.ConnectionError as e:
self._set_connction_status(False, e)
def _stop(self):
try:
self._call('stop')
except mpd.ConnectionError as e:
self._set_connction_status(False, e)
# Playlist commands
def _load_playlist(self):
try:
playlist = []
for song in self._call('playlistinfo'):
try:
hash = MCGAlbum.hash(song['artist'], song['album'])
if len(playlist) == 0 or playlist[len(playlist)-1].get_hash() != hash:
date = ""
if 'date' in song:
date = song['date']
path = ""
if 'file' in song:
path = os.path.dirname(song['file'])
album = MCGAlbum(song['artist'], song['album'], date, path, self._host, self._image_dir)
playlist.append(album)
else:
album = playlist[len(playlist)-1]
track = MCGTrack(song['title'], song['track'], song['time'], song['file'])
album.add_track(track)
except KeyError:
pass
self._callback(MCGClient.SIGNAL_LOAD_PLAYLIST, playlist, None)
except mpd.ConnectionError as e:
self._set_connction_status(False, e)
# Database commands
def _load_albums(self):
"""Action: Performs the real update.
"""
try:
for song in self._call('listallinfo'):
try:
hash = MCGAlbum.hash(song['artist'], song['album'])
if hash in self._albums.keys():
album = self._albums[hash]
else:
date = ""
if 'date' in song:
date = song['date']
path = ""
if 'file' in song:
path = os.path.dirname(song['file'])
album = MCGAlbum(song['artist'], song['album'], date, path, self._host, self._image_dir)
self._albums[album.get_hash()] = album
track = MCGTrack(song['title'], song['track'], song['time'], song['file'])
album.add_track(track)
except KeyError:
pass
self._callback(MCGClient.SIGNAL_LOAD_ALBUMS, self._albums, None)
except mpd.ConnectionError as e:
self._set_connction_status(False, e)
def _update(self):
try:
self._call('update')
except mpd.ConnectionError as e:
self._set_connction_status(False, e)
def _set_connction_status(self, status, error):
self._connected = status
self._callback(MCGClient.SIGNAL_CONNECT, status, error)
if not status:
self._client_stop.set()
def _idle(self):
"""Reacts to idle events from MPD.
"""
try:
modules = self._call('idle')
if not modules:
return
if 'player' in modules:
self.get_status()
if 'mixer' in modules:
# TODO mixer
print("not implemented: idle mixer")
if 'playlist' in modules:
self.load_playlist()
if 'database' in modules:
self.load_albums()
self.load_playlist()
self.get_status()
if 'update' in modules:
self.load_albums()
self.load_playlist()
self.get_status()
except ConnectionResetError as e:
self._set_connction_status(False, e)
except mpd.ConnectionError as e:
self._set_connction_status(False, e)
class MCGAlbum:
SORT_BY_ARTIST = 'artist'
SORT_BY_TITLE = 'title'
SORT_BY_YEAR = 'year'
_file_names = ['folder', 'cover']
_file_exts = ['jpg', 'jpeg', 'png']
def __init__(self, artist, title, date, path, host, image_dir):
self._artist = artist
if type(self._artist) is list:
self._artist = self._artist[0]
self._title = title
self._date = date
self._path = path
self._host = host
self._image_dir = image_dir
self._tracks = []
self._cover = None
self._cover_searched = False
self._set_hash()
def get_artist(self):
return self._artist
def get_title(self):
return self._title
def get_date(self):
return self._date
def get_path(self):
return self._path
def add_track(self, track):
if track not in self._tracks:
self._tracks.append(track)
def get_tracks(self):
return self._tracks
def get_cover(self):
if self._cover is None and not self._cover_searched:
self._find_cover()
return self._cover
def hash(artist, title):
if type(artist) is list:
artist = artist[0]
return md5(artist.encode('utf-8')+title.encode('utf-8')).hexdigest()
def get_hash(self):
return self._hash
def filter(self, filter_string):
values = [self._artist, self._title, self._date]
values.extend(map(lambda track: track.get_title(), self._tracks))
for value in values:
if filter_string.lower() in value.lower():
return True
return False
def compare(album1, album2, criterion=None):
if criterion == None:
criterion = MCGAlbum.SORT_BY_TITLE
if criterion == MCGAlbum.SORT_BY_ARTIST:
value_function = "get_artist"
if criterion == MCGAlbum.SORT_BY_TITLE:
value_function = "get_title"
if criterion == MCGAlbum.SORT_BY_YEAR:
value_function = "get_date"
if getattr(album1, value_function)() < getattr(album2, value_function)():
return -1
elif getattr(album1, value_function)() == getattr(album2, value_function)():
return 0
else:
return 1
def _set_hash(self):
self._hash = MCGAlbum.hash(self._artist, self._title)
def _find_cover(self):
names = list(self._file_names)
names.append(self._title)
names.append(' - '.join([self._artist, self._title]))
if self._host == "localhost" or self._host == "127.0.0.1":
self._cover = self._find_cover_local(names)
else:
self._cover = self._find_cover_web(names)
self._cover_searched = True
def _find_cover_web(self, names):
for name in names:
for ext in self._file_exts:
url = '/'.join([
'http:/',
self._host,
urllib.request.quote(self._path),
urllib.request.quote('.'.join([name, ext]))
])
request = urllib.request.Request(url)
try:
response = urllib.request.urlopen(request)
return url
except urllib.error.URLError as e:
pass
def _find_cover_local(self, names):
for name in names:
for ext in self._file_exts:
filename = os.path.join(self._image_dir, self._path, '.'.join([name, ext]))
if os.path.isfile(filename):
return filename
class MCGTrack:
def __init__(self, title, track, time, file):
self._title = title
self._track = track
self._time = time
self._file = file
def get_title(self):
return self._title
def get_track(self):
return self._track
def get_time(self):
return self._time
def get_file(self):
return self._file
class MCGConfig(configparser.ConfigParser):
CONFIG_DIR = '~/.config/mcg/'
def __init__(self, filename):
configparser.ConfigParser.__init__(self)
self._filename = os.path.expanduser(os.path.join(MCGConfig.CONFIG_DIR, filename))
self._create_dir()
def load(self):
if os.path.isfile(self._filename):
self.read(self._filename)
def save(self):
with open(self._filename, 'w') as configfile:
self.write(configfile)
def _create_dir(self):
dirname = os.path.dirname(self._filename)
if not os.path.exists(dirname):
os.makedirs(dirname)
class MCGProfileConfig(MCGConfig):
CONFIG_FILE = 'profiles.conf'
def __init__(self):
MCGConfig.__init__(self, MCGProfileConfig.CONFIG_FILE)
self._profiles = []
def add_profile(self, profile):
self._profiles.append(profile)
def get_profiles(self):
return self._profiles
def load(self):
super().load()
count = 0
if self.has_section('profiles'):
if self.has_option('profiles', 'count'):
count = self.getint('profiles', 'count')
for index in range(count):
section = 'profile'+str(index+1)
if self.has_section(section):
profile = MCGProfile()
for attribute in profile.get_attributes():
if self.has_option(section, attribute):
profile.set(attribute, self.get(section, attribute))
self._profiles.append(profile)
def save(self):
if not self.has_section('profiles'):
self.add_section('profiles')
self.set('profiles', 'count', str(len(self._profiles)))
for index in range(len(self._profiles)):
profile = self._profiles[index]
section = 'profile'+str(index+1)
if not self.has_section(section):
self.add_section(section)
for attribute in profile.get_attributes():
self.set(section, attribute, str(profile.get(attribute)))
super().save()
class MCGConfigurable:
def __init__(self):
self._attributes = []
def get(self, attribute):
return getattr(self, attribute)
def set(self, attribute, value):
setattr(self, attribute, value)
if attribute not in self._attributes:
self._attributes.append(attribute)
def get_attributes(self):
return self._attributes
class MCGProfile(MCGConfigurable):
def __init__(self):
MCGConfigurable.__init__(self)
self.set('host', "localhost")
self.set('port', 6600)
self.set('password', "")
self.set('image_dir', "")
self.set('tags', "")
def get_tags(self):
return self.get('tags').split(',')
def set_tags(self, tags):
self.set('tags', ','.join(tags))
class MCGCache():
DIRNAME = '~/.cache/mcg/'
SIZE_FILENAME = 'size'
def __init__(self, host, size):
self._dirname = os.path.expanduser(os.path.join(MCGCache.DIRNAME, host))
if not os.path.exists(self._dirname):
os.makedirs(self._dirname)
self._size = size
self._read_size()
def create_filename(self, album):
return os.path.join(self._dirname, '-'.join([album.get_hash()]))
def _read_size(self):
size = 100
filename = os.path.join(self._dirname, MCGCache.SIZE_FILENAME)
if os.path.exists(filename):
with open(filename, 'r') as f:
size = int(f.readline())
if size != self._size:
self._clear()
with open(filename, 'w') as f:
f.write(str(self._size))
def _clear(self):
for filename in os.listdir(self._dirname):
path = os.path.join(self._dirname, filename)
try:
if os.path.isfile(path):
os.unlink(path)
except Exception as e:
print("clear:", e)