mirror of
https://github.com/home-assistant/core.git
synced 2025-04-27 18:57:57 +00:00
Add cover platform to Tessie (#105422)
* Add cover platform * fix case * Remove virtual key issue * Remove redundant logic * Fix logic that I missed * Add missing types * Add missing type * Update entity * Make window name better * Fix test * Update docstrings and comments
This commit is contained in:
parent
1170e72913
commit
23fa86cc23
@ -18,6 +18,7 @@ PLATFORMS = [
|
|||||||
Platform.BINARY_SENSOR,
|
Platform.BINARY_SENSOR,
|
||||||
Platform.BUTTON,
|
Platform.BUTTON,
|
||||||
Platform.CLIMATE,
|
Platform.CLIMATE,
|
||||||
|
Platform.COVER,
|
||||||
Platform.DEVICE_TRACKER,
|
Platform.DEVICE_TRACKER,
|
||||||
Platform.LOCK,
|
Platform.LOCK,
|
||||||
Platform.SELECT,
|
Platform.SELECT,
|
||||||
|
107
homeassistant/components/tessie/cover.py
Normal file
107
homeassistant/components/tessie/cover.py
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
"""Cover platform for Tessie integration."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
from tessie_api import (
|
||||||
|
close_charge_port,
|
||||||
|
close_windows,
|
||||||
|
open_unlock_charge_port,
|
||||||
|
vent_windows,
|
||||||
|
)
|
||||||
|
|
||||||
|
from homeassistant.components.cover import (
|
||||||
|
CoverDeviceClass,
|
||||||
|
CoverEntity,
|
||||||
|
CoverEntityFeature,
|
||||||
|
)
|
||||||
|
from homeassistant.config_entries import ConfigEntry
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
|
||||||
|
from .const import DOMAIN
|
||||||
|
from .coordinator import TessieDataUpdateCoordinator
|
||||||
|
from .entity import TessieEntity
|
||||||
|
|
||||||
|
|
||||||
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
|
"""Set up the Tessie sensor platform from a config entry."""
|
||||||
|
coordinators = hass.data[DOMAIN][entry.entry_id]
|
||||||
|
|
||||||
|
async_add_entities(
|
||||||
|
Entity(coordinator)
|
||||||
|
for Entity in (
|
||||||
|
TessieWindowEntity,
|
||||||
|
TessieChargePortEntity,
|
||||||
|
)
|
||||||
|
for coordinator in coordinators
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TessieWindowEntity(TessieEntity, CoverEntity):
|
||||||
|
"""Cover entity for current charge."""
|
||||||
|
|
||||||
|
_attr_device_class = CoverDeviceClass.WINDOW
|
||||||
|
_attr_supported_features = CoverEntityFeature.OPEN | CoverEntityFeature.CLOSE
|
||||||
|
|
||||||
|
def __init__(self, coordinator: TessieDataUpdateCoordinator) -> None:
|
||||||
|
"""Initialize the sensor."""
|
||||||
|
super().__init__(coordinator, "windows")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_closed(self) -> bool | None:
|
||||||
|
"""Return if the cover is closed or not."""
|
||||||
|
return (
|
||||||
|
self.get("vehicle_state_fd_window") == 0
|
||||||
|
and self.get("vehicle_state_fp_window") == 0
|
||||||
|
and self.get("vehicle_state_rd_window") == 0
|
||||||
|
and self.get("vehicle_state_rp_window") == 0
|
||||||
|
)
|
||||||
|
|
||||||
|
async def async_open_cover(self, **kwargs: Any) -> None:
|
||||||
|
"""Open windows."""
|
||||||
|
await self.run(vent_windows)
|
||||||
|
self.set(
|
||||||
|
("vehicle_state_fd_window", 1),
|
||||||
|
("vehicle_state_fp_window", 1),
|
||||||
|
("vehicle_state_rd_window", 1),
|
||||||
|
("vehicle_state_rp_window", 1),
|
||||||
|
)
|
||||||
|
|
||||||
|
async def async_close_cover(self, **kwargs: Any) -> None:
|
||||||
|
"""Close windows."""
|
||||||
|
await self.run(close_windows)
|
||||||
|
self.set(
|
||||||
|
("vehicle_state_fd_window", 0),
|
||||||
|
("vehicle_state_fp_window", 0),
|
||||||
|
("vehicle_state_rd_window", 0),
|
||||||
|
("vehicle_state_rp_window", 0),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TessieChargePortEntity(TessieEntity, CoverEntity):
|
||||||
|
"""Cover entity for the charge port."""
|
||||||
|
|
||||||
|
_attr_device_class = CoverDeviceClass.DOOR
|
||||||
|
_attr_supported_features = CoverEntityFeature.OPEN | CoverEntityFeature.CLOSE
|
||||||
|
|
||||||
|
def __init__(self, coordinator: TessieDataUpdateCoordinator) -> None:
|
||||||
|
"""Initialize the sensor."""
|
||||||
|
super().__init__(coordinator, "charge_state_charge_port_door_open")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_closed(self) -> bool | None:
|
||||||
|
"""Return if the cover is closed or not."""
|
||||||
|
return not self._value
|
||||||
|
|
||||||
|
async def async_open_cover(self, **kwargs: Any) -> None:
|
||||||
|
"""Open windows."""
|
||||||
|
await self.run(open_unlock_charge_port)
|
||||||
|
self.set((self.key, True))
|
||||||
|
|
||||||
|
async def async_close_cover(self, **kwargs: Any) -> None:
|
||||||
|
"""Close windows."""
|
||||||
|
await self.run(close_charge_port)
|
||||||
|
self.set((self.key, False))
|
@ -117,6 +117,14 @@
|
|||||||
"name": "Passenger temperature setting"
|
"name": "Passenger temperature setting"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"cover": {
|
||||||
|
"windows": {
|
||||||
|
"name": "Vent windows"
|
||||||
|
},
|
||||||
|
"charge_state_charge_port_door_open": {
|
||||||
|
"name": "Charge port door"
|
||||||
|
}
|
||||||
|
},
|
||||||
"select": {
|
"select": {
|
||||||
"climate_state_seat_heater_left": {
|
"climate_state_seat_heater_left": {
|
||||||
"name": "Seat heater left",
|
"name": "Seat heater left",
|
||||||
|
@ -35,6 +35,11 @@ ERROR_TIMEOUT = ClientResponseError(
|
|||||||
ERROR_UNKNOWN = ClientResponseError(
|
ERROR_UNKNOWN = ClientResponseError(
|
||||||
request_info=TEST_REQUEST_INFO, history=None, status=HTTPStatus.BAD_REQUEST
|
request_info=TEST_REQUEST_INFO, history=None, status=HTTPStatus.BAD_REQUEST
|
||||||
)
|
)
|
||||||
|
ERROR_VIRTUAL_KEY = ClientResponseError(
|
||||||
|
request_info=TEST_REQUEST_INFO,
|
||||||
|
history=None,
|
||||||
|
status=HTTPStatus.INTERNAL_SERVER_ERROR,
|
||||||
|
)
|
||||||
ERROR_CONNECTION = ClientConnectionError()
|
ERROR_CONNECTION = ClientConnectionError()
|
||||||
|
|
||||||
|
|
||||||
|
112
tests/components/tessie/test_cover.py
Normal file
112
tests/components/tessie/test_cover.py
Normal file
@ -0,0 +1,112 @@
|
|||||||
|
"""Test the Tessie cover platform."""
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from homeassistant.components.cover import (
|
||||||
|
DOMAIN as COVER_DOMAIN,
|
||||||
|
SERVICE_CLOSE_COVER,
|
||||||
|
SERVICE_OPEN_COVER,
|
||||||
|
STATE_CLOSED,
|
||||||
|
STATE_OPEN,
|
||||||
|
)
|
||||||
|
from homeassistant.const import ATTR_ENTITY_ID
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
from homeassistant.exceptions import HomeAssistantError
|
||||||
|
|
||||||
|
from .common import ERROR_UNKNOWN, TEST_RESPONSE, setup_platform
|
||||||
|
|
||||||
|
|
||||||
|
async def test_window(hass: HomeAssistant) -> None:
|
||||||
|
"""Tests that the window cover entity is correct."""
|
||||||
|
|
||||||
|
await setup_platform(hass)
|
||||||
|
|
||||||
|
entity_id = "cover.test_vent_windows"
|
||||||
|
assert hass.states.get(entity_id).state == STATE_CLOSED
|
||||||
|
|
||||||
|
# Test open windows
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.tessie.cover.vent_windows",
|
||||||
|
return_value=TEST_RESPONSE,
|
||||||
|
) as mock_set:
|
||||||
|
await hass.services.async_call(
|
||||||
|
COVER_DOMAIN,
|
||||||
|
SERVICE_OPEN_COVER,
|
||||||
|
{ATTR_ENTITY_ID: [entity_id]},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
mock_set.assert_called_once()
|
||||||
|
assert hass.states.get(entity_id).state == STATE_OPEN
|
||||||
|
|
||||||
|
# Test close windows
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.tessie.cover.close_windows",
|
||||||
|
return_value=TEST_RESPONSE,
|
||||||
|
) as mock_set:
|
||||||
|
await hass.services.async_call(
|
||||||
|
COVER_DOMAIN,
|
||||||
|
SERVICE_CLOSE_COVER,
|
||||||
|
{ATTR_ENTITY_ID: [entity_id]},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
mock_set.assert_called_once()
|
||||||
|
assert hass.states.get(entity_id).state == STATE_CLOSED
|
||||||
|
|
||||||
|
|
||||||
|
async def test_charge_port(hass: HomeAssistant) -> None:
|
||||||
|
"""Tests that the charge port cover entity is correct."""
|
||||||
|
|
||||||
|
await setup_platform(hass)
|
||||||
|
|
||||||
|
entity_id = "cover.test_charge_port_door"
|
||||||
|
assert hass.states.get(entity_id).state == STATE_OPEN
|
||||||
|
|
||||||
|
# Test close charge port
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.tessie.cover.close_charge_port",
|
||||||
|
return_value=TEST_RESPONSE,
|
||||||
|
) as mock_set:
|
||||||
|
await hass.services.async_call(
|
||||||
|
COVER_DOMAIN,
|
||||||
|
SERVICE_CLOSE_COVER,
|
||||||
|
{ATTR_ENTITY_ID: [entity_id]},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
mock_set.assert_called_once()
|
||||||
|
assert hass.states.get(entity_id).state == STATE_CLOSED
|
||||||
|
|
||||||
|
# Test open charge port
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.tessie.cover.open_unlock_charge_port",
|
||||||
|
return_value=TEST_RESPONSE,
|
||||||
|
) as mock_set:
|
||||||
|
await hass.services.async_call(
|
||||||
|
COVER_DOMAIN,
|
||||||
|
SERVICE_OPEN_COVER,
|
||||||
|
{ATTR_ENTITY_ID: [entity_id]},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
mock_set.assert_called_once()
|
||||||
|
assert hass.states.get(entity_id).state == STATE_OPEN
|
||||||
|
|
||||||
|
|
||||||
|
async def test_errors(hass: HomeAssistant) -> None:
|
||||||
|
"""Tests errors are handled."""
|
||||||
|
|
||||||
|
await setup_platform(hass)
|
||||||
|
entity_id = "cover.test_charge_port_door"
|
||||||
|
|
||||||
|
# Test setting cover open with unknown error
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.tessie.cover.open_unlock_charge_port",
|
||||||
|
side_effect=ERROR_UNKNOWN,
|
||||||
|
) as mock_set, pytest.raises(HomeAssistantError) as error:
|
||||||
|
await hass.services.async_call(
|
||||||
|
COVER_DOMAIN,
|
||||||
|
SERVICE_OPEN_COVER,
|
||||||
|
{ATTR_ENTITY_ID: [entity_id]},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
mock_set.assert_called_once()
|
||||||
|
assert error.from_exception == ERROR_UNKNOWN
|
Loading…
x
Reference in New Issue
Block a user