2019-04-03 15:40:03 +00:00
|
|
|
"""Sensor for Xbox Live account status."""
|
2016-09-03 23:43:33 +00:00
|
|
|
import logging
|
2019-09-27 06:02:58 +00:00
|
|
|
from datetime import timedelta
|
2016-10-30 08:58:34 +00:00
|
|
|
|
2016-09-03 23:43:33 +00:00
|
|
|
import voluptuous as vol
|
2019-09-27 06:02:58 +00:00
|
|
|
from xboxapi import xbox_api
|
2016-09-03 23:43:33 +00:00
|
|
|
|
|
|
|
from homeassistant.components.sensor import PLATFORM_SCHEMA
|
2019-09-27 06:02:58 +00:00
|
|
|
from homeassistant.const import CONF_API_KEY, CONF_SCAN_INTERVAL
|
|
|
|
from homeassistant.core import callback
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-09-03 23:43:33 +00:00
|
|
|
from homeassistant.helpers.entity import Entity
|
2019-09-27 06:02:58 +00:00
|
|
|
from homeassistant.helpers.event import async_track_time_interval
|
2016-09-03 23:43:33 +00:00
|
|
|
|
2016-10-30 08:58:34 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_XUID = "xuid"
|
2016-09-03 23:43:33 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ICON = "mdi:xbox"
|
2016-10-30 08:58:34 +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_XUID): vol.All(cv.ensure_list, [cv.string]),
|
|
|
|
}
|
|
|
|
)
|
2016-09-03 23:43:33 +00:00
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2016-10-30 08:58:34 +00:00
|
|
|
"""Set up the Xbox platform."""
|
2019-09-27 06:02:58 +00:00
|
|
|
api = xbox_api.XboxApi(config[CONF_API_KEY])
|
|
|
|
entities = []
|
2016-09-03 23:43:33 +00:00
|
|
|
|
2018-01-18 05:26:23 +00:00
|
|
|
# request personal profile to check api connection
|
|
|
|
profile = api.get_profile()
|
2019-07-31 19:25:30 +00:00
|
|
|
if profile.get("error_code") is not None:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Can't setup XboxAPI connection. Check your account or "
|
2019-09-27 06:02:58 +00:00
|
|
|
"api key on xboxapi.com. Code: %s Description: %s ",
|
|
|
|
profile.get("error_code", "unknown"),
|
|
|
|
profile.get("error_message", "unknown"),
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-01-18 05:26:23 +00:00
|
|
|
return
|
|
|
|
|
2019-09-27 06:02:58 +00:00
|
|
|
users = config[CONF_XUID]
|
|
|
|
|
|
|
|
interval = timedelta(minutes=1 * len(users))
|
|
|
|
interval = config.get(CONF_SCAN_INTERVAL, interval)
|
|
|
|
|
|
|
|
for xuid in users:
|
|
|
|
gamercard = get_user_gamercard(api, xuid)
|
|
|
|
if gamercard is None:
|
|
|
|
continue
|
|
|
|
entities.append(XboxSensor(api, xuid, gamercard, interval))
|
|
|
|
|
|
|
|
if entities:
|
|
|
|
add_entities(entities, True)
|
|
|
|
|
2016-09-03 23:43:33 +00:00
|
|
|
|
2019-09-27 06:02:58 +00:00
|
|
|
def get_user_gamercard(api, xuid):
|
|
|
|
"""Get profile info."""
|
|
|
|
gamercard = api.get_user_gamercard(xuid)
|
|
|
|
_LOGGER.debug("User gamercard: %s", gamercard)
|
|
|
|
|
|
|
|
if gamercard.get("success", True) and gamercard.get("code") is None:
|
|
|
|
return gamercard
|
|
|
|
_LOGGER.error(
|
|
|
|
"Can't get user profile %s. Error Code: %s Description: %s",
|
|
|
|
xuid,
|
|
|
|
gamercard.get("code", "unknown"),
|
|
|
|
gamercard.get("description", "unknown"),
|
|
|
|
)
|
|
|
|
return None
|
2016-09-03 23:43:33 +00:00
|
|
|
|
|
|
|
|
|
|
|
class XboxSensor(Entity):
|
|
|
|
"""A class for the Xbox account."""
|
|
|
|
|
2019-09-27 06:02:58 +00:00
|
|
|
def __init__(self, api, xuid, gamercard, interval):
|
2016-09-03 23:43:33 +00:00
|
|
|
"""Initialize the sensor."""
|
2019-01-24 07:20:20 +00:00
|
|
|
self._state = None
|
2019-09-27 06:02:58 +00:00
|
|
|
self._presence = []
|
2016-09-03 23:43:33 +00:00
|
|
|
self._xuid = xuid
|
|
|
|
self._api = api
|
2019-09-27 06:02:58 +00:00
|
|
|
self._gamertag = gamercard.get("gamertag")
|
|
|
|
self._gamerscore = gamercard.get("gamerscore")
|
|
|
|
self._interval = interval
|
|
|
|
self._picture = gamercard.get("gamerpicSmallSslImagePath")
|
|
|
|
self._tier = gamercard.get("tier")
|
2016-09-03 23:43:33 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._gamertag
|
|
|
|
|
2019-09-27 06:02:58 +00:00
|
|
|
@property
|
|
|
|
def should_poll(self):
|
|
|
|
"""Return False as this entity has custom polling."""
|
|
|
|
return False
|
|
|
|
|
2016-09-03 23:43:33 +00:00
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the sensor."""
|
|
|
|
return self._state
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the state attributes."""
|
|
|
|
attributes = {}
|
2019-07-31 19:25:30 +00:00
|
|
|
attributes["gamerscore"] = self._gamerscore
|
|
|
|
attributes["tier"] = self._tier
|
2018-03-04 01:38:51 +00:00
|
|
|
|
2016-09-03 23:43:33 +00:00
|
|
|
for device in self._presence:
|
2019-07-31 19:25:30 +00:00
|
|
|
for title in device.get("titles"):
|
2016-09-03 23:43:33 +00:00
|
|
|
attributes[
|
2019-09-27 06:02:58 +00:00
|
|
|
f'{device.get("type")} {title.get("placement")}'
|
2019-07-31 19:25:30 +00:00
|
|
|
] = title.get("name")
|
2016-09-03 23:43:33 +00:00
|
|
|
|
|
|
|
return attributes
|
|
|
|
|
|
|
|
@property
|
|
|
|
def entity_picture(self):
|
|
|
|
"""Avatar of the account."""
|
|
|
|
return self._picture
|
|
|
|
|
|
|
|
@property
|
|
|
|
def icon(self):
|
|
|
|
"""Return the icon to use in the frontend."""
|
|
|
|
return ICON
|
|
|
|
|
2019-09-27 06:02:58 +00:00
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Start custom polling."""
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_update(event_time=None):
|
|
|
|
"""Update the entity."""
|
|
|
|
self.async_schedule_update_ha_state(True)
|
|
|
|
|
|
|
|
async_track_time_interval(self.hass, async_update, self._interval)
|
|
|
|
|
2016-09-03 23:43:33 +00:00
|
|
|
def update(self):
|
|
|
|
"""Update state data from Xbox API."""
|
|
|
|
presence = self._api.get_user_presence(self._xuid)
|
2019-09-27 06:02:58 +00:00
|
|
|
_LOGGER.debug("User presence: %s", presence)
|
2019-07-31 19:25:30 +00:00
|
|
|
self._state = presence.get("state")
|
2019-09-27 06:02:58 +00:00
|
|
|
self._presence = presence.get("devices", [])
|