Add QNAP QSW Button platform (#70980)

Co-authored-by: J. Nick Koston <nick@koston.org>
This commit is contained in:
Álvaro Fernández Rojas 2022-05-14 02:24:29 +02:00 committed by GitHub
parent e8a8d35289
commit abe78b1212
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 122 additions and 1 deletions

View File

@ -11,7 +11,7 @@ from homeassistant.helpers import aiohttp_client
from .const import DOMAIN from .const import DOMAIN
from .coordinator import QswUpdateCoordinator from .coordinator import QswUpdateCoordinator
PLATFORMS: list[Platform] = [Platform.BINARY_SENSOR, Platform.SENSOR] PLATFORMS: list[Platform] = [Platform.BINARY_SENSOR, Platform.BUTTON, Platform.SENSOR]
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:

View File

@ -0,0 +1,77 @@
"""Support for the QNAP QSW buttons."""
from __future__ import annotations
from collections.abc import Awaitable, Callable
from dataclasses import dataclass
from typing import Final
from aioqsw.localapi import QnapQswApi
from homeassistant.components.button import (
ButtonDeviceClass,
ButtonEntity,
ButtonEntityDescription,
)
from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity import EntityCategory
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from .const import DOMAIN, QSW_REBOOT
from .coordinator import QswUpdateCoordinator
from .entity import QswEntity
@dataclass
class QswButtonDescriptionMixin:
"""Mixin to describe a Button entity."""
press_action: Callable[[QnapQswApi], Awaitable[bool]]
@dataclass
class QswButtonDescription(ButtonEntityDescription, QswButtonDescriptionMixin):
"""Class to describe a Button entity."""
BUTTON_TYPES: Final[tuple[QswButtonDescription, ...]] = (
QswButtonDescription(
device_class=ButtonDeviceClass.RESTART,
entity_category=EntityCategory.CONFIG,
key=QSW_REBOOT,
name="Reboot",
press_action=lambda qsw: qsw.reboot(),
),
)
async def async_setup_entry(
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
) -> None:
"""Add QNAP QSW buttons from a config_entry."""
coordinator: QswUpdateCoordinator = hass.data[DOMAIN][entry.entry_id]
async_add_entities(
QswButton(coordinator, description, entry) for description in BUTTON_TYPES
)
class QswButton(QswEntity, ButtonEntity):
"""Define a QNAP QSW button."""
entity_description: QswButtonDescription
def __init__(
self,
coordinator: QswUpdateCoordinator,
description: QswButtonDescription,
entry: ConfigEntry,
) -> None:
"""Initialize."""
super().__init__(coordinator, entry)
self._attr_name = f"{self.product} {description.name}"
self._attr_unique_id = f"{entry.unique_id}_{description.key}"
self.entity_description = description
async def async_press(self) -> None:
"""Triggers the QNAP QSW button action."""
await self.entity_description.press_action(self.coordinator.qsw)

View File

@ -10,4 +10,5 @@ MANUFACTURER: Final = "QNAP"
RPM: Final = "rpm" RPM: Final = "rpm"
QSW_REBOOT = "reboot"
QSW_TIMEOUT_SEC: Final = 25 QSW_TIMEOUT_SEC: Final = 25

View File

@ -0,0 +1,37 @@
"""The sensor tests for the QNAP QSW platform."""
from unittest.mock import patch
from homeassistant.components.button.const import DOMAIN as BUTTON_DOMAIN, SERVICE_PRESS
from homeassistant.const import ATTR_ENTITY_ID, STATE_UNKNOWN
from homeassistant.core import HomeAssistant
from .util import SYSTEM_COMMAND_MOCK, USERS_VERIFICATION_MOCK, async_init_integration
async def test_qnap_buttons(hass: HomeAssistant) -> None:
"""Test buttons."""
await async_init_integration(hass)
state = hass.states.get("button.qsw_m408_4c_reboot")
assert state
assert state.state == STATE_UNKNOWN
with patch(
"homeassistant.components.qnap_qsw.QnapQswApi.get_users_verification",
return_value=USERS_VERIFICATION_MOCK,
) as mock_users_verification, patch(
"homeassistant.components.qnap_qsw.QnapQswApi.post_system_command",
return_value=SYSTEM_COMMAND_MOCK,
) as mock_post_system_command:
await hass.services.async_call(
BUTTON_DOMAIN,
SERVICE_PRESS,
{ATTR_ENTITY_ID: "button.qsw_m408_4c_reboot"},
blocking=True,
)
await hass.async_block_till_done()
mock_users_verification.assert_called_once()
mock_post_system_command.assert_called_once()

View File

@ -90,6 +90,12 @@ FIRMWARE_INFO_MOCK = {
}, },
} }
SYSTEM_COMMAND_MOCK = {
API_ERROR_CODE: 200,
API_ERROR_MESSAGE: "OK",
API_RESULT: "None",
}
SYSTEM_SENSOR_MOCK = { SYSTEM_SENSOR_MOCK = {
API_ERROR_CODE: 200, API_ERROR_CODE: 200,
API_ERROR_MESSAGE: "OK", API_ERROR_MESSAGE: "OK",