Add connect and disconnect functions

This commit is contained in:
cyr-ius
2020-03-29 15:57:24 +02:00
parent 5114707300
commit ff4d19d6d9
2 changed files with 70 additions and 73 deletions

View File

@@ -3,9 +3,9 @@ import asyncio
import logging import logging
import async_timeout import async_timeout
from roomba import Roomba from roomba import Roomba, RoombaConnectionError
from homeassistant import config_entries from homeassistant import config_entries, exceptions
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME
from .const import COMPONENTS, CONF_CERT, CONF_CONTINUOUS, CONF_DELAY, DOMAIN from .const import COMPONENTS, CONF_CERT, CONF_CONTINUOUS, CONF_DELAY, DOMAIN
@@ -45,48 +45,35 @@ async def async_setup_entry(hass, config_entry):
if DOMAIN not in hass.data: if DOMAIN not in hass.data:
hass.data[DOMAIN] = {} hass.data[DOMAIN] = {}
config_entry.add_update_listener(async_connect) if "roomba" not in hass.data[DOMAIN]:
if not await async_connect(hass, config_entry):
return False roomba = Roomba(
address=config_entry.data[CONF_HOST],
blid=config_entry.data[CONF_USERNAME],
password=config_entry.data[CONF_PASSWORD],
cert_name=config_entry.options[CONF_CERT],
continuous=config_entry.options[CONF_CONTINUOUS],
delay=config_entry.options[CONF_DELAY],
)
if not await async_connect_or_timeout(hass, roomba):
return False
for component in COMPONENTS: for component in COMPONENTS:
hass.async_create_task( hass.async_create_task(
hass.config_entries.async_forward_entry_setup(config_entry, component) hass.config_entries.async_forward_entry_setup(config_entry, component)
) )
config_entry.add_update_listener(async_update_options)
return True return True
async def async_unload_entry(hass, config_entry): async def async_connect_or_timeout(hass, roomba):
"""Unload a config entry."""
for component in COMPONENTS:
hass.async_create_task(
hass.config_entries.async_forward_entry_unload(config_entry, component)
)
roomba = hass.data[DOMAIN]["roomba"]
await hass.async_add_job(roomba.disconnect)
return True
async def async_connect(hass, config_entry):
"""Connect to vacuum.""" """Connect to vacuum."""
# Check if triggered listener
if "roomba" in hass.data[DOMAIN]:
await hass.async_add_job(hass.data[DOMAIN]["roomba"].disconnect)
await asyncio.sleep(1)
roomba = Roomba(
address=config_entry.data[CONF_HOST],
blid=config_entry.data[CONF_USERNAME],
password=config_entry.data[CONF_PASSWORD],
cert_name=config_entry.options[CONF_CERT],
continuous=config_entry.options[CONF_CONTINUOUS],
delay=config_entry.options[CONF_DELAY],
)
try: try:
name = None hass.data[DOMAIN]["name"] = name = None
with async_timeout.timeout(9): with async_timeout.timeout(10):
await hass.async_add_job(roomba.connect) await hass.async_add_job(roomba.connect)
while not roomba.roomba_connected or name is None: while not roomba.roomba_connected or name is None:
# Waiting for connection and check datas ready # Waiting for connection and check datas ready
@@ -96,11 +83,47 @@ async def async_connect(hass, config_entry):
.get("name", None) .get("name", None)
) )
await asyncio.sleep(0.5) await asyncio.sleep(0.5)
hass.data[DOMAIN]["roomba"] = roomba hass.data[DOMAIN]["roomba"] = roomba
hass.data[DOMAIN]["name"] = name
except RoombaConnectionError:
_LOGGER.error("Error to connect to vacuum")
raise CannotConnect
except asyncio.TimeoutError: except asyncio.TimeoutError:
# api looping if user or password incorrect and roomba exist # api looping if user or password incorrect and roomba exist
_LOGGER.error("Timeout exceeded") await async_disconnect_or_timeout(hass, roomba)
await hass.async_add_job(roomba.disconnect) _LOGGER.exception("Timeout expired, user or password incorrect")
return False raise InvalidAuth
return True return True
async def async_disconnect_or_timeout(hass, roomba):
"""Disconnect to vacuum."""
await hass.async_add_job(roomba.disconnect)
await asyncio.sleep(1)
return True
async def async_update_options(hass, config_entry):
"""Update options."""
roomba = hass.data[DOMAIN]["roomba"]
await async_disconnect_or_timeout(hass, roomba)
await async_connect_or_timeout(hass, roomba)
async def async_unload_entry(hass, config_entry):
"""Unload a config entry."""
for component in COMPONENTS:
hass.async_create_task(
hass.config_entries.async_forward_entry_unload(config_entry, component)
)
roomba = hass.data[DOMAIN]["roomba"]
return await async_disconnect_or_timeout(hass, roomba)
class CannotConnect(exceptions.HomeAssistantError):
"""Error to indicate we cannot connect."""
class InvalidAuth(exceptions.HomeAssistantError):
"""Error to indicate there is invalid auth."""

View File

@@ -1,15 +1,14 @@
"""Config flow to configure demo component.""" """Config flow to configure demo component."""
import asyncio
import logging import logging
import async_timeout from roomba import Roomba
from roomba import Roomba, RoombaConnectionError
import voluptuous as vol import voluptuous as vol
from homeassistant import config_entries, core, exceptions from homeassistant import config_entries, core
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME
from homeassistant.core import callback from homeassistant.core import callback
from . import CannotConnect, InvalidAuth, async_connect_or_timeout
from .const import ( from .const import (
CONF_CERT, CONF_CERT,
CONF_CONTINUOUS, CONF_CONTINUOUS,
@@ -39,7 +38,7 @@ async def validate_input(hass: core.HomeAssistant, data):
Data has the keys from DATA_SCHEMA with values provided by the user. Data has the keys from DATA_SCHEMA with values provided by the user.
""" """
api = Roomba( roomba = Roomba(
address=data[CONF_HOST], address=data[CONF_HOST],
blid=data[CONF_USERNAME], blid=data[CONF_USERNAME],
password=data[CONF_PASSWORD], password=data[CONF_PASSWORD],
@@ -47,26 +46,8 @@ async def validate_input(hass: core.HomeAssistant, data):
continuous=data[CONF_CONTINUOUS], continuous=data[CONF_CONTINUOUS],
delay=data[CONF_DELAY], delay=data[CONF_DELAY],
) )
try:
name = None
with async_timeout.timeout(10):
await hass.async_add_job(api.connect)
while not api.roomba_connected or name is None:
name = (
api.master_state.get("state", {})
.get("reported", {})
.get("name", None)
)
await asyncio.sleep(0.5)
hass.data[DOMAIN]["roomba"] = api
except RoombaConnectionError:
raise CannotConnect
except asyncio.TimeoutError:
# Api looping if user or password incorrect and roomba exist
await hass.async_add_job(api.disconnect)
raise InvalidAuth
return {"title": name} await async_connect_or_timeout(hass, roomba)
class RoombaConfigFlow(config_entries.ConfigFlow, domain=DOMAIN): class RoombaConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
@@ -93,17 +74,18 @@ class RoombaConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
if user_input is not None: if user_input is not None:
try: try:
info = await validate_input(self.hass, user_input) await validate_input(self.hass, user_input)
except CannotConnect: except CannotConnect:
errors = {"base": "cannot_connect"} errors = {"base": "cannot_connect"}
except InvalidAuth: except InvalidAuth:
errors = {"base": "invalid_auth"} errors = {"base": "invalid_auth"}
except Exception: # pylint: disable=broad-except except Exception: # pylint: disable=broad-except
_LOGGER.exception("Unexpected exception") errors = {"base": "unknown"}
errors["base"] = "unknown"
if "base" not in errors: if "base" not in errors:
return self.async_create_entry(title=info["title"], data=user_input) return self.async_create_entry(
title=self.hass.data[DOMAIN]["name"], data=user_input
)
# If there was no user input, do not show the errors. # If there was no user input, do not show the errors.
if user_input is None: if user_input is None:
@@ -149,11 +131,3 @@ class OptionsFlowHandler(config_entries.OptionsFlow):
} }
), ),
) )
class CannotConnect(exceptions.HomeAssistantError):
"""Error to indicate we cannot connect."""
class InvalidAuth(exceptions.HomeAssistantError):
"""Error to indicate there is invalid auth."""