mirror of
https://github.com/home-assistant/core.git
synced 2025-07-23 05:07:41 +00:00
Nibe Heat Pump after merge fixups (#78931)
This commit is contained in:
parent
9f62a29928
commit
27d1c1f471
@ -1,6 +1,7 @@
|
|||||||
"""The Nibe Heat Pump integration."""
|
"""The Nibe Heat Pump integration."""
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from collections import defaultdict
|
||||||
from datetime import timedelta
|
from datetime import timedelta
|
||||||
|
|
||||||
from nibe.coil import Coil
|
from nibe.coil import Coil
|
||||||
@ -11,14 +12,20 @@ from nibe.heatpump import HeatPump, Model
|
|||||||
from tenacity import RetryError, retry, retry_if_exception_type, stop_after_attempt
|
from tenacity import RetryError, retry, retry_if_exception_type, stop_after_attempt
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import CONF_IP_ADDRESS, CONF_MODEL, Platform
|
from homeassistant.const import (
|
||||||
|
CONF_IP_ADDRESS,
|
||||||
|
CONF_MODEL,
|
||||||
|
EVENT_HOMEASSISTANT_STOP,
|
||||||
|
Platform,
|
||||||
|
)
|
||||||
from homeassistant.core import CALLBACK_TYPE, HomeAssistant
|
from homeassistant.core import CALLBACK_TYPE, HomeAssistant
|
||||||
from homeassistant.exceptions import ConfigEntryNotReady, HomeAssistantError
|
from homeassistant.exceptions import HomeAssistantError
|
||||||
from homeassistant.helpers import device_registry as dr
|
from homeassistant.helpers import device_registry as dr
|
||||||
from homeassistant.helpers.entity import DeviceInfo, async_generate_entity_id
|
from homeassistant.helpers.entity import DeviceInfo, async_generate_entity_id
|
||||||
from homeassistant.helpers.update_coordinator import (
|
from homeassistant.helpers.update_coordinator import (
|
||||||
CoordinatorEntity,
|
CoordinatorEntity,
|
||||||
DataUpdateCoordinator,
|
DataUpdateCoordinator,
|
||||||
|
UpdateFailed,
|
||||||
)
|
)
|
||||||
|
|
||||||
from .const import (
|
from .const import (
|
||||||
@ -33,6 +40,7 @@ from .const import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
PLATFORMS: list[Platform] = [Platform.SENSOR]
|
PLATFORMS: list[Platform] = [Platform.SENSOR]
|
||||||
|
COIL_READ_RETRIES = 5
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||||
@ -40,7 +48,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||||||
|
|
||||||
heatpump = HeatPump(Model[entry.data[CONF_MODEL]])
|
heatpump = HeatPump(Model[entry.data[CONF_MODEL]])
|
||||||
heatpump.word_swap = entry.data[CONF_WORD_SWAP]
|
heatpump.word_swap = entry.data[CONF_WORD_SWAP]
|
||||||
heatpump.initialize()
|
await hass.async_add_executor_job(heatpump.initialize)
|
||||||
|
|
||||||
connection_type = entry.data[CONF_CONNECTION_TYPE]
|
connection_type = entry.data[CONF_CONNECTION_TYPE]
|
||||||
|
|
||||||
@ -56,17 +64,16 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||||||
raise HomeAssistantError(f"Connection type {connection_type} is not supported.")
|
raise HomeAssistantError(f"Connection type {connection_type} is not supported.")
|
||||||
|
|
||||||
await connection.start()
|
await connection.start()
|
||||||
|
|
||||||
|
entry.async_on_unload(
|
||||||
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, connection.stop)
|
||||||
|
)
|
||||||
|
|
||||||
coordinator = Coordinator(hass, heatpump, connection)
|
coordinator = Coordinator(hass, heatpump, connection)
|
||||||
|
|
||||||
data = hass.data.setdefault(DOMAIN, {})
|
data = hass.data.setdefault(DOMAIN, {})
|
||||||
data[entry.entry_id] = coordinator
|
data[entry.entry_id] = coordinator
|
||||||
|
|
||||||
try:
|
|
||||||
await coordinator.async_config_entry_first_refresh()
|
|
||||||
except ConfigEntryNotReady:
|
|
||||||
await connection.stop()
|
|
||||||
raise
|
|
||||||
|
|
||||||
reg = dr.async_get(hass)
|
reg = dr.async_get(hass)
|
||||||
reg.async_get_or_create(
|
reg.async_get_or_create(
|
||||||
config_entry_id=entry.entry_id,
|
config_entry_id=entry.entry_id,
|
||||||
@ -139,13 +146,8 @@ class Coordinator(DataUpdateCoordinator[dict[int, Coil]]):
|
|||||||
return float(value)
|
return float(value)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
async def async_write_coil(
|
async def async_write_coil(self, coil: Coil, value: int | float | str) -> None:
|
||||||
self, coil: Coil | None, value: int | float | str
|
|
||||||
) -> None:
|
|
||||||
"""Write coil and update state."""
|
"""Write coil and update state."""
|
||||||
if not coil:
|
|
||||||
raise HomeAssistantError("No coil available")
|
|
||||||
|
|
||||||
coil.value = value
|
coil.value = value
|
||||||
coil = await self.connection.write_coil(coil)
|
coil = await self.connection.write_coil(coil)
|
||||||
|
|
||||||
@ -155,16 +157,17 @@ class Coordinator(DataUpdateCoordinator[dict[int, Coil]]):
|
|||||||
|
|
||||||
async def _async_update_data(self) -> dict[int, Coil]:
|
async def _async_update_data(self) -> dict[int, Coil]:
|
||||||
@retry(
|
@retry(
|
||||||
retry=retry_if_exception_type(CoilReadException), stop=stop_after_attempt(2)
|
retry=retry_if_exception_type(CoilReadException),
|
||||||
|
stop=stop_after_attempt(COIL_READ_RETRIES),
|
||||||
)
|
)
|
||||||
async def read_coil(coil: Coil):
|
async def read_coil(coil: Coil):
|
||||||
return await self.connection.read_coil(coil)
|
return await self.connection.read_coil(coil)
|
||||||
|
|
||||||
callbacks: dict[int, list[CALLBACK_TYPE]] = {}
|
callbacks: dict[int, list[CALLBACK_TYPE]] = defaultdict(list)
|
||||||
for update_callback, context in list(self._listeners.values()):
|
for update_callback, context in list(self._listeners.values()):
|
||||||
assert isinstance(context, set)
|
assert isinstance(context, set)
|
||||||
for address in context:
|
for address in context:
|
||||||
callbacks.setdefault(address, []).append(update_callback)
|
callbacks[address].append(update_callback)
|
||||||
|
|
||||||
result: dict[int, Coil] = {}
|
result: dict[int, Coil] = {}
|
||||||
|
|
||||||
@ -173,7 +176,7 @@ class Coordinator(DataUpdateCoordinator[dict[int, Coil]]):
|
|||||||
coil = self.heatpump.get_coil_by_address(address)
|
coil = self.heatpump.get_coil_by_address(address)
|
||||||
self.data[coil.address] = result[coil.address] = await read_coil(coil)
|
self.data[coil.address] = result[coil.address] = await read_coil(coil)
|
||||||
except (CoilReadException, RetryError) as exception:
|
except (CoilReadException, RetryError) as exception:
|
||||||
self.logger.warning("Failed to update: %s", exception)
|
raise UpdateFailed(f"Failed to update: {exception}") from exception
|
||||||
except CoilNotFoundException as exception:
|
except CoilNotFoundException as exception:
|
||||||
self.logger.debug("Skipping missing coil: %s", exception)
|
self.logger.debug("Skipping missing coil: %s", exception)
|
||||||
|
|
||||||
|
@ -115,7 +115,7 @@ class ConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
|
|||||||
try:
|
try:
|
||||||
info = await validate_input(self.hass, user_input)
|
info = await validate_input(self.hass, user_input)
|
||||||
except FieldError as exception:
|
except FieldError as exception:
|
||||||
LOGGER.exception("Validation error")
|
LOGGER.debug("Validation error %s", exception)
|
||||||
errors[exception.field] = exception.error
|
errors[exception.field] = exception.error
|
||||||
except Exception: # pylint: disable=broad-except
|
except Exception: # pylint: disable=broad-except
|
||||||
LOGGER.exception("Unexpected exception")
|
LOGGER.exception("Unexpected exception")
|
||||||
|
@ -7,6 +7,7 @@ from homeassistant.components.sensor import (
|
|||||||
ENTITY_ID_FORMAT,
|
ENTITY_ID_FORMAT,
|
||||||
SensorDeviceClass,
|
SensorDeviceClass,
|
||||||
SensorEntity,
|
SensorEntity,
|
||||||
|
SensorEntityDescription,
|
||||||
SensorStateClass,
|
SensorStateClass,
|
||||||
)
|
)
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
@ -20,7 +21,6 @@ from homeassistant.const import (
|
|||||||
ENERGY_WATT_HOUR,
|
ENERGY_WATT_HOUR,
|
||||||
TEMP_CELSIUS,
|
TEMP_CELSIUS,
|
||||||
TEMP_FAHRENHEIT,
|
TEMP_FAHRENHEIT,
|
||||||
TEMP_KELVIN,
|
|
||||||
TIME_HOURS,
|
TIME_HOURS,
|
||||||
)
|
)
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
@ -29,6 +29,78 @@ from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|||||||
|
|
||||||
from . import DOMAIN, CoilEntity, Coordinator
|
from . import DOMAIN, CoilEntity, Coordinator
|
||||||
|
|
||||||
|
UNIT_DESCRIPTIONS = {
|
||||||
|
"°C": SensorEntityDescription(
|
||||||
|
key="°C",
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
device_class=SensorDeviceClass.TEMPERATURE,
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
native_unit_of_measurement=TEMP_CELSIUS,
|
||||||
|
),
|
||||||
|
"°F": SensorEntityDescription(
|
||||||
|
key="°F",
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
device_class=SensorDeviceClass.TEMPERATURE,
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
native_unit_of_measurement=TEMP_FAHRENHEIT,
|
||||||
|
),
|
||||||
|
"A": SensorEntityDescription(
|
||||||
|
key="A",
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
device_class=SensorDeviceClass.CURRENT,
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
native_unit_of_measurement=ELECTRIC_CURRENT_AMPERE,
|
||||||
|
),
|
||||||
|
"mA": SensorEntityDescription(
|
||||||
|
key="mA",
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
device_class=SensorDeviceClass.CURRENT,
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
native_unit_of_measurement=ELECTRIC_CURRENT_MILLIAMPERE,
|
||||||
|
),
|
||||||
|
"V": SensorEntityDescription(
|
||||||
|
key="V",
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
device_class=SensorDeviceClass.CURRENT,
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
native_unit_of_measurement=ELECTRIC_POTENTIAL_VOLT,
|
||||||
|
),
|
||||||
|
"mV": SensorEntityDescription(
|
||||||
|
key="mV",
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
device_class=SensorDeviceClass.CURRENT,
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
native_unit_of_measurement=ELECTRIC_POTENTIAL_MILLIVOLT,
|
||||||
|
),
|
||||||
|
"Wh": SensorEntityDescription(
|
||||||
|
key="Wh",
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
device_class=SensorDeviceClass.POWER,
|
||||||
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
||||||
|
native_unit_of_measurement=ENERGY_WATT_HOUR,
|
||||||
|
),
|
||||||
|
"kWh": SensorEntityDescription(
|
||||||
|
key="kWh",
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
device_class=SensorDeviceClass.POWER,
|
||||||
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
||||||
|
native_unit_of_measurement=ENERGY_KILO_WATT_HOUR,
|
||||||
|
),
|
||||||
|
"MWh": SensorEntityDescription(
|
||||||
|
key="MWh",
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
device_class=SensorDeviceClass.POWER,
|
||||||
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
||||||
|
native_unit_of_measurement=ENERGY_MEGA_WATT_HOUR,
|
||||||
|
),
|
||||||
|
"h": SensorEntityDescription(
|
||||||
|
key="h",
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
device_class=SensorDeviceClass.DURATION,
|
||||||
|
native_unit_of_measurement=TIME_HOURS,
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(
|
async def async_setup_entry(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
@ -40,38 +112,27 @@ async def async_setup_entry(
|
|||||||
coordinator: Coordinator = hass.data[DOMAIN][config_entry.entry_id]
|
coordinator: Coordinator = hass.data[DOMAIN][config_entry.entry_id]
|
||||||
|
|
||||||
async_add_entities(
|
async_add_entities(
|
||||||
Sensor(coordinator, coil)
|
Sensor(coordinator, coil, UNIT_DESCRIPTIONS.get(coil.unit))
|
||||||
for coil in coordinator.coils
|
for coil in coordinator.coils
|
||||||
if not coil.is_writable and not coil.is_boolean
|
if not coil.is_writable and not coil.is_boolean
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class Sensor(SensorEntity, CoilEntity):
|
class Sensor(CoilEntity, SensorEntity):
|
||||||
"""Sensor entity."""
|
"""Sensor entity."""
|
||||||
|
|
||||||
_attr_entity_category = EntityCategory.DIAGNOSTIC
|
def __init__(
|
||||||
|
self,
|
||||||
def __init__(self, coordinator: Coordinator, coil: Coil) -> None:
|
coordinator: Coordinator,
|
||||||
|
coil: Coil,
|
||||||
|
entity_description: SensorEntityDescription | None,
|
||||||
|
) -> None:
|
||||||
"""Initialize entity."""
|
"""Initialize entity."""
|
||||||
super().__init__(coordinator, coil, ENTITY_ID_FORMAT)
|
super().__init__(coordinator, coil, ENTITY_ID_FORMAT)
|
||||||
self._attr_native_unit_of_measurement = coil.unit
|
if entity_description:
|
||||||
|
self.entity_description = entity_description
|
||||||
unit = self.native_unit_of_measurement
|
|
||||||
if unit in {TEMP_CELSIUS, TEMP_FAHRENHEIT, TEMP_KELVIN}:
|
|
||||||
self._attr_device_class = SensorDeviceClass.TEMPERATURE
|
|
||||||
elif unit in {ELECTRIC_CURRENT_AMPERE, ELECTRIC_CURRENT_MILLIAMPERE}:
|
|
||||||
self._attr_device_class = SensorDeviceClass.CURRENT
|
|
||||||
elif unit in {ELECTRIC_POTENTIAL_VOLT, ELECTRIC_POTENTIAL_MILLIVOLT}:
|
|
||||||
self._attr_device_class = SensorDeviceClass.VOLTAGE
|
|
||||||
elif unit in {ENERGY_WATT_HOUR, ENERGY_KILO_WATT_HOUR, ENERGY_MEGA_WATT_HOUR}:
|
|
||||||
self._attr_device_class = SensorDeviceClass.ENERGY
|
|
||||||
elif unit in {TIME_HOURS}:
|
|
||||||
self._attr_device_class = SensorDeviceClass.DURATION
|
|
||||||
else:
|
else:
|
||||||
self._attr_device_class = None
|
self._attr_native_unit_of_measurement = coil.unit
|
||||||
|
|
||||||
if unit:
|
|
||||||
self._attr_state_class = SensorStateClass.MEASUREMENT
|
|
||||||
|
|
||||||
def _async_read_coil(self, coil: Coil):
|
def _async_read_coil(self, coil: Coil):
|
||||||
self._attr_native_value = coil.value
|
self._attr_native_value = coil.value
|
||||||
|
@ -17,9 +17,6 @@
|
|||||||
"address_in_use": "The selected listening port is already in use on this system.",
|
"address_in_use": "The selected listening port is already in use on this system.",
|
||||||
"model": "The model selected doesn't seem to support modbus40",
|
"model": "The model selected doesn't seem to support modbus40",
|
||||||
"unknown": "[%key:common::config_flow::error::unknown%]"
|
"unknown": "[%key:common::config_flow::error::unknown%]"
|
||||||
},
|
|
||||||
"abort": {
|
|
||||||
"already_configured": "[%key:common::config_flow::abort::already_configured_device%]"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,11 +1,9 @@
|
|||||||
{
|
{
|
||||||
"config": {
|
"config": {
|
||||||
"abort": {
|
|
||||||
"already_configured": "Device is already configured"
|
|
||||||
},
|
|
||||||
"error": {
|
"error": {
|
||||||
"address": "Invalid remote IP address specified. Address must be a IPV4 address.",
|
"address": "Invalid remote IP address specified. Address must be a IPV4 address.",
|
||||||
"address_in_use": "The selected listening port is already in use on this system. Reconfigure your gateway device to use a different address if the conflict can not be resolved.",
|
"address_in_use": "The selected listening port is already in use on this system.",
|
||||||
|
"model": "The model selected doesn't seem to support modbus40",
|
||||||
"read": "Error on read request from pump. Verify your `Remote read port` or `Remote IP address`.",
|
"read": "Error on read request from pump. Verify your `Remote read port` or `Remote IP address`.",
|
||||||
"unknown": "Unexpected error",
|
"unknown": "Unexpected error",
|
||||||
"write": "Error on write request to pump. Verify your `Remote write port` or `Remote IP address`."
|
"write": "Error on write request to pump. Verify your `Remote write port` or `Remote IP address`."
|
||||||
|
@ -91,6 +91,16 @@ async def test_address_inuse(hass: HomeAssistant, mock_connection: Mock) -> None
|
|||||||
assert result2["type"] == FlowResultType.FORM
|
assert result2["type"] == FlowResultType.FORM
|
||||||
assert result2["errors"] == {"listening_port": "address_in_use"}
|
assert result2["errors"] == {"listening_port": "address_in_use"}
|
||||||
|
|
||||||
|
error.errno = errno.EACCES
|
||||||
|
mock_connection.return_value.start.side_effect = error
|
||||||
|
|
||||||
|
result2 = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"], MOCK_FLOW_USERDATA
|
||||||
|
)
|
||||||
|
|
||||||
|
assert result2["type"] == FlowResultType.FORM
|
||||||
|
assert result2["errors"] == {"base": "unknown"}
|
||||||
|
|
||||||
|
|
||||||
async def test_read_timeout(hass: HomeAssistant, mock_connection: Mock) -> None:
|
async def test_read_timeout(hass: HomeAssistant, mock_connection: Mock) -> None:
|
||||||
"""Test we handle cannot connect error."""
|
"""Test we handle cannot connect error."""
|
||||||
|
Loading…
x
Reference in New Issue
Block a user