184 lines
5.9 KiB
Python
184 lines
5.9 KiB
Python
"""Lock platform for Tessie integration."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from typing import Any
|
|
|
|
from tessie_api import (
|
|
disable_speed_limit,
|
|
enable_speed_limit,
|
|
lock,
|
|
open_unlock_charge_port,
|
|
unlock,
|
|
)
|
|
|
|
from homeassistant.components.automation import automations_with_entity
|
|
from homeassistant.components.lock import ATTR_CODE, LockEntity
|
|
from homeassistant.components.script import scripts_with_entity
|
|
from homeassistant.const import Platform
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.exceptions import ServiceValidationError
|
|
from homeassistant.helpers import entity_registry as er, issue_registry as ir
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from . import TessieConfigEntry
|
|
from .const import DOMAIN, TessieChargeCableLockStates
|
|
from .entity import TessieEntity
|
|
from .models import TessieVehicleData
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant,
|
|
entry: TessieConfigEntry,
|
|
async_add_entities: AddEntitiesCallback,
|
|
) -> None:
|
|
"""Set up the Tessie sensor platform from a config entry."""
|
|
data = entry.runtime_data
|
|
|
|
entities: list[TessieEntity] = [
|
|
klass(vehicle)
|
|
for klass in (TessieLockEntity, TessieCableLockEntity)
|
|
for vehicle in data.vehicles
|
|
]
|
|
|
|
ent_reg = er.async_get(hass)
|
|
|
|
for vehicle in data.vehicles:
|
|
entity_id = ent_reg.async_get_entity_id(
|
|
Platform.LOCK,
|
|
DOMAIN,
|
|
f"{vehicle.vin}-vehicle_state_speed_limit_mode_active",
|
|
)
|
|
if entity_id:
|
|
entity_entry = ent_reg.async_get(entity_id)
|
|
assert entity_entry
|
|
if entity_entry.disabled:
|
|
ent_reg.async_remove(entity_id)
|
|
else:
|
|
entities.append(TessieSpeedLimitEntity(vehicle))
|
|
|
|
entity_automations = automations_with_entity(hass, entity_id)
|
|
entity_scripts = scripts_with_entity(hass, entity_id)
|
|
for item in entity_automations + entity_scripts:
|
|
ir.async_create_issue(
|
|
hass,
|
|
DOMAIN,
|
|
f"deprecated_speed_limit_{entity_id}_{item}",
|
|
breaks_in_ha_version="2024.11.0",
|
|
is_fixable=True,
|
|
is_persistent=False,
|
|
severity=ir.IssueSeverity.WARNING,
|
|
translation_key="deprecated_speed_limit_entity",
|
|
translation_placeholders={
|
|
"entity": entity_id,
|
|
"info": item,
|
|
},
|
|
)
|
|
async_add_entities(entities)
|
|
|
|
|
|
class TessieLockEntity(TessieEntity, LockEntity):
|
|
"""Lock entity for Tessie."""
|
|
|
|
def __init__(
|
|
self,
|
|
vehicle: TessieVehicleData,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
super().__init__(vehicle, "vehicle_state_locked")
|
|
|
|
@property
|
|
def is_locked(self) -> bool | None:
|
|
"""Return the state of the Lock."""
|
|
return self._value
|
|
|
|
async def async_lock(self, **kwargs: Any) -> None:
|
|
"""Set new value."""
|
|
await self.run(lock)
|
|
self.set((self.key, True))
|
|
|
|
async def async_unlock(self, **kwargs: Any) -> None:
|
|
"""Set new value."""
|
|
await self.run(unlock)
|
|
self.set((self.key, False))
|
|
|
|
|
|
class TessieSpeedLimitEntity(TessieEntity, LockEntity):
|
|
"""Speed Limit with PIN entity for Tessie."""
|
|
|
|
_attr_code_format = r"^\d\d\d\d$"
|
|
|
|
def __init__(
|
|
self,
|
|
vehicle: TessieVehicleData,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
super().__init__(vehicle, "vehicle_state_speed_limit_mode_active")
|
|
|
|
@property
|
|
def is_locked(self) -> bool | None:
|
|
"""Return the state of the Lock."""
|
|
return self._value
|
|
|
|
async def async_lock(self, **kwargs: Any) -> None:
|
|
"""Enable speed limit with pin."""
|
|
ir.async_create_issue(
|
|
self.coordinator.hass,
|
|
DOMAIN,
|
|
"deprecated_speed_limit_locked",
|
|
breaks_in_ha_version="2024.11.0",
|
|
is_fixable=True,
|
|
is_persistent=False,
|
|
severity=ir.IssueSeverity.WARNING,
|
|
translation_key="deprecated_speed_limit_locked",
|
|
)
|
|
code: str | None = kwargs.get(ATTR_CODE)
|
|
if code:
|
|
await self.run(enable_speed_limit, pin=code)
|
|
self.set((self.key, True))
|
|
|
|
async def async_unlock(self, **kwargs: Any) -> None:
|
|
"""Disable speed limit with pin."""
|
|
ir.async_create_issue(
|
|
self.coordinator.hass,
|
|
DOMAIN,
|
|
"deprecated_speed_limit_unlocked",
|
|
breaks_in_ha_version="2024.11.0",
|
|
is_fixable=True,
|
|
is_persistent=False,
|
|
severity=ir.IssueSeverity.WARNING,
|
|
translation_key="deprecated_speed_limit_unlocked",
|
|
)
|
|
code: str | None = kwargs.get(ATTR_CODE)
|
|
if code:
|
|
await self.run(disable_speed_limit, pin=code)
|
|
self.set((self.key, False))
|
|
|
|
|
|
class TessieCableLockEntity(TessieEntity, LockEntity):
|
|
"""Cable Lock entity for Tessie."""
|
|
|
|
def __init__(
|
|
self,
|
|
vehicle: TessieVehicleData,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
super().__init__(vehicle, "charge_state_charge_port_latch")
|
|
|
|
@property
|
|
def is_locked(self) -> bool | None:
|
|
"""Return the state of the Lock."""
|
|
return self._value == TessieChargeCableLockStates.ENGAGED
|
|
|
|
async def async_lock(self, **kwargs: Any) -> None:
|
|
"""Charge cable Lock cannot be manually locked."""
|
|
raise ServiceValidationError(
|
|
translation_domain=DOMAIN,
|
|
translation_key="no_cable",
|
|
)
|
|
|
|
async def async_unlock(self, **kwargs: Any) -> None:
|
|
"""Unlock charge cable lock."""
|
|
await self.run(open_unlock_charge_port)
|
|
self.set((self.key, TessieChargeCableLockStates.DISENGAGED))
|