2019-11-26 19:17:11 +00:00
|
|
|
"""StarLine Account."""
|
2021-03-18 13:31:38 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-09-29 14:19:06 +00:00
|
|
|
from collections.abc import Callable
|
2019-12-08 20:09:48 +00:00
|
|
|
from datetime import datetime, timedelta
|
2021-09-29 14:19:06 +00:00
|
|
|
from typing import Any
|
2019-12-08 20:09:48 +00:00
|
|
|
|
2019-11-26 19:17:11 +00:00
|
|
|
from starline import StarlineApi, StarlineDevice
|
|
|
|
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant
|
2021-05-11 05:11:51 +00:00
|
|
|
from homeassistant.helpers.entity import DeviceInfo
|
2019-11-26 19:17:11 +00:00
|
|
|
from homeassistant.helpers.event import async_track_time_interval
|
|
|
|
|
|
|
|
from .const import (
|
2020-10-17 02:24:08 +00:00
|
|
|
_LOGGER,
|
2019-12-08 20:09:48 +00:00
|
|
|
DATA_EXPIRES,
|
|
|
|
DATA_SLID_TOKEN,
|
|
|
|
DATA_SLNET_TOKEN,
|
|
|
|
DATA_USER_ID,
|
|
|
|
DEFAULT_SCAN_INTERVAL,
|
2021-01-13 19:44:24 +00:00
|
|
|
DEFAULT_SCAN_OBD_INTERVAL,
|
2019-11-26 19:17:11 +00:00
|
|
|
DOMAIN,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class StarlineAccount:
|
|
|
|
"""StarLine Account class."""
|
|
|
|
|
2021-05-20 13:58:17 +00:00
|
|
|
def __init__(self, hass: HomeAssistant, config_entry: ConfigEntry) -> None:
|
2019-12-10 22:25:06 +00:00
|
|
|
"""Initialize StarLine account."""
|
2019-11-26 19:17:11 +00:00
|
|
|
self._hass: HomeAssistant = hass
|
|
|
|
self._config_entry: ConfigEntry = config_entry
|
|
|
|
self._update_interval: int = DEFAULT_SCAN_INTERVAL
|
2021-01-13 19:44:24 +00:00
|
|
|
self._update_obd_interval: int = DEFAULT_SCAN_OBD_INTERVAL
|
2021-03-18 13:31:38 +00:00
|
|
|
self._unsubscribe_auto_updater: Callable | None = None
|
|
|
|
self._unsubscribe_auto_obd_updater: Callable | None = None
|
2019-11-26 19:17:11 +00:00
|
|
|
self._api: StarlineApi = StarlineApi(
|
|
|
|
config_entry.data[DATA_USER_ID], config_entry.data[DATA_SLNET_TOKEN]
|
|
|
|
)
|
|
|
|
|
2021-01-13 19:44:24 +00:00
|
|
|
def _check_slnet_token(self, interval: int) -> None:
|
2019-11-26 19:17:11 +00:00
|
|
|
"""Check SLNet token expiration and update if needed."""
|
|
|
|
now = datetime.now().timestamp()
|
|
|
|
slnet_token_expires = self._config_entry.data[DATA_EXPIRES]
|
|
|
|
|
2021-01-13 19:44:24 +00:00
|
|
|
if now + interval > slnet_token_expires:
|
2019-11-26 19:17:11 +00:00
|
|
|
self._update_slnet_token()
|
|
|
|
|
|
|
|
def _update_slnet_token(self) -> None:
|
|
|
|
"""Update SLNet token."""
|
|
|
|
slid_token = self._config_entry.data[DATA_SLID_TOKEN]
|
|
|
|
|
|
|
|
try:
|
|
|
|
slnet_token, slnet_token_expires, user_id = self._api.get_user_id(
|
|
|
|
slid_token
|
|
|
|
)
|
|
|
|
self._api.set_slnet_token(slnet_token)
|
|
|
|
self._api.set_user_id(user_id)
|
|
|
|
self._hass.config_entries.async_update_entry(
|
|
|
|
self._config_entry,
|
|
|
|
data={
|
|
|
|
**self._config_entry.data,
|
|
|
|
DATA_SLNET_TOKEN: slnet_token,
|
|
|
|
DATA_EXPIRES: slnet_token_expires,
|
|
|
|
DATA_USER_ID: user_id,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
except Exception as err: # pylint: disable=broad-except
|
2020-10-17 02:24:08 +00:00
|
|
|
_LOGGER.error("Error updating SLNet token: %s", err)
|
2019-11-26 19:17:11 +00:00
|
|
|
|
|
|
|
def _update_data(self):
|
|
|
|
"""Update StarLine data."""
|
2021-01-13 19:44:24 +00:00
|
|
|
self._check_slnet_token(self._update_interval)
|
2019-11-26 19:17:11 +00:00
|
|
|
self._api.update()
|
|
|
|
|
2021-01-13 19:44:24 +00:00
|
|
|
def _update_obd_data(self):
|
|
|
|
"""Update StarLine OBD data."""
|
|
|
|
self._check_slnet_token(self._update_obd_interval)
|
|
|
|
self._api.update_obd()
|
|
|
|
|
2019-11-26 19:17:11 +00:00
|
|
|
@property
|
|
|
|
def api(self) -> StarlineApi:
|
|
|
|
"""Return the instance of the API."""
|
|
|
|
return self._api
|
|
|
|
|
|
|
|
async def update(self, unused=None):
|
|
|
|
"""Update StarLine data."""
|
|
|
|
await self._hass.async_add_executor_job(self._update_data)
|
|
|
|
|
2021-01-13 19:44:24 +00:00
|
|
|
async def update_obd(self, unused=None):
|
|
|
|
"""Update StarLine OBD data."""
|
|
|
|
await self._hass.async_add_executor_job(self._update_obd_data)
|
|
|
|
|
2019-11-26 19:17:11 +00:00
|
|
|
def set_update_interval(self, interval: int) -> None:
|
|
|
|
"""Set StarLine API update interval."""
|
2020-10-17 02:24:08 +00:00
|
|
|
_LOGGER.debug("Setting update interval: %ds", interval)
|
2019-11-26 19:17:11 +00:00
|
|
|
self._update_interval = interval
|
|
|
|
if self._unsubscribe_auto_updater is not None:
|
|
|
|
self._unsubscribe_auto_updater()
|
|
|
|
|
|
|
|
delta = timedelta(seconds=interval)
|
|
|
|
self._unsubscribe_auto_updater = async_track_time_interval(
|
|
|
|
self._hass, self.update, delta
|
|
|
|
)
|
|
|
|
|
2021-01-13 19:44:24 +00:00
|
|
|
def set_update_obd_interval(self, interval: int) -> None:
|
|
|
|
"""Set StarLine API OBD update interval."""
|
|
|
|
_LOGGER.debug("Setting OBD update interval: %ds", interval)
|
|
|
|
self._update_obd_interval = interval
|
|
|
|
if self._unsubscribe_auto_obd_updater is not None:
|
|
|
|
self._unsubscribe_auto_obd_updater()
|
|
|
|
|
|
|
|
delta = timedelta(seconds=interval)
|
|
|
|
self._unsubscribe_auto_obd_updater = async_track_time_interval(
|
|
|
|
self._hass, self.update_obd, delta
|
|
|
|
)
|
|
|
|
|
2019-11-26 19:17:11 +00:00
|
|
|
def unload(self):
|
|
|
|
"""Unload StarLine API."""
|
2021-03-19 14:26:36 +00:00
|
|
|
_LOGGER.debug("Unloading StarLine API")
|
2019-11-26 19:17:11 +00:00
|
|
|
if self._unsubscribe_auto_updater is not None:
|
|
|
|
self._unsubscribe_auto_updater()
|
|
|
|
self._unsubscribe_auto_updater = None
|
2021-01-13 19:44:24 +00:00
|
|
|
if self._unsubscribe_auto_obd_updater is not None:
|
|
|
|
self._unsubscribe_auto_obd_updater()
|
|
|
|
self._unsubscribe_auto_obd_updater = None
|
2019-11-26 19:17:11 +00:00
|
|
|
|
|
|
|
@staticmethod
|
2021-05-11 05:11:51 +00:00
|
|
|
def device_info(device: StarlineDevice) -> DeviceInfo:
|
2019-11-26 19:17:11 +00:00
|
|
|
"""Device information for entities."""
|
2021-10-28 07:40:01 +00:00
|
|
|
return DeviceInfo(
|
|
|
|
identifiers={(DOMAIN, device.device_id)},
|
|
|
|
manufacturer="StarLine",
|
|
|
|
model=device.typename,
|
|
|
|
name=device.name,
|
|
|
|
sw_version=device.fw_version,
|
|
|
|
)
|
2019-11-26 19:17:11 +00:00
|
|
|
|
|
|
|
@staticmethod
|
2021-03-18 13:31:38 +00:00
|
|
|
def gps_attrs(device: StarlineDevice) -> dict[str, Any]:
|
2019-11-26 19:17:11 +00:00
|
|
|
"""Attributes for device tracker."""
|
|
|
|
return {
|
|
|
|
"updated": datetime.utcfromtimestamp(device.position["ts"]).isoformat(),
|
|
|
|
"online": device.online,
|
|
|
|
}
|
|
|
|
|
|
|
|
@staticmethod
|
2021-03-18 13:31:38 +00:00
|
|
|
def balance_attrs(device: StarlineDevice) -> dict[str, Any]:
|
2019-11-26 19:17:11 +00:00
|
|
|
"""Attributes for balance sensor."""
|
|
|
|
return {
|
|
|
|
"operator": device.balance.get("operator"),
|
|
|
|
"state": device.balance.get("state"),
|
|
|
|
"updated": device.balance.get("ts"),
|
|
|
|
}
|
|
|
|
|
|
|
|
@staticmethod
|
2021-03-18 13:31:38 +00:00
|
|
|
def gsm_attrs(device: StarlineDevice) -> dict[str, Any]:
|
2019-11-26 19:17:11 +00:00
|
|
|
"""Attributes for GSM sensor."""
|
|
|
|
return {
|
|
|
|
"raw": device.gsm_level,
|
|
|
|
"imei": device.imei,
|
|
|
|
"phone": device.phone,
|
|
|
|
"online": device.online,
|
|
|
|
}
|
|
|
|
|
|
|
|
@staticmethod
|
2021-03-18 13:31:38 +00:00
|
|
|
def engine_attrs(device: StarlineDevice) -> dict[str, Any]:
|
2019-11-26 19:17:11 +00:00
|
|
|
"""Attributes for engine switch."""
|
|
|
|
return {
|
|
|
|
"autostart": device.car_state.get("r_start"),
|
|
|
|
"ignition": device.car_state.get("run"),
|
|
|
|
}
|
2021-01-13 19:44:24 +00:00
|
|
|
|
|
|
|
@staticmethod
|
2021-03-18 13:31:38 +00:00
|
|
|
def errors_attrs(device: StarlineDevice) -> dict[str, Any]:
|
2021-01-13 19:44:24 +00:00
|
|
|
"""Attributes for errors sensor."""
|
|
|
|
return {"errors": device.errors.get("errors")}
|