2021-07-28 19:41:11 +00:00
|
|
|
"""Proxy to handle account communication with Renault servers."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
import asyncio
|
2021-09-29 12:06:51 +00:00
|
|
|
from collections.abc import Awaitable, Callable
|
2021-08-31 09:06:54 +00:00
|
|
|
from dataclasses import dataclass
|
2021-07-28 19:41:11 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
import logging
|
2021-09-29 12:06:51 +00:00
|
|
|
from typing import cast
|
2021-07-28 19:41:11 +00:00
|
|
|
|
|
|
|
from renault_api.kamereon import models
|
|
|
|
from renault_api.renault_vehicle import RenaultVehicle
|
|
|
|
|
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from homeassistant.helpers.entity import DeviceInfo
|
|
|
|
|
|
|
|
from .const import DOMAIN
|
|
|
|
from .renault_coordinator import RenaultDataUpdateCoordinator
|
|
|
|
|
|
|
|
LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2021-08-31 09:06:54 +00:00
|
|
|
@dataclass
|
|
|
|
class RenaultCoordinatorDescription:
|
|
|
|
"""Class describing Renault coordinators."""
|
|
|
|
|
|
|
|
endpoint: str
|
|
|
|
key: str
|
|
|
|
update_method: Callable[
|
|
|
|
[RenaultVehicle],
|
|
|
|
Callable[[], Awaitable[models.KamereonVehicleDataAttributes]],
|
|
|
|
]
|
|
|
|
# Optional keys
|
|
|
|
requires_electricity: bool = False
|
|
|
|
|
|
|
|
|
2021-07-28 19:41:11 +00:00
|
|
|
class RenaultVehicleProxy:
|
|
|
|
"""Handle vehicle communication with Renault servers."""
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
hass: HomeAssistant,
|
|
|
|
vehicle: RenaultVehicle,
|
|
|
|
details: models.KamereonVehicleDetails,
|
|
|
|
scan_interval: timedelta,
|
|
|
|
) -> None:
|
|
|
|
"""Initialise vehicle proxy."""
|
|
|
|
self.hass = hass
|
|
|
|
self._vehicle = vehicle
|
|
|
|
self._details = details
|
2021-10-25 16:26:03 +00:00
|
|
|
self._device_info = DeviceInfo(
|
|
|
|
identifiers={(DOMAIN, cast(str, details.vin))},
|
|
|
|
manufacturer=(details.get_brand_label() or "").capitalize(),
|
|
|
|
model=(details.get_model_label() or "").capitalize(),
|
|
|
|
name=details.registrationNumber or "",
|
|
|
|
sw_version=details.get_model_code() or "",
|
|
|
|
)
|
2021-07-28 19:41:11 +00:00
|
|
|
self.coordinators: dict[str, RenaultDataUpdateCoordinator] = {}
|
|
|
|
self.hvac_target_temperature = 21
|
|
|
|
self._scan_interval = scan_interval
|
|
|
|
|
|
|
|
@property
|
|
|
|
def details(self) -> models.KamereonVehicleDetails:
|
|
|
|
"""Return the specs of the vehicle."""
|
|
|
|
return self._details
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_info(self) -> DeviceInfo:
|
|
|
|
"""Return a device description for device registry."""
|
|
|
|
return self._device_info
|
|
|
|
|
2021-09-01 09:23:54 +00:00
|
|
|
@property
|
|
|
|
def vehicle(self) -> RenaultVehicle:
|
|
|
|
"""Return the underlying vehicle."""
|
|
|
|
return self._vehicle
|
|
|
|
|
2021-07-28 19:41:11 +00:00
|
|
|
async def async_initialise(self) -> None:
|
2021-08-31 09:06:54 +00:00
|
|
|
"""Load available coordinators."""
|
|
|
|
self.coordinators = {
|
|
|
|
coord.key: RenaultDataUpdateCoordinator(
|
2021-07-28 19:41:11 +00:00
|
|
|
self.hass,
|
|
|
|
LOGGER,
|
|
|
|
# Name of the data. For logging purposes.
|
2021-08-31 09:06:54 +00:00
|
|
|
name=f"{self.details.vin} {coord.key}",
|
|
|
|
update_method=coord.update_method(self._vehicle),
|
2021-07-28 19:41:11 +00:00
|
|
|
# Polling interval. Will only be polled if there are subscribers.
|
|
|
|
update_interval=self._scan_interval,
|
|
|
|
)
|
2021-08-31 09:06:54 +00:00
|
|
|
for coord in COORDINATORS
|
|
|
|
if (
|
|
|
|
self.details.supports_endpoint(coord.endpoint)
|
|
|
|
and (not coord.requires_electricity or self.details.uses_electricity())
|
2021-07-28 19:41:11 +00:00
|
|
|
)
|
2021-08-31 09:06:54 +00:00
|
|
|
}
|
2021-07-28 19:41:11 +00:00
|
|
|
# Check all coordinators
|
|
|
|
await asyncio.gather(
|
|
|
|
*(
|
|
|
|
coordinator.async_config_entry_first_refresh()
|
|
|
|
for coordinator in self.coordinators.values()
|
|
|
|
)
|
|
|
|
)
|
|
|
|
for key in list(self.coordinators):
|
|
|
|
# list() to avoid Runtime iteration error
|
|
|
|
coordinator = self.coordinators[key]
|
|
|
|
if coordinator.not_supported:
|
|
|
|
# Remove endpoint as it is not supported for this vehicle.
|
2021-08-16 11:49:04 +00:00
|
|
|
LOGGER.warning(
|
2021-07-28 19:41:11 +00:00
|
|
|
"Ignoring endpoint %s as it is not supported for this vehicle: %s",
|
|
|
|
coordinator.name,
|
|
|
|
coordinator.last_exception,
|
|
|
|
)
|
|
|
|
del self.coordinators[key]
|
|
|
|
elif coordinator.access_denied:
|
|
|
|
# Remove endpoint as it is denied for this vehicle.
|
2021-08-16 11:49:04 +00:00
|
|
|
LOGGER.warning(
|
2021-07-28 19:41:11 +00:00
|
|
|
"Ignoring endpoint %s as it is denied for this vehicle: %s",
|
|
|
|
coordinator.name,
|
|
|
|
coordinator.last_exception,
|
|
|
|
)
|
|
|
|
del self.coordinators[key]
|
|
|
|
|
|
|
|
|
2021-08-31 09:06:54 +00:00
|
|
|
COORDINATORS: tuple[RenaultCoordinatorDescription, ...] = (
|
|
|
|
RenaultCoordinatorDescription(
|
|
|
|
endpoint="cockpit",
|
|
|
|
key="cockpit",
|
|
|
|
update_method=lambda x: x.get_cockpit,
|
|
|
|
),
|
|
|
|
RenaultCoordinatorDescription(
|
|
|
|
endpoint="hvac-status",
|
|
|
|
key="hvac_status",
|
|
|
|
update_method=lambda x: x.get_hvac_status,
|
|
|
|
),
|
2021-09-01 11:10:48 +00:00
|
|
|
RenaultCoordinatorDescription(
|
|
|
|
endpoint="location",
|
|
|
|
key="location",
|
|
|
|
update_method=lambda x: x.get_location,
|
|
|
|
),
|
2021-08-31 09:06:54 +00:00
|
|
|
RenaultCoordinatorDescription(
|
|
|
|
endpoint="battery-status",
|
|
|
|
key="battery",
|
|
|
|
requires_electricity=True,
|
|
|
|
update_method=lambda x: x.get_battery_status,
|
|
|
|
),
|
|
|
|
RenaultCoordinatorDescription(
|
|
|
|
endpoint="charge-mode",
|
|
|
|
key="charge_mode",
|
|
|
|
requires_electricity=True,
|
|
|
|
update_method=lambda x: x.get_charge_mode,
|
|
|
|
),
|
|
|
|
)
|