2019-02-14 21:09:22 +00:00
|
|
|
"""Sensor for Last.fm account status."""
|
2020-01-26 13:28:42 +00:00
|
|
|
import hashlib
|
2018-09-08 16:33:41 +00:00
|
|
|
import logging
|
2016-05-15 20:11:41 +00:00
|
|
|
import re
|
2016-09-03 23:30:21 +00:00
|
|
|
|
2019-10-22 05:38:01 +00:00
|
|
|
import pylast as lastfm
|
|
|
|
from pylast import WSError
|
2016-09-03 23:30:21 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.components.sensor import PLATFORM_SCHEMA
|
2019-10-22 05:38:01 +00:00
|
|
|
from homeassistant.const import ATTR_ATTRIBUTION, CONF_API_KEY
|
2016-09-03 23:30:21 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2018-01-15 22:25:00 +00:00
|
|
|
from homeassistant.helpers.entity import Entity
|
2016-09-03 23:30:21 +00:00
|
|
|
|
2018-09-08 16:33:41 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_LAST_PLAYED = "last_played"
|
|
|
|
ATTR_PLAY_COUNT = "play_count"
|
|
|
|
ATTR_TOP_PLAYED = "top_played"
|
2019-02-14 21:09:22 +00:00
|
|
|
ATTRIBUTION = "Data provided by Last.fm"
|
2016-09-03 23:30:21 +00:00
|
|
|
|
2020-04-05 16:03:13 +00:00
|
|
|
STATE_NOT_SCROBBLING = "Not Scrobbling"
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_USERS = "users"
|
2016-05-15 20:11:41 +00:00
|
|
|
|
2020-07-14 20:17:14 +00:00
|
|
|
ICON = "mdi:radio-fm"
|
2016-05-15 20:11:41 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_API_KEY): cv.string,
|
|
|
|
vol.Required(CONF_USERS, default=[]): vol.All(cv.ensure_list, [cv.string]),
|
|
|
|
}
|
|
|
|
)
|
2016-05-15 20:11:41 +00:00
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2018-09-08 16:33:41 +00:00
|
|
|
"""Set up the Last.fm sensor platform."""
|
|
|
|
api_key = config[CONF_API_KEY]
|
|
|
|
users = config.get(CONF_USERS)
|
|
|
|
|
|
|
|
lastfm_api = lastfm.LastFMNetwork(api_key=api_key)
|
|
|
|
|
|
|
|
entities = []
|
|
|
|
for username in users:
|
|
|
|
try:
|
|
|
|
lastfm_api.get_user(username).get_image()
|
|
|
|
entities.append(LastfmSensor(username, lastfm_api))
|
|
|
|
except WSError as error:
|
|
|
|
_LOGGER.error(error)
|
|
|
|
return
|
2016-09-03 23:30:21 +00:00
|
|
|
|
2018-09-08 16:33:41 +00:00
|
|
|
add_entities(entities, True)
|
2016-05-15 20:11:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
class LastfmSensor(Entity):
|
|
|
|
"""A class for the Last.fm account."""
|
|
|
|
|
2019-10-22 05:38:01 +00:00
|
|
|
def __init__(self, user, lastfm_api):
|
2016-05-15 20:11:41 +00:00
|
|
|
"""Initialize the sensor."""
|
2020-01-26 13:28:42 +00:00
|
|
|
self._unique_id = hashlib.sha256(user.encode("utf-8")).hexdigest()
|
2019-10-22 05:38:01 +00:00
|
|
|
self._user = lastfm_api.get_user(user)
|
2016-05-15 20:11:41 +00:00
|
|
|
self._name = user
|
2019-10-22 05:38:01 +00:00
|
|
|
self._lastfm = lastfm_api
|
2016-05-15 20:11:41 +00:00
|
|
|
self._state = "Not Scrobbling"
|
|
|
|
self._playcount = None
|
|
|
|
self._lastplayed = None
|
|
|
|
self._topplayed = None
|
|
|
|
self._cover = None
|
|
|
|
|
2020-01-26 13:28:42 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return the unique ID of the sensor."""
|
|
|
|
return self._unique_id
|
|
|
|
|
2016-05-15 20:11:41 +00:00
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the sensor."""
|
|
|
|
return self._state
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Update device state."""
|
|
|
|
self._cover = self._user.get_image()
|
|
|
|
self._playcount = self._user.get_playcount()
|
2020-04-05 16:03:13 +00:00
|
|
|
|
|
|
|
recent_tracks = self._user.get_recent_tracks(limit=2)
|
|
|
|
if recent_tracks:
|
|
|
|
last = recent_tracks[0]
|
|
|
|
self._lastplayed = f"{last.track.artist} - {last.track.title}"
|
|
|
|
|
|
|
|
top_tracks = self._user.get_top_tracks(limit=1)
|
|
|
|
if top_tracks:
|
|
|
|
top = top_tracks[0]
|
|
|
|
toptitle = re.search("', '(.+?)',", str(top))
|
|
|
|
topartist = re.search("'(.+?)',", str(top))
|
|
|
|
self._topplayed = f"{topartist.group(1)} - {toptitle.group(1)}"
|
|
|
|
|
|
|
|
now_playing = self._user.get_now_playing()
|
|
|
|
if now_playing is None:
|
|
|
|
self._state = STATE_NOT_SCROBBLING
|
2016-05-15 20:11:41 +00:00
|
|
|
return
|
2020-04-05 16:03:13 +00:00
|
|
|
|
|
|
|
self._state = f"{now_playing.artist} - {now_playing.title}"
|
2016-05-15 20:11:41 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the state attributes."""
|
2017-02-14 12:10:38 +00:00
|
|
|
return {
|
2019-02-14 21:09:22 +00:00
|
|
|
ATTR_ATTRIBUTION: ATTRIBUTION,
|
2017-02-14 12:10:38 +00:00
|
|
|
ATTR_LAST_PLAYED: self._lastplayed,
|
|
|
|
ATTR_PLAY_COUNT: self._playcount,
|
|
|
|
ATTR_TOP_PLAYED: self._topplayed,
|
|
|
|
}
|
2016-05-15 20:11:41 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def entity_picture(self):
|
|
|
|
"""Avatar of the user."""
|
|
|
|
return self._cover
|
|
|
|
|
|
|
|
@property
|
|
|
|
def icon(self):
|
|
|
|
"""Return the icon to use in the frontend."""
|
|
|
|
return ICON
|