mcg/mcg.py
2013-03-04 17:55:17 +01:00

773 lines
16 KiB
Python

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""MPDCoverGrid is a client for the Music Player Daemon, focused on albums instead of single tracks."""
__author__ = "coderkun"
__email__ = "<olli@coderkun.de>"
__license__ = "GPL"
__version__ = "0.2"
__status__ = "Development"
import configparser
import glob
import os
import queue
import threading
import urllib.request
from hashlib import md5
import mpd
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'
# Signal: error
SIGNAL_ERROR = 'error'
def __init__(self):
"""Sets class variables and instantiates the MPDClient.
"""
MCGBase.__init__(self)
mpd.MPDClient.__init__(self)
self._connected = False
self._state = None
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
def set_volume(self, volume):
self._add_action(self._set_volume, volume)
# 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)
def clear_playlist(self):
"""Clear the current playlist"""
self._add_action(self._clear_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._call('noidle')
except BrokenPipeError:
pass
except ConnectionResetError as e:
self._set_connection_status(False, e)
except mpd.ConnectionError as e:
self._set_connection_status(False, e)
def _work(self, action):
method = action[0]
params = action[1]
method(*params)
def _call(self, command, *args):
try:
return getattr(super(), command)(*args)
except mpd.CommandError as e:
self._callback(MCGClient.SIGNAL_ERROR, e)
except mpd.ConnectionError as e:
self._set_connection_status(False, e)
except ConnectionResetError as e:
self._set_connection_status(False, e)
except BrokenPipeError:
pass
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:
try:
self._call('password', password)
except mpd.CommandError as e:
self._disconnect()
raise e
self._set_connection_status(True)
except OSError as e:
self._set_connection_status(False, e)
def _disconnect(self):
self._call('noidle')
self._call('disconnect')
self._set_connection_status(False)
# Status commands
def _get_status(self):
"""Action: Performs the real status determination
"""
# current status
self._call('noidle')
status = self._call('status')
state = status['state']
volume = int(status['volume'])
error = None
if 'error' in status:
error = status['error']
# current song
self._call('noidle')
song = self._call('currentsong')
album = None
pos = None
if song:
hash = MCGAlbum.hash(song['album'], song['date'])
if hash in self._albums:
album = self._albums[hash]
pos = song['track']
if type(pos) is list:
pos = pos[0]
if '/' in pos:
pos = pos[0: pos.index('/')]
pos = int(pos) - 1
self._state = state
self._callback(MCGClient.SIGNAL_STATUS, state, album, pos, volume, error)
# Playback option commants
def _set_volume(self, volume):
self._call('setvol', volume)
# 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):
if album not in self._albums:
return
track_ids = []
for track in self._albums[album].get_tracks():
track_id = self._call('addid', track.get_file())
track_ids.append(track_id)
if self._state != 'play':
self._call('playid', track_ids[0])
def _stop(self):
self._call('stop')
# Playlist commands
def _load_playlist(self):
playlist = []
for song in self._call('playlistinfo'):
try:
hash = MCGAlbum.hash(song['album'], song['date'])
if len(playlist) == 0 or playlist[len(playlist)-1].get_hash() != hash:
date = ""
if 'date' in song:
date = song['date']
album = MCGAlbum(song['album'], date, self._host, self._image_dir)
playlist.append(album)
else:
album = playlist[len(playlist)-1]
track = MCGTrack(song['artist'], song['title'], song['track'], song['time'], song['file'])
album.add_track(track)
except KeyError:
pass
self._callback(MCGClient.SIGNAL_LOAD_PLAYLIST, playlist, None)
def _clear_playlist(self):
"""Action: Performs the real clearing of the current
playlist.
"""
self._call('clear')
# Database commands
def _load_albums(self):
"""Action: Performs the real update.
"""
for song in self._call('listallinfo'):
try:
hash = MCGAlbum.hash(song['album'], song['date'])
if hash in self._albums.keys():
album = self._albums[hash]
else:
date = ""
if 'date' in song:
date = song['date']
album = MCGAlbum(song['album'], date, self._host, self._image_dir)
self._albums[album.get_hash()] = album
track = MCGTrack(song['artist'], song['title'], song['track'], song['time'], song['file'])
album.add_track(track)
except KeyError:
pass
self._callback(MCGClient.SIGNAL_LOAD_ALBUMS, self._albums, None)
def _update(self):
self._call('update')
def _set_connection_status(self, status, error=None):
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.
"""
modules = self._call('idle')
if not modules:
return
if 'player' in modules:
self.get_status()
if 'mixer' in modules:
self.get_status()
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()
class MCGAlbum:
SORT_BY_ARTIST = 'artist'
SORT_BY_TITLE = 'title'
SORT_BY_YEAR = 'year'
_FILE_NAMES = ['folder', 'cover']
_FILE_EXTS = ['jpg', 'png', 'jpeg']
def __init__(self, title, date, host, image_dir):
self._artists = []
self._pathes = []
if type(title) is list:
title = title[0]
self._title = title
if type(date) is list:
date = date[0]
self._date = date
self._host = host
self._image_dir = image_dir
self._tracks = []
self._cover = None
self._cover_searched = False
self._set_hash()
def get_artists(self):
return self._artists
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)
for artist in track.get_artists():
if artist not in self._artists:
self._artists.append(artist)
path = os.path.dirname(track.get_file())
if path not in self._pathes:
self._pathes.append(path)
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(title, date):
if type(title) is list:
title = title[0]
if type(date) is list:
date = date[0]
return md5(title.encode('utf-8')+date.encode('utf-8')).hexdigest()
def get_hash(self):
return self._hash
def filter(self, filter_string):
values = self._artists + [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_artists"
elif criterion == MCGAlbum.SORT_BY_TITLE:
value_function = "get_title"
elif 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._title, self._date)
def _find_cover(self):
names = list(MCGAlbum._FILE_NAMES)
names.append(self._title)
names.append(' - '.join([self._artists[0], 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 path in self._pathes:
for name in names:
for ext in self._FILE_EXTS:
url = '/'.join([
'http:/',
self._host,
urllib.request.quote(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 path in self._pathes:
for name in names:
for ext in self._FILE_EXTS:
filename = os.path.join(self._image_dir, path, '.'.join([name, ext]))
if os.path.isfile(filename):
return filename
return self._find_cover_local_fallback()
def _find_cover_local_fallback(self):
for path in self._pathes:
for ext in self._FILE_EXTS:
filename = os.path.join(self._image_dir, path, "*."+ext)
files = glob.glob(filename)
if len(files) > 0:
return files[0]
class MCGTrack:
def __init__(self, artists, title, track, time, file):
if type(artists) is not list:
artists = [artists]
self._artists = artists
if type(title) is list:
title = title[0]
self._title = title
if type(track) is list:
track = track[0]
self._track = track
self._time = time
self._file = file
def get_artists(self):
return self._artists
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 delete_profile(self, profile):
if profile in self._profiles:
self._profiles.remove(profile)
self._force_default_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)
self._force_default_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()
def _force_default_profile(self):
if len(self._profiles) == 0:
self._profiles.append(MCGProfile())
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 __str__(self):
return self.get("host")
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)