2022-01-01 21:23:10 +00:00
|
|
|
"""The unifiprotect integration models."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
2022-01-10 04:37:24 +00:00
|
|
|
from collections.abc import Callable, Coroutine
|
2022-01-01 21:23:10 +00:00
|
|
|
from dataclasses import dataclass
|
2022-06-21 16:17:29 +00:00
|
|
|
from enum import Enum
|
2022-01-10 04:37:24 +00:00
|
|
|
import logging
|
2022-06-22 20:57:21 +00:00
|
|
|
from typing import Any, Generic, TypeVar, Union
|
2022-01-10 04:37:24 +00:00
|
|
|
|
2022-06-22 20:57:21 +00:00
|
|
|
from pyunifiprotect.data import NVR, ProtectAdoptableDeviceModel
|
2022-01-10 04:37:24 +00:00
|
|
|
|
|
|
|
from homeassistant.helpers.entity import EntityDescription
|
|
|
|
|
|
|
|
from .utils import get_nested_attr
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2022-01-01 21:23:10 +00:00
|
|
|
|
2022-06-22 20:57:21 +00:00
|
|
|
T = TypeVar("T", bound=Union[ProtectAdoptableDeviceModel, NVR])
|
2022-01-17 20:51:55 +00:00
|
|
|
|
2022-01-01 21:23:10 +00:00
|
|
|
|
2022-06-21 16:17:29 +00:00
|
|
|
class PermRequired(int, Enum):
|
|
|
|
"""Type of permission level required for entity."""
|
|
|
|
|
|
|
|
NO_WRITE = 1
|
|
|
|
WRITE = 2
|
2022-08-25 23:54:52 +00:00
|
|
|
DELETE = 3
|
2022-06-21 16:17:29 +00:00
|
|
|
|
|
|
|
|
2022-01-01 21:23:10 +00:00
|
|
|
@dataclass
|
2022-03-11 23:57:38 +00:00
|
|
|
class ProtectRequiredKeysMixin(EntityDescription, Generic[T]):
|
2022-01-01 21:23:10 +00:00
|
|
|
"""Mixin for required keys."""
|
|
|
|
|
|
|
|
ufp_required_field: str | None = None
|
|
|
|
ufp_value: str | None = None
|
2022-01-17 20:51:55 +00:00
|
|
|
ufp_value_fn: Callable[[T], Any] | None = None
|
2022-01-13 03:54:22 +00:00
|
|
|
ufp_enabled: str | None = None
|
2022-06-21 16:17:29 +00:00
|
|
|
ufp_perm: PermRequired | None = None
|
2022-01-10 04:37:24 +00:00
|
|
|
|
2022-01-17 20:51:55 +00:00
|
|
|
def get_ufp_value(self, obj: T) -> Any:
|
2022-01-10 04:37:24 +00:00
|
|
|
"""Return value from UniFi Protect device."""
|
|
|
|
if self.ufp_value is not None:
|
|
|
|
return get_nested_attr(obj, self.ufp_value)
|
|
|
|
if self.ufp_value_fn is not None:
|
|
|
|
return self.ufp_value_fn(obj)
|
|
|
|
|
|
|
|
# reminder for future that one is required
|
|
|
|
raise RuntimeError( # pragma: no cover
|
|
|
|
"`ufp_value` or `ufp_value_fn` is required"
|
|
|
|
)
|
|
|
|
|
2022-01-17 20:51:55 +00:00
|
|
|
def get_ufp_enabled(self, obj: T) -> bool:
|
2022-01-13 03:54:22 +00:00
|
|
|
"""Return value from UniFi Protect device."""
|
|
|
|
if self.ufp_enabled is not None:
|
|
|
|
return bool(get_nested_attr(obj, self.ufp_enabled))
|
|
|
|
return True
|
|
|
|
|
2022-01-10 04:37:24 +00:00
|
|
|
|
|
|
|
@dataclass
|
2022-04-26 17:15:47 +00:00
|
|
|
class ProtectSetableKeysMixin(ProtectRequiredKeysMixin[T]):
|
2022-01-17 20:51:55 +00:00
|
|
|
"""Mixin for settable values."""
|
2022-01-10 04:37:24 +00:00
|
|
|
|
|
|
|
ufp_set_method: str | None = None
|
2022-01-17 20:51:55 +00:00
|
|
|
ufp_set_method_fn: Callable[[T, Any], Coroutine[Any, Any, None]] | None = None
|
2022-01-10 04:37:24 +00:00
|
|
|
|
2022-01-17 20:51:55 +00:00
|
|
|
async def ufp_set(self, obj: T, value: Any) -> None:
|
2022-01-10 04:37:24 +00:00
|
|
|
"""Set value for UniFi Protect device."""
|
2022-06-22 20:57:21 +00:00
|
|
|
_LOGGER.debug("Setting %s to %s for %s", self.name, value, obj.display_name)
|
2022-01-10 04:37:24 +00:00
|
|
|
if self.ufp_set_method is not None:
|
|
|
|
await getattr(obj, self.ufp_set_method)(value)
|
|
|
|
elif self.ufp_set_method_fn is not None:
|
|
|
|
await self.ufp_set_method_fn(obj, value)
|