2019-02-13 20:21:14 +00:00
|
|
|
"""Support for Waterfurnaces."""
|
2018-01-20 21:51:59 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
import logging
|
|
|
|
import threading
|
2019-12-09 13:47:53 +00:00
|
|
|
import time
|
2018-01-20 21:51:59 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
2019-10-11 16:30:27 +00:00
|
|
|
from waterfurnace.waterfurnace import WaterFurnace, WFCredentialError, WFException
|
2018-01-20 21:51:59 +00:00
|
|
|
|
2019-12-09 13:47:53 +00:00
|
|
|
from homeassistant.const import CONF_PASSWORD, CONF_USERNAME, EVENT_HOMEASSISTANT_STOP
|
2018-01-20 21:51:59 +00:00
|
|
|
from homeassistant.core import callback
|
2019-12-09 13:47:53 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv, discovery
|
2018-01-20 21:51:59 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN = "waterfurnace"
|
2020-04-05 14:01:41 +00:00
|
|
|
UPDATE_TOPIC = f"{DOMAIN}_update"
|
2018-01-20 21:51:59 +00:00
|
|
|
SCAN_INTERVAL = timedelta(seconds=10)
|
2018-08-14 11:49:04 +00:00
|
|
|
ERROR_INTERVAL = timedelta(seconds=300)
|
|
|
|
MAX_FAILS = 10
|
2019-07-31 19:25:30 +00:00
|
|
|
NOTIFICATION_ID = "waterfurnace_website_notification"
|
|
|
|
NOTIFICATION_TITLE = "WaterFurnace website status"
|
|
|
|
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
DOMAIN: vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_PASSWORD): cv.string,
|
|
|
|
vol.Required(CONF_USERNAME): cv.string,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
2018-01-20 21:51:59 +00:00
|
|
|
|
|
|
|
|
|
|
|
def setup(hass, base_config):
|
2018-08-19 20:29:08 +00:00
|
|
|
"""Set up waterfurnace platform."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2018-01-20 21:51:59 +00:00
|
|
|
config = base_config.get(DOMAIN)
|
|
|
|
|
|
|
|
username = config.get(CONF_USERNAME)
|
|
|
|
password = config.get(CONF_PASSWORD)
|
|
|
|
|
2019-10-11 16:30:27 +00:00
|
|
|
wfconn = WaterFurnace(username, password)
|
2018-01-20 21:51:59 +00:00
|
|
|
# NOTE(sdague): login will throw an exception if this doesn't
|
|
|
|
# work, which will abort the setup.
|
|
|
|
try:
|
|
|
|
wfconn.login()
|
2019-10-11 16:30:27 +00:00
|
|
|
except WFCredentialError:
|
2020-07-05 21:04:19 +00:00
|
|
|
_LOGGER.error("Invalid credentials for waterfurnace login")
|
2018-01-20 21:51:59 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
hass.data[DOMAIN] = WaterFurnaceData(hass, wfconn)
|
|
|
|
hass.data[DOMAIN].start()
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
discovery.load_platform(hass, "sensor", DOMAIN, {}, config)
|
2018-01-20 21:51:59 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
class WaterFurnaceData(threading.Thread):
|
|
|
|
"""WaterFurnace Data collector.
|
|
|
|
|
|
|
|
This is implemented as a dedicated thread polling a websocket in a
|
|
|
|
tight loop. The websocket will shut itself from the server side if
|
|
|
|
a packet is not sent at least every 30 seconds. The reading is
|
|
|
|
cheap, the login is less cheap, so keeping this open and polling
|
|
|
|
on a very regular cadence is actually the least io intensive thing
|
|
|
|
to do.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, hass, client):
|
|
|
|
"""Initialize the data object."""
|
|
|
|
super().__init__()
|
|
|
|
self.hass = hass
|
|
|
|
self.client = client
|
2018-12-05 12:03:27 +00:00
|
|
|
self.unit = self.client.gwid
|
2018-01-20 21:51:59 +00:00
|
|
|
self.data = None
|
|
|
|
self._shutdown = False
|
2018-08-14 11:49:04 +00:00
|
|
|
self._fails = 0
|
|
|
|
|
|
|
|
def _reconnect(self):
|
|
|
|
"""Reconnect on a failure."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2018-08-14 11:49:04 +00:00
|
|
|
self._fails += 1
|
|
|
|
if self._fails > MAX_FAILS:
|
2020-07-05 21:04:19 +00:00
|
|
|
_LOGGER.error("Failed to refresh login credentials. Thread stopped")
|
2018-08-14 11:49:04 +00:00
|
|
|
self.hass.components.persistent_notification.create(
|
|
|
|
"Error:<br/>Connection to waterfurnace website failed "
|
2020-07-05 21:04:19 +00:00
|
|
|
"the maximum number of times. Thread has stopped",
|
2018-08-14 11:49:04 +00:00
|
|
|
title=NOTIFICATION_TITLE,
|
2019-07-31 19:25:30 +00:00
|
|
|
notification_id=NOTIFICATION_ID,
|
|
|
|
)
|
2018-08-14 11:49:04 +00:00
|
|
|
|
|
|
|
self._shutdown = True
|
|
|
|
return
|
|
|
|
|
|
|
|
# sleep first before the reconnect attempt
|
|
|
|
_LOGGER.debug("Sleeping for fail # %s", self._fails)
|
|
|
|
time.sleep(self._fails * ERROR_INTERVAL.seconds)
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.client.login()
|
|
|
|
self.data = self.client.read()
|
2019-10-11 16:30:27 +00:00
|
|
|
except WFException:
|
2018-08-14 11:49:04 +00:00
|
|
|
_LOGGER.exception("Failed to reconnect attempt %s", self._fails)
|
|
|
|
else:
|
|
|
|
_LOGGER.debug("Reconnected to furnace")
|
|
|
|
self._fails = 0
|
2018-01-20 21:51:59 +00:00
|
|
|
|
|
|
|
def run(self):
|
|
|
|
"""Thread run loop."""
|
2018-02-06 11:12:35 +00:00
|
|
|
|
2018-01-20 21:51:59 +00:00
|
|
|
@callback
|
|
|
|
def register():
|
|
|
|
"""Connect to hass for shutdown."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2018-01-20 21:51:59 +00:00
|
|
|
def shutdown(event):
|
|
|
|
"""Shutdown the thread."""
|
2020-07-05 21:04:19 +00:00
|
|
|
_LOGGER.debug("Signaled to shutdown")
|
2018-01-20 21:51:59 +00:00
|
|
|
self._shutdown = True
|
|
|
|
self.join()
|
|
|
|
|
|
|
|
self.hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, shutdown)
|
|
|
|
|
|
|
|
self.hass.add_job(register)
|
|
|
|
|
|
|
|
# This does a tight loop in sending read calls to the
|
|
|
|
# websocket. That's a blocking call, which returns pretty
|
|
|
|
# quickly (1 second). It's important that we do this
|
|
|
|
# frequently though, because if we don't call the websocket at
|
|
|
|
# least every 30 seconds the server side closes the
|
|
|
|
# connection.
|
|
|
|
while True:
|
|
|
|
if self._shutdown:
|
|
|
|
_LOGGER.debug("Graceful shutdown")
|
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.data = self.client.read()
|
|
|
|
|
2019-10-11 16:30:27 +00:00
|
|
|
except WFException:
|
2018-02-06 11:12:35 +00:00
|
|
|
# WFExceptions are things the WF library understands
|
|
|
|
# that pretty much can all be solved by logging in and
|
|
|
|
# back out again.
|
|
|
|
_LOGGER.exception("Failed to read data, attempting to recover")
|
2018-08-14 11:49:04 +00:00
|
|
|
self._reconnect()
|
2018-01-20 21:51:59 +00:00
|
|
|
|
|
|
|
else:
|
|
|
|
self.hass.helpers.dispatcher.dispatcher_send(UPDATE_TOPIC)
|
|
|
|
time.sleep(SCAN_INTERVAL.seconds)
|