Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

Align configuration classes with Home Assistant standards #270

Merged
122 changes: 50 additions & 72 deletions custom_components/mbapi2020/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,11 @@
from __future__ import annotations

import asyncio
from collections.abc import Callable, Coroutine
from dataclasses import dataclass
from datetime import datetime
import time
from typing import Protocol
from typing import Any

import aiohttp
import voluptuous as vol
Expand All @@ -26,12 +27,13 @@
from custom_components.mbapi2020.errors import WebsocketError
from custom_components.mbapi2020.helper import LogHelper as loghelper
from custom_components.mbapi2020.services import setup_services
from homeassistant.components.switch import SwitchEntityDescription
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import EntityCategory
from homeassistant.core import HomeAssistant, callback
from homeassistant.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady, HomeAssistantError
from homeassistant.helpers.device_registry import DeviceInfo
from homeassistant.helpers.entity import Entity
from homeassistant.helpers.entity import Entity, EntityDescription
from homeassistant.helpers.typing import ConfigType
from homeassistant.helpers.update_coordinator import CoordinatorEntity
from homeassistant.util import slugify
Expand Down Expand Up @@ -107,7 +109,10 @@ async def async_setup_entry(hass: HomeAssistant, config_entry: ConfigEntry):
try:
capabilities = await coordinator.client.webapi.get_car_capabilities_commands(vin)
hass.async_add_executor_job(
coordinator.client.write_debug_json_output, capabilities, f"ca-{loghelper.Mask_VIN(vin)}-", True
coordinator.client.write_debug_json_output,
capabilities,
f"ca-{loghelper.Mask_VIN(vin)}-",
True,
)
if capabilities:
for feature in capabilities.get("commands"):
Expand All @@ -116,7 +121,6 @@ async def async_setup_entry(hass: HomeAssistant, config_entry: ConfigEntry):
capabilityInformation = feature.get("capabilityInformation", None)
if capabilityInformation and len(capabilityInformation) > 0:
features[feature.get("capabilityInformation")[0]] = bool(feature.get("isAvailable"))

ReneNulschDE marked this conversation as resolved.
Show resolved Hide resolved
except aiohttp.ClientError:
# For some cars a HTTP401 is raised when asking for capabilities, see github issue #83
# We just ignore the capabilities
Expand All @@ -134,7 +138,9 @@ async def async_setup_entry(hass: HomeAssistant, config_entry: ConfigEntry):
rcp_supported_settings = await coordinator.client.webapi.get_car_rcp_supported_settings(vin)
if rcp_supported_settings:
hass.async_add_executor_job(
coordinator.client.write_debug_json_output, rcp_supported_settings, "rcs"
coordinator.client.write_debug_json_output,
rcp_supported_settings,
"rcs",
)
if rcp_supported_settings.get("data"):
if rcp_supported_settings.get("data").get("attributes"):
Expand All @@ -159,7 +165,9 @@ async def async_setup_entry(hass: HomeAssistant, config_entry: ConfigEntry):
setting_result = await coordinator.client.webapi.get_car_rcp_settings(vin, setting)
if setting_result is not None:
hass.async_add_executor_job(
coordinator.client.write_debug_json_output, setting_result, f"rcs_{setting}"
coordinator.client.write_debug_json_output,
setting_result,
f"rcs_{setting}",
)

current_car = Car(vin)
Expand Down Expand Up @@ -231,45 +239,12 @@ async def async_unload_entry(hass: HomeAssistant, config_entry: ConfigEntry):
return unload_ok


class CapabilityCheckFunc(Protocol):
"""Protocol for a callable that checks if a capability is available for a given car."""

def __call__(self, car: Car) -> bool:
"""Check if the capability is available for the specified car."""


@dataclass(frozen=True)
class MercedesMeEntityConfig:
@dataclass(frozen=True, kw_only=True)
class MercedesMeEntityDescription(SwitchEntityDescription):
"""Configuration class for MercedesMe entities."""

id: str
entity_name: str
feature_name: str
object_name: str
attribute_name: str

attributes: list[str] | None = None
icon: str | None = None
device_class: str | None = None
entity_category: EntityCategory | None = None

capability_check: CapabilityCheckFunc | None = None

def __repr__(self) -> str:
"""Return a string representation of the MercedesMeEntityConfig instance."""
return (
f"{self.__class__.__name__}("
f"internal_name={self.id!r}, "
f"entity_name={self.entity_name!r}, "
f"feature_name={self.feature_name!r}, "
f"object_name={self.object_name!r}, "
f"attribute_name={self.attribute_name!r}, "
f"capability_check={self.capability_check!r}, "
f"attributes={self.attributes!r}, "
f"device_class={self.device_class!r}, "
f"icon={self.icon!r}, "
f"entity_category={self.entity_category!r})"
)
check_capability_fn: Callable[[Car], Callable[[], Coroutine[Any, Any, bool]]]


class MercedesMeEntity(CoordinatorEntity[MBAPI2020DataUpdateCoordinator], Entity):
Expand All @@ -280,38 +255,31 @@ class MercedesMeEntity(CoordinatorEntity[MBAPI2020DataUpdateCoordinator], Entity
def __init__(
self,
internal_name: str,
config: list | MercedesMeEntityConfig,
config: list | EntityDescription,
vin: str,
coordinator: MBAPI2020DataUpdateCoordinator,
should_poll: bool = False,
) -> None:
"""Initialize the MercedesMe entity."""
super().__init__(coordinator)

self._hass = coordinator.hass
self._coordinator = coordinator
self._vin = vin
self._internal_name = internal_name
self._sensor_config = config
self._car = self._coordinator.client.cars[self._vin]
self._feature_name = None
self._object_name = None
self._attrib_name = None

self._flip_result = False
self._state = None

# Temporary workaround: If PR get's approved, all entity types should be migrated to the new config classes
if isinstance(config, MercedesMeEntityConfig):
self._sensor_name = config.entity_name
self._internal_unit = None
self._feature_name = config.feature_name
self._object_name = config.object_name
self._attrib_name = config.attribute_name
self._flip_result = False
self._attr_device_class = config.device_class
self._attr_icon = config.icon
self._attr_state_class = None
self._attr_entity_category = config.entity_category
if isinstance(config, EntityDescription):
self._attributes = config.attributes
self.entity_description = config
else:
self._sensor_name = config[scf.DISPLAY_NAME.value]
self._internal_unit = config[scf.UNIT_OF_MEASUREMENT.value]
self._feature_name = config[scf.OBJECT_NAME.value]
self._object_name = config[scf.ATTRIBUTE_NAME.value]
self._attrib_name = config[scf.VALUE_FIELD_NAME.value]
Expand All @@ -321,25 +289,23 @@ def __init__(
self._attr_state_class = self._sensor_config[scf.STATE_CLASS.value]
self._attr_entity_category = self._sensor_config[scf.ENTITY_CATEGORY.value]
self._attributes = self._sensor_config[scf.EXTENDED_ATTRIBUTE_LIST.value]

self._car = self._coordinator.client.cars[self._vin]
self._use_chinese_location_data: bool = self._coordinator.config_entry.options.get(
CONF_ENABLE_CHINA_GCJ_02, False
)

self._name = f"{self._car.licenseplate} {self._sensor_name}"
self._attr_native_unit_of_measurement = self.unit_of_measurement
self._use_chinese_location_data: bool = self._coordinator.config_entry.options.get(
CONF_ENABLE_CHINA_GCJ_02, False
)
self._attr_translation_key = self._internal_name.lower()
self._attr_name = config[scf.DISPLAY_NAME.value]
self._name = f"{self._car.licenseplate} {config[scf.DISPLAY_NAME.value]}"

self._attr_device_info = {"identifiers": {(DOMAIN, self._vin)}}
self._attr_should_poll = should_poll

self._attr_native_unit_of_measurement = self.unit_of_measurement
self._attr_translation_key = self._internal_name.lower()
self._attr_unique_id = slugify(f"{self._vin}_{self._internal_name}")
self._attr_name = self._sensor_name

super().__init__(coordinator)

def device_retrieval_status(self):
"""Return the retrieval_status of the sensor."""
if self._sensor_name == "Car":
if self._internal_name == "car":
return "VALID"

return self._get_car_value(self._feature_name, self._object_name, "retrievalstatus", "error")
Expand Down Expand Up @@ -404,15 +370,27 @@ def unit_of_measurement(self):
)
return reported_unit

if isinstance(self._sensor_config, MercedesMeEntityConfig):
if isinstance(self._sensor_config, EntityDescription):
return None
return self._sensor_config[scf.UNIT_OF_MEASUREMENT.value]

def update(self):
"""Get the latest data and updates the states."""
if not self.enabled:
return

if isinstance(self._sensor_config, EntityDescription):
try:
self._mercedes_me_update()
except Exception as err:
LOGGER.error("Error while updating entity %s: %s", self.name, err)
else:
self._state = self._get_car_value(self._feature_name, self._object_name, self._attrib_name, "error")
self.async_write_ha_state()

self._state = self._get_car_value(self._feature_name, self._object_name, self._attrib_name, "error")
self.async_write_ha_state()
def _mercedes_me_update(self) -> None:
"""Update Mercedes Me entity."""
raise NotImplementedError

def _get_car_value(self, feature, object_name, attrib_name, default_value):
value = None
Expand Down
4 changes: 4 additions & 0 deletions custom_components/mbapi2020/car.py
Original file line number Diff line number Diff line change
Expand Up @@ -295,6 +295,10 @@ def publish_updates(self):
for callback in self._update_listeners:
callback()

def check_capabilities(self, required_capabilities: list[str]) -> bool:
"""Check if the car has the required capabilities."""
return all(self.features.get(capability) is True for capability in required_capabilities)


@dataclass(init=False)
class Tires:
Expand Down
7 changes: 7 additions & 0 deletions custom_components/mbapi2020/client.py
Original file line number Diff line number Diff line change
Expand Up @@ -1261,6 +1261,13 @@ async def preheat_start_immediate(self, vin: str):
await self.websocket.call(message.SerializeToString())
LOGGER.info("End preheat_start_immediate for vin %s", loghelper.Mask_VIN(vin))

async def preheat_start_universal(self,vin: str) -> None:
"""Turn on preheat universally for any car model."""
if self._is_car_feature_available(vin, "precondNow"):
await self.preheat_start(vin)
else:
await self.preheat_start_immediate(vin)

async def preheat_start_departure_time(self, vin: str, departure_time: int):
"""Send a preconditioning start by time command to the car."""
LOGGER.info("Start preheat_start_departure_time for vin %s", loghelper.Mask_VIN(vin))
Expand Down
5 changes: 0 additions & 5 deletions custom_components/mbapi2020/helper.py
Original file line number Diff line number Diff line change
Expand Up @@ -229,8 +229,3 @@ def default(self, o) -> Union[str, dict]: # noqa: D102
retval.update({p: getattr(o, p) for p in get_class_property_names(o)})
return {k: v for k, v in retval.items() if k not in JSON_EXPORT_IGNORED_KEYS}
return str(o)


def check_capabilities(car, required_capabilities):
"""Check if the car has the required capabilities."""
return all(car.features.get(capability) is True for capability in required_capabilities)
Loading
Loading