2019-04-03 15:40:03 +00:00
|
|
|
"""Support for Xeoma Cameras."""
|
2021-08-11 00:33:06 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2018-01-25 22:48:38 +00:00
|
|
|
import logging
|
|
|
|
|
2019-11-25 23:39:54 +00:00
|
|
|
from pyxeoma.xeoma import Xeoma, XeomaError
|
2019-12-09 13:58:51 +00:00
|
|
|
import voluptuous as vol
|
2018-01-25 22:48:38 +00:00
|
|
|
|
|
|
|
from homeassistant.components.camera import PLATFORM_SCHEMA, Camera
|
2019-07-31 19:25:30 +00:00
|
|
|
from homeassistant.const import CONF_HOST, CONF_NAME, CONF_PASSWORD, CONF_USERNAME
|
2022-01-03 12:22:41 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2018-01-25 22:48:38 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv
|
2022-01-03 12:22:41 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
2018-01-25 22:48:38 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_CAMERAS = "cameras"
|
|
|
|
CONF_HIDE = "hide"
|
|
|
|
CONF_IMAGE_NAME = "image_name"
|
|
|
|
CONF_NEW_VERSION = "new_version"
|
|
|
|
CONF_VIEWER_PASSWORD = "viewer_password"
|
|
|
|
CONF_VIEWER_USERNAME = "viewer_username"
|
|
|
|
|
|
|
|
CAMERAS_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_IMAGE_NAME): cv.string,
|
|
|
|
vol.Optional(CONF_HIDE, default=False): cv.boolean,
|
|
|
|
vol.Optional(CONF_NAME): cv.string,
|
|
|
|
},
|
|
|
|
required=False,
|
|
|
|
)
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Optional(CONF_CAMERAS): vol.Schema(
|
|
|
|
vol.All(cv.ensure_list, [CAMERAS_SCHEMA])
|
|
|
|
),
|
|
|
|
vol.Optional(CONF_NEW_VERSION, default=True): cv.boolean,
|
|
|
|
vol.Optional(CONF_PASSWORD): cv.string,
|
|
|
|
vol.Optional(CONF_USERNAME): cv.string,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-01-03 12:22:41 +00:00
|
|
|
async def async_setup_platform(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config: ConfigType,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
|
|
) -> None:
|
2018-01-25 22:48:38 +00:00
|
|
|
"""Discover and setup Xeoma Cameras."""
|
|
|
|
|
|
|
|
host = config[CONF_HOST]
|
|
|
|
login = config.get(CONF_USERNAME)
|
|
|
|
password = config.get(CONF_PASSWORD)
|
|
|
|
|
2018-02-06 18:47:38 +00:00
|
|
|
xeoma = Xeoma(host, login, password)
|
2018-01-25 22:48:38 +00:00
|
|
|
|
|
|
|
try:
|
2018-03-23 17:22:01 +00:00
|
|
|
await xeoma.async_test_connection()
|
|
|
|
discovered_image_names = await xeoma.async_get_image_names()
|
2018-01-25 22:48:38 +00:00
|
|
|
discovered_cameras = [
|
|
|
|
{
|
|
|
|
CONF_IMAGE_NAME: image_name,
|
|
|
|
CONF_HIDE: False,
|
2018-02-06 18:47:38 +00:00
|
|
|
CONF_NAME: image_name,
|
|
|
|
CONF_VIEWER_USERNAME: username,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_VIEWER_PASSWORD: pw,
|
2018-01-25 22:48:38 +00:00
|
|
|
}
|
2018-02-06 18:47:38 +00:00
|
|
|
for image_name, username, pw in discovered_image_names
|
2018-01-25 22:48:38 +00:00
|
|
|
]
|
|
|
|
|
2018-02-17 09:29:14 +00:00
|
|
|
for cam in config.get(CONF_CAMERAS, []):
|
2018-01-25 22:48:38 +00:00
|
|
|
camera = next(
|
2019-07-31 19:25:30 +00:00
|
|
|
(
|
|
|
|
dc
|
|
|
|
for dc in discovered_cameras
|
|
|
|
if dc[CONF_IMAGE_NAME] == cam[CONF_IMAGE_NAME]
|
|
|
|
),
|
|
|
|
None,
|
|
|
|
)
|
2018-01-25 22:48:38 +00:00
|
|
|
|
|
|
|
if camera is not None:
|
|
|
|
if CONF_NAME in cam:
|
|
|
|
camera[CONF_NAME] = cam[CONF_NAME]
|
|
|
|
if CONF_HIDE in cam:
|
|
|
|
camera[CONF_HIDE] = cam[CONF_HIDE]
|
|
|
|
|
|
|
|
cameras = list(filter(lambda c: not c[CONF_HIDE], discovered_cameras))
|
2018-08-24 14:37:30 +00:00
|
|
|
async_add_entities(
|
2019-07-31 19:25:30 +00:00
|
|
|
[
|
|
|
|
XeomaCamera(
|
|
|
|
xeoma,
|
|
|
|
camera[CONF_IMAGE_NAME],
|
|
|
|
camera[CONF_NAME],
|
|
|
|
camera[CONF_VIEWER_USERNAME],
|
|
|
|
camera[CONF_VIEWER_PASSWORD],
|
|
|
|
)
|
|
|
|
for camera in cameras
|
|
|
|
]
|
|
|
|
)
|
2018-01-25 22:48:38 +00:00
|
|
|
except XeomaError as err:
|
|
|
|
_LOGGER.error("Error: %s", err.message)
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
|
|
class XeomaCamera(Camera):
|
|
|
|
"""Implementation of a Xeoma camera."""
|
|
|
|
|
2018-02-06 18:47:38 +00:00
|
|
|
def __init__(self, xeoma, image, name, username, password):
|
2018-01-25 22:48:38 +00:00
|
|
|
"""Initialize a Xeoma camera."""
|
|
|
|
super().__init__()
|
|
|
|
self._xeoma = xeoma
|
|
|
|
self._name = name
|
|
|
|
self._image = image
|
2018-02-06 18:47:38 +00:00
|
|
|
self._username = username
|
|
|
|
self._password = password
|
2018-01-25 22:48:38 +00:00
|
|
|
self._last_image = None
|
|
|
|
|
2021-08-11 00:33:06 +00:00
|
|
|
async def async_camera_image(
|
|
|
|
self, width: int | None = None, height: int | None = None
|
|
|
|
) -> bytes | None:
|
2018-01-25 22:48:38 +00:00
|
|
|
"""Return a still image response from the camera."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2018-01-25 22:48:38 +00:00
|
|
|
try:
|
2018-03-23 17:22:01 +00:00
|
|
|
image = await self._xeoma.async_get_camera_image(
|
2019-07-31 19:25:30 +00:00
|
|
|
self._image, self._username, self._password
|
|
|
|
)
|
2018-01-25 22:48:38 +00:00
|
|
|
self._last_image = image
|
|
|
|
except XeomaError as err:
|
|
|
|
_LOGGER.error("Error fetching image: %s", err.message)
|
|
|
|
|
|
|
|
return self._last_image
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of this device."""
|
|
|
|
return self._name
|