From 16ef181c3531c0322e9526805ce527a9c08f9f1f Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Thu, 29 Jun 2023 02:00:18 +0200 Subject: [PATCH 01/11] select for load balancing mode --- custom_components/alfen_wallbox/__init__.py | 11 +- custom_components/alfen_wallbox/alfen.py | 5 +- .../alfen_wallbox/config_flow.py | 46 ++----- custom_components/alfen_wallbox/const.py | 1 + custom_components/alfen_wallbox/entity.py | 122 ++++++++++++++++++ custom_components/alfen_wallbox/select.py | 99 ++++++++++++++ custom_components/alfen_wallbox/sensor.py | 7 +- 7 files changed, 241 insertions(+), 50 deletions(-) create mode 100644 custom_components/alfen_wallbox/entity.py create mode 100644 custom_components/alfen_wallbox/select.py diff --git a/custom_components/alfen_wallbox/__init__.py b/custom_components/alfen_wallbox/__init__.py index f9dff5d..a62e207 100644 --- a/custom_components/alfen_wallbox/__init__.py +++ b/custom_components/alfen_wallbox/__init__.py @@ -3,26 +3,21 @@ import asyncio from datetime import timedelta import logging -from typing import Any, Dict +from typing import Dict from aiohttp import ClientConnectionError from async_timeout import timeout -from homeassistant.components.sensor import DOMAIN as SENSOR_DOMAIN from homeassistant.config_entries import ConfigEntry from homeassistant.const import ( - ATTR_NAME, CONF_HOST, CONF_NAME, CONF_USERNAME, CONF_PASSWORD, + Platform, ) from homeassistant.core import HomeAssistant from homeassistant.exceptions import ConfigEntryNotReady -from homeassistant.helpers import device_registry as dr -from homeassistant.helpers.aiohttp_client import async_get_clientsession -from homeassistant.helpers.entity import Entity -from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed from homeassistant.helpers.typing import HomeAssistantType from .alfen import AlfenDevice @@ -32,7 +27,7 @@ TIMEOUT, ) -PLATFORMS = [SENSOR_DOMAIN] +PLATFORMS = [Platform.SENSOR, Platform.SELECT] SCAN_INTERVAL = timedelta(seconds=60) _LOGGER = logging.getLogger(__name__) diff --git a/custom_components/alfen_wallbox/alfen.py b/custom_components/alfen_wallbox/alfen.py index 04f481b..48de76e 100644 --- a/custom_components/alfen_wallbox/alfen.py +++ b/custom_components/alfen_wallbox/alfen.py @@ -458,7 +458,8 @@ def __init__(self, response, prev_status): elif prop["id"] == "2110_0": self.gprs_signal_strength = prop["value"] elif prop["id"] == "3280_1": - self.lb_solar_charging_mode = self.solar_charging_mode(prop["value"]) + self.lb_solar_charging_mode = self.solar_charging_mode( + prop["value"]) elif prop["id"] == "3280_2": self.lb_solar_charging_green_share = prop["value"] elif prop["id"] == "3280_3": @@ -487,4 +488,4 @@ def __init__(self, response): self.model = f"{ALFEN_PRODUCT_MAP[self.model_id]} ({self.model_id})" self.object_id = response["ObjectId"] - self.type = response["Type"] \ No newline at end of file + self.type = response["Type"] diff --git a/custom_components/alfen_wallbox/config_flow.py b/custom_components/alfen_wallbox/config_flow.py index 186a6e0..2ef6247 100644 --- a/custom_components/alfen_wallbox/config_flow.py +++ b/custom_components/alfen_wallbox/config_flow.py @@ -15,7 +15,6 @@ _LOGGER = logging.getLogger(__name__) - @config_entries.HANDLERS.register("alfen_wallbox") class FlowHandler(config_entries.ConfigFlow): """Handle a config flow.""" @@ -30,26 +29,14 @@ async def _create_entry(self, host, name, username, password): if entry.data[KEY_IP] == host: return self.async_abort(reason="already_configured") - return self.async_create_entry( - title=host, - data={ - CONF_HOST: host, - CONF_NAME: name, - CONF_USERNAME: username, - CONF_PASSWORD: password, - }, - ) + return self.async_create_entry(title=host, data={CONF_HOST: host, CONF_NAME: name, CONF_USERNAME: username, CONF_PASSWORD: password}) async def _create_device(self, host, name, username, password): """Create device.""" try: device = AlfenDevice( - host, - name, - self.hass.helpers.aiohttp_client.async_get_clientsession(), - username, - password, + host, name, self.hass.helpers.aiohttp_client.async_get_clientsession(), username, password ) with timeout(TIMEOUT): await device.init() @@ -68,31 +55,18 @@ async def async_step_user(self, user_input=None): """User initiated config flow.""" if user_input is None: return self.async_show_form( - step_id="user", - data_schema=vol.Schema( - { - vol.Required(CONF_HOST): str, - vol.Required(CONF_USERNAME, default="admin"): str, - vol.Required(CONF_PASSWORD): str, - vol.Optional(CONF_NAME): str, - } - ), + step_id="user", data_schema=vol.Schema({ + vol.Required(CONF_HOST): str, + vol.Required(CONF_USERNAME, default="admin"): str, + vol.Required(CONF_PASSWORD): str, + vol.Optional(CONF_NAME): str + }) ) - return await self._create_device( - user_input[CONF_HOST], - user_input[CONF_NAME], - user_input[CONF_USERNAME], - user_input[CONF_PASSWORD], - ) + return await self._create_device(user_input[CONF_HOST], user_input[CONF_NAME], user_input[CONF_USERNAME], user_input[CONF_PASSWORD]) async def async_step_import(self, user_input): """Import a config entry.""" host = user_input.get(CONF_HOST) if not host: return await self.async_step_user() - return await self._create_device( - host, - user_input[CONF_NAME], - user_input[CONF_USERNAME], - user_input[CONF_PASSWORD], - ) + return await self._create_device(host, user_input[CONF_NAME], user_input[CONF_USERNAME], user_input[CONF_PASSWORD]) diff --git a/custom_components/alfen_wallbox/const.py b/custom_components/alfen_wallbox/const.py index 41e0664..d93f829 100644 --- a/custom_components/alfen_wallbox/const.py +++ b/custom_components/alfen_wallbox/const.py @@ -1,3 +1,4 @@ + DOMAIN = "alfen_wallbox" KEY_MAC = "mac" diff --git a/custom_components/alfen_wallbox/entity.py b/custom_components/alfen_wallbox/entity.py new file mode 100644 index 0000000..b00ec50 --- /dev/null +++ b/custom_components/alfen_wallbox/entity.py @@ -0,0 +1,122 @@ +import logging + +import ssl +from config.custom_components.alfen_wallbox.alfen import HEADER_JSON, POST_HEADER_JSON, AlfenDevice, AlfenStatus +from homeassistant.core import DOMAIN +from homeassistant.helpers.entity import DeviceInfo, Entity + +_LOGGER = logging.getLogger(__name__) + + +class AlfenEntity(Entity): + + def __init__(self, device: AlfenDevice) -> None: + self._host = device.host + self._attr_name = device.name + self._status = device.status + self._session = device._session + self._username = device.username + if self._username is None: + self._username = "admin" + self._password = device.password + # Default ciphers needed as of python 3.10 + context = ssl.create_default_context() + context.set_ciphers("DEFAULT") + context.check_hostname = False + context.verify_mode = ssl.CERT_NONE + self.ssl = context + self.device = device + + self._attr_device_info = DeviceInfo( + identifiers={(DOMAIN, self.device.id)}, + manufacturer="Alfen", + model=self.device.info.model, + name=self.name, + sw_version=self.device.info.firmware_version, + ) + + async def async_added_to_hass(self) -> None: + """Add listener for state changes.""" + await super().async_added_to_hass() + + async def _on_update(self) -> None: + await self.login() + + # max 32 ids each time + response = await self._session.request( + ssl=self.ssl, + method="GET", + headers=HEADER_JSON, + url=self.__get_url( + "prop?ids=2060_0,2056_0,2221_3,2221_4,2221_5,2221_A,2221_B,2221_C,2221_16,2201_0,2501_2,2221_22,2129_0,2126_0,2068_0,2069_0,2062_0,2064_0,212B_0,212D_0,2185_0,2053_0,2067_0,212F_1,212F_2,212F_3,2100_0,2101_0,2102_0,2104_0,2105_0" + ), + ) + _LOGGER.debug(f"Status Response {response}") + + response_json = await response.json(content_type=None) + _LOGGER.debug(response_json) + + response2 = await self._session.request( + ssl=self.ssl, + method="GET", + headers=HEADER_JSON, + url=self.__get_url( + "prop?ids=2057_0,2112_0,2071_1,2071_2,2072_1,2073_1,2074_1,2075_1,2076_0,2078_1,2078_2,2079_1,207A_1,207B_1,207C_1,207D_1,207E_1,207F_1,2080_1,2081_0,2082_0,2110_0,3280_1,3280_2,3280_3,3280_4" + ), + ) + _LOGGER.debug(f"Status Response {response2}") + + response_json2 = await response2.json(content_type=None) + _LOGGER.debug(response_json2) + + await self.logout() + + response_json_combined = response_json + if response_json2 is not None: + response_json_combined["properties"] = ( + response_json["properties"] + response_json2["properties"] + ) + response_json_combined["total"] = ( + response_json["total"] + response_json2["total"] + ) + + _LOGGER.debug(response_json_combined) + + self._status = AlfenStatus(response_json_combined, self._status) + self.async_write_ha_state() + + def __get_url(self, action): + return "https://{}/api/{}".format(self._host, action) + + async def login(self): + await self._session.request( + ssl=self.ssl, + method="POST", + headers=HEADER_JSON, + url=self.__get_url("login"), + json={"username": self._username, "password": self._password}, + ) + + async def logout(self): + await self._session.request( + ssl=self.ssl, + method="POST", + headers=HEADER_JSON, + url=self.__get_url("logout"), + ) + + async def update_state(self, api_param, value): + """Get the state of the entity.""" + + await self.login() + + response = await self._session.request( + ssl=self.ssl, + method="POST", + headers=POST_HEADER_JSON, + url=self.__get_url("prop"), + json={api_param: {"id": api_param, "value": value}}, + ) + _LOGGER.info(f"Set {api_param} value {value} response {response}") + + await self.logout() diff --git a/custom_components/alfen_wallbox/select.py b/custom_components/alfen_wallbox/select.py new file mode 100644 index 0000000..b2ab4e9 --- /dev/null +++ b/custom_components/alfen_wallbox/select.py @@ -0,0 +1,99 @@ +import logging +from typing import Coroutine, Final, Any + +from dataclasses import dataclass +from config.custom_components.alfen_wallbox.alfen import AlfenDevice +from config.custom_components.alfen_wallbox.entity import AlfenEntity + +from homeassistant.config_entries import ConfigEntry +from homeassistant.helpers.entity_platform import AddEntitiesCallback + +from .alfen import AlfenDevice + +from homeassistant.components.select import ( + SelectEntity, + SelectEntityDescription, +) + +from homeassistant.core import HomeAssistant, callback +from . import DOMAIN as ALFEN_DOMAIN + +_LOGGER = logging.getLogger(__name__) + + +@dataclass +class AlfenSelectDescriptionMixin: + """Define an entity description mixin for select entities.""" + + api_param: str + options_dict: dict[str, int] + + +@dataclass +class AlfenSelectDescription(SelectEntityDescription, AlfenSelectDescriptionMixin): + """Class to describe an Alfen select entity.""" + + +CHARGING_MODE_DICT: Final[dict[str, int]] = { + "Disable": 0, "Comfort": 1, "Green": 2} + + +SELECT_TYPES: Final[tuple[AlfenSelectDescription, ...]] = ( + AlfenSelectDescription( + key="lb_solar_charging_mode", + name="Solar Charging Mode", + icon="mdi:solar-power", + options=list(CHARGING_MODE_DICT), + options_dict=CHARGING_MODE_DICT, + api_param="3280_1", + ), +) + + +async def async_setup_entry( + hass: HomeAssistant, + entry: ConfigEntry, + async_add_entities: AddEntitiesCallback +) -> None: + """Add Alfen Select from a config_entry""" + + device = hass.data[ALFEN_DOMAIN][entry.entry_id] + async_add_entities([AlfenSelect(device, description) + for description in SELECT_TYPES]) + + +class AlfenSelect(AlfenEntity, SelectEntity): + """Define Alfen select.""" + + values_dict: dict[int, str] + + def __init__(self, device: AlfenDevice, description: AlfenSelectDescription) -> None: + super().__init__(device) + self._attr_name = f"{description.name}" + self._attr_options = description.options + self.entity_description = description + self._attr_unique_id = f"{self._attr_unique_id}_{description.key}" + self._device = device + self.values_dict = {v: k for k, v in description.options_dict.items()} + + self._async_update_attrs() + + async def async_select_option(self, option: str) -> None: + value = {v: k for k, v in self.values_dict.items()}[option] + await self.update_state(self.entity_description.api_param, value) + self.async_write_ha_state() + + @property + def current_option(self) -> str | None: + return self._get_current_option() + + def _get_current_option(self) -> str | None: + return getattr(self._device.status, self.entity_description.key) + + async def async_update(self): + await self._device.async_update() + + @callback + def _async_update_attrs(self) -> None: + """Update select attributes.""" + self._attr_current_option = self._get_current_option() diff --git a/custom_components/alfen_wallbox/sensor.py b/custom_components/alfen_wallbox/sensor.py index 53dc52e..f907a35 100644 --- a/custom_components/alfen_wallbox/sensor.py +++ b/custom_components/alfen_wallbox/sensor.py @@ -2,17 +2,16 @@ import voluptuous as vol -from homeassistant.const import CONF_ICON, CONF_NAME, TEMP_CELSIUS +from homeassistant.const import TEMP_CELSIUS from homeassistant.helpers.entity import Entity from homeassistant.components.sensor import ( DEVICE_CLASS_ENERGY, DEVICE_CLASS_POWER, - PLATFORM_SCHEMA, STATE_CLASS_TOTAL_INCREASING, STATE_CLASS_MEASUREMENT, SensorEntity, ) -from homeassistant.helpers import config_validation as cv, entity_platform, service +from homeassistant.helpers import config_validation as cv, entity_platform from . import DOMAIN as ALFEN_DOMAIN @@ -121,7 +120,7 @@ async def async_setup_entry(hass, entry, async_add_entities): AlfenSensor(device, "Wired DNS 2", "comm_wired_dns_2"), AlfenSensor(device, "Protocol Name", "comm_protocol_name"), AlfenSensor(device, "Protocol Version", "comm_protocol_version"), - AlfenSensor(device, "Solar Charging Mode", "lb_solar_charging_mode"), + #AlfenSensor(device, "Solar Charging Mode", "lb_solar_charging_mode"), AlfenSensor( device, "Solar Charging Green Share %", From ba5e977ccb7051a76883e1ce38f33885ed2ab70d Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Thu, 29 Jun 2023 20:48:37 +0200 Subject: [PATCH 02/11] rework refactor sensor and select --- custom_components/alfen_wallbox/__init__.py | 36 +- custom_components/alfen_wallbox/alfen.py | 236 ++----- custom_components/alfen_wallbox/entity.py | 112 +-- custom_components/alfen_wallbox/select.py | 54 +- custom_components/alfen_wallbox/sensor.py | 739 ++++++++++++++------ 5 files changed, 670 insertions(+), 507 deletions(-) diff --git a/custom_components/alfen_wallbox/__init__.py b/custom_components/alfen_wallbox/__init__.py index a62e207..91cf7cb 100644 --- a/custom_components/alfen_wallbox/__init__.py +++ b/custom_components/alfen_wallbox/__init__.py @@ -3,7 +3,6 @@ import asyncio from datetime import timedelta import logging -from typing import Dict from aiohttp import ClientConnectionError from async_timeout import timeout @@ -18,7 +17,6 @@ ) from homeassistant.core import HomeAssistant from homeassistant.exceptions import ConfigEntryNotReady -from homeassistant.helpers.typing import HomeAssistantType from .alfen import AlfenDevice @@ -33,42 +31,36 @@ _LOGGER = logging.getLogger(__name__) -async def async_setup(hass: HomeAssistant, config: Dict) -> bool: +async def async_setup(hass: HomeAssistant, config: dict) -> bool: """Set up the Alfen Wallbox component.""" hass.data.setdefault(DOMAIN, {}) return True -async def async_setup_entry(hass: HomeAssistantType, entry: ConfigEntry): +async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry): conf = entry.data device = await alfen_setup( hass, conf[CONF_HOST], conf[CONF_NAME], conf[CONF_USERNAME], conf[CONF_PASSWORD] ) if not device: return False - hass.data.setdefault(DOMAIN, {}).update({entry.entry_id: device}) - for component in PLATFORMS: - hass.async_create_task( - hass.config_entries.async_forward_entry_setup(entry, component) - ) - - """device_registry = await dr.async_get_registry(hass) - device_registry.async_get_or_create(**device.device_info)""" + + hass.data.setdefault(DOMAIN, {}) + hass.data[DOMAIN][entry.entry_id] = device + + # hass.data.setdefault(DOMAIN, {}).update({entry.entry_id: device}) + + await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS) + return True -async def async_unload_entry(hass, config_entry): +async def async_unload_entry(hass: HomeAssistant, config_entry: ConfigEntry): """Unload a config entry.""" _LOGGER.debug("async_unload_entry: %s", config_entry) - unload_ok = all( - await asyncio.gather( - *[ - hass.config_entries.async_forward_entry_unload(config_entry, component) - for component in PLATFORMS - ] - ) - ) + unload_ok = await hass.config_entries.async_unload_platforms(config_entry, PLATFORMS) + hass.data[DOMAIN].pop(config_entry.entry_id) if not hass.data[DOMAIN]: hass.data.pop(DOMAIN) @@ -76,7 +68,7 @@ async def async_unload_entry(hass, config_entry): return unload_ok -async def alfen_setup(hass, host, name, username, password): +async def alfen_setup(hass: HomeAssistant, host: str, name: str, username: str, password: str): """Create a Alfen instance only once.""" session = hass.helpers.aiohttp_client.async_get_clientsession() diff --git a/custom_components/alfen_wallbox/alfen.py b/custom_components/alfen_wallbox/alfen.py index 48de76e..201e9c8 100644 --- a/custom_components/alfen_wallbox/alfen.py +++ b/custom_components/alfen_wallbox/alfen.py @@ -1,4 +1,5 @@ import logging +from aiohttp import ClientSession import requests import time @@ -19,15 +20,23 @@ class AlfenDevice: - def __init__(self, host, name, session, username, password): + def __init__(self, + host: str, + name: str, + session: ClientSession, + username: str, + password: str) -> None: self.host = host self.name = name self._status = None self._session = session self.username = username + self.info = None + self.id = None if self.username is None: self.username = "admin" self.password = password + self.properties = [] # Default ciphers needed as of python 3.10 context = ssl.create_default_context() context.set_ciphers("DEFAULT") @@ -37,7 +46,7 @@ def __init__(self, host, name, session, username, password): async def init(self): await self.async_get_info() - self.id = "alfen_{}".format(self.info.identity) + self.id = "alfen_{}".format(self.name) if self.name is None: self.name = f"{self.info.identity} ({self.host})" await self.async_update() @@ -50,7 +59,7 @@ def status(self): def device_info(self): """Return a device description for device registry.""" return { - "identifiers": {(DOMAIN, self.id)}, + "identifiers": {(DOMAIN, self.name)}, "manufacturer": "Alfen", "model": self.info.model, "name": self.name, @@ -64,8 +73,8 @@ def _request(self, parameter_list): async def async_update(self): await self._do_update() - async def _do_update(self): - await self._session.request( + async def login(self): + response = await self._session.request( ssl=self.ssl, method="POST", headers=HEADER_JSON, @@ -73,52 +82,55 @@ async def _do_update(self): json={"username": self.username, "password": self.password}, ) - # max 32 ids each time - response = await self._session.request( - ssl=self.ssl, - method="GET", - headers=HEADER_JSON, - url=self.__get_url( - "prop?ids=2060_0,2056_0,2221_3,2221_4,2221_5,2221_A,2221_B,2221_C,2221_16,2201_0,2501_2,2221_22,2129_0,2126_0,2068_0,2069_0,2062_0,2064_0,212B_0,212D_0,2185_0,2053_0,2067_0,212F_1,212F_2,212F_3,2100_0,2101_0,2102_0,2104_0,2105_0" - ), - ) - _LOGGER.debug(f"Status Response {response}") - - response_json = await response.json(content_type=None) - _LOGGER.debug(response_json) + _LOGGER.debug(f"Login response {response}") - response2 = await self._session.request( + async def logout(self): + response = await self._session.request( ssl=self.ssl, - method="GET", + method="POST", headers=HEADER_JSON, - url=self.__get_url( - "prop?ids=2057_0,2112_0,2071_1,2071_2,2072_1,2073_1,2074_1,2075_1,2076_0,2078_1,2078_2,2079_1,207A_1,207B_1,207C_1,207D_1,207E_1,207F_1,2080_1,2081_0,2082_0,2110_0,3280_1,3280_2,3280_3,3280_4" - ), + url=self.__get_url("logout"), ) - _LOGGER.debug(f"Status Response {response2}") - - response_json2 = await response2.json(content_type=None) - _LOGGER.debug(response_json2) + _LOGGER.debug(f"Logout response {response}") - await self._session.request( + async def update_value(self, api_param, value): + response = await self._session.request( ssl=self.ssl, method="POST", - headers=HEADER_JSON, - url=self.__get_url("logout"), + headers=POST_HEADER_JSON, + url=self.__get_url("prop"), + json={api_param: {"id": api_param, "value": value}}, ) + _LOGGER.info(f"Set {api_param} value {value} response {response}") - response_json_combined = response_json - if response_json2 is not None: - response_json_combined["properties"] = ( - response_json["properties"] + response_json2["properties"] - ) - response_json_combined["total"] = ( - response_json["total"] + response_json2["total"] - ) - - _LOGGER.debug(response_json_combined) - - self._status = AlfenStatus(response_json_combined, self._status) + async def _do_update(self): + await self.login() + + properties = [] + for i in ["generic", "generic2", "meter1", "states", "temp"]: + nextRequest = True + offset = 0 + while (nextRequest): + response = await self._session.request( + ssl=self.ssl, + method="GET", + headers=HEADER_JSON, + url=self.__get_url( + "prop?cat={}&offset={}".format(i, offset) + ), + ) + _LOGGER.debug(f"Status Response {response}") + + response_json = await response.json(content_type=None) + if response_json is not None: + properties += response_json["properties"] + nextRequest = response_json["total"] > ( + offset + len(response_json["properties"])) + offset += len(response_json["properties"]) + + await self.logout() + + self.properties = properties async def async_get_info(self): response = await self._session.request( @@ -343,141 +355,17 @@ def __get_url(self, action): return "https://{}/api/{}".format(self.host, action) -class AlfenStatus: - def __init__(self, response, prev_status): - for prop in response["properties"]: - _LOGGER.debug("Prop") - _LOGGER.debug(prop) - - if prop["id"] == "2060_0": - self.uptime = max(0, prop["value"] / 1000 * 60) - elif prop["id"] == "2056_0": - self.bootups = prop["value"] - elif prop["id"] == "2221_3": - self.voltage_l1 = round(prop["value"], 2) - elif prop["id"] == "2221_4": - self.voltage_l2 = round(prop["value"], 2) - elif prop["id"] == "2221_5": - self.voltage_l3 = round(prop["value"], 2) - elif prop["id"] == "2221_A": - self.current_l1 = round(prop["value"], 2) - elif prop["id"] == "2221_B": - self.current_l2 = round(prop["value"], 2) - elif prop["id"] == "2221_C": - self.current_l3 = round(prop["value"], 2) - elif prop["id"] == "2221_16": - self.active_power_total = round(prop["value"], 2) - elif prop["id"] == "2201_0": - self.temperature = round(prop["value"], 2) - elif prop["id"] == "2501_2": - self.status = prop["value"] - elif prop["id"] == "2221_22": - self.meter_reading = round((prop["value"] / 1000), 2) - elif prop["id"] == "2129_0": - self.current_limit = prop["value"] - elif prop["id"] == "2126_0": - self.auth_mode = self.auth_mode_as_str(prop["value"]) - elif prop["id"] == "2068_0": - self.alb_safe_current = prop["value"] - elif prop["id"] == "2069_0": - self.alb_phase_connection = prop["value"] - elif prop["id"] == "2062_0": - self.max_station_current = prop["value"] - elif prop["id"] == "2064_0": - self.load_balancing_mode = prop["value"] - elif prop["id"] == "212B_0": - self.main_static_lb_max_current = round(prop["value"], 2) - elif prop["id"] == "212D_0": - self.main_active_lb_max_current = round(prop["value"], 2) - elif prop["id"] == "2185_0": - self.enable_phase_switching = prop["value"] - elif prop["id"] == "2053_0": - self.charging_box_identifier = prop["value"] - elif prop["id"] == "2057_0": - self.boot_reason = prop["value"] - elif prop["id"] == "2067_0": - self.max_smart_meter_current = prop["value"] - elif prop["id"] == "212F_1": - self.p1_measurements_1 = round(prop["value"], 2) - elif prop["id"] == "212F_2": - self.p1_measurements_2 = round(prop["value"], 2) - elif prop["id"] == "212F_3": - self.p1_measurements_3 = round(prop["value"], 2) - elif prop["id"] == "2100_0": - self.gprs_apn_name = prop["value"] - elif prop["id"] == "2101_0": - self.gprs_apn_user = prop["value"] - elif prop["id"] == "2102_0": - self.gprs_apn_password = prop["value"] - elif prop["id"] == "2104_0": - self.gprs_sim_imsi = prop["value"] - elif prop["id"] == "2105_0": - self.gprs_sim_iccid = prop["value"] - elif prop["id"] == "2112_0": - self.gprs_provider = prop["value"] - elif prop["id"] == "2104_0": - self.p1_measurements_3 = prop["value"] - elif prop["id"] == "2071_1": - self.comm_bo_url_wired_server_domain_and_port = prop["value"] - elif prop["id"] == "2071_2": - self.comm_bo_url_wired_server_path = prop["value"] - elif prop["id"] == "2072_1": - self.comm_dhcp_address_1 = prop["value"] - elif prop["id"] == "2073_1": - self.comm_netmask_address_1 = prop["value"] - elif prop["id"] == "2074_1": - self.comm_gateway_address_1 = prop["value"] - elif prop["id"] == "2075_1": - self.comm_ip_address_1 = prop["value"] - elif prop["id"] == "2076_0": - self.comm_bo_short_name = prop["value"] - elif prop["id"] == "2078_1": - self.comm_bo_url_gprs_server_domain_and_port = prop["value"] - elif prop["id"] == "2078_2": - self.comm_bo_url_gprs_server_path = prop["value"] - elif prop["id"] == "2079_1": - self.comm_gprs_dns_1 = prop["value"] - elif prop["id"] == "207A_1": - self.comm_dhcp_address_2 = prop["value"] - elif prop["id"] == "207B_1": - self.comm_netmask_address_2 = prop["value"] - elif prop["id"] == "207C_1": - self.comm_gateway_address_2 = prop["value"] - elif prop["id"] == "207D_1": - self.comm_ip_address_2 = prop["value"] - elif prop["id"] == "207E_1": - self.comm_wired_dns_1 = prop["value"] - elif prop["id"] == "207F_1": - self.comm_wired_dns_2 = prop["value"] - elif prop["id"] == "2080_1": - self.comm_gprs_dns_2 = prop["value"] - elif prop["id"] == "2081_0": - self.comm_protocol_name = prop["value"] - elif prop["id"] == "2082_0": - self.comm_protocol_version = prop["value"] - elif prop["id"] == "2110_0": - self.gprs_signal_strength = prop["value"] - elif prop["id"] == "3280_1": - self.lb_solar_charging_mode = self.solar_charging_mode( - prop["value"]) - elif prop["id"] == "3280_2": - self.lb_solar_charging_green_share = prop["value"] - elif prop["id"] == "3280_3": - self.lb_solar_charging_comfort_level = prop["value"] - elif prop["id"] == "3280_4": - self.lb_solar_charging_boost = prop["value"] - - def auth_mode_as_str(self, code): - switcher = {0: "Plug and Charge", 2: "RFID"} - return switcher.get(code, "Unknown") - - def solar_charging_mode(self, code): - switcher = {0: "Disable", 1: "Comfort", 2: "Green"} - return switcher.get(code, "Unknown") +# def auth_mode_as_str(self, code): +# switcher = {0: "Plug and Charge", 2: "RFID"} +# return switcher.get(code, "Unknown") + +# # def solar_charging_mode(self, code): +# # switcher = {0: "Disable", 1: "Comfort", 2: "Green"} +# # return switcher.get(code, "Unknown") class AlfenDeviceInfo: - def __init__(self, response): + def __init__(self, response) -> None: self.identity = response["Identity"] self.firmware_version = response["FWVersion"] self.model_id = response["Model"] diff --git a/custom_components/alfen_wallbox/entity.py b/custom_components/alfen_wallbox/entity.py index b00ec50..e0eddd8 100644 --- a/custom_components/alfen_wallbox/entity.py +++ b/custom_components/alfen_wallbox/entity.py @@ -1,122 +1,40 @@ +from datetime import timedelta import logging import ssl -from config.custom_components.alfen_wallbox.alfen import HEADER_JSON, POST_HEADER_JSON, AlfenDevice, AlfenStatus -from homeassistant.core import DOMAIN +from config.custom_components.alfen_wallbox.alfen import HEADER_JSON, POST_HEADER_JSON, AlfenDevice +from homeassistant.util import Throttle +from .const import DOMAIN as ALFEN_DOMAIN from homeassistant.helpers.entity import DeviceInfo, Entity _LOGGER = logging.getLogger(__name__) +MIN_TIME_BETWEEN_UPDATES = timedelta(seconds=30) + class AlfenEntity(Entity): def __init__(self, device: AlfenDevice) -> None: - self._host = device.host - self._attr_name = device.name - self._status = device.status - self._session = device._session - self._username = device.username - if self._username is None: - self._username = "admin" - self._password = device.password - # Default ciphers needed as of python 3.10 - context = ssl.create_default_context() - context.set_ciphers("DEFAULT") - context.check_hostname = False - context.verify_mode = ssl.CERT_NONE - self.ssl = context - self.device = device + """Initialize the Alfen entity.""" + self._device = device self._attr_device_info = DeviceInfo( - identifiers={(DOMAIN, self.device.id)}, + identifiers={(ALFEN_DOMAIN, self._device.name)}, manufacturer="Alfen", - model=self.device.info.model, - name=self.name, - sw_version=self.device.info.firmware_version, + model=self._device.info.model, + name=device.name, + sw_version=self._device.info.firmware_version, ) async def async_added_to_hass(self) -> None: """Add listener for state changes.""" await super().async_added_to_hass() - async def _on_update(self) -> None: - await self.login() - - # max 32 ids each time - response = await self._session.request( - ssl=self.ssl, - method="GET", - headers=HEADER_JSON, - url=self.__get_url( - "prop?ids=2060_0,2056_0,2221_3,2221_4,2221_5,2221_A,2221_B,2221_C,2221_16,2201_0,2501_2,2221_22,2129_0,2126_0,2068_0,2069_0,2062_0,2064_0,212B_0,212D_0,2185_0,2053_0,2067_0,212F_1,212F_2,212F_3,2100_0,2101_0,2102_0,2104_0,2105_0" - ), - ) - _LOGGER.debug(f"Status Response {response}") - - response_json = await response.json(content_type=None) - _LOGGER.debug(response_json) - - response2 = await self._session.request( - ssl=self.ssl, - method="GET", - headers=HEADER_JSON, - url=self.__get_url( - "prop?ids=2057_0,2112_0,2071_1,2071_2,2072_1,2073_1,2074_1,2075_1,2076_0,2078_1,2078_2,2079_1,207A_1,207B_1,207C_1,207D_1,207E_1,207F_1,2080_1,2081_0,2082_0,2110_0,3280_1,3280_2,3280_3,3280_4" - ), - ) - _LOGGER.debug(f"Status Response {response2}") - - response_json2 = await response2.json(content_type=None) - _LOGGER.debug(response_json2) - - await self.logout() - - response_json_combined = response_json - if response_json2 is not None: - response_json_combined["properties"] = ( - response_json["properties"] + response_json2["properties"] - ) - response_json_combined["total"] = ( - response_json["total"] + response_json2["total"] - ) - - _LOGGER.debug(response_json_combined) - - self._status = AlfenStatus(response_json_combined, self._status) - self.async_write_ha_state() - - def __get_url(self, action): - return "https://{}/api/{}".format(self._host, action) - - async def login(self): - await self._session.request( - ssl=self.ssl, - method="POST", - headers=HEADER_JSON, - url=self.__get_url("login"), - json={"username": self._username, "password": self._password}, - ) - - async def logout(self): - await self._session.request( - ssl=self.ssl, - method="POST", - headers=HEADER_JSON, - url=self.__get_url("logout"), - ) - async def update_state(self, api_param, value): """Get the state of the entity.""" - await self.login() + await self._device.login() - response = await self._session.request( - ssl=self.ssl, - method="POST", - headers=POST_HEADER_JSON, - url=self.__get_url("prop"), - json={api_param: {"id": api_param, "value": value}}, - ) - _LOGGER.info(f"Set {api_param} value {value} response {response}") + await self._device.update_value(api_param, value) - await self.logout() + await self._device.logout() diff --git a/custom_components/alfen_wallbox/select.py b/custom_components/alfen_wallbox/select.py index b2ab4e9..bae820a 100644 --- a/custom_components/alfen_wallbox/select.py +++ b/custom_components/alfen_wallbox/select.py @@ -1,5 +1,5 @@ import logging -from typing import Coroutine, Final, Any +from typing import Final, Any from dataclasses import dataclass from config.custom_components.alfen_wallbox.alfen import AlfenDevice @@ -37,8 +37,10 @@ class AlfenSelectDescription(SelectEntityDescription, AlfenSelectDescriptionMixi CHARGING_MODE_DICT: Final[dict[str, int]] = { "Disable": 0, "Comfort": 1, "Green": 2} +ON_OFF_DICT: Final[dict[str, int]] = { + "Off": 0, "On": 1} -SELECT_TYPES: Final[tuple[AlfenSelectDescription, ...]] = ( +ALFEN_SELECT_TYPES: Final[tuple[AlfenSelectDescription, ...]] = ( AlfenSelectDescription( key="lb_solar_charging_mode", name="Solar Charging Mode", @@ -47,6 +49,22 @@ class AlfenSelectDescription(SelectEntityDescription, AlfenSelectDescriptionMixi options_dict=CHARGING_MODE_DICT, api_param="3280_1", ), + AlfenSelectDescription( + key="enable_phase_switching", + name="Enable Phase Switching", + icon="mdi:ev-station", + options=list(ON_OFF_DICT), + options_dict=ON_OFF_DICT, + api_param="2185_0", + ), + AlfenSelectDescription( + key="lb_solar_charging_boost", + name="Solar Charging Boost", + icon="mdi:ev-station", + options=list(ON_OFF_DICT), + options_dict=ON_OFF_DICT, + api_param="3280_4", + ), ) @@ -58,8 +76,11 @@ async def async_setup_entry( """Add Alfen Select from a config_entry""" device = hass.data[ALFEN_DOMAIN][entry.entry_id] - async_add_entities([AlfenSelect(device, description) - for description in SELECT_TYPES]) + selects = [ + AlfenSelect(device, description) for description in ALFEN_SELECT_TYPES + ] + + async_add_entities(selects) class AlfenSelect(AlfenEntity, SelectEntity): @@ -67,30 +88,41 @@ class AlfenSelect(AlfenEntity, SelectEntity): values_dict: dict[int, str] - def __init__(self, device: AlfenDevice, description: AlfenSelectDescription) -> None: + def __init__(self, + device: AlfenDevice, + description: AlfenSelectDescription) -> None: + """Initialize.""" super().__init__(device) - self._attr_name = f"{description.name}" + self._device = device + self._attr_name = f"{device.name} {description.name}" + + self._attr_unique_id = f"{self._attr_unique_id}_{description.key}" self._attr_options = description.options self.entity_description = description - self._attr_unique_id = f"{self._attr_unique_id}_{description.key}" - self._device = device self.values_dict = {v: k for k, v in description.options_dict.items()} - self._async_update_attrs() async def async_select_option(self, option: str) -> None: + """Change the selected option.""" value = {v: k for k, v in self.values_dict.items()}[option] await self.update_state(self.entity_description.api_param, value) self.async_write_ha_state() @property def current_option(self) -> str | None: - return self._get_current_option() + """Return the current option.""" + value = self._get_current_option() + return self.values_dict.get(value) def _get_current_option(self) -> str | None: - return getattr(self._device.status, self.entity_description.key) + """Return the current option.""" + for prop in self._device.properties: + if prop['id'] == self.entity_description.api_param: + return prop['value'] + return None async def async_update(self): + """Update the entity.""" await self._device.async_update() @callback diff --git a/custom_components/alfen_wallbox/sensor.py b/custom_components/alfen_wallbox/sensor.py index f907a35..ffa51ac 100644 --- a/custom_components/alfen_wallbox/sensor.py +++ b/custom_components/alfen_wallbox/sensor.py @@ -1,20 +1,29 @@ import logging +from typing import Final +from dataclasses import dataclass import voluptuous as vol +from config.custom_components.alfen_wallbox.entity import AlfenEntity +from homeassistant import const +from homeassistant.config_entries import ConfigEntry +from homeassistant.const import UnitOfElectricCurrent, UnitOfElectricPotential, UnitOfEnergy, UnitOfInformation, UnitOfPower, UnitOfSpeed, UnitOfTemperature -from homeassistant.const import TEMP_CELSIUS +from homeassistant.core import HomeAssistant, callback from homeassistant.helpers.entity import Entity from homeassistant.components.sensor import ( DEVICE_CLASS_ENERGY, DEVICE_CLASS_POWER, - STATE_CLASS_TOTAL_INCREASING, - STATE_CLASS_MEASUREMENT, SensorEntity, + SensorEntityDescription, + SensorStateClass ) +from homeassistant.helpers.entity_platform import AddEntitiesCallback + from homeassistant.helpers import config_validation as cv, entity_platform from . import DOMAIN as ALFEN_DOMAIN + from .alfen import AlfenDevice from .const import ( SERVICE_REBOOT_WALLBOX, @@ -31,111 +40,469 @@ _LOGGER = logging.getLogger(__name__) -async def async_setup_platform(hass, config, async_add_entities, discovery_info=None): +@dataclass +class AlfenSensorDescriptionMixin: + """Define an entity description mixin for sensor entities.""" + + api_param: str + unit: str + + +@dataclass +class AlfenSensorDescription( + SensorEntityDescription, AlfenSensorDescriptionMixin +): + """Class to describe an Alfen sensor entity.""" + + +STATUS_DICT: Final[dict[str, int]] = { + 0: "Unknown", + 1: "Off", + 2: "Booting", + 3: "Booting Check Mains", + 4: "Available", + 5: "Prep. Authorising", + 6: "Prep. Authorised", + 7: "Prep. Cable connected", + 8: "Prep EV Connected", + 9: "Charging Preparing", + 10: "Vehicle connected", + 11: "Charging Active Normal", + 12: "Charging Active Simplified", + 13: "Charging Suyspended Over Current", + 14: "Charging Suspended HF Switching", + 15: "Charging Suspended EV Disconnected", + 16: "Finish Wait Vehicle", + 17: "Finished Wait Disconnect", + 18: "Error Protective Earth", + 19: "Error Powerline Fault", + 20: "Error Contactor Fault", + 21: "Error Charging", + 22: "Error Power Failure", + 23: "Error Temperature", + 24: "Error Illegal CP Value", + 25: "Error Illegal PP Value", + 26: "Error Too Many Restarts", + 27: "Error", + 28: "Error Message", + 29: "Error Message Not Authorised", + 30: "Error Message Cable Not Supported", + 31: "Error Message S2 Not Opened", + 32: "Error Message Time Out", + 33: "Reserved", + 34: "Inoperative", + 35: "Load Balancing Limited", + 36: "Load Balancing Forced Off", + 38: "Not Charging", + 39: "Solar Charging Wait", + 41: "Solar Charging", + 42: "Charge Point Ready, Waiting For Power", + 43: "Partial Solar Charging", +} + +ALFEN_SENSOR_TYPES: Final[tuple[AlfenSensorDescription, ...]] = ( + AlfenSensorDescription( + key="status", + name="Status Code", + icon="mdi:ev-station", + api_param="2501_2", + unit=None, + ), + AlfenSensorDescription( + key="uptime", + name="Uptime", + icon="mdi:timer-outline", + api_param="2060_0", + unit=None, + ), + AlfenSensorDescription( + key="bootups", + name="Bootups", + icon="mdi:reload", + api_param="2056_0", + unit=None, + ), + AlfenSensorDescription( + key="voltage_l1", + name="Voltage L1", + icon="mdi:flash", + api_param="2221_3", + unit=UnitOfElectricPotential.VOLT, + ), + AlfenSensorDescription( + key="voltage_l2", + name="Voltage L2", + icon="mdi:flash", + api_param="2221_4", + unit=UnitOfElectricPotential.VOLT, + ), + AlfenSensorDescription( + key="voltage_l3", + name="Voltage L3", + icon="mdi:flash", + api_param="2221_5", + unit=UnitOfElectricPotential.VOLT, + ), + AlfenSensorDescription( + key="current_l1", + name="Current L1", + icon="mdi:current-ac", + api_param="2221_A", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="current_l2", + name="Current L2", + icon="mdi:current-ac", + api_param="2221_B", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="current_l3", + name="Current L3", + icon="mdi:current-ac", + api_param="2221_C", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="active_power_total", + name="Active Power Total", + icon="mdi:circle-slice-3", + api_param="2221_16", + unit=UnitOfPower.WATT, + ), + AlfenSensorDescription( + key="meter_reading", + name="Meter Reading", + icon=None, + api_param="2221_22", + unit=UnitOfEnergy.KILO_WATT_HOUR, + ), + AlfenSensorDescription( + key="temperature", + name="Temperature", + icon="mdi:thermometer", + api_param="2201_0", + unit=UnitOfTemperature.CELSIUS, + ), + AlfenSensorDescription( + key="current_limit", + name="Current Limit", + icon="mdi:current-ac", + api_param="2129_0", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="auth_mode", + name="Authorization Mode", + icon=None, + api_param="2126_0", + unit=None, + ), + AlfenSensorDescription( + key="alb_safe_current", + name="Active Load Balancing Safe Current", + icon="mdi:current-ac", + api_param="2068_0", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="alb_phase_connection", + name="Active Load Balancing Phase Connection", + icon=None, + api_param="2069_0", + unit=None, + ), + AlfenSensorDescription( + key="max_station_current", + name="Maximum Smart Meter current", + icon="mdi:current-ac", + api_param="2062_0", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="load_balancing_mode", + name="Load Balancing Mode", + icon=None, + api_param="2064_0", + unit=None, + ), + AlfenSensorDescription( + key="main_static_lb_max_current", + name="Main Static Load Balancing Max Current", + icon="mdi:current-ac", + api_param="212B_0", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="main_active_lb_max_current", + name="Main Active Load Balancing Max Current", + icon="mdi:current-ac", + api_param="212D_0", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="charging_box_identifier", + name="Charging Box Identifier", + icon=None, + api_param="2053_0", + unit=None, + ), + AlfenSensorDescription( + key="boot_reason", + name="System Boot Reason", + icon=None, + api_param="2057_0", + unit=None, + ), + AlfenSensorDescription( + key="max_smart_meter_current", + name="Max Smart Meter Current", + icon="mdi:current-ac", + api_param="2067_0", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="p1_measurements_1", + name="P1 Meter Phase 1 Current", + icon="mdi:current-ac", + api_param="212F_1", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="p1_measurements_2", + name="P1 Meter Phase 2 Current", + icon="mdi:current-ac", + api_param="212F_2", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="p1_measurements_3", + name="P1 Meter Phase 3 Current", + icon="mdi:current-ac", + api_param="212F_3", + unit=UnitOfElectricCurrent.AMPERE, + ), + AlfenSensorDescription( + key="gprs_apn_name", + name="GPRS APN Name", + icon="mdi:antenna", + api_param="2100_0", + unit=None, + ), + AlfenSensorDescription( + key="gprs_apn_user", + name="GPRS APN User", + icon="mdi:antenna", + api_param="2101_0", + unit=None, + ), + AlfenSensorDescription( + key="gprs_apn_password", + name="GPRS APN Password", + icon="mdi:antenna", + api_param="2102_0", + unit=None, + ), + AlfenSensorDescription( + key="gprs_sim_imsi", + name="GPRS SIM IMSI", + icon="mdi:antenna", + api_param="2104_0", + unit=None, + ), + AlfenSensorDescription( + key="gprs_sim_iccid", + name="GPRS SIM Serial", + icon="mdi:antenna", + api_param="", + unit=None, + ), + AlfenSensorDescription( + key="gprs_provider", + name="GPRS Provider", + icon="mdi:antenna", + api_param="", + unit=None, + ), + AlfenSensorDescription( + key="comm_bo_url_wired_server_domain_and_port", + name="Wired Url Server Domain And Port", + icon="mdi:cable-data", + api_param="2071_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_bo_url_wired_server_path", + name="Wired Url Wired Server Path", + icon="mdi:cable-data", + api_param="2071_2", + unit=None, + ), + AlfenSensorDescription( + key="comm_dhcp_address_1", + name="GPRS DHCP Address", + icon="mdi:antenna", + api_param="2072_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_netmask_address_1", + name="GPRS Netmask", + icon="mdi:antenna", + api_param="2073_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_gateway_address_1", + name="GPRS Gateway Address", + icon="mdi:antenna", + api_param="2074_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_ip_address_1", + name="GPRS IP Address", + icon="mdi:antenna", + api_param="2075_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_bo_short_name", + name="Backoffice Short Name", + icon=None, + api_param="2076_0", + unit=None, + ), + AlfenSensorDescription( + key="comm_bo_url_gprs_server_domain_and_port", + name="GPRS Url Server Domain And Port", + icon="mdi:antenna", + api_param="2078_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_bo_url_gprs_server_path", + name="GPRS Url Server Path", + icon="mdi:antenna", + api_param="2078_2", + unit=None, + ), + AlfenSensorDescription( + key="comm_gprs_dns_1", + name="GPRS DNS 1", + icon="mdi:antenna", + api_param="2079_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_gprs_dns_2", + name="GPRS DNS 2", + icon="mdi:antenna", + api_param="2080_1", + unit=None, + ), + AlfenSensorDescription( + key="gprs_signal_strength", + name="GPRS Signal", + icon="mdi:antenna", + api_param="2110_0", + unit=const.SIGNAL_STRENGTH_DECIBELS, + ), + AlfenSensorDescription( + key="comm_dhcp_address_2", + name="Wired DHCP", + icon="mdi:cable-data", + api_param="207A_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_netmask_address_2", + name="Wired Netmask", + icon="mdi:cable-data", + api_param="207B_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_gateway_address_2", + name="Wired Gateway Address", + icon="mdi:cable-data", + api_param="207C_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_ip_address_2", + name="Wired IP Address", + icon="mdi:cable-data", + api_param="207D_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_wired_dns_1", + name="Wired DNS 1", + icon="mdi:cable-data", + api_param="207E_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_wired_dns_2", + name="Wired DNS 2", + icon="mdi:cable-data", + api_param="207F_1", + unit=None, + ), + AlfenSensorDescription( + key="comm_protocol_name", + name="Protocol Name", + icon=None, + api_param="2081_0", + unit=None, + ), + AlfenSensorDescription( + key="comm_protocol_version", + name="Protocol Version", + icon=None, + api_param="2082_0", + unit=None, + ), + AlfenSensorDescription( + key="lb_solar_charging_green_share", + name="Solar Charging Green Share %", + icon=None, + api_param="3280_2", + unit=const.PERCENTAGE, + ), + AlfenSensorDescription( + key="lb_solar_charging_comfort_level", + name="Solar Charging Comfort Level w", + icon=None, + api_param="3280_3", + unit=UnitOfPower.WATT, + ), +) + + +async def async_setup_platform( + hass: HomeAssistant, + config: ConfigEntry, + async_add_entities: AddEntitiesCallback, + discovery_info=None): pass -async def async_setup_entry(hass, entry, async_add_entities): +async def async_setup_entry( + hass: HomeAssistant, + entry: ConfigEntry, + async_add_entities: AddEntitiesCallback): """Set up using config_entry.""" - device = hass.data[ALFEN_DOMAIN].get(entry.entry_id) - async_add_entities( - [ - AlfenMainSensor(device), - AlfenSensor(device, "Status Code", "status"), - AlfenSensor(device, "Uptime", "uptime", "s"), - AlfenSensor(device, "Bootups", "bootups"), - AlfenSensor(device, "Voltage L1", "voltage_l1", "V"), - AlfenSensor(device, "Voltage L2", "voltage_l2", "V"), - AlfenSensor(device, "Voltage L3", "voltage_l3", "V"), - AlfenSensor(device, "Current L1", "current_l1", "A"), - AlfenSensor(device, "Current L2", "current_l2", "A"), - AlfenSensor(device, "Current L3", "current_l3", "A"), - AlfenSensor(device, "Active Power Total", "active_power_total", "W"), - AlfenSensor(device, "Meter Reading", "meter_reading", "kWh"), - AlfenSensor(device, "Temperature", "temperature", TEMP_CELSIUS), - AlfenSensor(device, "Current Limit", "current_limit", "A"), - AlfenSensor(device, "Authorization Mode", "auth_mode"), - AlfenSensor( - device, "Active Load Balancing Safe Current", "alb_safe_current", "A" - ), - AlfenSensor( - device, "Active Load Balancing Phase Connection", "alb_phase_connection" - ), - AlfenSensor( - device, "Maximum Smart Meter current", "max_station_current", "A" - ), - AlfenSensor(device, "Load Balancing Mode", "load_balancing_mode"), - AlfenSensor( - device, - "Main Static Load Balancing Max Current", - "main_static_lb_max_current", - "A", - ), - AlfenSensor( - device, - "Main Active Load Balancing Max Current", - "main_active_lb_max_current", - "A", - ), - AlfenSensor(device, "Enable Phase Switching", "enable_phase_switching"), - AlfenSensor(device, "Charging Box Identifier", "charging_box_identifier"), - AlfenSensor(device, "System Boot Reason", "boot_reason"), - AlfenSensor(device, "Max Smart Meter Current", "max_smart_meter_current"), - AlfenSensor(device, "P1 Meter Phase 1 Current", "p1_measurements_1", "A"), - AlfenSensor(device, "P1 Meter Phase 2 Current", "p1_measurements_2", "A"), - AlfenSensor(device, "P1 Meter Phase 3 Current", "p1_measurements_3", "A"), - AlfenSensor(device, "GPRS APN Name", "gprs_apn_name"), - AlfenSensor(device, "GPRS APN User", "gprs_apn_user"), - AlfenSensor(device, "GPRS APN Password", "gprs_apn_password"), - AlfenSensor(device, "GPRS SIM IMSI", "gprs_sim_imsi"), - AlfenSensor(device, "GPRS SIM Serial", "gprs_sim_iccid"), - AlfenSensor(device, "GPRS Provider", "gprs_provider"), - AlfenSensor( - device, - "Wired Url Server Domain And Port", - "comm_bo_url_wired_server_domain_and_port", - ), - AlfenSensor( - device, "Wired Url Wired Server Path", "comm_bo_url_wired_server_path" - ), - AlfenSensor(device, "GPRS DHCP Address", "comm_dhcp_address_1"), - AlfenSensor(device, "GPRS Netmask", "comm_netmask_address_1"), - AlfenSensor(device, "GPRS Gateway Address", "comm_gateway_address_1"), - AlfenSensor(device, "GPRS IP Address", "comm_ip_address_1"), - AlfenSensor(device, "Backoffice Short Name", "comm_bo_short_name"), - AlfenSensor( - device, - "GPRS Url Server Domain And Port", - "comm_bo_url_gprs_server_domain_and_port", - ), - AlfenSensor(device, "GPRS Url Server Path", "comm_bo_url_gprs_server_path"), - AlfenSensor(device, "GPRS DNS 1", "comm_gprs_dns_1"), - AlfenSensor(device, "GPRS DNS 2", "comm_gprs_dns_2"), - AlfenSensor(device, "GPRS Signal", "gprs_signal_strength"), - AlfenSensor(device, "Wired DHCP", "comm_dhcp_address_2"), - AlfenSensor(device, "Wired Netmask", "comm_netmask_address_2"), - AlfenSensor(device, "Wired Gateway Address", "comm_gateway_address_2"), - AlfenSensor(device, "Wired IP Address", "comm_ip_address_2"), - AlfenSensor(device, "Wired DNS 1", "comm_wired_dns_1"), - AlfenSensor(device, "Wired DNS 2", "comm_wired_dns_2"), - AlfenSensor(device, "Protocol Name", "comm_protocol_name"), - AlfenSensor(device, "Protocol Version", "comm_protocol_version"), - #AlfenSensor(device, "Solar Charging Mode", "lb_solar_charging_mode"), - AlfenSensor( - device, - "Solar Charging Green Share %", - "lb_solar_charging_green_share", - "%", - ), - AlfenSensor( - device, - "Solar Charging Comfort Level w", - "lb_solar_charging_comfort_level", - "W", - ), - AlfenSensor(device, "Solar Charging Boost", "lb_solar_charging_boost"), - ] - ) + device = hass.data[ALFEN_DOMAIN][entry.entry_id] + + sensors = [ + AlfenSensor(device, description) for description in ALFEN_SENSOR_TYPES + ] + + async_add_entities(sensors) + async_add_entities([AlfenMainSensor(device)]) platform = entity_platform.current_platform.get() @@ -202,11 +569,12 @@ async def async_setup_entry(hass, entry, async_add_entities): ) -class AlfenMainSensor(Entity): - def __init__(self, device: AlfenDevice): +class AlfenMainSensor(AlfenEntity): + def __init__(self, device: AlfenDevice) -> None: """Initialize the sensor.""" + super().__init__(device) self._device = device - self._name = f"{device.name}" + self._attr_name = f"{device.name}" self._sensor = "sensor" @property @@ -214,13 +582,9 @@ def unique_id(self): """Return a unique ID.""" return f"{self._device.id}-{self._sensor}" - @property - def name(self): - """Return the name of the sensor.""" - return self._name - @property def icon(self): + """Return the icon.""" return "mdi:car-electric" @property @@ -231,33 +595,43 @@ def state(self): return None async def async_reboot_wallbox(self): + """Reboot the wallbox.""" await self._device.reboot_wallbox() async def async_set_current_limit(self, limit): + """Set the current limit.""" await self._device.set_current_limit(limit) async def async_enable_rfid_auth_mode(self): + """Enable RFID authorization mode.""" await self._device.set_rfid_auth_mode(True) async def async_disable_rfid_auth_mode(self): + """Disable RFID authorization mode.""" await self._device.set_rfid_auth_mode(False) async def async_update(self): + """Update the sensor.""" await self._device.async_update() async def async_set_current_phase(self, phase): + """Set the current phase.""" await self._device.set_current_phase(phase) async def async_enable_phase_switching(self): + """Enable phase switching.""" await self._device.set_phase_switching(True) async def async_disable_phase_switching(self): + """Disable phase switching.""" await self._device.set_phase_switching(False) async def async_set_green_share(self, value): + """Set the green share.""" await self._device.set_green_share(value) async def async_set_comfort_power(self, value): + """Set the comfort power.""" await self._device.set_comfort_power(value) @property @@ -266,109 +640,65 @@ def device_info(self): return self._device.device_info def status_as_str(self): - switcher = { - 0: "Unknown", - 1: "Off", - 2: "Booting", - 3: "Booting Check Mains", - 4: "Available", - 5: "Prep. Authorising", - 6: "Prep. Authorised", - 7: "Prep. Cable connected", - 8: "Prep EV Connected", - 9: "Charging Preparing", - 10: "Vehicle connected", - 11: "Charging Active Normal", - 12: "Charging Active Simplified", - 13: "Charging Suyspended Over Current", - 14: "Charging Suspended HF Switching", - 15: "Charging Suspended EV Disconnected", - 16: "Finish Wait Vehicle", - 17: "Finished Wait Disconnect", - 18: "Error Protective Earth", - 19: "Error Powerline Fault", - 20: "Error Contactor Fault", - 21: "Error Charging", - 22: "Error Power Failure", - 23: "Error Temperature", - 24: "Error Illegal CP Value", - 25: "Error Illegal PP Value", - 26: "Error Too Many Restarts", - 27: "Error", - 28: "Error Message", - 29: "Error Message Not Authorised", - 30: "Error Message Cable Not Supported", - 31: "Error Message S2 Not Opened", - 32: "Error Message Time Out", - 33: "Reserved", - 34: "Inoperative", - 35: "Load Balancing Limited", - 36: "Load Balancing Forced Off", - 38: "Not Charging", - 39: "Solar Charging Wait", - 41: "Solar Charging", - 42: "Charge Point Ready, Waiting For Power", - 43: "Partial Solar Charging", - } - return switcher.get(self._device.status.status, "Unknown") - - -class AlfenSensor(SensorEntity): - def __init__(self, device: AlfenDevice, name, sensor, unit=None): + """Return the status as a string.""" + return STATUS_DICT.get(self._device.status.status, "Unknown") + + +class AlfenSensor(AlfenEntity, SensorEntity): + """Representation of a Alfen Sensor.""" + + entity_description: AlfenSensorDescription + + def __init__(self, + device: AlfenDevice, + description: AlfenSensorDescription) -> None: """Initialize the sensor.""" + super().__init__(device) self._device = device - self._name = f"{device.name} {name}" - self._sensor = sensor - self._unit = unit - if self._sensor == "active_power_total": - _LOGGER.info(f"Initiating State sensors {self._name}") + self._attr_name = f"{device.name} {description.name}" + _LOGGER.info("Initiating State sensors %s", self._attr_name) + self._attr_unique_id = f"{self._attr_unique_id}-{description.key}" + self.entity_description = description + if self.entity_description.key == "active_power_total": + _LOGGER.info(f"Initiating State sensors {self._attr_name}") self._attr_device_class = DEVICE_CLASS_POWER - self._attr_state_class = STATE_CLASS_MEASUREMENT - elif self._sensor == "uptime": - _LOGGER.info(f"Initiating State sensors {self._name}") - self._attr_state_class = STATE_CLASS_TOTAL_INCREASING - elif self._sensor == "meter_reading": - _LOGGER.info(f"Initiating State sensors {self._name}") + self._attr_state_class = SensorStateClass.MEASUREMENT + elif self.entity_description.key == "uptime": + _LOGGER.info(f"Initiating State sensors {self._attr_name}") + self._attr_state_class = SensorStateClass.TOTAL_INCREASING + elif self.entity_description.key == "meter_reading": + _LOGGER.info(f"Initiating State sensors {self._attr_name}") self._attr_device_class = DEVICE_CLASS_ENERGY - self._attr_state_class = STATE_CLASS_TOTAL_INCREASING + self._attr_state_class = SensorStateClass.TOTAL_INCREASING + self._async_update_attrs() + + def _get_current_value(self): + """Get the current value.""" + for prop in self._device.properties: + if prop['id'] == self.entity_description.api_param: + return prop['value'] + return None +# return getattr(self._device.properties, self.entity_description.key) + + @callback + def _async_update_attrs(self) -> None: + """Update the state and attributes.""" + self._attr_native_value = self._get_current_value() @property def unique_id(self): """Return a unique ID.""" - return f"{self._device.id}-{self._sensor}" + return f"{self._device.id}-{self.entity_description.key}" @property def name(self): """Return the name of the sensor.""" - return self._name + return self._attr_name @property def icon(self): """Return the icon of the sensor.""" - icon = None - if self._sensor == "temperature": - icon = "mdi:thermometer" - elif ( - (self._sensor.startswith("current_")) - | (self._sensor.startswith("p1_measurements")) - | ("_current" in self._sensor) - ): - icon = "mdi:current-ac" - elif self._sensor.startswith("voltage_"): - icon = "mdi:flash" - elif self._sensor == "uptime": - icon = "mdi:timer-outline" - elif self._sensor == "bootups": - icon = "mdi:reload" - elif self._sensor == "active_power_total": - icon = "mdi:circle-slice-3" - elif ("gprs_" in self._sensor) | ("_address_1" in self._sensor): - icon = "mdi:antenna" - elif ("wired_" in self._sensor) | ("_address_2" in self._sensor): - icon = "mdi:cable-data" - elif self._sensor.startswith("lb_solar_charging"): - icon = "mdi:solar-power" - return icon + return self.entity_description.icon @property def native_value(self): @@ -378,19 +708,22 @@ def native_value(self): @property def native_unit_of_measurement(self): """Return the unit the value is expressed in.""" - return self._unit + return self.entity_description.unit @property def state(self): """Return the state of the sensor.""" - if self._sensor in self._device.status.__dict__: - return self._device.status.__dict__[self._sensor] + for prop in self._device.properties: + if prop['id'] == self.entity_description.api_param: + return prop['value'] @property def unit_of_measurement(self): - return self._unit + """Return the unit of measurement.""" + return self.entity_description.unit async def async_update(self): + """Get the latest data and updates the states.""" await self._device.async_update() @property From 56da2d5949a35149d13f949984b90dd98f9a1259 Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Thu, 29 Jun 2023 21:44:17 +0200 Subject: [PATCH 03/11] fix push to dev --- custom_components/alfen_wallbox/entity.py | 4 +--- custom_components/alfen_wallbox/select.py | 3 +-- custom_components/alfen_wallbox/sensor.py | 3 +-- 3 files changed, 3 insertions(+), 7 deletions(-) diff --git a/custom_components/alfen_wallbox/entity.py b/custom_components/alfen_wallbox/entity.py index e0eddd8..f0c11ef 100644 --- a/custom_components/alfen_wallbox/entity.py +++ b/custom_components/alfen_wallbox/entity.py @@ -1,9 +1,7 @@ from datetime import timedelta import logging -import ssl -from config.custom_components.alfen_wallbox.alfen import HEADER_JSON, POST_HEADER_JSON, AlfenDevice -from homeassistant.util import Throttle +from .alfen import AlfenDevice from .const import DOMAIN as ALFEN_DOMAIN from homeassistant.helpers.entity import DeviceInfo, Entity diff --git a/custom_components/alfen_wallbox/select.py b/custom_components/alfen_wallbox/select.py index bae820a..67197e7 100644 --- a/custom_components/alfen_wallbox/select.py +++ b/custom_components/alfen_wallbox/select.py @@ -2,8 +2,7 @@ from typing import Final, Any from dataclasses import dataclass -from config.custom_components.alfen_wallbox.alfen import AlfenDevice -from config.custom_components.alfen_wallbox.entity import AlfenEntity +from .entity import AlfenEntity from homeassistant.config_entries import ConfigEntry from homeassistant.helpers.entity_platform import AddEntitiesCallback diff --git a/custom_components/alfen_wallbox/sensor.py b/custom_components/alfen_wallbox/sensor.py index ffa51ac..400bdbe 100644 --- a/custom_components/alfen_wallbox/sensor.py +++ b/custom_components/alfen_wallbox/sensor.py @@ -3,7 +3,7 @@ from dataclasses import dataclass import voluptuous as vol -from config.custom_components.alfen_wallbox.entity import AlfenEntity +from .entity import AlfenEntity from homeassistant import const from homeassistant.config_entries import ConfigEntry from homeassistant.const import UnitOfElectricCurrent, UnitOfElectricPotential, UnitOfEnergy, UnitOfInformation, UnitOfPower, UnitOfSpeed, UnitOfTemperature @@ -656,7 +656,6 @@ def __init__(self, super().__init__(device) self._device = device self._attr_name = f"{device.name} {description.name}" - _LOGGER.info("Initiating State sensors %s", self._attr_name) self._attr_unique_id = f"{self._attr_unique_id}-{description.key}" self.entity_description = description if self.entity_description.key == "active_power_total": From 4729d57ed7fb3ee2eebbe602067bf33d840b6fda Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Thu, 29 Jun 2023 22:55:27 +0200 Subject: [PATCH 04/11] round numbers, wallbox status, auth_mode --- custom_components/alfen_wallbox/alfen.py | 13 +--- custom_components/alfen_wallbox/sensor.py | 92 ++++++++++++++++++++--- 2 files changed, 84 insertions(+), 21 deletions(-) diff --git a/custom_components/alfen_wallbox/alfen.py b/custom_components/alfen_wallbox/alfen.py index 201e9c8..ca8c2e3 100644 --- a/custom_components/alfen_wallbox/alfen.py +++ b/custom_components/alfen_wallbox/alfen.py @@ -101,13 +101,13 @@ async def update_value(self, api_param, value): url=self.__get_url("prop"), json={api_param: {"id": api_param, "value": value}}, ) - _LOGGER.info(f"Set {api_param} value {value} response {response}") + _LOGGER.debug(f"Set {api_param} value {value} response {response}") async def _do_update(self): await self.login() properties = [] - for i in ["generic", "generic2", "meter1", "states", "temp"]: + for i in ("generic", "generic2", "meter1", "states", "temp"): nextRequest = True offset = 0 while (nextRequest): @@ -355,15 +355,6 @@ def __get_url(self, action): return "https://{}/api/{}".format(self.host, action) -# def auth_mode_as_str(self, code): -# switcher = {0: "Plug and Charge", 2: "RFID"} -# return switcher.get(code, "Unknown") - -# # def solar_charging_mode(self, code): -# # switcher = {0: "Disable", 1: "Comfort", 2: "Green"} -# # return switcher.get(code, "Unknown") - - class AlfenDeviceInfo: def __init__(self, response) -> None: self.identity = response["Identity"] diff --git a/custom_components/alfen_wallbox/sensor.py b/custom_components/alfen_wallbox/sensor.py index 400bdbe..b849fc5 100644 --- a/custom_components/alfen_wallbox/sensor.py +++ b/custom_components/alfen_wallbox/sensor.py @@ -46,6 +46,7 @@ class AlfenSensorDescriptionMixin: api_param: str unit: str + round_digits: int | None @dataclass @@ -55,7 +56,13 @@ class AlfenSensorDescription( """Class to describe an Alfen sensor entity.""" -STATUS_DICT: Final[dict[str, int]] = { +AUTH_MODE_DICT: Final[dict[int, str]] = { + 0: "Plug and Charge", + 2: "RFID" +} + + +STATUS_DICT: Final[dict[int, str]] = { 0: "Unknown", 1: "Off", 2: "Booting", @@ -107,6 +114,7 @@ class AlfenSensorDescription( icon="mdi:ev-station", api_param="2501_2", unit=None, + round_digits=None, ), AlfenSensorDescription( key="uptime", @@ -114,6 +122,7 @@ class AlfenSensorDescription( icon="mdi:timer-outline", api_param="2060_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="bootups", @@ -121,6 +130,7 @@ class AlfenSensorDescription( icon="mdi:reload", api_param="2056_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="voltage_l1", @@ -128,6 +138,7 @@ class AlfenSensorDescription( icon="mdi:flash", api_param="2221_3", unit=UnitOfElectricPotential.VOLT, + round_digits=2, ), AlfenSensorDescription( key="voltage_l2", @@ -135,6 +146,7 @@ class AlfenSensorDescription( icon="mdi:flash", api_param="2221_4", unit=UnitOfElectricPotential.VOLT, + round_digits=2, ), AlfenSensorDescription( key="voltage_l3", @@ -142,6 +154,7 @@ class AlfenSensorDescription( icon="mdi:flash", api_param="2221_5", unit=UnitOfElectricPotential.VOLT, + round_digits=2, ), AlfenSensorDescription( key="current_l1", @@ -149,6 +162,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="2221_A", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="current_l2", @@ -156,6 +170,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="2221_B", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="current_l3", @@ -163,6 +178,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="2221_C", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="active_power_total", @@ -170,6 +186,7 @@ class AlfenSensorDescription( icon="mdi:circle-slice-3", api_param="2221_16", unit=UnitOfPower.WATT, + round_digits=None, ), AlfenSensorDescription( key="meter_reading", @@ -177,6 +194,7 @@ class AlfenSensorDescription( icon=None, api_param="2221_22", unit=UnitOfEnergy.KILO_WATT_HOUR, + round_digits=None, ), AlfenSensorDescription( key="temperature", @@ -184,6 +202,7 @@ class AlfenSensorDescription( icon="mdi:thermometer", api_param="2201_0", unit=UnitOfTemperature.CELSIUS, + round_digits=2, ), AlfenSensorDescription( key="current_limit", @@ -191,6 +210,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="2129_0", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="auth_mode", @@ -198,6 +218,7 @@ class AlfenSensorDescription( icon=None, api_param="2126_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="alb_safe_current", @@ -205,6 +226,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="2068_0", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="alb_phase_connection", @@ -212,6 +234,7 @@ class AlfenSensorDescription( icon=None, api_param="2069_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="max_station_current", @@ -219,6 +242,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="2062_0", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="load_balancing_mode", @@ -226,6 +250,7 @@ class AlfenSensorDescription( icon=None, api_param="2064_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="main_static_lb_max_current", @@ -233,6 +258,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="212B_0", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="main_active_lb_max_current", @@ -240,6 +266,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="212D_0", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="charging_box_identifier", @@ -247,6 +274,7 @@ class AlfenSensorDescription( icon=None, api_param="2053_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="boot_reason", @@ -254,6 +282,7 @@ class AlfenSensorDescription( icon=None, api_param="2057_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="max_smart_meter_current", @@ -261,6 +290,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="2067_0", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="p1_measurements_1", @@ -268,6 +298,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="212F_1", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="p1_measurements_2", @@ -275,6 +306,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="212F_2", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="p1_measurements_3", @@ -282,6 +314,7 @@ class AlfenSensorDescription( icon="mdi:current-ac", api_param="212F_3", unit=UnitOfElectricCurrent.AMPERE, + round_digits=2, ), AlfenSensorDescription( key="gprs_apn_name", @@ -289,6 +322,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2100_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="gprs_apn_user", @@ -296,6 +330,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2101_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="gprs_apn_password", @@ -303,6 +338,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2102_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="gprs_sim_imsi", @@ -310,6 +346,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2104_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="gprs_sim_iccid", @@ -317,6 +354,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="", unit=None, + round_digits=None, ), AlfenSensorDescription( key="gprs_provider", @@ -324,6 +362,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_bo_url_wired_server_domain_and_port", @@ -331,6 +370,7 @@ class AlfenSensorDescription( icon="mdi:cable-data", api_param="2071_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_bo_url_wired_server_path", @@ -338,6 +378,7 @@ class AlfenSensorDescription( icon="mdi:cable-data", api_param="2071_2", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_dhcp_address_1", @@ -345,6 +386,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2072_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_netmask_address_1", @@ -352,6 +394,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2073_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_gateway_address_1", @@ -359,6 +402,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2074_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_ip_address_1", @@ -366,6 +410,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2075_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_bo_short_name", @@ -373,6 +418,7 @@ class AlfenSensorDescription( icon=None, api_param="2076_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_bo_url_gprs_server_domain_and_port", @@ -380,6 +426,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2078_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_bo_url_gprs_server_path", @@ -387,6 +434,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2078_2", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_gprs_dns_1", @@ -394,6 +442,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2079_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_gprs_dns_2", @@ -401,6 +450,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2080_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="gprs_signal_strength", @@ -408,6 +458,7 @@ class AlfenSensorDescription( icon="mdi:antenna", api_param="2110_0", unit=const.SIGNAL_STRENGTH_DECIBELS, + round_digits=None, ), AlfenSensorDescription( key="comm_dhcp_address_2", @@ -415,6 +466,7 @@ class AlfenSensorDescription( icon="mdi:cable-data", api_param="207A_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_netmask_address_2", @@ -422,6 +474,7 @@ class AlfenSensorDescription( icon="mdi:cable-data", api_param="207B_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_gateway_address_2", @@ -429,6 +482,7 @@ class AlfenSensorDescription( icon="mdi:cable-data", api_param="207C_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_ip_address_2", @@ -436,6 +490,7 @@ class AlfenSensorDescription( icon="mdi:cable-data", api_param="207D_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_wired_dns_1", @@ -443,6 +498,7 @@ class AlfenSensorDescription( icon="mdi:cable-data", api_param="207E_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_wired_dns_2", @@ -450,6 +506,7 @@ class AlfenSensorDescription( icon="mdi:cable-data", api_param="207F_1", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_protocol_name", @@ -457,6 +514,7 @@ class AlfenSensorDescription( icon=None, api_param="2081_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="comm_protocol_version", @@ -464,6 +522,7 @@ class AlfenSensorDescription( icon=None, api_param="2082_0", unit=None, + round_digits=None, ), AlfenSensorDescription( key="lb_solar_charging_green_share", @@ -471,6 +530,7 @@ class AlfenSensorDescription( icon=None, api_param="3280_2", unit=const.PERCENTAGE, + round_digits=None, ), AlfenSensorDescription( key="lb_solar_charging_comfort_level", @@ -478,6 +538,7 @@ class AlfenSensorDescription( icon=None, api_param="3280_3", unit=UnitOfPower.WATT, + round_digits=None, ), ) @@ -502,7 +563,10 @@ async def async_setup_entry( ] async_add_entities(sensors) - async_add_entities([AlfenMainSensor(device)]) + + # get the first type in the typle + + async_add_entities([AlfenMainSensor(device, ALFEN_SENSOR_TYPES[0])]) platform = entity_platform.current_platform.get() @@ -570,12 +634,13 @@ async def async_setup_entry( class AlfenMainSensor(AlfenEntity): - def __init__(self, device: AlfenDevice) -> None: + def __init__(self, device: AlfenDevice, description: AlfenSensorDescription) -> None: """Initialize the sensor.""" super().__init__(device) self._device = device self._attr_name = f"{device.name}" self._sensor = "sensor" + self.entity_description = description @property def unique_id(self): @@ -590,9 +655,12 @@ def icon(self): @property def state(self): """Return the state of the sensor.""" - if self._device.status is not None: - return self.status_as_str() - return None + for prop in self._device.properties: + if prop['id'] == self.entity_description.api_param: + if self.entity_description.round_digits is not None: + return round(prop['value'], self.entity_description.round_digits) + return STATUS_DICT.get(prop['value'], 'Unknown') + return 'Unknown' async def async_reboot_wallbox(self): """Reboot the wallbox.""" @@ -639,10 +707,6 @@ def device_info(self): """Return a device description for device registry.""" return self._device.device_info - def status_as_str(self): - """Return the status as a string.""" - return STATUS_DICT.get(self._device.status.status, "Unknown") - class AlfenSensor(AlfenEntity, SensorEntity): """Representation of a Alfen Sensor.""" @@ -714,6 +778,14 @@ def state(self): """Return the state of the sensor.""" for prop in self._device.properties: if prop['id'] == self.entity_description.api_param: + if self.entity_description.round_digits is not None: + return round(prop['value'], self.entity_description.round_digits) + + # some exception of return value + # auth_mode + if self.entity_description.api_param == "2126_0": + return AUTH_MODE_DICT.get(prop['value'], 'Unknown') + return prop['value'] @property From 26c72e77310e93d6a28f666683541ec780c23694 Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Fri, 30 Jun 2023 09:57:01 +0200 Subject: [PATCH 05/11] add more sensor exception return value --- custom_components/alfen_wallbox/sensor.py | 15 ++++++++++++--- 1 file changed, 12 insertions(+), 3 deletions(-) diff --git a/custom_components/alfen_wallbox/sensor.py b/custom_components/alfen_wallbox/sensor.py index b849fc5..13349fd 100644 --- a/custom_components/alfen_wallbox/sensor.py +++ b/custom_components/alfen_wallbox/sensor.py @@ -7,6 +7,7 @@ from homeassistant import const from homeassistant.config_entries import ConfigEntry from homeassistant.const import UnitOfElectricCurrent, UnitOfElectricPotential, UnitOfEnergy, UnitOfInformation, UnitOfPower, UnitOfSpeed, UnitOfTemperature +import datetime from homeassistant.core import HomeAssistant, callback from homeassistant.helpers.entity import Entity @@ -778,14 +779,22 @@ def state(self): """Return the state of the sensor.""" for prop in self._device.properties: if prop['id'] == self.entity_description.api_param: - if self.entity_description.round_digits is not None: - return round(prop['value'], self.entity_description.round_digits) - # some exception of return value # auth_mode if self.entity_description.api_param == "2126_0": return AUTH_MODE_DICT.get(prop['value'], 'Unknown') + # meter_reading from w to kWh + if self.entity_description.api_param == "2221_22": + return round((prop["value"] / 1000), 2) + + # change milliseconds to HH:MM:SS + if self.entity_description.api_param == "2060_0": + return str(datetime.timedelta(milliseconds=prop['value'])).split('.', maxsplit=1)[0] + + if self.entity_description.round_digits is not None: + return round(prop['value'], self.entity_description.round_digits) + return prop['value'] @property From f990daae5e1402ce9890122487350aa0967d484b Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Fri, 30 Jun 2023 10:38:13 +0200 Subject: [PATCH 06/11] fix active power rounding --- custom_components/alfen_wallbox/alfen.py | 1 - custom_components/alfen_wallbox/sensor.py | 2 +- 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/custom_components/alfen_wallbox/alfen.py b/custom_components/alfen_wallbox/alfen.py index ca8c2e3..2b2f81c 100644 --- a/custom_components/alfen_wallbox/alfen.py +++ b/custom_components/alfen_wallbox/alfen.py @@ -129,7 +129,6 @@ async def _do_update(self): offset += len(response_json["properties"]) await self.logout() - self.properties = properties async def async_get_info(self): diff --git a/custom_components/alfen_wallbox/sensor.py b/custom_components/alfen_wallbox/sensor.py index 13349fd..b0c7066 100644 --- a/custom_components/alfen_wallbox/sensor.py +++ b/custom_components/alfen_wallbox/sensor.py @@ -187,7 +187,7 @@ class AlfenSensorDescription( icon="mdi:circle-slice-3", api_param="2221_16", unit=UnitOfPower.WATT, - round_digits=None, + round_digits=2, ), AlfenSensorDescription( key="meter_reading", From d255d9528151fda71bf998e6a4934cf059a7c25d Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Sat, 1 Jul 2023 22:44:08 +0200 Subject: [PATCH 07/11] add binary_sensor and switch --- custom_components/alfen_wallbox/__init__.py | 5 +- custom_components/alfen_wallbox/alfen.py | 3 +- .../alfen_wallbox/binary_sensor.py | 82 +++++++++++++++++ custom_components/alfen_wallbox/select.py | 92 ++++++++++++++++--- custom_components/alfen_wallbox/sensor.py | 92 ++++++++----------- custom_components/alfen_wallbox/switch.py | 92 +++++++++++++++++++ 6 files changed, 295 insertions(+), 71 deletions(-) create mode 100644 custom_components/alfen_wallbox/binary_sensor.py create mode 100644 custom_components/alfen_wallbox/switch.py diff --git a/custom_components/alfen_wallbox/__init__.py b/custom_components/alfen_wallbox/__init__.py index 91cf7cb..dde1094 100644 --- a/custom_components/alfen_wallbox/__init__.py +++ b/custom_components/alfen_wallbox/__init__.py @@ -25,7 +25,10 @@ TIMEOUT, ) -PLATFORMS = [Platform.SENSOR, Platform.SELECT] +PLATFORMS = [Platform.SENSOR, + Platform.SELECT, + Platform.BINARY_SENSOR, + Platform.SWITCH] SCAN_INTERVAL = timedelta(seconds=60) _LOGGER = logging.getLogger(__name__) diff --git a/custom_components/alfen_wallbox/alfen.py b/custom_components/alfen_wallbox/alfen.py index 2b2f81c..aa532a8 100644 --- a/custom_components/alfen_wallbox/alfen.py +++ b/custom_components/alfen_wallbox/alfen.py @@ -101,7 +101,7 @@ async def update_value(self, api_param, value): url=self.__get_url("prop"), json={api_param: {"id": api_param, "value": value}}, ) - _LOGGER.debug(f"Set {api_param} value {value} response {response}") + _LOGGER.info(f"Set {api_param} value {value} response {response}") async def _do_update(self): await self.login() @@ -129,6 +129,7 @@ async def _do_update(self): offset += len(response_json["properties"]) await self.logout() + self.properties = properties async def async_get_info(self): diff --git a/custom_components/alfen_wallbox/binary_sensor.py b/custom_components/alfen_wallbox/binary_sensor.py new file mode 100644 index 0000000..b11b1a6 --- /dev/null +++ b/custom_components/alfen_wallbox/binary_sensor.py @@ -0,0 +1,82 @@ +import logging + +from dataclasses import dataclass +from typing import Final +from config.custom_components.alfen_wallbox.alfen import AlfenDevice +from config.custom_components.alfen_wallbox.entity import AlfenEntity + +from homeassistant.components.binary_sensor import BinarySensorDeviceClass, BinarySensorEntity, BinarySensorEntityDescription +from homeassistant.config_entries import ConfigEntry +from homeassistant.core import HomeAssistant +from homeassistant.helpers.entity_platform import AddEntitiesCallback + +from . import DOMAIN as ALFEN_DOMAIN + +_LOGGER = logging.getLogger(__name__) + + +@dataclass +class AlfenBinaryDescriptionMixin: + """Define an entity description mixin for binary sensor entities.""" + + api_param: str + + +@dataclass +class AlfenBinaryDescription(BinarySensorEntityDescription, AlfenBinaryDescriptionMixin): + """Class to describe an Alfen binary sensor entity.""" + + +ALFEN_BINARY_SENSOR_TYPES: Final[tuple[AlfenBinaryDescription, ...]] = ( + AlfenBinaryDescription( + key="system_date_light_savings", + name="System Daylight Savings", + device_class=None, + api_param="205B_0", + ), +) + + +async def async_setup_entry( + hass: HomeAssistant, + entry: ConfigEntry, + async_add_entities: AddEntitiesCallback, +) -> None: + """Set up Alfen binary sensor entities from a config entry.""" + device = hass.data[ALFEN_DOMAIN][entry.entry_id] + binaries = [AlfenBinarySensor(device, description) + for description in ALFEN_BINARY_SENSOR_TYPES] + + async_add_entities(binaries) + + +class AlfenBinarySensor(AlfenEntity, BinarySensorEntity): + """Define an Alfen binary sensor.""" + + # entity_description: AlfenBinaryDescriptionMixin + + def __init__(self, + device: AlfenDevice, + description: AlfenBinaryDescription + ) -> None: + """Initialize.""" + super().__init__(device) + self._device = device + self._attr_name = f"{device.name} {description.name}" + self._attr_unique_id = f"{self._attr_unique_id}_{description.key}" + self.entity_description = description + + @property + def available(self) -> bool: + for prop in self._device.properties: + if prop["id"] == self.entity_description.api_param: + return True + return False + + @property + def is_on(self) -> bool: + for prop in self._device.properties: + if prop["id"] == self.entity_description.api_param: + return prop["value"] == 1 + + return False diff --git a/custom_components/alfen_wallbox/select.py b/custom_components/alfen_wallbox/select.py index 67197e7..4aaa633 100644 --- a/custom_components/alfen_wallbox/select.py +++ b/custom_components/alfen_wallbox/select.py @@ -36,8 +36,43 @@ class AlfenSelectDescription(SelectEntityDescription, AlfenSelectDescriptionMixi CHARGING_MODE_DICT: Final[dict[str, int]] = { "Disable": 0, "Comfort": 1, "Green": 2} -ON_OFF_DICT: Final[dict[str, int]] = { - "Off": 0, "On": 1} +ON_OFF_DICT: Final[dict[str, int]] = {"Off": 0, "On": 1} + +PHASE_ROTATION_DICT: Final[dict[str, str]] = { + "L1": "L1", + "L2": "L2", + "L3": "L3", + "L1,L2,L3": "L1L2L3", + "L1,L3,L2": "L1L3L2", + "L2,L1,L3": "L2L1L3", + "L2,L3,L1": "L2L3L1", + "L3,L1,L2": "L3L1L2", + "L3,L2,L1": "L3L2L1", +} + +SAFE_AMPS_DICT: Final[dict[str, int]] = { + "1 A": 1, + "2 A": 2, + "3 A": 3, + "4 A": 4, + "5 A": 5, + "6 A": 6, + "7 A": 7, + "8 A": 8, + "9 A": 9, + "10 A": 10, +} + +AUTH_MODE_DICT: Final[dict[str, int]] = { + "Plug and Charge": 0, + "RFID": 2 +} + +LOAD_BALANCE_MODE_DICT: Final[dict[str, int]] = { + "Modbus TCP/IP": 0, + "DSMR4.x / SMR5.0 (P1)": 3 +} + ALFEN_SELECT_TYPES: Final[tuple[AlfenSelectDescription, ...]] = ( AlfenSelectDescription( @@ -64,20 +99,51 @@ class AlfenSelectDescription(SelectEntityDescription, AlfenSelectDescriptionMixi options_dict=ON_OFF_DICT, api_param="3280_4", ), + AlfenSelectDescription( + key="alb_phase_connection", + name="Active Load Balancing Phase Connection", + icon=None, + options=list(PHASE_ROTATION_DICT), + options_dict=PHASE_ROTATION_DICT, + api_param="2069_0", + ), + AlfenSelectDescription( + key="alb_safe_current", + name="Active Load Balancing Safe Current", + icon="mdi:current-ac", + options=list(SAFE_AMPS_DICT), + options_dict=SAFE_AMPS_DICT, + api_param="2068_0", + ), + + AlfenSelectDescription( + key="auth_mode", + name="Authorization Mode", + icon="mdi:key", + options=list(AUTH_MODE_DICT), + options_dict=AUTH_MODE_DICT, + api_param="2126_0", + ), + + AlfenSelectDescription( + key="load_balancing_mode", + name="Load Balancing Mode", + icon="mdi:scale-balance", + options=list(LOAD_BALANCE_MODE_DICT), + options_dict=LOAD_BALANCE_MODE_DICT, + api_param="2064_0", + ), ) async def async_setup_entry( - hass: HomeAssistant, - entry: ConfigEntry, - async_add_entities: AddEntitiesCallback + hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback ) -> None: """Add Alfen Select from a config_entry""" device = hass.data[ALFEN_DOMAIN][entry.entry_id] - selects = [ - AlfenSelect(device, description) for description in ALFEN_SELECT_TYPES - ] + selects = [AlfenSelect(device, description) + for description in ALFEN_SELECT_TYPES] async_add_entities(selects) @@ -87,9 +153,9 @@ class AlfenSelect(AlfenEntity, SelectEntity): values_dict: dict[int, str] - def __init__(self, - device: AlfenDevice, - description: AlfenSelectDescription) -> None: + def __init__( + self, device: AlfenDevice, description: AlfenSelectDescription + ) -> None: """Initialize.""" super().__init__(device) self._device = device @@ -116,8 +182,8 @@ def current_option(self) -> str | None: def _get_current_option(self) -> str | None: """Return the current option.""" for prop in self._device.properties: - if prop['id'] == self.entity_description.api_param: - return prop['value'] + if prop["id"] == self.entity_description.api_param: + return prop["value"] return None async def async_update(self): diff --git a/custom_components/alfen_wallbox/sensor.py b/custom_components/alfen_wallbox/sensor.py index b0c7066..0ae567f 100644 --- a/custom_components/alfen_wallbox/sensor.py +++ b/custom_components/alfen_wallbox/sensor.py @@ -3,6 +3,8 @@ from dataclasses import dataclass import voluptuous as vol + +from homeassistant.components.sensor.const import SensorDeviceClass from .entity import AlfenEntity from homeassistant import const from homeassistant.config_entries import ConfigEntry @@ -57,12 +59,6 @@ class AlfenSensorDescription( """Class to describe an Alfen sensor entity.""" -AUTH_MODE_DICT: Final[dict[int, str]] = { - 0: "Plug and Charge", - 2: "RFID" -} - - STATUS_DICT: Final[dict[int, str]] = { 0: "Unknown", 1: "Off", @@ -125,6 +121,24 @@ class AlfenSensorDescription( unit=None, round_digits=None, ), + AlfenSensorDescription( + key="last_modify_datetime", + name="Last Modify Config datetime", + icon="mdi:timer-outline", + api_param="2187_0", + unit=None, + state_class=SensorDeviceClass.DATE, + round_digits=None, + ), + # too much logging data + # AlfenSensorDescription( + # key="system_date_time", + # name="System Datetime", + # icon="mdi:timer-outline", + # api_param="2059_0", + # unit=None, + # round_digits=None, + # ), AlfenSensorDescription( key="bootups", name="Bootups", @@ -192,7 +206,7 @@ class AlfenSensorDescription( AlfenSensorDescription( key="meter_reading", name="Meter Reading", - icon=None, + icon="mdi:counter", api_param="2221_22", unit=UnitOfEnergy.KILO_WATT_HOUR, round_digits=None, @@ -213,46 +227,15 @@ class AlfenSensorDescription( unit=UnitOfElectricCurrent.AMPERE, round_digits=2, ), - AlfenSensorDescription( - key="auth_mode", - name="Authorization Mode", - icon=None, - api_param="2126_0", - unit=None, - round_digits=None, - ), - AlfenSensorDescription( - key="alb_safe_current", - name="Active Load Balancing Safe Current", - icon="mdi:current-ac", - api_param="2068_0", - unit=UnitOfElectricCurrent.AMPERE, - round_digits=2, - ), - AlfenSensorDescription( - key="alb_phase_connection", - name="Active Load Balancing Phase Connection", - icon=None, - api_param="2069_0", - unit=None, - round_digits=None, - ), + AlfenSensorDescription( key="max_station_current", - name="Maximum Smart Meter current", + name="Maximum Smart Meter Current", icon="mdi:current-ac", api_param="2062_0", unit=UnitOfElectricCurrent.AMPERE, round_digits=2, ), - AlfenSensorDescription( - key="load_balancing_mode", - name="Load Balancing Mode", - icon=None, - api_param="2064_0", - unit=None, - round_digits=None, - ), AlfenSensorDescription( key="main_static_lb_max_current", name="Main Static Load Balancing Max Current", @@ -272,7 +255,7 @@ class AlfenSensorDescription( AlfenSensorDescription( key="charging_box_identifier", name="Charging Box Identifier", - icon=None, + icon="mdi:ev-station", api_param="2053_0", unit=None, round_digits=None, @@ -280,7 +263,7 @@ class AlfenSensorDescription( AlfenSensorDescription( key="boot_reason", name="System Boot Reason", - icon=None, + icon="mdi:reload", api_param="2057_0", unit=None, round_digits=None, @@ -353,7 +336,7 @@ class AlfenSensorDescription( key="gprs_sim_iccid", name="GPRS SIM Serial", icon="mdi:antenna", - api_param="", + api_param="2105_0", unit=None, round_digits=None, ), @@ -361,7 +344,7 @@ class AlfenSensorDescription( key="gprs_provider", name="GPRS Provider", icon="mdi:antenna", - api_param="", + api_param="2112_0", unit=None, round_digits=None, ), @@ -416,7 +399,7 @@ class AlfenSensorDescription( AlfenSensorDescription( key="comm_bo_short_name", name="Backoffice Short Name", - icon=None, + icon="mdi:antenna", api_param="2076_0", unit=None, round_digits=None, @@ -512,7 +495,7 @@ class AlfenSensorDescription( AlfenSensorDescription( key="comm_protocol_name", name="Protocol Name", - icon=None, + icon="mdi:information-outline", api_param="2081_0", unit=None, round_digits=None, @@ -520,7 +503,7 @@ class AlfenSensorDescription( AlfenSensorDescription( key="comm_protocol_version", name="Protocol Version", - icon=None, + icon="mdi:information-outline", api_param="2082_0", unit=None, round_digits=None, @@ -528,7 +511,7 @@ class AlfenSensorDescription( AlfenSensorDescription( key="lb_solar_charging_green_share", name="Solar Charging Green Share %", - icon=None, + icon="mdi:solar-power", api_param="3280_2", unit=const.PERCENTAGE, round_digits=None, @@ -536,7 +519,7 @@ class AlfenSensorDescription( AlfenSensorDescription( key="lb_solar_charging_comfort_level", name="Solar Charging Comfort Level w", - icon=None, + icon="mdi:solar-power", api_param="3280_3", unit=UnitOfPower.WATT, round_digits=None, @@ -564,9 +547,6 @@ async def async_setup_entry( ] async_add_entities(sensors) - - # get the first type in the typle - async_add_entities([AlfenMainSensor(device, ALFEN_SENSOR_TYPES[0])]) platform = entity_platform.current_platform.get() @@ -742,7 +722,6 @@ def _get_current_value(self): if prop['id'] == self.entity_description.api_param: return prop['value'] return None -# return getattr(self._device.properties, self.entity_description.key) @callback def _async_update_attrs(self) -> None: @@ -780,9 +759,6 @@ def state(self): for prop in self._device.properties: if prop['id'] == self.entity_description.api_param: # some exception of return value - # auth_mode - if self.entity_description.api_param == "2126_0": - return AUTH_MODE_DICT.get(prop['value'], 'Unknown') # meter_reading from w to kWh if self.entity_description.api_param == "2221_22": @@ -792,6 +768,10 @@ def state(self): if self.entity_description.api_param == "2060_0": return str(datetime.timedelta(milliseconds=prop['value'])).split('.', maxsplit=1)[0] + # change milliseconds to d/m/y HH:MM:SS + if self.entity_description.api_param == "2187_0" or self.entity_description.api_param == "2059_0": + return datetime.datetime.fromtimestamp(prop['value'] / 1000).strftime("%d/%m/%Y %H:%M:%S") + if self.entity_description.round_digits is not None: return round(prop['value'], self.entity_description.round_digits) diff --git a/custom_components/alfen_wallbox/switch.py b/custom_components/alfen_wallbox/switch.py new file mode 100644 index 0000000..40557a2 --- /dev/null +++ b/custom_components/alfen_wallbox/switch.py @@ -0,0 +1,92 @@ +import logging + +from dataclasses import dataclass +from typing import Any, Final +from config.custom_components.alfen_wallbox.alfen import AlfenDevice +from config.custom_components.alfen_wallbox.entity import AlfenEntity + +from homeassistant.components.switch import SwitchEntity, SwitchEntityDescription +from homeassistant.config_entries import ConfigEntry +from homeassistant.core import HomeAssistant +from homeassistant.helpers.entity_platform import AddEntitiesCallback + +from . import DOMAIN as ALFEN_DOMAIN + +_LOGGER = logging.getLogger(__name__) + + +@dataclass +class AlfenSwitchDescriptionMixin: + """Define an entity description mixin for binary sensor entities.""" + + api_param: str + + +@dataclass +class AlfenSwitchDescription(SwitchEntityDescription, AlfenSwitchDescriptionMixin): + """Class to describe an Alfen binary sensor entity.""" + + +ALFEN_BINARY_SENSOR_TYPES: Final[tuple[AlfenSwitchDescription, ...]] = ( + AlfenSwitchDescription( + key="enable_phase_switching", + name="Enable Phase Switching", + api_param="2185_0", + ), +) + + +async def async_setup_entry( + hass: HomeAssistant, + entry: ConfigEntry, + async_add_entities: AddEntitiesCallback, +) -> None: + """Set up Alfen switch entities from a config entry.""" + device = hass.data[ALFEN_DOMAIN][entry.entry_id] + switches = [AlfenSwitchSensor(device, description) + for description in ALFEN_BINARY_SENSOR_TYPES] + + async_add_entities(switches) + + +class AlfenSwitchSensor(AlfenEntity, SwitchEntity): + """Define an Alfen binary sensor.""" + + # entity_description: AlfenSwitchDescriptionMixin + + def __init__(self, + device: AlfenDevice, + description: AlfenSwitchDescription + ) -> None: + """Initialize.""" + super().__init__(device) + self._device = device + self._attr_name = f"{device.name} {description.name}" + self._attr_unique_id = f"{self._attr_unique_id}_{description.key}" + self.entity_description = description + + @property + def available(self) -> bool: + for prop in self._device.properties: + if prop["id"] == self.entity_description.api_param: + return True + return False + + @property + def is_on(self) -> bool: + for prop in self._device.properties: + if prop["id"] == self.entity_description.api_param: + return prop["value"] == 1 + + return False + + async def async_turn_on(self, **kwargs: Any) -> None: + """Turn the light on.""" + # Do the turning on. + await self.update_state(self.entity_description.api_param, 1) + await self._device.async_update() + + async def async_turn_off(self, **kwargs: Any) -> None: + """Turn the entity off.""" + await self.update_state(self.entity_description.api_param, 0) + await self._device.async_update() From 03d3c28df36ef52c390b64c18bdbd4041c8f815a Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Sun, 2 Jul 2023 17:58:26 +0200 Subject: [PATCH 08/11] add number --- custom_components/alfen_wallbox/__init__.py | 11 +- custom_components/alfen_wallbox/number.py | 188 ++++++++++++++++++++ custom_components/alfen_wallbox/select.py | 43 +++-- custom_components/alfen_wallbox/sensor.py | 73 ++------ 4 files changed, 239 insertions(+), 76 deletions(-) create mode 100644 custom_components/alfen_wallbox/number.py diff --git a/custom_components/alfen_wallbox/__init__.py b/custom_components/alfen_wallbox/__init__.py index dde1094..29efd75 100644 --- a/custom_components/alfen_wallbox/__init__.py +++ b/custom_components/alfen_wallbox/__init__.py @@ -25,10 +25,13 @@ TIMEOUT, ) -PLATFORMS = [Platform.SENSOR, - Platform.SELECT, - Platform.BINARY_SENSOR, - Platform.SWITCH] +PLATFORMS = [ + Platform.SENSOR, + Platform.SELECT, + Platform.BINARY_SENSOR, + Platform.SWITCH, + Platform.NUMBER, +] SCAN_INTERVAL = timedelta(seconds=60) _LOGGER = logging.getLogger(__name__) diff --git a/custom_components/alfen_wallbox/number.py b/custom_components/alfen_wallbox/number.py new file mode 100644 index 0000000..09c2aa4 --- /dev/null +++ b/custom_components/alfen_wallbox/number.py @@ -0,0 +1,188 @@ +from homeassistant.components.number import NumberDeviceClass, NumberEntity, NumberEntityDescription, NumberMode +from homeassistant.components.number.const import DEVICE_CLASS_UNITS +from homeassistant.config_entries import ConfigEntry +from homeassistant.helpers.entity_platform import AddEntitiesCallback +from . import DOMAIN as ALFEN_DOMAIN +from homeassistant.core import HomeAssistant, callback +import logging +from typing import Final, Any +from dataclasses import dataclass +from .entity import AlfenEntity +from .alfen import AlfenDevice +from homeassistant.const import ( + PERCENTAGE, + UnitOfElectricCurrent, + UnitOfPower, +) + + +_LOGGER = logging.getLogger(__name__) + + +@dataclass +class AlfenNumberDescriptionMixin: + """Define an entity description mixin for select entities.""" + assumed_state: bool + state: float + api_param: str + + +@dataclass +class AlfenNumberDescription(NumberEntityDescription, AlfenNumberDescriptionMixin): + """Class to describe an Alfen select entity.""" + + +ALFEN_NUMBER_TYPES: Final[tuple[AlfenNumberDescription, ...]] = ( + AlfenNumberDescription( + key="alb_safe_current", + name="ALB Safe A.", + state=None, + icon="mdi:current-ac", + assumed_state=False, + device_class=NumberDeviceClass.CURRENT, + native_min_value=1, + native_max_value=16, + native_step=1, + mode=NumberMode.BOX, + unit_of_measurement=UnitOfElectricCurrent.AMPERE, + api_param="2068_0", + ), + AlfenNumberDescription( + key="current_limit", + name="Current Limit", + state=None, + icon="mdi:current-ac", + assumed_state=False, + device_class=NumberDeviceClass.CURRENT, + native_min_value=1, + native_max_value=16, + native_step=1, + mode=NumberMode.BOX, + unit_of_measurement=UnitOfElectricCurrent.AMPERE, + api_param="2129_0", + ), + AlfenNumberDescription( + key="max_station_current", + name="Max. Station A.", + state=None, + icon="mdi:current-ac", + assumed_state=False, + device_class=NumberDeviceClass.CURRENT, + native_min_value=1, + native_max_value=16, + native_step=1, + mode=NumberMode.BOX, + unit_of_measurement=UnitOfElectricCurrent.AMPERE, + api_param="2062_0", + ), + AlfenNumberDescription( + key="max_smart_meter_current", + name="Max. Meter A.", + state=None, + icon="mdi:current-ac", + assumed_state=False, + device_class=NumberDeviceClass.CURRENT, + native_min_value=1, + native_max_value=16, + native_step=1, + mode=NumberMode.SLIDER, + unit_of_measurement=UnitOfElectricCurrent.AMPERE, + api_param="2067_0", + ), + AlfenNumberDescription( + key="lb_solar_charging_green_share", + name="Solar Green Share", + state=None, + icon="mdi:current-ac", + assumed_state=False, + device_class=NumberDeviceClass.POWER_FACTOR, + native_min_value=0, + native_max_value=100, + native_step=1, + mode=NumberMode.BOX, + unit_of_measurement=PERCENTAGE, + api_param="3280_2", + ), + AlfenNumberDescription( + key="lb_solar_charging_comfort_level", + name="Solar Comfort Level", + state=None, + icon="mdi:current-ac", + assumed_state=False, + device_class=NumberDeviceClass.POWER_FACTOR, + native_min_value=1400, + native_max_value=3500, + native_step=100, + mode=NumberMode.BOX, + unit_of_measurement=UnitOfPower.WATT, + api_param="3280_3", + ), +) + + +async def async_setup_entry( + hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback +) -> None: + """Set up Alfen select entities from a config entry.""" + device = hass.data[ALFEN_DOMAIN][entry.entry_id] + numbers = [AlfenNumber(device, description) + for description in ALFEN_NUMBER_TYPES] + + async_add_entities(numbers) + + +class AlfenNumber(AlfenEntity, NumberEntity): + """Define an Alfen select entity.""" + + _attr_has_entity_name = True + _attr_name = None + _attr_should_poll = False + + def __init__( + self, + device: AlfenDevice, + description: AlfenNumberDescription, + ) -> None: + """Initialize the Demo Number entity.""" + super().__init__(device) + self._device = device + self._attr_name = f"{description.name}" + self._attr_unique_id = f"{description.key}" + self._attr_assumed_state = description.assumed_state + self._attr_device_class = description.device_class + self._attr_icon = description.icon + self._attr_mode = description.mode + self._attr_native_unit_of_measurement = description.unit_of_measurement + self._attr_native_value = description.state + self.entity_description = description + + if description.native_min_value is not None: + self._attr_native_min_value = description.native_min_value + if description.native_max_value is not None: + self._attr_native_max_value = description.native_max_value + if description.native_step is not None: + self._attr_native_step = description.native_step + + async def async_set_native_value(self, value: float) -> None: + """Update the current value.""" + self._attr_native_value = value + self.async_write_ha_state() + self._async_update_attrs() + + @property + def native_value(self) -> float | None: + """Return the entity value to represent the entity state.""" + for prop in self._device.properties: + if prop["id"] == self.entity_description.api_param: + return prop["value"] + return None + + async def async_set_native_value(self, value: float) -> None: + """Update the current value.""" + await self.update_state(self.entity_description.api_param, value) + self.async_write_ha_state() + + @callback + def _async_update_attrs(self) -> None: + """Update attrs from device.""" + self._attr_native_value = self._get_current_option() diff --git a/custom_components/alfen_wallbox/select.py b/custom_components/alfen_wallbox/select.py index 4aaa633..0084bce 100644 --- a/custom_components/alfen_wallbox/select.py +++ b/custom_components/alfen_wallbox/select.py @@ -73,6 +73,16 @@ class AlfenSelectDescription(SelectEntityDescription, AlfenSelectDescriptionMixi "DSMR4.x / SMR5.0 (P1)": 3 } +LOAD_BALANCE_DATA_SOURCE_DICT: Final[dict[str, int]] = { + "Meter": 0, + "Energy Management System": 3 +} + +LOAD_BALANCE_RECEIVED_MEASUREMENTS_DICT: Final[dict[str, int]] = { + "Exclude Charging Ev": 0, + "Include Charging Ev": 1 +} + ALFEN_SELECT_TYPES: Final[tuple[AlfenSelectDescription, ...]] = ( AlfenSelectDescription( @@ -83,14 +93,6 @@ class AlfenSelectDescription(SelectEntityDescription, AlfenSelectDescriptionMixi options_dict=CHARGING_MODE_DICT, api_param="3280_1", ), - AlfenSelectDescription( - key="enable_phase_switching", - name="Enable Phase Switching", - icon="mdi:ev-station", - options=list(ON_OFF_DICT), - options_dict=ON_OFF_DICT, - api_param="2185_0", - ), AlfenSelectDescription( key="lb_solar_charging_boost", name="Solar Charging Boost", @@ -107,14 +109,14 @@ class AlfenSelectDescription(SelectEntityDescription, AlfenSelectDescriptionMixi options_dict=PHASE_ROTATION_DICT, api_param="2069_0", ), - AlfenSelectDescription( - key="alb_safe_current", - name="Active Load Balancing Safe Current", - icon="mdi:current-ac", - options=list(SAFE_AMPS_DICT), - options_dict=SAFE_AMPS_DICT, - api_param="2068_0", - ), + # AlfenSelectDescription( + # key="alb_safe_current", + # name="Active Load Balancing Safe Current", + # icon="mdi:current-ac", + # options=list(SAFE_AMPS_DICT), + # options_dict=SAFE_AMPS_DICT, + # api_param="2068_0", + # ), AlfenSelectDescription( key="auth_mode", @@ -133,6 +135,15 @@ class AlfenSelectDescription(SelectEntityDescription, AlfenSelectDescriptionMixi options_dict=LOAD_BALANCE_MODE_DICT, api_param="2064_0", ), + AlfenSelectDescription( + key="lb_active_balancing_received_measurements", + name="Load Balancing Received Measurements", + icon="mdi:scale-balance", + options=list(LOAD_BALANCE_RECEIVED_MEASUREMENTS_DICT), + options_dict=LOAD_BALANCE_RECEIVED_MEASUREMENTS_DICT, + api_param="206F_0", + ), + ) diff --git a/custom_components/alfen_wallbox/sensor.py b/custom_components/alfen_wallbox/sensor.py index 0ae567f..7a8846d 100644 --- a/custom_components/alfen_wallbox/sensor.py +++ b/custom_components/alfen_wallbox/sensor.py @@ -219,23 +219,7 @@ class AlfenSensorDescription( unit=UnitOfTemperature.CELSIUS, round_digits=2, ), - AlfenSensorDescription( - key="current_limit", - name="Current Limit", - icon="mdi:current-ac", - api_param="2129_0", - unit=UnitOfElectricCurrent.AMPERE, - round_digits=2, - ), - AlfenSensorDescription( - key="max_station_current", - name="Maximum Smart Meter Current", - icon="mdi:current-ac", - api_param="2062_0", - unit=UnitOfElectricCurrent.AMPERE, - round_digits=2, - ), AlfenSensorDescription( key="main_static_lb_max_current", name="Main Static Load Balancing Max Current", @@ -268,14 +252,7 @@ class AlfenSensorDescription( unit=None, round_digits=None, ), - AlfenSensorDescription( - key="max_smart_meter_current", - name="Max Smart Meter Current", - icon="mdi:current-ac", - api_param="2067_0", - unit=UnitOfElectricCurrent.AMPERE, - round_digits=2, - ), + AlfenSensorDescription( key="p1_measurements_1", name="P1 Meter Phase 1 Current", @@ -364,14 +341,14 @@ class AlfenSensorDescription( unit=None, round_digits=None, ), - AlfenSensorDescription( - key="comm_dhcp_address_1", - name="GPRS DHCP Address", - icon="mdi:antenna", - api_param="2072_1", - unit=None, - round_digits=None, - ), + # AlfenSensorDescription( + # key="comm_dhcp_address_1", + # name="GPRS DHCP Address", + # icon="mdi:antenna", + # api_param="2072_1", + # unit=None, + # round_digits=None, + # ), AlfenSensorDescription( key="comm_netmask_address_1", name="GPRS Netmask", @@ -444,14 +421,14 @@ class AlfenSensorDescription( unit=const.SIGNAL_STRENGTH_DECIBELS, round_digits=None, ), - AlfenSensorDescription( - key="comm_dhcp_address_2", - name="Wired DHCP", - icon="mdi:cable-data", - api_param="207A_1", - unit=None, - round_digits=None, - ), + # AlfenSensorDescription( + # key="comm_dhcp_address_2", + # name="Wired DHCP", + # icon="mdi:cable-data", + # api_param="207A_1", + # unit=None, + # round_digits=None, + # ), AlfenSensorDescription( key="comm_netmask_address_2", name="Wired Netmask", @@ -508,22 +485,6 @@ class AlfenSensorDescription( unit=None, round_digits=None, ), - AlfenSensorDescription( - key="lb_solar_charging_green_share", - name="Solar Charging Green Share %", - icon="mdi:solar-power", - api_param="3280_2", - unit=const.PERCENTAGE, - round_digits=None, - ), - AlfenSensorDescription( - key="lb_solar_charging_comfort_level", - name="Solar Charging Comfort Level w", - icon="mdi:solar-power", - api_param="3280_3", - unit=UnitOfPower.WATT, - round_digits=None, - ), ) From 7687eb35532e7f44e28982e774771734ee440774 Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Sun, 2 Jul 2023 18:15:39 +0200 Subject: [PATCH 09/11] change A to current --- custom_components/alfen_wallbox/number.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/custom_components/alfen_wallbox/number.py b/custom_components/alfen_wallbox/number.py index 09c2aa4..ccebb18 100644 --- a/custom_components/alfen_wallbox/number.py +++ b/custom_components/alfen_wallbox/number.py @@ -35,7 +35,7 @@ class AlfenNumberDescription(NumberEntityDescription, AlfenNumberDescriptionMixi ALFEN_NUMBER_TYPES: Final[tuple[AlfenNumberDescription, ...]] = ( AlfenNumberDescription( key="alb_safe_current", - name="ALB Safe A.", + name="ALB Safe Current", state=None, icon="mdi:current-ac", assumed_state=False, @@ -63,7 +63,7 @@ class AlfenNumberDescription(NumberEntityDescription, AlfenNumberDescriptionMixi ), AlfenNumberDescription( key="max_station_current", - name="Max. Station A.", + name="Max. Station Current", state=None, icon="mdi:current-ac", assumed_state=False, @@ -77,7 +77,7 @@ class AlfenNumberDescription(NumberEntityDescription, AlfenNumberDescriptionMixi ), AlfenNumberDescription( key="max_smart_meter_current", - name="Max. Meter A.", + name="Max. Meter Current", state=None, icon="mdi:current-ac", assumed_state=False, @@ -85,7 +85,7 @@ class AlfenNumberDescription(NumberEntityDescription, AlfenNumberDescriptionMixi native_min_value=1, native_max_value=16, native_step=1, - mode=NumberMode.SLIDER, + mode=NumberMode.BOX, unit_of_measurement=UnitOfElectricCurrent.AMPERE, api_param="2067_0", ), From 7c6b3d6400565371797b225665e2185afd53ba08 Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Sun, 2 Jul 2023 18:17:27 +0200 Subject: [PATCH 10/11] Update README.md --- README.md | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index 09ab5dc..834b08d 100644 --- a/README.md +++ b/README.md @@ -1,14 +1,16 @@ + # Alfen Wallbox - HomeAssistant Integration This is a custom component to allow control of Alfen Wallboxes in [HomeAssistant](https://home-assistant.io). The component is a fork of the [Garo Wallbox custom integration](https://github.com/sockless-coding/garo_wallbox). -![image](https://github.com/leeyuentuen/alfen_wallbox/assets/1487966/a25af9bc-a6b3-496d-9c04-6812825cb375) +![Screenshot 2023-07-02 at 18 09 47](https://github.com/leeyuentuen/alfen_wallbox/assets/1487966/322e9e05-117f-4adc-b159-7177533fde01) + +![Screenshot 2023-07-02 at 18 09 58](https://github.com/leeyuentuen/alfen_wallbox/assets/1487966/310f0537-9bc4-49a0-9552-0c8414b97425) -Add Solar charging data: +![Screenshot 2023-07-02 at 18 10 13](https://github.com/leeyuentuen/alfen_wallbox/assets/1487966/f5e2670d-4bd8-40d2-bbbe-f0628cff6273) -image Example of running in Services: Note; the name of the configured charging point is "wallbox" in these examples. From fe71f716aea4f00051269664fbe73de05ab641a3 Mon Sep 17 00:00:00 2001 From: Tuen Lee Date: Sun, 2 Jul 2023 18:42:49 +0200 Subject: [PATCH 11/11] fix dev to prod --- custom_components/alfen_wallbox/alfen.py | 1 + custom_components/alfen_wallbox/binary_sensor.py | 6 +++--- custom_components/alfen_wallbox/number.py | 1 - custom_components/alfen_wallbox/sensor.py | 7 +++---- custom_components/alfen_wallbox/switch.py | 4 ++-- 5 files changed, 9 insertions(+), 10 deletions(-) diff --git a/custom_components/alfen_wallbox/alfen.py b/custom_components/alfen_wallbox/alfen.py index c0394b1..aa532a8 100644 --- a/custom_components/alfen_wallbox/alfen.py +++ b/custom_components/alfen_wallbox/alfen.py @@ -354,6 +354,7 @@ async def set_comfort_power(self, value): def __get_url(self, action): return "https://{}/api/{}".format(self.host, action) + class AlfenDeviceInfo: def __init__(self, response) -> None: self.identity = response["Identity"] diff --git a/custom_components/alfen_wallbox/binary_sensor.py b/custom_components/alfen_wallbox/binary_sensor.py index b11b1a6..6eb0d24 100644 --- a/custom_components/alfen_wallbox/binary_sensor.py +++ b/custom_components/alfen_wallbox/binary_sensor.py @@ -2,10 +2,10 @@ from dataclasses import dataclass from typing import Final -from config.custom_components.alfen_wallbox.alfen import AlfenDevice -from config.custom_components.alfen_wallbox.entity import AlfenEntity +from .alfen import AlfenDevice +from .entity import AlfenEntity -from homeassistant.components.binary_sensor import BinarySensorDeviceClass, BinarySensorEntity, BinarySensorEntityDescription +from homeassistant.components.binary_sensor import BinarySensorEntity, BinarySensorEntityDescription from homeassistant.config_entries import ConfigEntry from homeassistant.core import HomeAssistant from homeassistant.helpers.entity_platform import AddEntitiesCallback diff --git a/custom_components/alfen_wallbox/number.py b/custom_components/alfen_wallbox/number.py index ccebb18..cf0227f 100644 --- a/custom_components/alfen_wallbox/number.py +++ b/custom_components/alfen_wallbox/number.py @@ -1,5 +1,4 @@ from homeassistant.components.number import NumberDeviceClass, NumberEntity, NumberEntityDescription, NumberMode -from homeassistant.components.number.const import DEVICE_CLASS_UNITS from homeassistant.config_entries import ConfigEntry from homeassistant.helpers.entity_platform import AddEntitiesCallback from . import DOMAIN as ALFEN_DOMAIN diff --git a/custom_components/alfen_wallbox/sensor.py b/custom_components/alfen_wallbox/sensor.py index 7a8846d..324bd5c 100644 --- a/custom_components/alfen_wallbox/sensor.py +++ b/custom_components/alfen_wallbox/sensor.py @@ -4,21 +4,20 @@ import voluptuous as vol -from homeassistant.components.sensor.const import SensorDeviceClass from .entity import AlfenEntity from homeassistant import const from homeassistant.config_entries import ConfigEntry -from homeassistant.const import UnitOfElectricCurrent, UnitOfElectricPotential, UnitOfEnergy, UnitOfInformation, UnitOfPower, UnitOfSpeed, UnitOfTemperature +from homeassistant.const import UnitOfElectricCurrent, UnitOfElectricPotential, UnitOfEnergy, UnitOfPower, UnitOfTemperature import datetime from homeassistant.core import HomeAssistant, callback -from homeassistant.helpers.entity import Entity from homeassistant.components.sensor import ( DEVICE_CLASS_ENERGY, DEVICE_CLASS_POWER, SensorEntity, SensorEntityDescription, - SensorStateClass + SensorStateClass, + SensorDeviceClass ) from homeassistant.helpers.entity_platform import AddEntitiesCallback diff --git a/custom_components/alfen_wallbox/switch.py b/custom_components/alfen_wallbox/switch.py index 40557a2..c66fdba 100644 --- a/custom_components/alfen_wallbox/switch.py +++ b/custom_components/alfen_wallbox/switch.py @@ -2,8 +2,8 @@ from dataclasses import dataclass from typing import Any, Final -from config.custom_components.alfen_wallbox.alfen import AlfenDevice -from config.custom_components.alfen_wallbox.entity import AlfenEntity +from .alfen import AlfenDevice +from .entity import AlfenEntity from homeassistant.components.switch import SwitchEntity, SwitchEntityDescription from homeassistant.config_entries import ConfigEntry