2019-04-03 15:40:03 +00:00
|
|
|
"""Support for the Tank Utility propane monitor."""
|
2017-08-30 20:21:54 +00:00
|
|
|
|
|
|
|
import datetime
|
|
|
|
import logging
|
|
|
|
|
|
|
|
import requests
|
2019-12-13 09:31:53 +00:00
|
|
|
from tank_utility import auth, device as tank_monitor
|
2017-08-30 20:21:54 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2021-03-22 18:47:44 +00:00
|
|
|
from homeassistant.components.sensor import PLATFORM_SCHEMA, SensorEntity
|
2020-09-05 19:09:14 +00:00
|
|
|
from homeassistant.const import CONF_DEVICES, CONF_EMAIL, CONF_PASSWORD, PERCENTAGE
|
2019-11-27 00:02:36 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2017-08-30 20:21:54 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
SCAN_INTERVAL = datetime.timedelta(hours=1)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_EMAIL): cv.string,
|
|
|
|
vol.Required(CONF_PASSWORD): cv.string,
|
|
|
|
vol.Required(CONF_DEVICES): vol.All(cv.ensure_list, vol.Length(min=1)),
|
|
|
|
}
|
|
|
|
)
|
2017-08-30 20:21:54 +00:00
|
|
|
|
|
|
|
SENSOR_TYPE = "tank"
|
|
|
|
SENSOR_ROUNDING_PRECISION = 1
|
|
|
|
SENSOR_ATTRS = [
|
|
|
|
"name",
|
|
|
|
"address",
|
|
|
|
"capacity",
|
|
|
|
"fuelType",
|
|
|
|
"orientation",
|
|
|
|
"status",
|
|
|
|
"time",
|
2019-07-31 19:25:30 +00:00
|
|
|
"time_iso",
|
2017-08-30 20:21:54 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2017-08-30 20:21:54 +00:00
|
|
|
"""Set up the Tank Utility sensor."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2017-08-30 20:21:54 +00:00
|
|
|
email = config.get(CONF_EMAIL)
|
|
|
|
password = config.get(CONF_PASSWORD)
|
|
|
|
devices = config.get(CONF_DEVICES)
|
|
|
|
|
|
|
|
try:
|
2019-12-13 09:31:53 +00:00
|
|
|
token = auth.get_token(email, password)
|
2017-08-30 20:21:54 +00:00
|
|
|
except requests.exceptions.HTTPError as http_error:
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
2019-07-31 19:46:17 +00:00
|
|
|
http_error.response.status_code
|
|
|
|
== requests.codes.unauthorized # pylint: disable=no-member
|
|
|
|
):
|
2017-08-30 20:21:54 +00:00
|
|
|
_LOGGER.error("Invalid credentials")
|
|
|
|
return
|
|
|
|
|
|
|
|
all_sensors = []
|
|
|
|
for device in devices:
|
|
|
|
sensor = TankUtilitySensor(email, password, token, device)
|
|
|
|
all_sensors.append(sensor)
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(all_sensors, True)
|
2017-08-30 20:21:54 +00:00
|
|
|
|
|
|
|
|
2021-03-22 18:47:44 +00:00
|
|
|
class TankUtilitySensor(SensorEntity):
|
2017-08-30 20:21:54 +00:00
|
|
|
"""Representation of a Tank Utility sensor."""
|
|
|
|
|
|
|
|
def __init__(self, email, password, token, device):
|
|
|
|
"""Initialize the sensor."""
|
|
|
|
self._email = email
|
|
|
|
self._password = password
|
|
|
|
self._token = token
|
|
|
|
self._device = device
|
2019-01-24 07:20:20 +00:00
|
|
|
self._state = None
|
2020-01-03 13:47:06 +00:00
|
|
|
self._name = f"Tank Utility {self.device}"
|
2020-09-05 19:09:14 +00:00
|
|
|
self._unit_of_measurement = PERCENTAGE
|
2017-08-30 20:21:54 +00:00
|
|
|
self._attributes = {}
|
|
|
|
|
2018-08-26 19:25:39 +00:00
|
|
|
@property
|
|
|
|
def device(self):
|
|
|
|
"""Return the device identifier."""
|
|
|
|
return self._device
|
|
|
|
|
2017-08-30 20:21:54 +00:00
|
|
|
@property
|
2021-08-11 16:57:50 +00:00
|
|
|
def native_value(self):
|
2017-08-30 20:21:54 +00:00
|
|
|
"""Return the state of the device."""
|
|
|
|
return self._state
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the device."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
2021-08-11 16:57:50 +00:00
|
|
|
def native_unit_of_measurement(self):
|
2017-08-30 20:21:54 +00:00
|
|
|
"""Return the unit of measurement of the device."""
|
|
|
|
return self._unit_of_measurement
|
|
|
|
|
|
|
|
@property
|
2021-03-11 19:16:26 +00:00
|
|
|
def extra_state_attributes(self):
|
2017-08-30 20:21:54 +00:00
|
|
|
"""Return the attributes of the device."""
|
|
|
|
return self._attributes
|
|
|
|
|
|
|
|
def get_data(self):
|
|
|
|
"""Get data from the device.
|
|
|
|
|
|
|
|
Flatten dictionary to map device to map of device data.
|
|
|
|
|
|
|
|
"""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2017-08-30 20:21:54 +00:00
|
|
|
data = {}
|
|
|
|
try:
|
2019-12-13 09:31:53 +00:00
|
|
|
data = tank_monitor.get_device_data(self._token, self.device)
|
2017-08-30 20:21:54 +00:00
|
|
|
except requests.exceptions.HTTPError as http_error:
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
2019-07-31 19:46:17 +00:00
|
|
|
http_error.response.status_code
|
|
|
|
== requests.codes.unauthorized # pylint: disable=no-member
|
2019-12-10 18:25:22 +00:00
|
|
|
or http_error.response.status_code
|
|
|
|
== requests.codes.bad_request # pylint: disable=no-member
|
2019-07-31 19:46:17 +00:00
|
|
|
):
|
2017-08-30 20:21:54 +00:00
|
|
|
_LOGGER.info("Getting new token")
|
2019-12-13 09:31:53 +00:00
|
|
|
self._token = auth.get_token(self._email, self._password, force=True)
|
|
|
|
data = tank_monitor.get_device_data(self._token, self.device)
|
2017-08-30 20:21:54 +00:00
|
|
|
else:
|
|
|
|
raise http_error
|
|
|
|
data.update(data.pop("device", {}))
|
|
|
|
data.update(data.pop("lastReading", {}))
|
|
|
|
return data
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Set the device state and attributes."""
|
|
|
|
data = self.get_data()
|
|
|
|
self._state = round(data[SENSOR_TYPE], SENSOR_ROUNDING_PRECISION)
|
|
|
|
self._attributes = {k: v for k, v in data.items() if k in SENSOR_ATTRS}
|