Add Reolink sensor platform (#96323)

* Add Reolink sensor platform

* fix styling

* Add state class

* Add Event connection sensor

* Update homeassistant/components/reolink/sensor.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* Use translation keys

* fix json

* fix json 2

* fix json 3

* Apply suggestions from code review

Co-authored-by: G Johansson <goran.johansson@shiftit.se>

---------

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
Co-authored-by: G Johansson <goran.johansson@shiftit.se>
This commit is contained in:
starkillerOG 2023-07-11 21:36:44 +02:00 committed by GitHub
parent c431fc2297
commit bc9b9048f0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 145 additions and 0 deletions

View File

@ -991,6 +991,7 @@ omit =
homeassistant/components/reolink/light.py
homeassistant/components/reolink/number.py
homeassistant/components/reolink/select.py
homeassistant/components/reolink/sensor.py
homeassistant/components/reolink/siren.py
homeassistant/components/reolink/switch.py
homeassistant/components/reolink/update.py

View File

@ -30,6 +30,7 @@ PLATFORMS = [
Platform.LIGHT,
Platform.NUMBER,
Platform.SELECT,
Platform.SENSOR,
Platform.SIREN,
Platform.SWITCH,
Platform.UPDATE,

View File

@ -432,6 +432,15 @@ class ReolinkHost:
webhook.async_unregister(self._hass, self.webhook_id)
self.webhook_id = None
@property
def event_connection(self) -> str:
"""Return the event connection type."""
if self._webhook_reachable:
return "onvif_push"
if self._long_poll_received:
return "onvif_long_poll"
return "fast_poll"
async def _async_long_polling(self, *_) -> None:
"""Use ONVIF long polling to immediately receive events."""
# This task will be cancelled once _async_stop_long_polling is called

View File

@ -0,0 +1,121 @@
"""Component providing support for Reolink sensors."""
from __future__ import annotations
from collections.abc import Callable
from dataclasses import dataclass
from datetime import date, datetime
from decimal import Decimal
from reolink_aio.api import Host
from homeassistant.components.sensor import (
SensorDeviceClass,
SensorEntity,
SensorEntityDescription,
SensorStateClass,
)
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import EntityCategory
from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.typing import StateType
from . import ReolinkData
from .const import DOMAIN
from .entity import ReolinkHostCoordinatorEntity
@dataclass
class ReolinkHostSensorEntityDescriptionMixin:
"""Mixin values for Reolink host sensor entities."""
value: Callable[[Host], bool]
@dataclass
class ReolinkHostSensorEntityDescription(
SensorEntityDescription, ReolinkHostSensorEntityDescriptionMixin
):
"""A class that describes host sensor entities."""
supported: Callable[[Host], bool] = lambda host: True
HOST_SENSORS = (
ReolinkHostSensorEntityDescription(
key="wifi_signal",
translation_key="wifi_signal",
icon="mdi:wifi",
state_class=SensorStateClass.MEASUREMENT,
entity_category=EntityCategory.DIAGNOSTIC,
entity_registry_enabled_default=False,
value=lambda api: api.wifi_signal,
supported=lambda api: api.supported(None, "wifi") and api.wifi_connection,
),
)
async def async_setup_entry(
hass: HomeAssistant,
config_entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up a Reolink IP Camera."""
reolink_data: ReolinkData = hass.data[DOMAIN][config_entry.entry_id]
entities: list[ReolinkHostSensorEntity | EventConnectionSensorEntity] = [
ReolinkHostSensorEntity(reolink_data, entity_description)
for entity_description in HOST_SENSORS
if entity_description.supported(reolink_data.host.api)
]
entities.append(EventConnectionSensorEntity(reolink_data))
async_add_entities(entities)
class ReolinkHostSensorEntity(ReolinkHostCoordinatorEntity, SensorEntity):
"""Base sensor class for Reolink host sensors."""
entity_description: ReolinkHostSensorEntityDescription
def __init__(
self,
reolink_data: ReolinkData,
entity_description: ReolinkHostSensorEntityDescription,
) -> None:
"""Initialize Reolink binary sensor."""
super().__init__(reolink_data)
self.entity_description = entity_description
self._attr_unique_id = f"{self._host.unique_id}_{entity_description.key}"
@property
def native_value(self) -> StateType | date | datetime | Decimal:
"""Return the value reported by the sensor."""
return self.entity_description.value(self._host.api)
class EventConnectionSensorEntity(ReolinkHostCoordinatorEntity, SensorEntity):
"""Reolink Event connection sensor."""
def __init__(
self,
reolink_data: ReolinkData,
) -> None:
"""Initialize Reolink binary sensor."""
super().__init__(reolink_data)
self.entity_description = SensorEntityDescription(
key="event_connection",
translation_key="event_connection",
icon="mdi:swap-horizontal",
device_class=SensorDeviceClass.ENUM,
entity_category=EntityCategory.DIAGNOSTIC,
entity_registry_enabled_default=False,
options=["onvif_push", "onvif_long_poll", "fast_poll"],
)
self._attr_unique_id = f"{self._host.unique_id}_{self.entity_description.key}"
@property
def native_value(self) -> str:
"""Return the value reported by the sensor."""
return self._host.event_connection

View File

@ -93,6 +93,19 @@
"alwaysonatnight": "Auto & always on at night"
}
}
},
"sensor": {
"event_connection": {
"name": "Event connection",
"state": {
"onvif_push": "ONVIF push",
"onvif_long_poll": "ONVIF long poll",
"fast_poll": "Fast poll"
}
},
"wifi_signal": {
"name": "Wi-Fi signal"
}
}
}
}