Merge branch 'gstreamer' of git://github.com/knutz3n/mopidy into gstreamer
This commit is contained in:
commit
ded033fd4c
@ -1,4 +1,5 @@
|
|||||||
import datetime as dt
|
import datetime as dt
|
||||||
|
import gobject
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
import multiprocessing
|
import multiprocessing
|
||||||
@ -14,8 +15,7 @@ from mopidy.backends.base import (BaseBackend, BaseCurrentPlaylistController,
|
|||||||
BaseStoredPlaylistsController)
|
BaseStoredPlaylistsController)
|
||||||
from mopidy.models import Artist, Album, Track, Playlist
|
from mopidy.models import Artist, Album, Track, Playlist
|
||||||
|
|
||||||
import alsaaudio
|
import gst
|
||||||
|
|
||||||
logger = logging.getLogger('mopidy.backends.libspotify')
|
logger = logging.getLogger('mopidy.backends.libspotify')
|
||||||
|
|
||||||
ENCODING = 'utf-8'
|
ENCODING = 'utf-8'
|
||||||
@ -44,8 +44,7 @@ class LibspotifyBackend(BaseBackend):
|
|||||||
self.stored_playlists = LibspotifyStoredPlaylistsController(
|
self.stored_playlists = LibspotifyStoredPlaylistsController(
|
||||||
backend=self)
|
backend=self)
|
||||||
self.uri_handlers = [u'spotify:', u'http://open.spotify.com/']
|
self.uri_handlers = [u'spotify:', u'http://open.spotify.com/']
|
||||||
self.audio_controller_class = kwargs.get(
|
self.gstreamer_pipeline = gst.Pipeline("spotify_pipeline")
|
||||||
'audio_controller_class', AlsaController)
|
|
||||||
self.spotify = self._connect()
|
self.spotify = self._connect()
|
||||||
|
|
||||||
def _connect(self):
|
def _connect(self):
|
||||||
@ -53,7 +52,7 @@ class LibspotifyBackend(BaseBackend):
|
|||||||
spotify = LibspotifySessionManager(
|
spotify = LibspotifySessionManager(
|
||||||
settings.SPOTIFY_USERNAME, settings.SPOTIFY_PASSWORD,
|
settings.SPOTIFY_USERNAME, settings.SPOTIFY_PASSWORD,
|
||||||
core_queue=self.core_queue,
|
core_queue=self.core_queue,
|
||||||
audio_controller_class=self.audio_controller_class)
|
gstreamer_pipeline=self.gstreamer_pipeline)
|
||||||
spotify.start()
|
spotify.start()
|
||||||
return spotify
|
return spotify
|
||||||
|
|
||||||
@ -96,10 +95,12 @@ class LibspotifyLibraryController(BaseLibraryController):
|
|||||||
|
|
||||||
class LibspotifyPlaybackController(BasePlaybackController):
|
class LibspotifyPlaybackController(BasePlaybackController):
|
||||||
def _pause(self):
|
def _pause(self):
|
||||||
# TODO
|
result = self.backend.gstreamer_pipeline.set_state(gst.STATE_PAUSED)
|
||||||
return False
|
logger.debug('Changed gstreamer state to paused. Result was: %s' % result)
|
||||||
|
return result == gst.STATE_CHANGE_SUCCESS
|
||||||
|
|
||||||
def _play(self, track):
|
def _play(self, track):
|
||||||
|
self.backend.gstreamer_pipeline.set_state(gst.STATE_READY)
|
||||||
if self.state == self.PLAYING:
|
if self.state == self.PLAYING:
|
||||||
self.stop()
|
self.stop()
|
||||||
if track.uri is None:
|
if track.uri is None:
|
||||||
@ -108,19 +109,22 @@ class LibspotifyPlaybackController(BasePlaybackController):
|
|||||||
self.backend.spotify.session.load(
|
self.backend.spotify.session.load(
|
||||||
Link.from_string(track.uri).as_track())
|
Link.from_string(track.uri).as_track())
|
||||||
self.backend.spotify.session.play(1)
|
self.backend.spotify.session.play(1)
|
||||||
|
self.backend.gstreamer_pipeline.set_state(gst.STATE_PLAYING)
|
||||||
return True
|
return True
|
||||||
except SpotifyError as e:
|
except SpotifyError as e:
|
||||||
logger.warning('Play %s failed: %s', track.uri, e)
|
logger.warning('Play %s failed: %s', track.uri, e)
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def _resume(self):
|
def _resume(self):
|
||||||
# TODO
|
result = self.backend.gstreamer_pipeline.set_state(gst.STATE_PLAYING)
|
||||||
return False
|
logger.debug('Changed gstreamer state to playing. Result was: %s' % result)
|
||||||
|
return result == gst.STATE_CHANGE_SUCCESS
|
||||||
|
|
||||||
def _seek(self, time_position):
|
def _seek(self, time_position):
|
||||||
pass # TODO
|
pass # TODO
|
||||||
|
|
||||||
def _stop(self):
|
def _stop(self):
|
||||||
|
self.backend.gstreamer_pipeline.set_state(gst.STATE_READY)
|
||||||
self.backend.spotify.session.play(0)
|
self.backend.spotify.session.play(0)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@ -195,6 +199,22 @@ class LibspotifyTranslator(object):
|
|||||||
tracks=[cls.to_mopidy_track(t) for t in spotify_playlist],
|
tracks=[cls.to_mopidy_track(t) for t in spotify_playlist],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
class GstreamerMessageBusProcess(threading.Thread):
|
||||||
|
def __init__(self, core_queue, pipeline):
|
||||||
|
super(GstreamerMessageBusProcess, self).__init__()
|
||||||
|
self.core_queue = core_queue
|
||||||
|
self.bus = pipeline.get_bus()
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
loop = gobject.MainLoop()
|
||||||
|
gobject.threads_init()
|
||||||
|
context = loop.get_context()
|
||||||
|
while True:
|
||||||
|
message = self.bus.pop_filtered(gst.MESSAGE_EOS)
|
||||||
|
if message is not None:
|
||||||
|
self.core_queue.put({'command': 'end_of_track'})
|
||||||
|
logger.debug('Got and handled Gstreamer message of type: %s' % message.type)
|
||||||
|
context.iteration(True)
|
||||||
|
|
||||||
class LibspotifySessionManager(SpotifySessionManager, threading.Thread):
|
class LibspotifySessionManager(SpotifySessionManager, threading.Thread):
|
||||||
cache_location = os.path.expanduser(settings.SPOTIFY_LIB_CACHE)
|
cache_location = os.path.expanduser(settings.SPOTIFY_LIB_CACHE)
|
||||||
@ -202,13 +222,34 @@ class LibspotifySessionManager(SpotifySessionManager, threading.Thread):
|
|||||||
appkey_file = os.path.expanduser(settings.SPOTIFY_LIB_APPKEY)
|
appkey_file = os.path.expanduser(settings.SPOTIFY_LIB_APPKEY)
|
||||||
user_agent = 'Mopidy %s' % get_version()
|
user_agent = 'Mopidy %s' % get_version()
|
||||||
|
|
||||||
def __init__(self, username, password, core_queue, audio_controller_class):
|
def __init__(self, username, password, core_queue, gstreamer_pipeline):
|
||||||
SpotifySessionManager.__init__(self, username, password)
|
SpotifySessionManager.__init__(self, username, password)
|
||||||
threading.Thread.__init__(self)
|
threading.Thread.__init__(self)
|
||||||
self.core_queue = core_queue
|
self.core_queue = core_queue
|
||||||
self.connected = threading.Event()
|
self.connected = threading.Event()
|
||||||
self.audio = audio_controller_class(alsaaudio.PCM_NORMAL)
|
|
||||||
self.session = None
|
self.session = None
|
||||||
|
self.gstreamer_pipeline = gstreamer_pipeline
|
||||||
|
|
||||||
|
cap_string = """audio/x-raw-int,
|
||||||
|
endianness=(int)1234,
|
||||||
|
channels=(int)2,
|
||||||
|
width=(int)16,
|
||||||
|
depth=(int)16,
|
||||||
|
signed=True,
|
||||||
|
rate=(int)44100"""
|
||||||
|
caps = gst.caps_from_string(cap_string)
|
||||||
|
|
||||||
|
self.gsrc = gst.element_factory_make("appsrc", "app-source")
|
||||||
|
self.gsrc.set_property('caps', caps)
|
||||||
|
|
||||||
|
self.gsink = gst.element_factory_make("autoaudiosink", "autosink")
|
||||||
|
|
||||||
|
self.gstreamer_pipeline.add(self.gsrc, self.gsink)
|
||||||
|
|
||||||
|
gst.element_link_many(self.gsrc, self.gsink)
|
||||||
|
|
||||||
|
message_process = GstreamerMessageBusProcess(self.core_queue, self.gstreamer_pipeline)
|
||||||
|
message_process.start()
|
||||||
|
|
||||||
def run(self):
|
def run(self):
|
||||||
self.connect()
|
self.connect()
|
||||||
@ -250,8 +291,17 @@ class LibspotifySessionManager(SpotifySessionManager, threading.Thread):
|
|||||||
def music_delivery(self, session, frames, frame_size, num_frames,
|
def music_delivery(self, session, frames, frame_size, num_frames,
|
||||||
sample_type, sample_rate, channels):
|
sample_type, sample_rate, channels):
|
||||||
"""Callback used by pyspotify"""
|
"""Callback used by pyspotify"""
|
||||||
self.audio.music_delivery(session, frames, frame_size, num_frames,
|
cap_string = """audio/x-raw-int,
|
||||||
sample_type, sample_rate, channels)
|
endianness=(int)1234,
|
||||||
|
channels=(int)2,
|
||||||
|
width=(int)16,
|
||||||
|
depth=(int)16,
|
||||||
|
signed=True,
|
||||||
|
rate=(int)44100"""
|
||||||
|
caps = gst.caps_from_string(cap_string)
|
||||||
|
b = gst.Buffer(frames)
|
||||||
|
b.set_caps(caps)
|
||||||
|
self.gsrc.emit('push-buffer', b)
|
||||||
|
|
||||||
def play_token_lost(self, session):
|
def play_token_lost(self, session):
|
||||||
"""Callback used by pyspotify"""
|
"""Callback used by pyspotify"""
|
||||||
@ -264,8 +314,9 @@ class LibspotifySessionManager(SpotifySessionManager, threading.Thread):
|
|||||||
|
|
||||||
def end_of_track(self, session):
|
def end_of_track(self, session):
|
||||||
"""Callback used by pyspotify"""
|
"""Callback used by pyspotify"""
|
||||||
logger.debug('End of track')
|
logger.debug('End of track.')
|
||||||
self.core_queue.put({'command': 'end_of_track'})
|
self.gsrc.emit('end-of-stream')
|
||||||
|
logger.debug('End of stream sent to gstreamer.')
|
||||||
|
|
||||||
def search(self, query, connection):
|
def search(self, query, connection):
|
||||||
"""Search method used by Mopidy backend"""
|
"""Search method used by Mopidy backend"""
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user