2017-03-17 06:32:52 +00:00
|
|
|
"""
|
|
|
|
Volumio Platform.
|
|
|
|
|
2018-02-06 18:46:44 +00:00
|
|
|
Volumio rest API: https://volumio.github.io/docs/API/REST_API.html
|
2017-03-17 06:32:52 +00:00
|
|
|
"""
|
2018-02-02 22:12:54 +00:00
|
|
|
from datetime import timedelta
|
2020-10-15 13:49:36 +00:00
|
|
|
import json
|
2017-03-26 13:50:40 +00:00
|
|
|
|
2020-07-27 07:19:19 +00:00
|
|
|
from homeassistant.components.media_player import MediaPlayerEntity
|
2019-02-08 22:18:18 +00:00
|
|
|
from homeassistant.components.media_player.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
MEDIA_TYPE_MUSIC,
|
2020-10-15 13:49:36 +00:00
|
|
|
SUPPORT_BROWSE_MEDIA,
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_CLEAR_PLAYLIST,
|
|
|
|
SUPPORT_NEXT_TRACK,
|
|
|
|
SUPPORT_PAUSE,
|
|
|
|
SUPPORT_PLAY,
|
2020-10-15 13:49:36 +00:00
|
|
|
SUPPORT_PLAY_MEDIA,
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_PREVIOUS_TRACK,
|
|
|
|
SUPPORT_SEEK,
|
|
|
|
SUPPORT_SELECT_SOURCE,
|
2019-12-09 13:42:53 +00:00
|
|
|
SUPPORT_SHUFFLE_SET,
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_STOP,
|
|
|
|
SUPPORT_VOLUME_MUTE,
|
|
|
|
SUPPORT_VOLUME_SET,
|
|
|
|
SUPPORT_VOLUME_STEP,
|
|
|
|
)
|
2017-03-17 06:32:52 +00:00
|
|
|
from homeassistant.const import (
|
2020-07-27 07:19:19 +00:00
|
|
|
CONF_ID,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_NAME,
|
|
|
|
STATE_IDLE,
|
|
|
|
STATE_PAUSED,
|
|
|
|
STATE_PLAYING,
|
|
|
|
)
|
2018-02-02 22:12:54 +00:00
|
|
|
from homeassistant.util import Throttle
|
2017-03-17 06:32:52 +00:00
|
|
|
|
2020-10-15 13:49:36 +00:00
|
|
|
from .browse_media import browse_node, browse_top_level
|
2020-07-27 07:19:19 +00:00
|
|
|
from .const import DATA_INFO, DATA_VOLUMIO, DOMAIN
|
|
|
|
|
2017-03-17 06:32:52 +00:00
|
|
|
_CONFIGURING = {}
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_VOLUMIO = (
|
|
|
|
SUPPORT_PAUSE
|
|
|
|
| SUPPORT_VOLUME_SET
|
|
|
|
| SUPPORT_VOLUME_MUTE
|
|
|
|
| SUPPORT_PREVIOUS_TRACK
|
|
|
|
| SUPPORT_NEXT_TRACK
|
|
|
|
| SUPPORT_SEEK
|
|
|
|
| SUPPORT_STOP
|
|
|
|
| SUPPORT_PLAY
|
2020-10-15 13:49:36 +00:00
|
|
|
| SUPPORT_PLAY_MEDIA
|
2019-07-31 19:25:30 +00:00
|
|
|
| SUPPORT_VOLUME_STEP
|
|
|
|
| SUPPORT_SELECT_SOURCE
|
2019-08-15 11:41:54 +00:00
|
|
|
| SUPPORT_SHUFFLE_SET
|
2019-07-31 19:25:30 +00:00
|
|
|
| SUPPORT_CLEAR_PLAYLIST
|
2020-10-15 13:49:36 +00:00
|
|
|
| SUPPORT_BROWSE_MEDIA
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2017-03-17 06:32:52 +00:00
|
|
|
|
2018-02-02 22:12:54 +00:00
|
|
|
PLAYLIST_UPDATE_INTERVAL = timedelta(seconds=15)
|
2017-03-17 06:32:52 +00:00
|
|
|
|
|
|
|
|
2020-07-27 07:19:19 +00:00
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
|
|
|
"""Set up the Volumio media player platform."""
|
2018-05-01 19:20:38 +00:00
|
|
|
|
2020-07-27 07:19:19 +00:00
|
|
|
data = hass.data[DOMAIN][config_entry.entry_id]
|
|
|
|
volumio = data[DATA_VOLUMIO]
|
|
|
|
info = data[DATA_INFO]
|
|
|
|
uid = config_entry.data[CONF_ID]
|
|
|
|
name = config_entry.data[CONF_NAME]
|
2018-05-01 19:20:38 +00:00
|
|
|
|
2020-07-30 14:51:46 +00:00
|
|
|
entity = Volumio(volumio, uid, name, info)
|
2018-08-24 14:37:30 +00:00
|
|
|
async_add_entities([entity])
|
2017-03-17 06:32:52 +00:00
|
|
|
|
|
|
|
|
2020-04-25 16:00:57 +00:00
|
|
|
class Volumio(MediaPlayerEntity):
|
2017-03-17 06:32:52 +00:00
|
|
|
"""Volumio Player Object."""
|
|
|
|
|
2020-07-30 14:51:46 +00:00
|
|
|
def __init__(self, volumio, uid, name, info):
|
2017-03-17 06:32:52 +00:00
|
|
|
"""Initialize the media player."""
|
2020-07-27 07:19:19 +00:00
|
|
|
self._volumio = volumio
|
|
|
|
self._uid = uid
|
2017-03-17 06:32:52 +00:00
|
|
|
self._name = name
|
2020-07-27 07:19:19 +00:00
|
|
|
self._info = info
|
2017-03-17 06:32:52 +00:00
|
|
|
self._state = {}
|
2018-02-02 22:12:54 +00:00
|
|
|
self._playlists = []
|
|
|
|
self._currentplaylist = None
|
2021-01-25 00:16:10 +00:00
|
|
|
self.thumbnail_cache = {}
|
2017-03-17 06:32:52 +00:00
|
|
|
|
2018-10-01 06:58:21 +00:00
|
|
|
async def async_update(self):
|
2017-03-17 06:32:52 +00:00
|
|
|
"""Update state."""
|
2020-07-27 07:19:19 +00:00
|
|
|
self._state = await self._volumio.get_state()
|
2018-10-01 06:58:21 +00:00
|
|
|
await self._async_update_playlists()
|
2020-07-27 07:19:19 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return the unique id for the entity."""
|
|
|
|
return self._uid
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the entity."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_info(self):
|
|
|
|
"""Return device info for this device."""
|
|
|
|
return {
|
|
|
|
"identifiers": {(DOMAIN, self.unique_id)},
|
|
|
|
"name": self.name,
|
|
|
|
"manufacturer": "Volumio",
|
|
|
|
"sw_version": self._info["systemversion"],
|
|
|
|
"model": self._info["hardware"],
|
|
|
|
}
|
2017-03-17 06:32:52 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_content_type(self):
|
|
|
|
"""Content type of current playing media."""
|
|
|
|
return MEDIA_TYPE_MUSIC
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the device."""
|
2019-07-31 19:25:30 +00:00
|
|
|
status = self._state.get("status", None)
|
|
|
|
if status == "pause":
|
2017-03-17 06:32:52 +00:00
|
|
|
return STATE_PAUSED
|
2019-07-31 19:25:30 +00:00
|
|
|
if status == "play":
|
2017-03-17 06:32:52 +00:00
|
|
|
return STATE_PLAYING
|
2017-07-06 06:30:01 +00:00
|
|
|
|
|
|
|
return STATE_IDLE
|
2017-03-17 06:32:52 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_title(self):
|
|
|
|
"""Title of current playing media."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._state.get("title", None)
|
2017-03-17 06:32:52 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_artist(self):
|
|
|
|
"""Artist of current playing media (Music track only)."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._state.get("artist", None)
|
2017-03-17 06:32:52 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_album_name(self):
|
|
|
|
"""Artist of current playing media (Music track only)."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._state.get("album", None)
|
2017-03-17 06:32:52 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_image_url(self):
|
|
|
|
"""Image url of current playing media."""
|
2019-07-31 19:25:30 +00:00
|
|
|
url = self._state.get("albumart", None)
|
2020-07-27 07:19:19 +00:00
|
|
|
return self._volumio.canonic_url(url)
|
2017-03-17 06:32:52 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_seek_position(self):
|
|
|
|
"""Time in seconds of current seek position."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._state.get("seek", None)
|
2017-03-17 06:32:52 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_duration(self):
|
|
|
|
"""Time in seconds of current song duration."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._state.get("duration", None)
|
2017-03-17 06:32:52 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def volume_level(self):
|
|
|
|
"""Volume level of the media player (0..1)."""
|
2019-07-31 19:25:30 +00:00
|
|
|
volume = self._state.get("volume", None)
|
2018-02-02 22:12:54 +00:00
|
|
|
if volume is not None and volume != "":
|
2019-01-19 06:12:56 +00:00
|
|
|
volume = int(volume) / 100
|
2017-03-17 06:32:52 +00:00
|
|
|
return volume
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_volume_muted(self):
|
|
|
|
"""Boolean if volume is currently muted."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._state.get("mute", None)
|
2017-03-17 06:32:52 +00:00
|
|
|
|
2019-08-15 11:41:54 +00:00
|
|
|
@property
|
|
|
|
def shuffle(self):
|
|
|
|
"""Boolean if shuffle is enabled."""
|
|
|
|
return self._state.get("random", False)
|
|
|
|
|
2018-02-02 22:12:54 +00:00
|
|
|
@property
|
|
|
|
def source_list(self):
|
|
|
|
"""Return the list of available input sources."""
|
|
|
|
return self._playlists
|
|
|
|
|
|
|
|
@property
|
|
|
|
def source(self):
|
|
|
|
"""Name of the current input source."""
|
|
|
|
return self._currentplaylist
|
|
|
|
|
2017-03-17 06:32:52 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Flag of media commands that are supported."""
|
|
|
|
return SUPPORT_VOLUMIO
|
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_media_next_track(self):
|
2017-03-17 06:32:52 +00:00
|
|
|
"""Send media_next command to media player."""
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.next()
|
2017-03-17 06:32:52 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_media_previous_track(self):
|
2017-03-17 06:32:52 +00:00
|
|
|
"""Send media_previous command to media player."""
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.previous()
|
2017-03-17 06:32:52 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_media_play(self):
|
2017-03-17 06:32:52 +00:00
|
|
|
"""Send media_play command to media player."""
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.play()
|
2017-03-17 06:32:52 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_media_pause(self):
|
2017-03-17 06:32:52 +00:00
|
|
|
"""Send media_pause command to media player."""
|
2020-12-22 11:32:56 +00:00
|
|
|
if self._state.get("trackType") == "webradio":
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.stop()
|
2020-01-29 21:59:45 +00:00
|
|
|
else:
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.pause()
|
2017-03-17 06:32:52 +00:00
|
|
|
|
2020-07-01 05:55:31 +00:00
|
|
|
async def async_media_stop(self):
|
|
|
|
"""Send media_stop command to media player."""
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.stop()
|
2020-07-01 05:55:31 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_set_volume_level(self, volume):
|
2017-03-17 06:32:52 +00:00
|
|
|
"""Send volume_up command to media player."""
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.set_volume_level(int(volume * 100))
|
2017-03-17 06:32:52 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_volume_up(self):
|
2018-02-02 22:12:54 +00:00
|
|
|
"""Service to send the Volumio the command for volume up."""
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.volume_up()
|
2018-02-02 22:12:54 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_volume_down(self):
|
2018-02-02 22:12:54 +00:00
|
|
|
"""Service to send the Volumio the command for volume down."""
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.volume_down()
|
2018-02-02 22:12:54 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_mute_volume(self, mute):
|
2017-03-17 06:32:52 +00:00
|
|
|
"""Send mute command to media player."""
|
|
|
|
if mute:
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.mute()
|
|
|
|
else:
|
|
|
|
await self._volumio.unmute()
|
2018-02-02 22:12:54 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_set_shuffle(self, shuffle):
|
2019-08-15 11:41:54 +00:00
|
|
|
"""Enable/disable shuffle mode."""
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.set_shuffle(shuffle)
|
2019-08-15 11:41:54 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_select_source(self, source):
|
2020-07-27 07:19:19 +00:00
|
|
|
"""Choose an available playlist and play it."""
|
|
|
|
await self._volumio.play_playlist(source)
|
2018-02-02 22:12:54 +00:00
|
|
|
self._currentplaylist = source
|
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_clear_playlist(self):
|
2018-02-02 22:12:54 +00:00
|
|
|
"""Clear players playlist."""
|
2020-07-27 07:19:19 +00:00
|
|
|
await self._volumio.clear_playlist()
|
2018-02-02 22:12:54 +00:00
|
|
|
self._currentplaylist = None
|
|
|
|
|
|
|
|
@Throttle(PLAYLIST_UPDATE_INTERVAL)
|
2018-03-10 03:38:51 +00:00
|
|
|
async def _async_update_playlists(self, **kwargs):
|
2018-02-02 22:12:54 +00:00
|
|
|
"""Update available Volumio playlists."""
|
2020-07-27 07:19:19 +00:00
|
|
|
self._playlists = await self._volumio.get_playlists()
|
2020-10-15 13:49:36 +00:00
|
|
|
|
|
|
|
async def async_play_media(self, media_type, media_id, **kwargs):
|
|
|
|
"""Send the play_media command to the media player."""
|
|
|
|
await self._volumio.replace_and_play(json.loads(media_id))
|
|
|
|
|
|
|
|
async def async_browse_media(self, media_content_type=None, media_content_id=None):
|
|
|
|
"""Implement the websocket media browsing helper."""
|
2021-01-25 00:16:10 +00:00
|
|
|
self.thumbnail_cache = {}
|
2020-10-15 13:49:36 +00:00
|
|
|
if media_content_type in [None, "library"]:
|
|
|
|
return await browse_top_level(self._volumio)
|
|
|
|
|
2021-01-25 00:16:10 +00:00
|
|
|
return await browse_node(
|
|
|
|
self, self._volumio, media_content_type, media_content_id
|
|
|
|
)
|
|
|
|
|
|
|
|
async def async_get_browse_image(
|
|
|
|
self, media_content_type, media_content_id, media_image_id=None
|
|
|
|
):
|
|
|
|
"""Get album art from Volumio."""
|
|
|
|
cached_url = self.thumbnail_cache.get(media_content_id)
|
|
|
|
image_url = self._volumio.canonic_url(cached_url)
|
|
|
|
return await self._async_fetch_image(image_url)
|