421 lines
8.7 KiB
Python
421 lines
8.7 KiB
Python
#!/usr/bin/python
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# Author: coderkun <olli@coderkun.de>
|
|
|
|
|
|
|
|
|
|
import mpd
|
|
import os
|
|
from hashlib import md5
|
|
from threading import Thread
|
|
|
|
class MCGClient:
|
|
"""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: general idle event
|
|
SIGNAL_IDLE = 'idle'
|
|
# Signal: status event
|
|
SIGNAL_STATUS = 'status'
|
|
# Signal: update event
|
|
SIGNAL_UPDATE = 'update'
|
|
|
|
|
|
def __init__(self):
|
|
"""Sets class variables and instantiates the MPDClient.
|
|
"""
|
|
self._connected = False
|
|
self._albums = {}
|
|
self._callbacks = {}
|
|
self._actions = []
|
|
self._worker = None
|
|
self._client = mpd.MPDClient()
|
|
self._go = True
|
|
|
|
|
|
def connect(self, host="localhost", port="6600", password=None):
|
|
"""Connects to MPD with the given host, port and password or
|
|
with standard values.
|
|
"""
|
|
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._add_action(self._disconnect)
|
|
|
|
|
|
def close(self):
|
|
"""Closes the connection and stops properly the worker thread.
|
|
This method is to stop the whole appliction.
|
|
"""
|
|
if not self.is_connected():
|
|
return
|
|
try:
|
|
self._go = False
|
|
self._client.noidle()
|
|
self._client.disconnect()
|
|
except TypeError as e:
|
|
pass
|
|
|
|
|
|
def update(self):
|
|
"""Updates the album list.
|
|
"""
|
|
self._add_action(self._update)
|
|
|
|
|
|
def get_status(self):
|
|
"""Determines the current status.
|
|
"""
|
|
self._add_action(self._get_status)
|
|
|
|
|
|
def play_album(self, album):
|
|
"""Plays the given album.
|
|
"""
|
|
self._add_action(self._play, album)
|
|
|
|
|
|
def playpause(self):
|
|
"""Plays or pauses the current state.
|
|
"""
|
|
self._add_action(self._playpause)
|
|
|
|
|
|
def next(self):
|
|
"""Plays the next album in the current order
|
|
"""
|
|
self._add_action(self._next)
|
|
|
|
|
|
def connect_signal(self, signal, callback):
|
|
"""Connects a callback function to a signal (event).
|
|
"""
|
|
self._callbacks[signal] = callback
|
|
|
|
|
|
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, *args):
|
|
"""Calls the callback function for a signal.
|
|
"""
|
|
if self._has_callback(signal):
|
|
callback = self._callbacks[signal]
|
|
callback(*args)
|
|
|
|
|
|
def _add_action(self, method, *args):
|
|
"""Adds an action to the action list.
|
|
"""
|
|
action = [method, args]
|
|
self._actions.append(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 = Thread(target=self._work, name='worker', args=())
|
|
self._worker.start()
|
|
else:
|
|
try:
|
|
self._client.noidle()
|
|
except TypeError as e:
|
|
pass
|
|
|
|
|
|
def _work(self):
|
|
"""Performs the next action or waits for an idle event.
|
|
"""
|
|
while True:
|
|
if self._actions:
|
|
action = self._actions.pop(0)
|
|
method = action[0]
|
|
params = action[1]
|
|
method(*params)
|
|
else:
|
|
if not self.is_connected():
|
|
break
|
|
modules = self._client.idle()
|
|
if not self._go:
|
|
break
|
|
self._idle(modules)
|
|
|
|
|
|
def _connect(self, host, port, password):
|
|
"""Action: Performs the real connect to MPD.
|
|
"""
|
|
try:
|
|
self._client.connect(host, port)
|
|
if password:
|
|
self._client.password(password)
|
|
# TODO Verbindung testen
|
|
self._connected = True
|
|
self._callback(self.SIGNAL_CONNECT, self._connected, None)
|
|
except IOError as e:
|
|
self._connected = False
|
|
self._callback(self.SIGNAL_CONNECT, self._connected, e)
|
|
|
|
|
|
def _disconnect(self):
|
|
"""Action: Performs the real disconnect from MPD.
|
|
"""
|
|
if not self.is_connected():
|
|
return
|
|
try:
|
|
#self._client.close()
|
|
self._client.disconnect()
|
|
except:
|
|
self._client = mpd.MPDClient()
|
|
self._connected = False
|
|
self._callback(self.SIGNAL_CONNECT, self._connected, None)
|
|
|
|
|
|
def _update(self):
|
|
"""Action: Performs the real update.
|
|
"""
|
|
for song in self._client.listallinfo():
|
|
try:
|
|
hash = MCGAlbum.hash(song['artist'], song['album'])
|
|
if hash in self._albums.keys():
|
|
album = self._albums[hash]
|
|
else:
|
|
album = MCGAlbum(song['artist'], song['album'], song['date'], os.path.dirname(song['file']))
|
|
self._albums[album.get_hash()] = album
|
|
track = MCGTrack(song['title'], song['track'], song['time'], song['file'])
|
|
album.add_track(track)
|
|
except KeyError:
|
|
pass
|
|
# TODO Alben sortieren
|
|
self._callback(self.SIGNAL_UPDATE, self._albums)
|
|
|
|
|
|
def _get_status(self):
|
|
"""Action: Performs the real status determination
|
|
"""
|
|
if not self._has_callback(self.SIGNAL_STATUS):
|
|
return
|
|
status = self._client.status()
|
|
state = status['state']
|
|
song = self._client.currentsong()
|
|
album = None
|
|
if song:
|
|
album = MCGAlbum(song['artist'], song['album'], song['date'], os.path.dirname(song['file']))
|
|
self._callback(self.SIGNAL_STATUS, state, album)
|
|
|
|
|
|
def _play(self, album):
|
|
"""Action: Performs the real play command.
|
|
"""
|
|
track_ids = []
|
|
for track in self._albums[album].get_tracks():
|
|
track_id = self._client.addid(track.get_file())
|
|
track_ids.append(track_id)
|
|
self._client.moveid(track_id, len(track_ids)-1)
|
|
self._client.playid(track_ids[0])
|
|
|
|
|
|
def _playpause(self):
|
|
"""Action: Performs the real play/pause command.
|
|
"""
|
|
status = self._client.status()
|
|
state = status['state']
|
|
|
|
if state == 'play':
|
|
self._client.pause()
|
|
else:
|
|
self._client.play()
|
|
|
|
|
|
def _next(self):
|
|
"""Action: Performs the real next command.
|
|
"""
|
|
song = self._client.currentsong()
|
|
if song:
|
|
current_album = MCGAlbum(song['artist'], song['album'], song['date'], os.path.dirname(song['file']))
|
|
# TODO _next()
|
|
|
|
|
|
def _idle(self, modules):
|
|
"""Reacts to idle events from MPD.
|
|
"""
|
|
if not modules:
|
|
return
|
|
|
|
if 'player' in modules:
|
|
self._get_status()
|
|
if 'database' in modules:
|
|
# TODO update DB
|
|
pass
|
|
if 'update' in modules:
|
|
# TODO update
|
|
pass
|
|
if 'mixer' in modules:
|
|
# TODO mixer
|
|
pass
|
|
|
|
|
|
def _idle_playlist(self):
|
|
""" Reacts on the playlist idle event.
|
|
"""
|
|
pass
|
|
|
|
|
|
|
|
|
|
class MCGAlbum:
|
|
_file_names = ['folder', 'cover']
|
|
_file_exts = ['jpg', 'jpeg', 'png']
|
|
SORT_BY_ARTIST = 'artist'
|
|
SORT_BY_TITLE = 'title'
|
|
SORT_BY_YEAR = 'year'
|
|
|
|
|
|
def __init__(self, artist, title, date, path):
|
|
self._artist = artist
|
|
if type(self._artist) is list:
|
|
self._artist = self._artist[0]
|
|
self._title = title
|
|
self._date = date
|
|
self._path = path
|
|
self._tracks = []
|
|
self._cover = None
|
|
|
|
self._set_hash()
|
|
self._find_cover()
|
|
|
|
|
|
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):
|
|
return self._cover
|
|
|
|
|
|
def _find_cover(self):
|
|
names = list(self._file_names)
|
|
names.append(self._title)
|
|
names.append(' - '.join((self._artist, self._title)))
|
|
|
|
|
|
for name in names:
|
|
for ext in self._file_exts:
|
|
filename = os.path.join('/home/oliver/Musik/', self._path, '.'.join([name, ext]))
|
|
if os.path.isfile(filename):
|
|
self._cover = filename
|
|
break
|
|
if self._cover is not None:
|
|
break
|
|
|
|
|
|
def hash(artist, title):
|
|
if type(artist) is list:
|
|
artist = artist[0]
|
|
return md5(artist.encode('utf-8')+title.encode('utf-8')).hexdigest()
|
|
|
|
|
|
def _set_hash(self):
|
|
self._hash = MCGAlbum.hash(self._artist, self._title)
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
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
|
|
|