2021-01-13 15:09:05 +00:00
|
|
|
"""Config flow for foscam integration."""
|
|
|
|
from libpyfoscam import FoscamCamera
|
2021-02-05 21:39:31 +00:00
|
|
|
from libpyfoscam.foscam import (
|
|
|
|
ERROR_FOSCAM_AUTH,
|
|
|
|
ERROR_FOSCAM_UNAVAILABLE,
|
|
|
|
FOSCAM_SUCCESS,
|
|
|
|
)
|
2021-01-13 15:09:05 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant import config_entries, exceptions
|
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_HOST,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_PASSWORD,
|
|
|
|
CONF_PORT,
|
|
|
|
CONF_USERNAME,
|
|
|
|
)
|
|
|
|
from homeassistant.data_entry_flow import AbortFlow
|
|
|
|
|
2021-03-02 08:02:04 +00:00
|
|
|
from .const import CONF_RTSP_PORT, CONF_STREAM, DOMAIN, LOGGER
|
2021-01-13 15:09:05 +00:00
|
|
|
|
|
|
|
STREAMS = ["Main", "Sub"]
|
|
|
|
|
|
|
|
DEFAULT_PORT = 88
|
2021-02-05 21:39:31 +00:00
|
|
|
DEFAULT_RTSP_PORT = 554
|
2021-01-13 15:09:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
DATA_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_HOST): str,
|
|
|
|
vol.Required(CONF_PORT, default=DEFAULT_PORT): int,
|
|
|
|
vol.Required(CONF_USERNAME): str,
|
|
|
|
vol.Required(CONF_PASSWORD): str,
|
|
|
|
vol.Required(CONF_STREAM, default=STREAMS[0]): vol.In(STREAMS),
|
2021-02-05 21:39:31 +00:00
|
|
|
vol.Required(CONF_RTSP_PORT, default=DEFAULT_RTSP_PORT): int,
|
2021-01-13 15:09:05 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class ConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
|
|
|
|
"""Handle a config flow for foscam."""
|
|
|
|
|
2021-02-05 21:39:31 +00:00
|
|
|
VERSION = 2
|
2021-01-13 15:09:05 +00:00
|
|
|
|
|
|
|
async def _validate_and_create(self, data):
|
|
|
|
"""Validate the user input allows us to connect.
|
|
|
|
|
|
|
|
Data has the keys from DATA_SCHEMA with values provided by the user.
|
|
|
|
"""
|
2021-05-11 20:00:12 +00:00
|
|
|
self._async_abort_entries_match(
|
|
|
|
{CONF_HOST: data[CONF_HOST], CONF_PORT: data[CONF_PORT]}
|
|
|
|
)
|
2021-02-05 21:39:31 +00:00
|
|
|
|
2021-01-13 15:09:05 +00:00
|
|
|
camera = FoscamCamera(
|
|
|
|
data[CONF_HOST],
|
|
|
|
data[CONF_PORT],
|
|
|
|
data[CONF_USERNAME],
|
|
|
|
data[CONF_PASSWORD],
|
|
|
|
verbose=False,
|
|
|
|
)
|
|
|
|
|
|
|
|
# Validate data by sending a request to the camera
|
2021-02-05 21:39:31 +00:00
|
|
|
ret, _ = await self.hass.async_add_executor_job(camera.get_product_all_info)
|
2021-01-13 15:09:05 +00:00
|
|
|
|
|
|
|
if ret == ERROR_FOSCAM_UNAVAILABLE:
|
|
|
|
raise CannotConnect
|
|
|
|
|
|
|
|
if ret == ERROR_FOSCAM_AUTH:
|
|
|
|
raise InvalidAuth
|
|
|
|
|
2021-02-05 21:39:31 +00:00
|
|
|
if ret != FOSCAM_SUCCESS:
|
|
|
|
LOGGER.error(
|
|
|
|
"Unexpected error code from camera %s:%s: %s",
|
|
|
|
data[CONF_HOST],
|
|
|
|
data[CONF_PORT],
|
|
|
|
ret,
|
|
|
|
)
|
|
|
|
raise InvalidResponse
|
|
|
|
|
|
|
|
# Try to get camera name (only possible with admin account)
|
|
|
|
ret, response = await self.hass.async_add_executor_job(camera.get_dev_info)
|
2021-01-13 15:09:05 +00:00
|
|
|
|
2021-02-05 21:39:31 +00:00
|
|
|
dev_name = response.get(
|
|
|
|
"devName", f"Foscam {data[CONF_HOST]}:{data[CONF_PORT]}"
|
|
|
|
)
|
|
|
|
|
|
|
|
name = data.pop(CONF_NAME, dev_name)
|
2021-01-13 15:09:05 +00:00
|
|
|
|
|
|
|
return self.async_create_entry(title=name, data=data)
|
|
|
|
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
|
|
"""Handle the initial step."""
|
|
|
|
errors = {}
|
|
|
|
|
|
|
|
if user_input is not None:
|
|
|
|
try:
|
|
|
|
return await self._validate_and_create(user_input)
|
|
|
|
|
|
|
|
except CannotConnect:
|
|
|
|
errors["base"] = "cannot_connect"
|
|
|
|
|
|
|
|
except InvalidAuth:
|
|
|
|
errors["base"] = "invalid_auth"
|
|
|
|
|
2021-02-05 21:39:31 +00:00
|
|
|
except InvalidResponse:
|
|
|
|
errors["base"] = "invalid_response"
|
|
|
|
|
2021-01-13 15:09:05 +00:00
|
|
|
except AbortFlow:
|
|
|
|
raise
|
|
|
|
|
|
|
|
except Exception: # pylint: disable=broad-except
|
|
|
|
LOGGER.exception("Unexpected exception")
|
|
|
|
errors["base"] = "unknown"
|
|
|
|
|
|
|
|
return self.async_show_form(
|
|
|
|
step_id="user", data_schema=DATA_SCHEMA, errors=errors
|
|
|
|
)
|
|
|
|
|
|
|
|
async def async_step_import(self, import_config):
|
|
|
|
"""Handle config import from yaml."""
|
|
|
|
try:
|
|
|
|
return await self._validate_and_create(import_config)
|
|
|
|
|
|
|
|
except CannotConnect:
|
2021-03-19 14:26:36 +00:00
|
|
|
LOGGER.error("Error importing foscam platform config: cannot connect")
|
2021-01-13 15:09:05 +00:00
|
|
|
return self.async_abort(reason="cannot_connect")
|
|
|
|
|
|
|
|
except InvalidAuth:
|
2021-03-19 14:26:36 +00:00
|
|
|
LOGGER.error("Error importing foscam platform config: invalid auth")
|
2021-01-13 15:09:05 +00:00
|
|
|
return self.async_abort(reason="invalid_auth")
|
|
|
|
|
2021-02-05 21:39:31 +00:00
|
|
|
except InvalidResponse:
|
|
|
|
LOGGER.exception(
|
2021-03-19 14:26:36 +00:00
|
|
|
"Error importing foscam platform config: invalid response from camera"
|
2021-02-05 21:39:31 +00:00
|
|
|
)
|
|
|
|
return self.async_abort(reason="invalid_response")
|
|
|
|
|
2021-01-13 15:09:05 +00:00
|
|
|
except AbortFlow:
|
|
|
|
raise
|
|
|
|
|
|
|
|
except Exception: # pylint: disable=broad-except
|
|
|
|
LOGGER.exception(
|
2021-03-19 14:26:36 +00:00
|
|
|
"Error importing foscam platform config: unexpected exception"
|
2021-01-13 15:09:05 +00:00
|
|
|
)
|
|
|
|
return self.async_abort(reason="unknown")
|
|
|
|
|
|
|
|
|
|
|
|
class CannotConnect(exceptions.HomeAssistantError):
|
|
|
|
"""Error to indicate we cannot connect."""
|
|
|
|
|
|
|
|
|
|
|
|
class InvalidAuth(exceptions.HomeAssistantError):
|
|
|
|
"""Error to indicate there is invalid auth."""
|
2021-02-05 21:39:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
class InvalidResponse(exceptions.HomeAssistantError):
|
|
|
|
"""Error to indicate there is invalid response."""
|